Користувальницький ходунок Nav відображає поточні пункти меню дітей або братів і сестер на дітей, які не мають дітей


14

Я годинами возився / шукав і досі не можу змусити це працювати, тому я нарешті поступаюся і прошу допомогу.

Я намагаюся написати користувальницьку прогулянкову програму, яка показує лише дітям поточні сторінки, або якщо дітей немає, вони відображають побратимів сторінок.

Наприклад, візьміть таке дерево меню:

  • 1,0
    • 1.2.0
      • 1.3.0
      • 1.3.1
      • 1.3.2
    • 1.2.1
    • 1.2.2
  • 2.0

Припустимо, що я на поточній сторінці 1.2.0. На цій сторінці я хочу показати дітей (1.3.0, 1.3.1, 1.3.2)

однак, якщо я на сторінці 1.2.2, оскільки у неї немає дітей, він повинен відображати своїх побратимів на поточному рівні, тому він повинен мені показувати (1.2.0, 1.2.1, 1.2.2).


4
Будь ласка, перенесіть своє рішення на відповідь, щоб воно було більш зрозумілим, а питання не переслідували сайт як невідповідальний.
Рарст

Що сказав @Rarst! Я ледь не пропустив, що ти придумав рішення.
Кріс Кричо

Некро відповідь. Я запитав більш-менш те саме питання щодо SO близько 2 років тому з дуже хорошою відповіддю. stackoverflow.com/questions/5826609 / ...
Stoosh

Відповідь переміщено всередині питання, щоб отримати окрему відповідь. ОП: Будь ласка, слідкуйте за цим.
кайзер

Відповіді:


4

Це ходунок, який я використовував для відображення лише дітей поточного пункту меню. Або пункти меню братів і сестер, якщо в них немає дітей.

У всьому класі є коментарі, що пояснюють кожен розділ

<?php

class SH_Child_Only_Walker extends Walker_Nav_Menu {

private $ID;
private $depth;
private $classes = array();
private $child_count = 0;
private $have_current = false;


// Don't start the top level
function start_lvl(&$output, $depth=0, $args=array()) {

    if( 0 == $depth || $this->depth != $depth )
        return;

    parent::start_lvl($output, $depth,$args);
}

// Don't end the top level
function end_lvl(&$output, $depth=0, $args=array()) {
    if( 0 == $depth || $this->depth != $depth )
        return;

    parent::end_lvl($output, $depth,$args);
}

// Don't print top-level elements
function start_el(&$output, $item, $depth=0, $args=array()) {

    $is_current = in_array('current-menu-item', $this->classes);

    if( 0 == $depth || ! $is_current )
        return;

    parent::start_el($output, $item, $depth, $args);
}

function end_el(&$output, $item, $depth=0, $args=array()) {
    if( 0 == $depth )
        return;

    parent::end_el($output, $item, $depth, $args);
}

// Only follow down one branch
function display_element( $element, &$children_elements, $max_depth, $depth=0, $args, &$output ) {

    // Check if element is in the current tree to display
    $current_element_markers = array( 'current-menu-item', 'current-menu-parent', 'current-menu-ancestor' );
    $this->classes = array_intersect( $current_element_markers, $element->classes );

    // If element has a 'current' class, it is an ancestor of the current element
    $ancestor_of_current = !empty($this->classes);

    // check if the element is the actual page element we are on.
    $is_current = in_array('current-menu-item', $this->classes);

    // if it is the current element
    if($is_current) {

        // set the count / ID / and depth to use in the other functions.
        $this->child_count = ( isset($children_elements[$element->ID]) ) ? count($children_elements[$element->ID]) : 0;
        $this->ID = $element->ID;
        $this->depth = $depth;
        $this->have_current = true;

        if($this->child_count > 0) {

            // if there are children loop through them and display the kids.
            foreach( $children_elements[$element->ID] as $child ) {
                parent::display_element( $child, $children_elements, $max_depth, $depth, $args, $output );
            }

        } else {
            // no children so loop through kids of parent item.
            foreach( $children_elements[$element->menu_item_parent] as $child ) {
                parent::display_element( $child, $children_elements, $max_depth, $depth, $args, $output );
            }

        }
    }

    // if depth is zero and not in current tree go to the next element
    if ( 0 == $depth && !$ancestor_of_current)
        return;

    // if we aren't on the current element proceed as normal
    if(! $this->have_current )
        parent::display_element( $element, $children_elements, $max_depth, $depth, $args, $output );
}
}

Прикріпіть його так, як і будь-який інший користувальницький ходунок на wp_nav_menu

<?php
wp_nav_menu( array(
    'menu' => 'primary-menu'
    ,'container' => 'nav'
    ,'container_class' => 'subpages'
    ,'depth' => 0
    ,'walker' => new SH_Child_Only_Walker()
 ));
?>

Я хочу вказати на коментар @ Стооша, що вказує тут. stackoverflow.com/questions/5826609/…, оскільки це ще одне хороше рішення
jchamb

0

У мене був подібний досвід. Можливо, ви захочете подумати над тим, як перемістити логіку сторінок з ходунки. В основному, складіть поточну ієрархію сторінок як об'єкт. Потім використовуйте параметр 'виключити' у функції wp_nav_menu. Тепер виключені сторінки залежатимуть від того, чи є поточна сторінка дітьми. Якщо дітей не показують брати; якщо діти && ці діти в кінці рядка, покажіть братів та дітей; якщо діти && та внуки існують, виключайте братів та показуйте дітей та онуків.


Який цей excludeпараметр, на який ви посилаєтесь? Я переглядаю документацію і не бачу жодної посилання на неї.
Кріс Кричо

1
Прошу вибачення, що помилився. Ви правильні, що немає параметра "виключити". Я мав на увазі використовувати функцію "wp_list_pages".
Стів Фішер

Дуже добре, і ніяких турбот. Мені було просто цікаво, чи є щось незадокументоване, але ззаду - я бачив, як це відбувалося раніше. Дякуємо, що очистили його! Я не думав використовувати wp_list_pages()в цьому контексті, тому це цікава ідея.
Кріс Кричо
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.