API настроек для сети сайтов (мультисайт)
В WordPress есть специальное API, которое позволяет без шума и пыли создавать страницы настроек (опций), например настройки плагина или темы. Данные таких опций записываться в таблицу опций. API позволяет создавать такие страницы быстро, с защитой и в логике и дизайне самого WordPress, т.е. с использованием API не нужно отдельно заботиться о HTML коде и CSS стилях формы, о необходимой защите и в целом логичности всего кода - все это API берет на себя.
Теме API настроек посвящена отдельная статья, где подробно объясняется как работает и как использовать это самое API.
Эта заметка является дополнением к вышеупомянутой статье и рассказывает, как создать страницу настроек в меню «сети сайтов» (админка мультисайта).
Дело в том, что при активации MU режима появляются отдельные сайты с привычной админкой и появляется раздел «Управление сетью» (админка сети), где настраивается вся сеть.
Так вот, API опций по умолчанию не рассчитан на использование в настройках сети. Точнее использовать основную логику API там можно, но «кое что» нужно будет учитывать и доделывать. Об этом «кое что» поговорим в этой заметке.
Как использовать «API опций» WordPress на страницах админки сети сайтов (Мультисайт)
При попытке решить такую задачу с помощью Google, как обычно напоролся на костыльные и не изящные решения, каким текущее решение, я полагаю, не является.
Суть проблемы
В стандартном варианте API регистрируется опция с помощью register_setting() и её значения отправляются POST запросом в файл /wp-admin/options.php. Далее WordPress все делает за нас: получает, защищает, обрабатывает и сохраняет данные опций в таблицу wp_options. Однако в сети мультисайт опции нужно сохранять в другую таблицу - wp_sitemeta, а на это логика API не рассчитана... По этому вопросу был создан тикет, но решения из коробки так и не появилось, впрочем оно не особо нужно, потому что задача легко решается, если подумать.
Напомню о различиях. В MU сборке WordPress опции отдельного сайта обрабатываются функциями get_option() и update_option(), а опции сети обрабатываются через get_network_option()/get_site_option() и update_network_option()/update_site_option().
Принципиальная разница в том, что данные сохраняются в разные таблицы, и настройки сайта у каждого свои, а настройки сети - это общие настройки для всех сайтов.
Чтобы не писать тут кучу букв от которых голова пойдет кругом, я покажу разницу в сравнении. Для этого покажу как создается стандартная страница настроек и как такая же страница создается для настроек сети. Меняется около 10% кода...
Обычная страница настроек, как предполагает API опций:
<?php ## Создаем страницу настроек плагина add_action( 'admin_menu', 'add_plugin_page' ); function add_plugin_page(){ add_options_page( 'Настройки плагина', 'Мой плагин', 'manage_options', 'myplug_slug', 'options_page_html' ); } ## HTML код страницы function options_page_html(){ ?> <div class="wrap"> <h2><?php echo get_admin_page_title() ?></h2> <form action="options.php" method="POST"> <?php settings_fields( 'option_group' ); // скрытые защитные поля - nonce do_settings_sections( 'myplug_page' ); // секции с настройками (опциями). У нас она всего одна 'section_id' submit_button(); ?> </form> </div> <?php } ## Регистрируем настройки. Настройки будут храниться в массиве, а не одна настройка = одна опция add_action( 'admin_init', 'my_plugin_settings' ); function my_plugin_settings(){ // создаем опцию register_setting( 'option_group', 'option_name', 'sanitize_callback' ); // создаем секцию с полями add_settings_section( 'section_id', 'Основные настройки', '', 'myplug_page' ); // создаем поля для секции $opt_name = 'my_option'; add_settings_field( $opt_name, 'Моя опция', 'fill_field', 'myplug_page', 'section_id', $opt_name ); $opt_name = 'my_option_two'; add_settings_field( $opt_name, 'Моя вторая опция', 'fill_field', 'myplug_page', 'section_id', $opt_name ); } ## Заполняем опцию 1 function fill_field( $opt_name ){ $opts = get_option( 'option_name' ); $name_attr = "option_name[$opt_name]"; $val = isset( $opts[ $opt_name ] ) ? $opts[ $opt_name ] : null; if( $opt_name === 'my_option' ){ echo '<input type="text" name="'. $name_attr .'" value="'. esc_attr( $val ) .'" />'; } if( $opt_name === 'my_option_two' ){ echo '<label><input type="checkbox" name="'. $name_attr .'" value="1" '. checked( 1, $val, 0 ) .' /> галка или нет?</label>'; } } ## Очистка сохраняемых данных function sanitize_callback( $options ){ foreach( $options as $name => & $val ){ if( $name == 'my_option' ) $val = sanitize_text_field( $val ); if( $name == 'my_option_two' ) $val = intval( $val ); } return $options; }
В результате получим такую страницу:
Такая же страница настроек, только для сети сайтов:
<?php ## NETWORK - Создаем страницу настроек плагина add_action( 'network_admin_menu', 'add_plugin_page' ); function add_plugin_page(){ add_submenu_page( 'settings.php', 'Настройки плагина', 'Мой плагин', 'manage_options', 'myplug_slug', 'options_page_html' ); } ## HTML код страницы function options_page_html(){ ?> <div class="wrap"> <h2><?php echo get_admin_page_title() ?></h2> <form action="edit.php?action=myplug_options" method="POST"> <?php wp_nonce_field( 'myplug_nonce' ); // NETWORK - settings_fields() не подходит для мультисайта... do_settings_sections( 'myplug_page' ); // секции с настройками (опциями). У нас она всего одна 'section_id' submit_button(); ?> </form> </div> <?php } ## Регистрируем настройки. Настройки будут храниться в массиве, а не одна настройка = одна опция add_action( 'admin_init', 'my_plugin_settings' ); function my_plugin_settings(){ // NETWORK - ловим обновления опций через хук 'network_admin_edit_(action)' if( is_multisite() ) add_action( 'network_admin_edit_'.'myplug_options', 'myplug_options_update' ); // создаем опцию register_setting( 'option_group', 'option_name', 'sanitize_callback' ); // создаем секцию с полями add_settings_section( 'section_id', 'Основные настройки', '', 'myplug_page' ); // создаем поля для секции $opt_name = 'my_option'; add_settings_field( $opt_name, 'Моя опция', 'fill_field', 'myplug_page', 'section_id', $opt_name ); $opt_name = 'my_option_two'; add_settings_field( $opt_name, 'Моя вторая опция', 'fill_field', 'myplug_page', 'section_id', $opt_name ); } ## Заполняем опцию 1 function fill_field( $opt_name ){ $opts = get_site_option( 'option_name' ); // NETWORK - не get_option() $name_attr = "option_name[$opt_name]"; $val = isset( $opts[ $opt_name ] ) ? $opts[ $opt_name ] : null; if( $opt_name === 'my_option' ){ echo '<input type="text" name="'. $name_attr .'" value="'. esc_attr( $val ) .'" />'; } if( $opt_name === 'my_option_two' ){ echo '<label><input type="checkbox" name="'. $name_attr .'" value="1" '. checked( 1, $val, 0 ) .' /> галка или нет?</label>'; } } ## Очистка сохраняемых данных function sanitize_callback( $options ){ foreach( $options as $name => & $val ){ if( $name == 'my_option' ) $val = sanitize_text_field( $val ); if( $name == 'my_option_two' ) $val = intval( $val ); } return $options; } ## NETWORK - обновляем опции в БД function myplug_options_update(){ // nonce check check_admin_referer( 'myplug_nonce' ); update_site_option( 'option_name', wp_unslash( $_POST['option_name'] ) ); wp_redirect( network_admin_url( 'settings.php?page=myplug_slug&updated=true' ) ); exit; }
В результате получим такую страницу:
Отличия
Распишу все отличия, чтобы не мучиться и не искать их, некоторые из которых вы в результате можете и не заметить.
-
Хук network_admin_menu вместо admin_menu для реги страницы опций.
-
add_submenu_page( 'settings.php'
вместоadd_options_page(
- создаем подстраницу в другом меню. -
<form action="edit.php?action=myplug_options"
вместо<form action="options.php"
- отправляем POST запрос в файл /network/edit.php - это специальный файл для таких задач. -
wp_nonce_field() вместо settings_fields() - для защиты. Родная функция API тут не подходит...
-
get_site_option() вместо get_option() - сохраняем опции в другую таблицу.
- Добавился хук network_admin_edit_(action) и функция для этого хука - myplug_options_update(), которая сохраняет наши опции с помощью update_site_option(). Для сети автоматический перехват запроса и его обработка, которую предусматривает API, не работает.
А вот так эти отличия выглядят:
<?php ## NETWORK - Создаем страницу настроек плагина add_action( 'network_admin_menu', 'add_plugin_page' ); function add_plugin_page(){ add_submenu_page( 'settings.php', 'Настройки плагина', 'Мой плагин', 'manage_options', 'myplug_slug', 'options_page_html' ); } ## HTML код страницы function options_page_html(){ ?> <form action="edit.php?action=myplug_options" method="POST"> <?php wp_nonce_field( 'myplug_nonce' ); // NETWORK - settings_fields() не подходит для мультисайта... // ... ?> </form> <?php } ## Регистрируем настройки. Настройки будут храниться в массиве, а не одна настройка = одна опция add_action( 'admin_init', 'my_plugin_settings' ); function my_plugin_settings(){ // NETWORK - ловим обновления опций через хук 'network_admin_edit_(action)' if( is_multisite() ) add_action( 'network_admin_edit_'.'myplug_options', 'myplug_options_update' ); // ... } ## Заполняем опцию 1 function fill_field( $opt_name ){ $opts = get_site_option( 'option_name' ); // NETWORK - не get_option() // ... } ## NETWORK - обновляем опции в БД function myplug_options_update(){ // nonce check check_admin_referer( 'myplug_nonce' ); update_site_option( 'option_name', $wp_unslash( $_POST['option_name'] ) ); wp_redirect( network_admin_url( 'settings.php?page='. 'myplug_slug' .'&updated=true' ) ); exit; }
-
Успехов в создании MU сайтов!