पृष्ठ पदानुक्रम में एक कस्टम पोस्ट प्रकार को एकीकृत करना


14

मैं टीम के सदस्यों के लिए एक कस्टम पोस्ट प्रकार के साथ एक थीम बना रहा हूं, मुझे निम्न पृष्ठ संरचना भी मिली है:

about  <-- this is a page
about/team-members  <-- this is a page, lists all the team members
about/team-members/joe-bloggs  <-- this is a custom post type (team member) entry

यहां तीसरी संरचना के बारे में और टीम के सदस्य पृष्ठों का उपयोग किया जाता है, लेकिन यह कस्टम पोस्ट प्रकार के स्लग का उपयोग करने के लिए जाता है ताकि यह देख सकें कि यह माता-पिता टीम के सदस्य हैं और इसके बारे में हैं। मैंने कस्टम पोस्ट प्रकार पर निम्न विकल्प सेट करके इसे प्राप्त किया है:

...
'rewrite' => array( 'slug' => 'about/team-members', 'with_front' => false)
...

यह बहुत अच्छा काम करता है, हालाँकि जब मैं टीम के सदस्य के बाद के स्तर पर उतरता हूँ तो मुझे वर्तमान-पृष्ठ, वर्तमान-पूर्वजों की कक्षाएं मूल पृष्ठों पर नहीं मिलती हैं। मुझे पता है कि ऐसा क्यों है, क्योंकि हम तकनीकी रूप से उन पेजों के पगले माता-पिता पर नहीं हैं, हालांकि एक ऐसा तरीका है जिससे मैं छल / फिक्स / चकमा दे सकता हूं ताकि माता-पिता के रूप में क्या न हो?

मैंने टीम के सदस्यों के लिए पृष्ठों का उपयोग करके इसे अच्छी तरह से हासिल किया था, हालांकि व्यवस्थापक के लिए उपयोग में आसान के बजाय एक कस्टम पोस्ट प्रकार चुना गया था।

धन्यवाद दोस्तों + लड़कियों!


आपको अपने कस्टम पोस्ट प्रकार post_parent के रूप में टीम के सदस्यों के पेज आईडी को सेट करने की आवश्यकता है।
बैनटर्नेट

register_post_typeदस्तावेज़ में मुझे वह विकल्प दिखाई नहीं देता , क्या आप सहायता कर सकते हैं?
बेन एवरार्ड

जवाबों:


6

जब आप पृष्ठों के साथ काम कर रहे हों, तो आप मूल पृष्ठ का चयन कर सकते हैं और post_parentडेटाबेस में बाल पृष्ठ के क्षेत्र में मूल पृष्ठ आईडी नंबर के रूप में उस मान को सहेजा जाता है ।

आपके मामले में, आप एक कस्टम पोस्ट प्रकार का उपयोग कर रहे हैं, इसलिए आपको मूल पृष्ठ के लिए अपना स्वयं का मेटाबेस बनाने की आवश्यकता होगी; कुछ इस तरह:

/* Define the custom box */
add_action('add_meta_boxes', 'child_cpt_add_custom_box');

/* Adds a box to the main column on the custom post type edit screens */
function child_cpt_add_custom_box() {
    add_meta_box('child_cpt', __( 'My child_cpt parent'),'team_member_inner_custom_box','team_member');
}

/* Prints the box content */
function team_member_inner_custom_box() {
    global $post;
    // Use nonce for verification
    wp_nonce_field( plugin_basename(__FILE__), 'team_member_inner_custom_box' );
    echo 'Select the parent page';
    $mypages = get_pages();
    echo '<select name="cpt_parent">';
    foreach($mypages as $page){     
        echo '<option value="'.$page->ID.'"';
        if ($page->ID == $post->post_parent) {echo ' selected';}
        echo '>"'.$page->post_title.'</option>';
    }
    echo '</select>';
}
/* Do something with the data entered */
add_action('wp_insert_post_data', 'myplugin_save_postdata');

