管理ダッシュにカスタム投稿タイプを追加し、投稿編集ウィンドウにカスタムメタボックスを追加する次のコードがあります。
function teasers_custom_init() {
$labels = array(
'name' => 'Teasers',
'singular_name' => 'Teaser',
'add_new' => 'Add New',
'add_new_item' => 'Add New Teasers',
'edit_item' => 'Edit Teaser',
'new_item' => 'New Teaser',
'all_items' => 'All Teasers',
'view_item' => 'View Teaser',
'search_items' => 'Search Teasers',
'not_found' => 'No teasers found',
'not_found_in_trash' => 'No teasers found in Trash',
'parent_item_colon' => 'Parent Page',
'menu_name' => 'Teasers'
);
$args = array(
'labels' => $labels,
'description' => 'set slider panels with loop times',
'public' => true,
'publicly_queryable' => true,
'show_ui' => true,
'show_in_menu' => true,
'query_var' => true,
'rewrite' => array( 'slug' => 'Teasers' ),
'capability_type' => 'page',
'has_archive' => true,
'hierarchical' => true,
'menu_position' => 60,
'supports' => array( 'title', 'thumbnail', 'page-attributes'),
);
register_post_type( 'teasers', $args );
}
add_action( 'init', 'teasers_custom_init' );
//adding the meta box when the admin panel initialises
add_action("admin_init", "admin_init");
// this adds the save teaser function on save post
add_action('save_post', 'save_teaser');
function admin_init(){
add_meta_box('teaser_loop', 'Loop Time', 'loop_meta', 'teasers', 'normal', 'default');
}
// callback function of add meta box that displays the meta box in the post edit screen
function loop_meta($post, $args){
$teaser_loop = get_post_meta($post->ID, 'teaser_loop', true);
?>
<label>Teaser Loop: </label><input type="text" name="teaser_loop" value="<?php echo $teaser_loop; ?>" /><br/>
<?php
}
// saving the teaser
function save_teaser(){
global $post;
update_post_meta($post->ID, 'teaser_loop', $_POST['teaser_loop']);
}
私の質問は、メタボックスを追加したい場合、最善のアプローチは何ですか?
Admin_init関数に別のadd_meta_box呼び出しを追加し、このメタボックスhtmlの追加のコールバック関数も作成しようとしましたが、フロントエンドで何も生成されませんでした。どんなポインタも素晴らしいでしょう。
編集:これは私が複数のメタボックスに対してそれを行った方法です(これは機能します):
//adding the meta box when the admin panel initialises
add_action("admin_init", "admin_init");
// this adds the save teaser function on save post
add_action('save_post', 'save_teaser');
function admin_init(){
add_meta_box('teaser_loop', 'Loop Time', 'loop_meta_1', 'teasers', 'normal', 'default');
add_meta_box('teaser_link', 'Teaser Link', 'loop_meta_2', 'teasers', 'normal', 'default');
}
// back function of add meta box that displays the meta box in the post edit screen
function loop_meta_1($post, $args){
$teaser_loop = get_post_meta($post->ID, 'teaser_loop', true);
?>
<label>Teaser Loop: </label><input type="text" name="teaser_loop" value="<?php echo $teaser_loop; ?>" /><br/>
<?php
}
function loop_meta_2($post, $args){
$teaser_link = get_post_meta($post->ID, 'teaser_link', true);
?>
<label>Teaser Link: </label><input type="text" name="teaser_link" value="<?php echo $teaser_link; ?>" /><br/>
<?php
}
// saving the teaser
function save_teaser(){
global $post;
update_post_meta($post->ID, 'teaser_loop', $_POST['teaser_loop']);
update_post_meta($post->ID, 'teaser_link', $_POST['teaser_link']);
}
クラスに完全にカプセル化できます。ここでは、カスタム投稿タイプの追加については扱っていません。テキストとチェックボックスの2つの単純な出力フィールドしかありません。本格的な作業コードは、必要な入力タイプのそれぞれを処理する必要があります。
<?php
/**
* Plugin Name: Sample Dynamic Meta Boxes
* Plugin URI: http://stackoverflow.com/q/13903529/1287812
* Author: brasofilo
*/
class B5F_Dynamic_Meta_Boxes
{
private $boxes;
# Safe to start up
public function __construct( $args )
{
$this->boxes = $args;
add_action( 'plugins_loaded', array( $this, 'start_up' ) );
}
public function start_up()
{
add_action( 'add_meta_boxes', array( $this, 'add_mb' ) );
}
public function add_mb()
{
foreach( $this->boxes as $box )
add_meta_box(
$box['id'],
$box['title'],
array( $this, 'mb_callback' ),
$box['post_type'],
isset( $box['context'] ) ? $box['context'] : 'normal',
isset( $box['priority'] ) ? $box['priority'] : 'default',
$box['args']
);
}
# Callback function, uses helper function to print each meta box
public function mb_callback( $post, $box )
{
switch( $box['args']['field'] )
{
case 'textfield':
$this->textfield( $box, $post->ID );
break;
case 'checkbox':
$this->checkbox( $box, $post->ID );
break;
}
}
private function textfield( $box, $post_id )
{
$post_meta = get_post_meta( $post_id, $box['id'], true );
printf(
'<label>%s: <input type="text" name="%s" value="%s" /></label> <small>%s</small><br/>',
$box['title'],
$box['id'],
$post_meta,
$box['args']['desc']
);
}
private function checkbox( $box, $post_id )
{
$post_meta = get_post_meta( $post_id, $box['id'], true );
printf(
'<label>%s: </label><input type="checkbox" name="%s" %s /> <small>%s</small><br/>',
$box['title'],
$box['id'],
checked( 1, $post_meta, false ),
$box['args']['desc']
);
}
}
# ADD TWO META BOXES - DIFFERENT POST TYPES - DIFFERENT CONTEXTS AND PRIORITIES
$args = array(
array(
'id' => 'teaser_loop',
'title' => 'Loop Time',
'post_type' => 'post',
'args' => array(
'desc' => 'Enter the time',
'field' => 'textfield',
)
),
array(
'id' => 'teaser_link',
'title' => 'Loop Link',
'post_type' => 'page',
'context' => 'side',
'priority' => 'high',
'args' => array(
'desc' => 'Open link',
'field' => 'checkbox',
)
),
);
new B5F_Dynamic_Meta_Boxes( $args );
# ADD ANOTHER META BOX TO ANOTHER POST TYPE
$more_args = array(
array(
'id' => 'extra_box',
'title' => 'And another one',
'post_type' => 'teaser',
'args' => array(
'desc' => 'Open link',
'field' => 'textfield',
)
),
);
new B5F_Dynamic_Meta_Boxes( $more_args );
これは単なるスケルトンであり、ここから書くべきことがたくさんあります。いくつかの例:
次のコードを使用して、メタボックスをカスタム投稿タイプに追加できます。
まず、メタボックスを作成します
add_action('admin_init', 'my_theme_on_admin_init');
function my_theme_on_admin_init() {
add_meta_box('my_metabox',
__('My metabox', 'textdomain'),
'my_metabox_render',
'my_custom_post_type', 'normal', 'high'
);
}
my_custom_post_type
はカスタム投稿タイプの名前です。my_metabox_render
-メタボックスをレンダリングする関数の名前。
レンダリング機能は、すべての必須フィールドを作成する必要があります
function my_metabox_render($post) {
$data = get_post_meta($post->ID, '_meta_key', true);
// Use nonce for verification
wp_nonce_field('add_my_meta', 'my_meta_nonce');
?>
<div class="inside">
<table class="form-table">
<tr valign="top">
<th scope="row"><label for="my_meta_value"><?php _e('My meta', 'textdomain'); ?></label></th>
<td><textarea id="my_meta_value" name="my_meta_value" cols="40" rows="5"><?php echo (isset($data)) ? $data : ''; ?></textarea></td>
</tr>
</table>
</div>
<?php
}
ユーザーが投稿を保存するときにメタデータを更新する必要があるより
add_action('wp_insert_post', 'save_my_meta', 10, 2);
function save_my_meta($id) {
if ( defined('DOING_AUTOSAVE') && DOING_AUTOSAVE )
return $id;
if (!current_user_can('edit_posts'))
return;
if (!isset($id))
$id = (int) $_REQUEST['post_ID'];
if (isset($_POST['my_meta_value']) && wp_verify_nonce($_POST['my_meta_value'], 'add_my_meta')) {
$data = $_POST['my_meta_value'];
if (isset($data)) {
update_post_meta($id, '_meta_key', $data);
}
else {
delete_post_meta($id, '_meta_key');
}
}
}
これは2つのメタボックスを追加する非常に不適切な方法ですか?それは動作しますが、非効率的なコード/ phpかもしれないと思います。オリジナルは https://typerocket.com に投稿されました
// Add form and basic text field
function press_meta_box(WP_Post $post) {
add_meta_box('press_meta', 'Press Release Date', function() use ($post) {
$field_name = 'press_date';
$field_name2 = 'press_link';
$field_value = get_post_meta($post->ID, $field_name, true);
$field_value2 = get_post_meta($post->ID, $field_name2, true);
wp_nonce_field('study_nonce', 'study_nonce');
?>
<table class="form-table">
<tr>
<th> <label for="<?php echo $field_name; ?>">Press Release Date (MM/DD/YYYY)</label></th>
<td>
<input id="<?php echo $field_name; ?>"
name="<?php echo $field_name; ?>"
type="text"
value="<?php echo esc_attr($field_value); ?>" />
</td>
</tr>
<tr>
<th> <label for="<?php echo $field_name2; ?>">Full link to pdf</label></th>
<td>
<input id="<?php echo $field_name2; ?>"
name="<?php echo $field_name2; ?>"
type="text"
value="<?php echo esc_attr($field_value2); ?>" />
</td>
</tr>
</table>
<?php
});
}
// Check for empty string allowing for a value of `0`
function empty_str( $str ) {
return ! isset( $str ) || $str === "";
}
// Save and delete meta but not when restoring a revision
add_action('save_post', function($post_id){
$post = get_post($post_id);
$is_revision = wp_is_post_revision($post_id);
$field_name = 'press_date';
$field_name2 = 'press_link';
// Do not save meta for a revision or on autosave
if ( $post->post_type != 'press_release' || $is_revision )
return;
// Do not save meta if fields are not present,
// like during a restore.
if( !isset($_POST[$field_name]) )
return;
// Secure with nonce field check
if( ! check_admin_referer('study_nonce', 'study_nonce') )
return;
// Clean up data
$field_value = trim($_POST[$field_name]);
$field_value2 = trim($_POST[$field_name2]);
// Do the saving and deleting
if( ! empty_str( $field_value ) ) {
update_post_meta($post_id, $field_name, $field_value);
} elseif( empty_str( $field_value ) ) {
delete_post_meta($post_id, $field_name);
}
// Do the saving and deleting 2
if( ! empty_str( $field_value2 ) ) {
update_post_meta($post_id, $field_name2, $field_value2);
} elseif( empty_str( $field_value2 ) ) {
delete_post_meta($post_id, $field_name2);
}
});