質問
私のメニューの各項目がどのような種類のメニュー項目であるか、つまりそれがページ、カテゴリ、またはカスタムリンクであることを確認したいと思います。
達成したいこと
カテゴリとページのリンクからなるメニューを作成したいです。メニュー内のカテゴリのうち、投稿を含むカテゴリのみを表示します。ページリンクは常に表示されます。
このメニューは、eコマースサイトでカタログが絶えず変化するのに使用されます。空のカテゴリを隠す動的メニューを作成すると、管理領域のメニューを常に更新する必要がなくなります。
私が今持っているもの
この SE post に基づいたカスタムのウォーカーメニュー。これはstart_el
関数にロジックを追加することで空のカテゴリへのリンクをうまく隠します。残念ながら私のロジックはページのリンクも隠しています
フルウォーカーメニュー機能:
function cs_modify_nav_menu_args( $args ){
if( 'primary-menu' == $args['theme_location'] ){
$args['walker'] = new cs_walker_nav_menu();
}
return $args;
}
class cs_walker_nav_menu extends Walker_Nav_Menu {
// filter empty categories and add main/sub classes to li's and links
function start_el( &$output, $item, $depth = 0, $args = array(), $id = 0 ) {
global $wp_query;
if(!is_page($item->ID)){
$non_empty_categories = get_categories(array('taxonomy' => 'product_cat'));
$empty_categories = array();
$is_empty = true;
// check menu items are for an empty category
foreach ( $non_empty_categories as $cat )
if ($item->object_id === $cat->term_id)
$is_empty = false;
// if it is empty add it to array
if ($is_empty)
$empty_categories[] = $item->ID;
// Don't build nav item for items in the is_empty array
foreach( $empty_categories as $category_to_skip )
if( $item->ID == $category_to_skip )
return $output;
}
$indent = ( $depth > 0 ? str_repeat( "\t", $depth ) : '' ); // code indent
// depth dependent classes
$depth_classes = array(
( $depth == 0 ? 'main-menu-item' : 'sub-menu-item' ),
( $depth >=2 ? 'sub-sub-menu-item' : '' ),
( $depth % 2 ? 'menu-item-odd' : 'menu-item-even' ),
'menu-item-depth-' . $depth
);
$depth_class_names = esc_attr( implode( ' ', $depth_classes ) );
// passed classes
$classes = empty( $item->classes ) ? array() : (array) $item->classes;
$class_names = esc_attr( implode( ' ', apply_filters( 'nav_menu_css_class', array_filter( $classes ), $item ) ) );
// build html
$output .= $indent . '<li id="nav-menu-item-'. $item->ID . '" class="' . $depth_class_names . ' ' . $class_names . '">';
// link attributes
$attributes = ! empty( $item->attr_title ) ? ' title="' . esc_attr( $item->attr_title ) .'"' : '';
$attributes .= ! empty( $item->target ) ? ' target="' . esc_attr( $item->target ) .'"' : '';
$attributes .= ! empty( $item->xfn ) ? ' rel="' . esc_attr( $item->xfn ) .'"' : '';
$attributes .= ! empty( $item->url ) ? ' href="' . esc_attr( $item->url ) .'"' : '';
$attributes .= ' class="menu-link ' . ( $depth > 0 ? 'sub-menu-link' : 'main-menu-link' ) . '"';
$item_output = sprintf( '%1$s<a%2$s>%3$s%4$s%5$s</a>%6$s',
$args->before,
$attributes,
$args->link_before,
apply_filters( 'the_title', $item->title, $item->ID ),
$args->link_after,
$args->after
);
// build html
$output .= apply_filters( 'walker_nav_menu_start_el', $item_output, $item, $depth, $args );
}
}
試みられた解決策
nav_menu_item
オブジェクトはメニュー項目タイプを提供しません。
Is_pageに$item->ID
を渡しても戻りません。 $item->ID
は投稿IDではなくメニュー項目IDだと思います。
幸い、これに対する簡単な修正があります。
Walkerの$item
はClassesを提供します、そしてメニュー項目がアーカイブであるならば、あなたはそれらのうちの1つを得ます:
と同様
このロジックをさまざまなアーカイブ(product_cat
だけでなく)で機能させたい場合は、分類法も変更する必要があります - ここでも、$item
は便利にあなたの$分類法を$item->object
として提供します。
今あなたはあなたのif
とあなたのWalker
の分類法を変える必要があるだけです
if( $item->type == 'taxonomy' ) {
$non_empty_categories = get_categories( array( 'taxonomy' => $item->object ) );
// ... The rest of your logic works fine
}
うまく動作するはずです:)