/* When the post is saved, saves our custom data */
function myplugin_save_postdata( $data, $postarr ) {
    global $post;
      // verify this came from the our screen and with proper authorization,
      // because save_post can be triggered at other times

      if ( !wp_verify_nonce( $_POST['team_member_inner_custom_box'], plugin_basename(__FILE__) ) )
          return $data;

      // verify if this is an auto save routine. 
      // If it is our form has not been submitted, so we dont want to do anything
      if ( defined('DOING_AUTOSAVE') && DOING_AUTOSAVE ) 
          return $data;
      // OK, we're authenticated: we need to find and save the data

      if ($post->post_type == "team_member")
          $data['post_parent'] = $_POST['cpt_parent'];

     return $data;
}

इसका कोई लेना-देना नहीं है register_post_type। आप यह सोचकर वर्डप्रेस को धोखा दे रहे हैं कि यह दूसरे पोस्ट टाइप (पेज) का चाइल्ड पेज है।


1
राइटो, इसलिए मैं देख सकता हूं कि यह "मूर्ख" वर्डप्रेस एक विशिष्ट पृष्ठ सोचने के लिए कैसे है, यह माता-पिता है, हालांकि यह पृष्ठ के मूल पृष्ठ को मूल पृष्ठ में नहीं जोड़ रहा है जब मैं wp_list_pages
बेन एवरार्ड

1
मैंने देखा है कि यह मेरी स्लग / पर्मलिंक संरचना के साथ भी खिलवाड़ करता है ...: एस
बेन एवरार्ड

2
मैं बेन के रूप में एक ही चीज को प्राप्त करने की कोशिश कर रहा हूं, लेकिन मैं उपयोग wp_nav_menuकरता हूं - post_parent के बारे में / टीम के सदस्य हैं लेकिन नेविगेशन मेरे "सामान्य" ब्लॉग पोस्ट के मूल आइटम को हाइलाइट करता है ... कोई अन्य विचार है कि मैं इसे कैसे ठीक कर सकता हूं?
pkyeck

@BenEverard: क्या आपको पर्मलिंक संरचना की गड़बड़ी का समाधान मिला?
अबुगम 22

0

मैं कुछ समान हासिल करने के लिए एक कस्टम वॉकर के साथ गया था ... कस्टम फ़ील्ड के लिए आवश्यकताओं से बचा जाता है, लेकिन एक प्रकार के सभी पदों को पेज ट्री में एक ही बिंदु से नीचे बैठना पड़ता है।

class Walker_Page_CustomPostTypeHack extends Walker_Page {
    function walk($elements, $max_depth) {
        $called_with = func_get_args();
        // current page is arg 3... see walk_page_tree for why
        $current_page = $called_with[3];

        // if there's no parent - see if we can find one.
        // some ACF options would be an easy way to make this configurable instad of constants
        if ($current_page === 0) {
            global $wp_query;
            $current_post = $wp_query->get_queried_object();
            switch ($current_post->post_type) {
                case 'course':
                    $current_page = POST_COURSES;
                    break;
                case 'project':
                    $current_page = POST_PROJECTS;
                    break;
                case 'story':
                    $current_page = POST_STORIES;
                    break;
            }
        }

        // now pass on into parent
        $called_with[3] = $current_page;
        return call_user_func_array(array('parent', 'walk'), $called_with);
    }

}

0

डिस्क्लेमर: इसे आजमाने के बाद यह मेरे लिए मौजूदा समस्या नहीं है, क्योंकि - कम से कम मेरे लिए - यह सिर्फ मेरे WP 3.9 पर काम करता है। हालांकि एक बग ट्रैकर के अनुसार नहीं मिल सका।


इसका परीक्षण करने के लिए मेरे पास थोड़ा सा प्लगइन है, जो किसी की मदद कर सकता है। लेकिन जैसा कि मैंने उपरोक्त अस्वीकरण में कहा था, मैं एक मौजूदा वर्डप्रेस इंस्टॉलेशन में समस्या को पुन: उत्पन्न नहीं कर सकता। मैंने प्लगइन को चार फाइलों में अलग कर दिया है, वे प्लगइन निर्देशिका के अंदर एक निर्देशिका में एक साथ जा रहे हैं।

