Темизация Drupal. Часть 3. Основы Drupal Forms API и темизация форм

Submitted by Ромка on Вс, 03/01/2010 - 22:58

Ромка аватар

Прежде чем говорить об изменении внешнего вида форм, ознакомимся с основами Drupal Forms API — программного интерфейса, используемого для генерации форм. Применение Forms API несколько сложнее создания HTML-форм вручную, так как требует изучения логики его работы, однако его использование обязательно, поскольку Forms API решает ряд важных задач:

  • любой разработчик может добавить или удалить элементы в форме, созданной другим разработчиком, не меняя ее исходного кода;
  • любой разработчик может добавить дополнительные функции проверки и обработки введенных пользователем данных без изменения исходной формы;
  • формы, созданные с использованием Forms API, защищены от атак, связанных с отправкой пользователем модифицированной формы;
  • любой разработчик может изменить внешний вид формы, не изменяя ее исходного кода.

Каждая форма в Drupal представляет собой функцию, возвращающую ассоциативный массив. Этот массив должен содержать информацию обо всех элементах формы, функциях проверки (валидаторы, validators) и обработки (сабмиттеры, submitters) введенных данных. Данная функция должна быть расположена в файле модуля, о разработке модуля говорилось в предыдущей статье.

На заметку: готовые решения

  • Темы оформления, позволяющие менять цветовую схему через интерфейс администратора: Pixture Reloaded, Dropshadow, Wabi, Garland (стандартная тема Drupal).
  • Темы оформления, обладающие множеством настроек и регионов: Deco, Acquia Marina.

