Можно ли заменить более одного элемента формы (оболочки), запущенного только одним триггерным элементом #ajax?

52
function ajax_example_simplest($form, &$form_state) {

  //This is my ajax trigger element
  $form['element_trigger'] = array(
    '#type' => 'select',
    '#options' => array(
      'one' => 'one',
      'two' => 'two',
      'three' => 'three',
    ),
    '#ajax' => array(
      'callback' => 'ajax_example_simplest_callback',

      /** Q: Can I somehow declare more than one wrapper? **/
      //Say for instance, something like:
      'wrapper' => array('replace_div_1', 'replace_div_2'),

     ),
  );

  //replace_div_1
  $form['element_to_be_replaced_1'] = array(
    '#type' => 'textfield',
    '#title' => t("My conditional field one"),
    '#prefix' => '<div id="replace_div_1">',
    '#suffix' => '</div>',
  );


 //... more form elements here

  //replace_div_2
  $form['element_to_be_replaced_2'] = array(
    '#type' => 'textfield',
    '#title' => t("My conditional field two"),
    '#prefix' => '<div id="replace_div_2">',
    '#suffix' => '</div>',
  );
  return $form;
}

function ajax_example_simplest_callback($form, $form_state) {

  //... do my stuff here


  //normally I would return only the form bit for replacing a single wrapper
  //declared in the trigger element, like this:
  return $form['element_to_be_replaced_blahblah'];

}

Можно ли вернуть более одного бита формы в функцию обратного вызова, сообщающую платформе AJAX, которая $form['element_to_be_replaced_1']должна заменить <div id="replace_div_1">и $form['element_to_be_replaced_2']должна заменить <div id="replace_div_2">?

Франциско Луз
источник

Ответы:

73

Вместо того, чтобы возвращать HTML отдельного элемента для обновления, ваш обратный вызов ajax может возвращать массив команд ajax . Поэтому он может вернуть два ajax_command_replace для замены каждого элемента.

function ajax_example_simplest_callback(&$form, $form_state) {
  return array(
    '#type' => 'ajax',
    '#commands' => array(
      ajax_command_replace("#replace_div_1", render($form['element_to_be_replaced_1'])),
      ajax_command_replace("#replace_div_2", render($form['element_to_be_replaced_2']))
    )
  );
}
Пьер Буйль
источник
2
Это круто! Сэкономил много времени. Большое спасибо :)
Сандеш Ядав
Щедрость прибывает на вашем пути через 24 часа.
АйешК
Пришлось изменить имя функции обратного вызова. Использование функции ajax_example_simplest_callback (& ​​$ form, $ form_state) {} дало мне белый экран смерти.
Готи
5

Drupal 8 альтернативный синтаксис

use Drupal\Core\Ajax\AjaxResponse;
use Drupal\Core\Ajax\ReplaceCommand;

class name extends FormBase{
   function ajax_example_simplest(array $form, FormStateInterface &$form_state) {
       $response = new AjaxResponse();
       $response->addCommand(new ReplaceCommand("#replace_div_1", ($form['element_to_be_replaced_1'])));
       $response->addCommand(new ReplaceCommand("#replace_div_2", ($form['element_to_be_replaced_2'])));
       return $response;
   }
}

Одним из различий является то, что команда рендеринга отброшена, потому что AjaxResponse реализует Drupal \ Core \ Render \ AttachmentsInterface

визуализация ($ form ['element_to_be_replaced_1'])

Добавление рендера все еще работает, но у меня были проблемы при обновлении таблицы TableSelect таким образом.

Johann
источник
Спасибо, это работает. Я думаю, что было бы неразумно использовать render (), потому что документация Drupal гласит, что он дублирован и не будет доступен в Drupal 9
Ngatia Frankline
4

Ответ Пьера Байля не сработал для меня. Однако что-то вроде следующего сработало.

function ajax_example_simplest_callback(&$form, $form_state) {
    $commands = array();
    $commands[] = ajax_command_replace("#replace_div_1", render($form['element_to_be_replaced_1']));
    $commands[] = ajax_command_replace("#replace_div_2", render($form['element_to_be_replaced_2']));
    $page = array('#type' => 'ajax', '#commands' => $commands);
    ajax_deliver($page);
}

Обратите внимание на вызов ajax_deliver () , а не на возвращение массива команд AJAX.

койот
источник
2
Я предполагаю, что вы используете #ajax ['path'] в вашей форме. В этом случае в вашей реализации hook_menu вы должны использовать свойство «доставить обратный вызов», это даст Drupal команду отображать результат обратного вызова вашей страницы как команду AJAX вместо разметки. Но используя его прямо в обратном вызове вашей страницы, вы обходите обычную доставку страниц Drupal.
Пьер Байл