प्लगइन- cpt_menu_hierarchy.php :

<?php
defined( 'ABSPATH' ) OR exit;
/**
 * Plugin Name: CPT Menu Hierarchy Fix?
 * Description: CPT Menu Hierarchy Fix?
 * Author:      ialocin
 * Author URL:  http://wordpress.stackexchange.com/users/22534/ialocin
 * Plugin URL:  http://wordpress.stackexchange.com/q/13308/22534
 */

// registering nonsense post type
include 'include-register_post_type.php';

// adding meta box to nosense custom post type
include 'include-cpt_parent_meta_box.php';

// menu highlighting fix
include 'include-menu_highlighting.php';

शामिल करें-register_post_type.php :

<?php
defined( 'ABSPATH' ) OR exit;

// See: http://codex.wordpress.org/Function_Reference/register_post_type
add_action( 'init', 'wpse13308_basic_reigister_post_type');
function wpse13308_basic_reigister_post_type() {
    $args = array(
        'public' => true,
        'label'  => 'Nonsense'
    );
    register_post_type( 'nonsense', $args );
}

शामिल करें- cpt_parent_meta_box.php :

<?php
defined( 'ABSPATH' ) OR exit;

// pretty much like @bainternet's answer

// Add Meta Box
add_action( 'add_meta_boxes', 'nonsense_add_meta_box' );
function nonsense_add_meta_box() {
    add_meta_box(
        'nonsense',
        __( 'Nonsense parent' ),
        'nonsense_inner_meta_box',
        'nonsense'
    );
}

// Meta Box Content
function nonsense_inner_meta_box() {
    global $post;

    wp_nonce_field(
        plugin_basename( __FILE__ ),
        'nonsense_inner_meta_box'
    );
    echo 'Parent Page:&nbsp;&nbsp;';
    $mypages = get_pages();
    echo '<select name="cpt_parent">';
    foreach($mypages as $page){     
        echo '<option value="'.$page->ID.'"';
        if ($page->ID == $post->post_parent) {echo ' selected';}
        echo '>'.$page->post_title.'</option>';
    }
    echo '</select>';
}

// Save Data From Meta Box
add_action( 'wp_insert_post_data', 'nonsense_save_meta_box_data' );
function nonsense_save_meta_box_data( $data, $postarr ) {
    global $post;

    if (
        ! wp_verify_nonce(
            $_POST['nonsense_inner_meta_box'],
            plugin_basename( __FILE__ )
        )
    ) {
        return $data;
    }

    if (
        defined('DOING_AUTOSAVE')
        && DOING_AUTOSAVE
    ) {
        return $data;
    }

    if ( $post->post_type == 'nonsense' ) {
        $data['post_parent'] = $_POST['cpt_parent'];
    }
    return $data;
}

शामिल करें- menu_highlighting.php :

<?php
defined( 'ABSPATH' ) OR exit;

// altering WordPress' nav menu classes via »nav_menu_css_class« filter
add_filter( 'nav_menu_css_class', 'wpse13308_fix_nav_menu_highlighting', 10, 2 );
function wpse13308_fix_nav_menu_highlighting( $classes, $item ) {
    // data of the current post
    global $post;

    // setting up some data from the current post
    $current_post_post_type = $post->post_type;
    $current_post_parent_id = $post->post_parent;
    // id of the post the current menu item represents
    $current_menu_item_id   = $item->object_id;

    // do this for a certain post type
    if( $current_post_post_type == 'nonsense' ) {
        // remove unwanted highlighting class via array_filter and callback
        // http://php.net/manual/de/function.array-filter.php
        $classes = array_filter(
            $classes,
            'wpse13308_remove_highlighting_classes'
        );
        // when the parents id equals the menu items id, we want to
        // highlight the parent menu item, so we check for:
        if( $current_post_parent_id == $current_menu_item_id ) {
            // use the css class used for highlighting
            $classes[] = 'replace-with-css-class';
        }
    }
    return $classes;
}