Рассмотрим простой пример.

  1. function test_form($form_state) {
  2. $form["example_text_field"] = array(
  3. '#type' => 'textfield',
  4. '#title' => 'Example text field',
  5. );
  6. $options = array(
  7. 0 => 'zero',
  8. 1 => 'one',
  9. 2 => 'two',
  10. );
  11. $form["example_select"] = array(
  12. '#type' => 'select',
  13. '#title' => 'Example select list',
  14. '#options' => $options,
  15. '#description' => t('You can select only value "one" in this
  16. form'),
  17. );
  18. $form["submit"] = array(
  19. '#type' => 'submit',
  20. '#value' => t('Submit'),
  21. );
  22. return $form;
  23. }

Приведенная выше функция генерирует форму, состоящую из текстового поля, выпадающего списка с тремя элементами и кнопкой для отправки данных. Имя этой функции — ее уникальный идентификатор ($form_id), который будет использоваться для отображения и изменения данной формы сторонними модулями. Чтобы вывести форму на экран, нужно через hook_menu создать страницу, где будет вызвана функция drupal_get_form, принимающая в качестве параметра $form_id формы, которая должна быть отображена на экране:

  1. function имя_модуля_menu() {
  2. $items = array();
  3. $items['test-form'] = array(
  4. 'title' => 'Test form',
  5. 'page callback' => 'test_form_page',
  6. 'access arguments' => array('access content'),
  7. 'type' => MENU_NORMAL_ITEM,
  8. );
  9. return $items;
  10. }
  11. function test_form_page() {
  12. return drupal_get_form('test_form');
  13. }

В массиве, возвращаемом функцией test_form($form_state), не определены процедуры проверки значений и заполнения полей (структур), поэтому ядро Drupal после нажатия на форме кнопки Submit попробует найти и выполнить функции form_id_validate и form_id_submit. В нашем случае, как легко догадаться, это будут функции с именами test_form_validate и test_form_submit:

  1. function test_form_validate($form, &$form_state) {
  2. if($form_state['values']['example_select'] != 1) {
  3. form_set_error('example_select', t('You must select value
  4. "one" in select list :)'));
  5. }
  6. }
  7. function test_form_submit($form, &$form_state) {
  8. drupal_set_message('Form sumitted! Values:');
  9. drupal_set_message("textbox: " .
  10. $form_state['values']['example_text']);
  11. drupal_set_message("selectlist: " .
  12. $form_state['values']['example_select']);
  13. }

Функция-валидатор проверяет выбранное в выпадающем списке значение и, если оно не one, посылает сообщение об ошибке с указанием, какой элемент формы вызвал ошибку. Функция-сабмиттер выводит на экран введенные пользователем значения. В реальном случае эта функция должна будет сохранить данные в базе данных.

При желании программист может в массиве $form функции test_form задать свойство #submit, содержащее массив обработчиков значений и свойство #validate с массивом валидаторов (см. листинг 4). Оранжевыми комментариями выделены строки, добавленные к ранее описанным функциям.

Листинг 4

  1. function test_form($form_state) {
  2. $form["example_text_field"] = array(
  3. '#type' => 'textfield',
  4. '#title' => 'Example text field',
  5. '#description' => 'Text must contain more then
  6. 3 symbols',
  7. );
  8. $options = array(
  9. 0 => 'zero',
  10. 1 => 'one',
  11. 2 => 'two',
  12. );
  13. $form["example_select"] = array(
  14. '#type' => 'select',
  15. '#title' => 'Example select list',
  16. '#options' => $options,
  17. '#description' => t('You can select only value "one"
  18. in this form'),
  19. );
  20. $form["submit"] = array(
  21. '#type' => 'submit',
  22. '#value' => t('Submit'),
  23. );
  24. //Добавлено
  25. $form["#validate"] = array('test_validate_first',
  26. 'test_validate_second');
  27. $form["#submit"] = array('test_submit_first',
  28. 'test_submit_second');
  29. //Конец добавления
  30. return $form;
  31. }

Код модуля также дополнится функциями из листинга 5.

Листинг 5

  1. function test_validate_first($form, &$form_state) {
  2. if(mb_strlen($form_state['values']['example_text_field']) < 3) {
  3. form_set_error('example_text_field', t('Text must contain more then 3 symbols'));
  4. }
  5. }
  6. function test_validate_second($form, &$form_state) {
  7. if($form_state['values']['example_select'] != 1) {
  8. form_set_error('example_select', t('You must select value "one" in select list :)'));
  9. }
  10. }
  11. function test_submit_first($form, &$form_state) {
  12. drupal_set_message('First submitter');
  13. drupal_set_message("textbox: " . $form_state['values']['example_text']);
  14. }
  15. function test_submit_second($form, &$form_state) {
  16. drupal_set_message('Second submitter');
  17. drupal_set_message("selectlist: " . $form_state['values']['example_select']);
  18. }

Здесь добавлен валидатор, который проверяет текст, введенный в текстовое поле; если его длина оказывается меньше трех символов, то генерируется сообщение об ошибке.

При создании форм всегда рекомендуется использовать не стандартные валидаторы и сабмиттеры, а объявлять их явно, так как в этом случае сторонние программисты смогут дополнить массивы #submit и #validate своими функциями. Если используются стандартные валидаторы и сабмиттеры, то сторонние программисты смогут только заменить существующие функции своими, а это не всегда удобно.

Теперь вернемся к основной теме статьи — темизации Drupal. Функция drupal_get_form, получив на вход $form_id, идентификатор формы, которую нужно вывести на экран, вызывает функцию form_builder, проверяющую права доступа текущего пользователя к каждому из полей формы, и при наличии этих прав выводит стандартный HTML-код для каждого элемента формы. Каждый созданный элемент формы имеет уникальный атрибут id. Самый простой способ переопределения внешнего вида элементов формы — создание CSS-файла с описанием стилей нужных элементов формы. Если этого недостаточно, элементам формы можно добавить параметры #prefix и #suffix, которые будут содержать HTML-код, выводимый до и после созданного элемента. Если и этого мало, можно определить параметр #theme, который должен содержать используемое имя функции темизации.

Персональные функции темизации можно задать для всей формы целиком или для каждого ее элемента. По умолчанию Drupal для темизации формы пробует найти функцию theme_form_id, поэтому использование параметра #theme не обязательно.

Все функции, определенные параметром #theme, должны быть также объявлены через hook_theme (описание этого хука было дано в предыдущей статье).

Давайте изменим внешний вид созданного текстового поля. Для этого сначала создадим реализацию хука hook_theme:

  1. function название_модуля_theme() {
  2. return array(
  3. 'example_text_field_theme_function' => array(
  4. 'arguments' => array('form' => NULL),
  5. ),
  6. );
  7. }

Затем модифицируем массив $form["example_text_field"], который создается в функции test_form, добавив в него параметр #theme:

  1. $form["example_text_field"] = array(
  2. '#type' => 'textfield',
  3. '#title' => 'Example text field',
  4. '#description' => 'Text must contain more then 3 symbols',
  5. '#theme' => 'example_text_field_theme_function',
  6. );

Теперь мы можем объявить функцию theme_example_text_field_theme_function и задать в ней любой HTML-код для отображения выбранного элемента:

  1. function theme_example_text_field_theme_function($element) {
  2. $class = "";
  3. if(isset($element["#needs_validation"])) {
  4. $class = " error";
  5. }
  6. $output = '<div id="' . $element["#id"] . '"
  7. class="form-item"><input id="edit-example-text-field"
  8. class="form-text' . $class . '" name="' . $element["#name"]
  9. . '"></div>';
  10. return $output;
  11. }

Кроме того, эту функцию можно переопределить, не изменяя кода модуля. Для этого в файле template.php, который находится в папке с текущей темой оформления, нужно создать копию этой функции, заменив в ней префикс theme на имя текущей темы.

Если в реализации hook_theme использовать параметр template, например, так:

  1. function название_модуля_theme() {
  2. return array(
  3. 'example_text_field_theme_function' => array(
  4. 'arguments' => array('form' => NULL),
  5. 'template' => 'example-text-field',
  6. ),
  7. );
  8. }

то HTML-код, ответственный за отображение элемента Web-страницы в браузере, можно будет задавать не в исходном тексте функции темизации, а в отдельном файле-шаблоне с соответствующим именем; в нашем примере это example-text-field.tpl.php. Такой подход удобен, если с сайтом должны работать дизайнеры, не имеющие опыта Web-программирования.

Если же возникла необходимость изменить внешний вид всей формы, а не только отдельных ее элементов, нужно проделать то же самое: указать значение параметра #theme формы, объявить в hook_theme функцию темизации и, наконец, реализовать ее. Давайте внесем необходимые изменения в наш код. Функция hook_theme будет выглядеть следующим образом:

  1. function название_модуля_theme() {
  2. return array(
  3. 'test_form_theme_function' => array(
  4. 'arguments' => array('form' => NULL),
  5. ),
  6. 'example_text_field_theme_function' => array(
  7. 'arguments' => array('form' => NULL),
  8. ),
  9. );
  10. }

Исходный текст обновленной функции test_form приводится в листинге 6.

Листинг 6

  1. function test_form($form_state) {
  2. $form['#theme'] = 'test_form_theme_function';
  3. $form["example_text_field"] = array(
  4. '#type' => 'textfield',
  5. '#title' => 'Example text field',
  6. '#description' => 'Text must contain more then
  7. 3 symbols',
  8. '#theme' => 'example_text_field_theme_function',//*/
  9. );
  10. $options = array(
  11. 0 => 'zero',
  12. 1 => 'one',
  13. 2 => 'two',
  14. );
  15. $form["example_select"] = array(
  16. '#type' => 'select',
  17. '#title' => 'Example select list',
  18. '#options' => $options,
  19. '#description' => t('You can select only value "one"
  20. in this form'),
  21. );
  22. $form["submit"] = array(
  23. '#type' => 'submit',
  24. '#value' => t('Submit'),
  25. );
  26. $form["#validate"] = array('test_validate_first',
  27. 'test_validate_second');
  28. $form["#submit"] = array('test_submit_first',
  29. 'test_submit_second');
  30. return $form;
  31. }

Также нам потребуется и сама функция темизации формы. Определим ее:

  1. function theme_test_form_theme_function($form) {
  2. $output = "Some additional text";
  3. // Выводим некоторые элменты отдельно с дополнительным
  4. // форматированием
  5. $output .= '<div style="background-color: #ccc; padding:
  6. 3px;">';
  7. $output .= drupal_render($form['example_text_field']);
  8. $output .= "</div>";
  9. // Выводим остальные элементы, которые не были выведены
  10. // ранее
  11. $output .= drupal_render($form);
  12. return $output;
  13. }

Как и с любой другой функцией темизации, ее содержимое можно вынести во внешний шаблон.

Ну а теперь осталось научиться модифицировать из внешнего модуля существующую форму. Для решения этой задачи нужно воспользоваться одним из двух хуков: или hook_form_alter, через который проходят массивы всех обрабатываемых форм и в котором каждый массив можно отредактировать, или hook_form_FORM_ID_alter, где FORM_ID должен быть заменен на идентификатор нужной формы. Через этот хук проходит только выбранная форма. На вход оба этих хука получают массив формы (в нашем примере это массив, который генерируется функцией test_form), и в этот массив могут быть добавлены или из него могут быть удалены любые параметры: #theme, #prefix, #suffix, #submit, #validate и другие.

Для примера добавим к форме контейнер, который может содержать в себе несколько полей. Переместим в него два поля и сменим заголовок одного из них:

  1. function название_модуля_form_test_form_alter(&$form,
  2. &$form_state) {
  3. $form["example_text_field"]["#title"] = "New title";
  4. $form["example_add_field"] = array(
  5. '#type' => 'fieldset',
  6. '#title' => 'new fieldset',
  7. '#collapsible' => TRUE,
  8. '#collapsed' => FALSE,
  9. '#weight' => 0,
  10. );
  11. foreach ($form as $name => $element) {
  12. if($element["#type"] == "select" || $element["#type"] ==
  13. "textfield") {
  14. $form["example_add_field"][$name] = $element;
  15. unset($form[$name]);
  16. }
  17. }
  18. $form["submit"]["#weight"] = 5;
  19. }

Вот и все.

Ссылки на другие части этой статьи:

Содержание всех статей: http://romka.eu/blog/my-drupal-articles

15 Comments

Спасибо!

Спасибо!

Немного подробнее, плз

Как без form_set_error объявить форму непрошедшей валидацию?
Заранее спасибо за ответ :)

свои настройки в теме

Расскажите как создать свои кастомные настройки для темы оформления, чтобы в админке пользователь мог поменять ширину колонок, отображать или нет автора ноды, выбрать бэкграунд для сайта и т.п.
На русском статей я не нашел, на Д.орг есть немного инфы но недостаточно абсолютно http://drupal.org/node/177868
Темы от http://fusiondrupalthemes.com используют skinr и все эти настройки, темы для Джумлы от Рокетов и прочих используют это уже давно, а для Друпала все очень плохо...

щас использую

щас использую http://drupal.org/project/style_settings для настроек, пришлось повозиться с мануалом чтобы все настроить.
меня не сложность беспокоит а то что вот вышла уже бета 1 для Д7 а шаблонов на него вообще нет от коммерческих производителей, очень уж медленно происходит становление отрасли шаблонов, качество их намного ниже чем жумловских а цены выше (да да знаю что до беты апи не устоялся и нельзя было делать шаблоны)

Не совсем то

не совсем то что искал...

http://yandex.ru/yandsearch?clid=48648&yasoft=barff&text=%D0%90%D0%94%D0%A0%D0%95%D0%A1+api+%D0%A1%D0%95%D0%A0%D0%92%D0%95%D0%A0%D0%90+%D0%94%D0%9B%D0%AF+%D0%94%D0%A0%D0%A3%D0%9F%D0%90%D0%9B&lr=10849

Странный какой то пример:

Странный какой то пример: theme_test_form_theme_function - повторно пересоздавать вывод, когда он уже создан, чтобы сместить элемент вперед. Корректней наверное череp hook_form_alter '#weight' позадавать.

Извиняюсь, понял, что не

Извиняюсь, понял, что не прав. Просто без точного понимания работы функции drupal_render не понятна тема, а в статье это не объясняется.
Например, первое что приходит в голову: $form['#theme'] переопределит стандартную функцию тем для формы theme_form, что на самом деле не так.

с чем?

с чем?

Добрый день ! Есть такая

Добрый день !

Есть такая проблема.

Вывожу несколько одинаковых форм на странице.
Каждая форма генерируется одной и той же функцией test
То есть я для каждой формы вызываю drupal_get_form с уникальным id - test_1, test_2 etc
а затем переопределяю функцию создания формы в hook_forms
('callback' => 'test',)
В функции test я прописываю в форме
'#theme' => 'test'
и создаю соответственно функцию темы. theme_test

При выводе тема вызывается, но в нее передается пустая форма - то есть видимо оно пытается найти форму test , а у меня имеются формі test_1, test_2 и тд...

Каким образом можно еще темизировать одинаковые формы с различными ID одной функцией темы ?

Заранее спасибо !

Не везде понятно, где код

Не везде понятно, где код должен прописываться в некоторых листингах.

Поясните пожалуйста.

Куда вставлять хук?

Подскажите пожалуйста, многоуважаемый гуру следующие вещи:
1) куда именон, в какой файл мне вставить hook_form_alter()
2) как мне узнать имя формы модуля Contact? (хочу изменить автозаполнение текстового поля в форме обратной связи модуля contact)