Добавление специального метабокса к произвольному типу записи
Мне удалось добавить метабокс к моему произвольному типу записи (см. код ниже). Теперь мне нужно добавить еще один метабокс. С помощью этого бокса я хочу отображать/редактировать значение из другой таблицы под названием wp_project_bids_mitglied. Эта таблица содержит строку с ID записей и строку со значениями (0,1,2), которые я хотел бы редактировать/отображать в админ-панели. Как мне нужно изменить код, чтобы достичь этой цели? Спасибо!
function add_custom_meta_box() {
add_meta_box(
'custom_meta_box', // $id
'Продолжительность', // $title
'show_custom_meta_box', // $callback
'project', // $page
'normal', // $context
'high'); // $priority
}
add_action('add_meta_boxes', 'add_custom_meta_box');
// Массив полей
$prefix = 'custom_';
$custom_meta_fields = array(
array(
'label'=> 'Поле выбора',
'desc' => 'Указать продолжительность фотозадания',
'id' => $prefix.'dauer',
'type' => 'select',
'options' => array (
'one' => array (
'label' => '1-3',
'value' => 'a'
),
'two' => array (
'label' => '3-6',
'value' => 'b'
),
'three' => array (
'label' => '6-9',
'value' => 'c'
),
'four' => array (
'label' => '>9',
'value' => 'd'
)
)
)
);
// Функция обратного вызова
function show_custom_meta_box() {
global $custom_meta_fields, $post;
// Используем nonce для верификации
echo '<input type="hidden" name="custom_meta_box_nonce" value="'.wp_create_nonce(basename(__FILE__)).'" />';
// Начинаем таблицу полей и цикл
echo '<table class="form-table">';
foreach ($custom_meta_fields as $field) {
// получаем значение этого поля, если оно существует для этой записи
$meta = get_post_meta($post->ID, $field['id'], true);
// начинаем строку таблицы
echo '<tr>
<th><label for="'.$field['id'].'">'.$field['label'].'</label></th>
<td>';
switch($field['type']) {
// сюда добавятся case элементы
// текст
case 'select':
echo '<select name="'.$field['id'].'" id="'.$field['id'].'">';
foreach ($field['options'] as $option) {
echo '<option', $meta == $option['value'] ? ' selected="selected"' : '', ' value="'.$option['value'].'">'.$option['label'].'</option>';
}
echo '</select><br /><span class="description">'.$field['desc'].'</span>';
break;
} //конец switch
echo '</td></tr>';
} // конец foreach
echo '</table>'; // конец таблицы
}
// Сохранение данных
function save_custom_meta($post_id) {
global $custom_meta_fields;
// проверка nonce
if (!wp_verify_nonce($_POST['custom_meta_box_nonce'], basename(__FILE__)))
return $post_id;
// проверка автосохранения
if (defined('DOING_AUTOSAVE') && DOING_AUTOSAVE)
return $post_id;
// проверка разрешений
if ('page' == $_POST['post_type']) {
if (!current_user_can('edit_page', $post_id))
return $post_id;
} elseif (!current_user_can('edit_post', $post_id)) {
return $post_id;
}
// проходим по полям и сохраняем данные
foreach ($custom_meta_fields as $field) {
$old = get_post_meta($post_id, $field['id'], true);
$new = $_POST[$field['id']];
if ($new && $new != $old) {
update_post_meta($post_id, $field['id'], $new);
} elseif ('' == $new && $old) {
delete_post_meta($post_id, $field['id'], $old);
}
} // конец foreach
}

Весь этот код предназначен для демонстрации процесса, а не для предоставления готового решения. Вам нужно адаптировать его под свои нужды. Я добавил комментарии к каждой строке — обратите на них внимание для лучшего понимания.
Шаг #1
Сначала создаем пользовательский метабокс с помощью этого кода:
<?php
//создаем метабокс (Примечание: метабокс != пользовательское поле)
function wpse_add_custom_meta_box_2() {
add_meta_box(
'custom_meta_box-2', // $id - идентификатор
'Длительность2', // $title - заголовок
'show_custom_meta_box_2', // $callback - функция вывода
'project', // $page - тип записи
'normal', // $context - расположение
'high' // $priority - приоритет
);
}
add_action('add_meta_boxes', 'wpse_add_custom_meta_box_2');
?>
Шаг #2
Теперь, когда метабокс готов, добавляем поля формы для ввода данных пользователя. Для этого используем функцию $callback
, которую объявили ранее:
<?php
//выводим пользовательские поля формы
function show_custom_meta_box_2() {
global $post;
// Используем nonce для проверки безопасности
wp_nonce_field( basename( __FILE__ ), 'wpse_our_nonce' );
?>
<!-- Поле для ввода значения -->
<input type="number" name="wpse_value" value="">
<?php
}
?>
Шаг #3
При сохранении записи данные из полей формы будут переданы методом POST. Теперь нам нужно сохранить их в нужном месте.
<?php
//сохраняем данные
function wpse_save_meta_fields( $post_id ) {
// проверяем nonce
if (!isset($_POST['wpse_our_nonce']) || !wp_verify_nonce($_POST['wpse_our_nonce'], basename(__FILE__)))
return 'nonce не подтвержден';
// проверяем автосохранение
if ( wp_is_post_autosave( $post_id ) )
return 'автосохранение';
// проверяем ревизию записи
if ( wp_is_post_revision( $post_id ) )
return 'ревизия';
// проверяем права пользователя
if ( 'project' == $_POST['post_type'] ) {
if ( ! current_user_can( 'edit_page', $post_id ) )
return 'недостаточно прав для редактирования страницы';
} elseif ( ! current_user_can( 'edit_post', $post_id ) ) {
return 'недостаточно прав для редактирования записи';
}
// получаем переданные данные из формы
$wpse_value = $_POST['wpse_value'];
// сохраняем данные
global $wpdb;
$table = $wpdb->base_prefix . 'project_bids_mitglied';
$wpdb->insert(
$table,
array(
'col_post_id' => $post_id, //идентификатор записи
'col_value' => intval( $wpse_value ) //значение (предполагаем числовое)
),
array(
'%d', //%s - строка, %d - число, %f - число с плавающей точкой
'%d', //%s - строка, %d - число, %f - число с плавающей точкой
)
);
}
add_action( 'save_post', 'wpse_save_meta_fields' );
add_action( 'new_to_publish', 'wpse_save_meta_fields' );
?>
Так как мы работаем с пользовательской таблицей, используем класс $wpdb для безопасного сохранения данных.
Обратите внимание: это не готовый код, а демонстрация процесса. Запомните два ключевых момента:
show_custom_meta_box_2()
— это место для вашей HTML-формы, рассматривайте этот участок как обычную HTML-форму.wpse_save_meta_fields()
, подключенная к соответствующим хукам, сохранит данные формы при публикации/сохранении/обновлении записи. Не забудьте о валидации и санитизации данных.
Надеюсь, это поможет. Благодарность @toscho за обсуждение. <3