// callback to remove highlighting classes
function wpse13308_remove_highlighting_classes( $class ) {
    return
        (
            // use the class(es) you need, overview over nav menu item css classes:
            // http://codex.wordpress.org/Function_Reference/wp_nav_menu#Menu_Item_CSS_Classes
            $class == 'highlight-class'
            // uncomment next line if you want to check for more then one class
            // repeat the line if you want to check for a third, fourth and so on
            // || $class == 'replace-with-css-class'
        ) 
        ? false
        : true
    ;
}



  • यह कुछ हद तक सामान्यीकृत कोड उदाहरण है।
  • इसे वास्तविक उपयोग के मामले में फिट किया जाना है।

0

एक संभावित समाधान यह है कि जब भी कस्टम पोस्ट प्रकार सहेजा जाता है, तो आप अपने 'माता-पिता को about/team-membersव्यावहारिक रूप से सेट कर सकते हैं ।

यहाँ कदम हैं:

  1. जब भी कोई पोस्ट को सहेजने का प्रयास करता है, तो आप save_post हुक का उपयोग कर सकते हैं ।
  2. यदि वह पोस्ट कस्टम पोस्ट प्रकार है जिसके बाद आप हैं, तो आगे बढ़ें।
  3. कस्टम पोस्ट के माता-पिता को अपने इच्छित पेज पर सेट करना सुनिश्चित करें (जब तक आप इसे डिलीट नहीं करते हैं, तब तक आप आईडी को हार्ड-कोड कर सकते हैं)। आप माता-पिता को बचाने के लिए wp_update_post का उपयोग कर सकते हैं (मैंने खुद इसकी कोशिश नहीं की है, लेकिन मैं यह नहीं देखता कि यह काम क्यों नहीं करना चाहिए)।

मैं इसके लिए कुछ कोड देखना चाहूंगा! यह सही होगा, लेकिन मैं इसे काम करने में असमर्थ हूं।
जोहान डाहल

0

मेरे पास खुद को खोदने के लिए कुछ और समय था (क्षमा करें यदि मैंने किसी का समय बर्बाद किया है), और मुझे लगा कि मेरे लिए, हाइलाइटिंग समस्या को हल करने का सबसे अच्छा तरीका थोड़े समय के लिए फिर से करना होगा _wp_menu_item_classes_by_context(), जो कि सभी पर है। मेनू आइटम के माता-पिता और पूर्वज, जो मेरे कस्टम पोस्ट प्रकार के माता-पिता के रूप में कार्य करते हैं, और उचित रूप से कक्षाएं जोड़ते हैं।

चूँकि मैं अपने कस्टम पोस्ट प्रकार के लिए मूल पृष्ठ को भी नियत करना चाहता था, और मूल परिवर्तन के बाद सभी पोस्ट को अपडेट किए बिना आसानी से परिवर्तनशील हो सकता है, इसलिए मैंने post_parentअपने कस्टम पोस्ट प्रकार के पोस्ट के क्षेत्र को आबाद करने के बजाय एक विकल्प का उपयोग करने का निर्णय लिया है । मैंने इसके लिए ACF का उपयोग किया है क्योंकि मैं इसे अपनी थीम में वैसे भी उपयोग कर रहा हूं, लेकिन डिफ़ॉल्ट वर्डप्रेस विकल्प कार्यक्षमता का उपयोग करना निश्चित रूप से यह भी करेगा।

अपनी आवश्यकताओं के लिए मैं wp_nav_menu_objectsफ़िल्टर का उपयोग कर सकता था । इसके अतिरिक्त मुझे विकल्प को फ़िल्टर करनाpage_for_posts पड़ा ताकि यह गलत / खाली मान लौटाए, यह डिफ़ॉल्ट पोस्ट पृष्ठ को हाइलाइट करने से भी बचता है।

ध्यान दें कि मैं सभी तरह से नहीं गया, फ़िल्टर केवल current-menu-ancestorऔर current-menu-parentकक्षाएं जोड़ता है , क्योंकि यह मेरी आवश्यकताओं के लिए पर्याप्त था!

/**
 * Filters the `page_for_posts` option on specific custom post types in
 * order to avoid the wrong menu item being marked as
 * `current-page-parent`.
 *
 * @see _wp_menu_item_classes_by_context()
 */
function wpse13308_pre_option_page_for_posts_filter()
{
    $types = array
    (
        'my_custom_post_type_x',
        'my_custom_post_type_y',
        'my_custom_post_type_z'
    );
    if(in_array(get_post_type(), $types))
    {
        return 0;
    }
    return false;
}
add_filter('pre_option_page_for_posts', 'wpse13308_pre_option_page_for_posts_filter');


/**
 * Returns the current posts parent page ID
 *
 * @return int
 */
function wpse13308_get_parent_page_id()
{
    $postType = get_post_type();
    $parentPageId = null;
    switch($postType)
    {
        case 'my_custom_post_type_x':
        case 'my_custom_post_type_y':
        case 'my_custom_post_type_z':
            $parentPageId = (int)get_field('page_for_' . $postType, 'options')->ID;
            break;

        case 'post':
            $parentPageId = (int)get_option('page_for_posts');
            break;
    }
    return $parentPageId;
}

/**
 * Adds proper context based classes so that the parent menu items are
 * being highlighted properly for custom post types and regular posts.
 *
 * @param array $menuItems
 * @return array
 *
 * @see _wp_menu_item_classes_by_context()
 */
function wpse13308_wp_nav_menu_objects_filter(array $menuItems)
{
    $parentPageId = wpse13308_get_parent_page_id();

    if($parentPageId !== null)
    {
        $activeAncestorItemIds = array();
        $activeParentItemIds = array();
        foreach($menuItems as $menuItem)
        {
            if((int)$parentPageId === (int)$menuItem->object_id)
            {
                $ancestorId = (int)$menuItem->db_id;

                while
                (
                    ($ancestorId = (int)get_post_meta($ancestorId, '_menu_item_menu_item_parent', true)) &&
                    !in_array($ancestorId, $activeAncestorItemIds)
                )
                {
                    $activeAncestorItemIds[] = $ancestorId;
                }
                $activeParentItemIds[] = (int)$menuItem->db_id;
            }
        }
        $activeAncestorItemIds = array_filter(array_unique($activeAncestorItemIds));
        $activeParentItemIds = array_filter(array_unique($activeParentItemIds));

        foreach($menuItems as $key => $menuItem)
        {
            $classes = $menuItems[$key]->classes;
            if(in_array(intval($menuItem->db_id), $activeAncestorItemIds))
            {
                $classes[] = 'current-menu-ancestor';
                $menuItems[$key]->current_item_ancestor = true;
            }

            if(in_array($menuItem->db_id, $activeParentItemIds))
            {
                $classes[] = 'current-menu-parent';
                $menuItems[$key]->current_item_parent = true;
            }

            $menuItems[$key]->classes = array_unique($classes);
        }
    }

    return $menuItems;
}
add_filter('wp_nav_menu_objects', 'wpse13308_wp_nav_menu_objects_filter');

पूर्णता की खातिर, जब विकल्प का उपयोग करने के बजाय पॉपुलेटिंगpost_parent ( @ Bainternet का उत्तर देखें ), तो मूल आईडी को पुनः प्राप्त करने से कुछ ऐसा दिखाई दे सकता है:

/**
 * Returns the current posts parent page ID
 *
 * @return int
 */
function wpse13308_get_parent_page_id()
{
    $parentPageId = null;
    $post = get_post();
    switch($post->post_type)
    {
        case 'my_custom_post_type_x':
        case 'my_custom_post_type_y':
        case 'my_custom_post_type_z':
            $parentPageId = (int)$post->post_parent;
            break;

        case 'post':
            $parentPageId = (int)get_option('page_for_posts');
            break;
    }
    return $parentPageId;
}

आपने मेरा समय बर्बाद नहीं किया है :) एक और बात, यकीन है कि यह अभी भी एक समस्या है? क्योंकि मेरे WP 3.9.2 स्थापना पर मैं इसे पुन: पेश नहीं कर सका। सही मेनू आइटम को हाइलाइट करना बॉक्स से बाहर काम करता है।
निकोलई

हाँ, यह निश्चित रूप से अभी भी एक समस्या है @ialocin। क्या ऐसा हो सकता है कि आप इसे 0 स्तर मेनू और डिफ़ॉल्ट पोस्ट प्रकार के साथ परीक्षण कर रहे हैं?
ndm

नहीं, मेरे उत्तर में पोस्ट किए गए कोड के साथ इसे आज़माया। तो एक कस्टम पोस्ट प्रकार के साथ और 1 और 2 के स्तर मेनू आइटम के अनुसार एक पृष्ठ के अनुसार पोस्ट प्रकार से। मैंने इसका परीक्षण करने के लिए वर्डप्रेस कोर बंडल विषयों का उपयोग किया।
निकोलाई

@ialocin सुनिश्चित नहीं है कि अगर मैं आपको सही तरीके से समझता हूं, क्योंकि " पोस्ट किए गए कोड के साथ " और " आउट ऑफ द बॉक्स " टाइप किया गया है? ;) क्या आप केवल कस्टम पोस्ट प्रकार की बात कर रहे हैं, हाइलाइटिंग फिक्स की नहीं?
ndm

सही :) ठीक है, सटीक होने के लिए, परिदृश्य के लिए एक सीपीटी की आवश्यकता है, इसलिए निश्चित रूप से मैंने इसे पंजीकृत किया। हाइलाइटिंग मेटा बॉक्स और हाइलाइटिंग फिक्स के उपयोग के बिना काम करता है। उदाहरण के लिए एक मेनू संरचना के साथ: दादा-दादी (पृष्ठ)> माता-पिता (पृष्ठ)> कुछ (पोस्ट)> दूसरी चीज (cpt)> एक-और-चीज (cpt) - प्रत्येक तत्व को सही css वर्ग (तों) मिलता है; यहाँ इस्तेमाल किया विषय बीस तेरह।
निकोलाई

-1
<?php
the_post();

// $postType holds all the information of the post type of the current post you are viewing
$postType = get_post_type_object(get_post_type());

// $postSlug is the slug you defined in the rewrite column: about/team-members
$postSlug = $postType->rewrite['slug'];

// $datas = { [0] => 'about', [1] => 'team-members' }
$datas = explode('/', $postSlug);

// $pageSlug = 'about'
$pageSlug = $datas[0];

// all the page information you require.
$page = get_page_by_path($pageSlug, OBJECT, 'page');
?>

http://codex.wordpress.org/Function_Reference/get_post_type_object http://codex.wordpress.org/Function_Reference/get_page_by_path

संपादित करें 1:

चूंकि संकेत काम नहीं करते हैं:

add_filter('wp_nav_menu_objects', 'my_menu_class_edit');
function my_menu_class_edit($items)
{
    if (is_single()) {
        $postType = get_post_type_object(get_post_type());
        $postSlug = $postType->rewrite['slug'];
        if($postSlug  != 'about/team-members')
            return $items;
        $datas = explode('/', $postSlug);
        $pageAbout = get_page_by_path($datas[0], OBJECT, 'page');
        $pageTeamMembers = get_page_by_path($datas[1], OBJECT, 'page');

        foreach ($items as $item) {
            if ($item->title == $pageAbout->post_title) {
                $item->classes[] = 'current-ancestor';
            } else if ($item->title == $pageTeamMembers->post_title) {
                $item->classes[] = 'current-page';
            }
        }
   }
    return $items;
}

तुम वहाँ जाओ। इसे wp_nav_menu_objects फ़िल्टर हुक के भीतर जोड़ा गया।
17
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.