プログラムでナビゲーションメニューとメニュー項目を追加する


41

API関数を使用して、新しいナビゲーションメニューを定義し、現在のテーマでそれを選択して、メニュー項目としていくつかのページを挿入します。これは、たとえばテーマのアクティブ化で実行されます。

ナビゲーションメニューとアイテムを手動でセットアップした後、データベースの挿入と更新をリバースエンジニアリングする(ある程度苦痛な)プロセスを経て、「footer-nav」はナビゲーションメニューIのスラッグIDです。 m作成:

if (!term_exists('footer-nav', 'nav_menu')) {

    $menu = wp_insert_term('Footer nav', 'nav_menu', array('slug' => 'footer-nav'));

    // Select this menu in the current theme
    update_option('theme_mods_'.get_current_theme(), array("nav_menu_locations" => array("primary" => $menu['term_id'])));

    // Insert new page
    $page = wp_insert_post(array('post_title' => 'Blog',
                                 'post_content' => '',
                                 'post_status' => 'publish',
                                 'post_type' => 'page'));

    // Insert new nav_menu_item
    $nav_item = wp_insert_post(array('post_title' => 'News',
                                     'post_content' => '',
                                     'post_status' => 'publish',
                                     'post_type' => 'nav_menu_item'));


    add_post_meta($nav_item, '_menu_item_type', 'post_type');
    add_post_meta($nav_item, '_menu_item_menu_item_parent', '0');
    add_post_meta($nav_item, '_menu_item_object_id', $page);
    add_post_meta($nav_item, '_menu_item_object', 'page');
    add_post_meta($nav_item, '_menu_item_target', '');
    add_post_meta($nav_item, '_menu_item_classes', 'a:1:{i:0;s:0:"";}');
    add_post_meta($nav_item, '_menu_item_xfn', '');
    add_post_meta($nav_item, '_menu_item_url', '');

    wp_set_object_terms($nav_item, 'footer-nav', 'nav_menu');
}

これはうまくいくようですが、:

  • 堅牢でエレガントな方法ですか?
  • これをすべて1行のコードで実行できる完全に明らかな何かを見逃していますか?

回答:


43

私はあなたを誤解しているかもしれませんが、なぜ使用しないのwp_create_nav_menu()ですか?

たとえば、BPがアクティブとして検出されたときにカスタムBuddyPressメニューを作成するためにこれを行います。

    $menuname = $lblg_themename . ' BuddyPress Menu';
$bpmenulocation = 'lblgbpmenu';
// Does the menu exist already?
$menu_exists = wp_get_nav_menu_object( $menuname );

// If it doesn't exist, let's create it.
if( !$menu_exists){
    $menu_id = wp_create_nav_menu($menuname);

    // Set up default BuddyPress links and add them to the menu.
    wp_update_nav_menu_item($menu_id, 0, array(
        'menu-item-title' =>  __('Home'),
        'menu-item-classes' => 'home',
        'menu-item-url' => home_url( '/' ), 
        'menu-item-status' => 'publish'));

    wp_update_nav_menu_item($menu_id, 0, array(
        'menu-item-title' =>  __('Activity'),
        'menu-item-classes' => 'activity',
        'menu-item-url' => home_url( '/activity/' ), 
        'menu-item-status' => 'publish'));

    wp_update_nav_menu_item($menu_id, 0, array(
        'menu-item-title' =>  __('Members'),
        'menu-item-classes' => 'members',
        'menu-item-url' => home_url( '/members/' ), 
        'menu-item-status' => 'publish'));

    wp_update_nav_menu_item($menu_id, 0, array(
        'menu-item-title' =>  __('Groups'),
        'menu-item-classes' => 'groups',
        'menu-item-url' => home_url( '/groups/' ), 
        'menu-item-status' => 'publish'));

    wp_update_nav_menu_item($menu_id, 0, array(
        'menu-item-title' =>  __('Forums'),
        'menu-item-classes' => 'forums',
        'menu-item-url' => home_url( '/forums/' ), 
        'menu-item-status' => 'publish'));

    // Grab the theme locations and assign our newly-created menu
    // to the BuddyPress menu location.
    if( !has_nav_menu( $bpmenulocation ) ){
        $locations = get_theme_mod('nav_menu_locations');
        $locations[$bpmenulocation] = $menu_id;
        set_theme_mod( 'nav_menu_locations', $locations );
    }

この機能については知りませんでした。はい、上記のコードがはるかに短くなると思います。API関数は、この場合のように低レベルであることが多いため、Codexを超えて実際のコードに飛び込む必要があると思います。ありがとう!
-julien_c

@julien_cこれが解決された場合、あなたの後に来る人がここでの経験から利益を得ることができるように、そのようにマークしてください。
mor7ifer

私は実際に実際にテストしたいので、私が望んでいることを確実にします。完了したらすぐに解決済みとしてマークすることを忘れないでください!
-julien_c

3
コーデックスにないこれらのような便利な関数
トムJノウェル

申し訳ありませんが、私の場合はうまくいくことを確認するのに長い時間がかかりました。回答が受け入れられました!また、カスタムリンクメニュー項目を定義している場合、ページリンクを定義するための回答を以下に追加しました(たとえば、URLの変更に対してより堅牢になります)。
julien_c

12

ZaMooseのanwserを補完するものとして、「カスタム」ではなく「ページタイプ」メニュー項目を作成する方法を次に示します。

wp_update_nav_menu_item($menu_id, 0, array('menu-item-title' => 'About',
                                           'menu-item-object' => 'page',
                                           'menu-item-object-id' => get_page_by_path('about')->ID,
                                           'menu-item-type' => 'post_type',
                                           'menu-item-status' => 'publish'));

たとえば、ページスラッグだけを知っていると仮定します。


9

私は受け入れられた答えにいくつかの問題があります-それは間違っていませんが、同じ質問をしたが同じことをしたかったので、私はいくつかの人々にとってより良い結果が得られると思う独自のコードを投稿しますコードの少ないもの。

まず、上記のコードは「URL」タイプのナビゲーション項目を作成しますが、これは一部の人には問題ありませんが、WordPressナビゲーションの重要な機能であるため、URLではなくPAGESにリンクします。 navアイテムタイプ。

また、投稿されたコードでは、子のフラット配列のみが処理されます。新しいnavアイテムを再帰的に宣言し、返されたメタデータ(主にループで作成された後のID)を格納する関数と、子を受け入れるためのパラメーターを作成しました。

編集するだけ$nav_items_to_addで、残りは再帰的に処理されます。各配列には3つの必須キーがあります。まず、配列キーはスラッグであるため、スラッグの'shop' => array( ... )あるページに必要なものはそうですshop['title']は、navアイテムがフロントエンドでラベル付けされる方法です。 pathページトップレベルの親であるならば、これはスラグと同一であり、そうであればWordPressのページ階層内のページへのパスは、あるshopの子だったhome、それは次のようになります'path' => 'home/shop'

最後のオプションの配列キーは['parent']、現在の配列の親として配列内の別のキーを宣言できる場所です。アイテムは再帰的に追加されるため、子を作成する前に親が存在する必要があることに注意することが重要です。これは、親navアイテムの宣言が子になる前に行われることを意味します。

    $locations = get_nav_menu_locations();

    if (isset($locations['primary_navigation'])) {
        $menu_id = $locations['primary_navigation'];

        $new_menu_obj = array();

        $nav_items_to_add = array(
                'shop' => array(
                    'title' => 'Shop',
                    'path' => 'shop',
                    ),
                'shop_l2' => array(
                    'title' => 'Shop',
                    'path' => 'shop',
                    'parent' => 'shop',
                    ),
                'cart' => array(
                    'title' => 'Cart',
                    'path' => 'shop/cart',
                    'parent' => 'shop',
                    ),
                'checkout' => array(
                    'title' => 'Checkout',
                    'path' => 'shop/checkout',
                    'parent' => 'shop',
                    ),
                'my-account' => array(
                    'title' => 'My Account',
                    'path' => 'shop/my-account',
                    'parent' => 'shop',
                    ),
                'lost-password' => array(
                    'title' => 'Lost Password',
                    'path' => 'shop/my-account/lost-password',
                    'parent' => 'my-account',
                    ),
                'edit-address' => array(
                    'title' => 'Edit My Address',
                    'path' => 'shop/my-account/edit-address',
                    'parent' => 'my-account',
                    ),
            );

    foreach ( $nav_items_to_add as $slug => $nav_item ) {
        $new_menu_obj[$slug] = array();
        if ( array_key_exists( 'parent', $nav_item ) )
            $new_menu_obj[$slug]['parent'] = $nav_item['parent'];
        $new_menu_obj[$slug]['id'] = wp_update_nav_menu_item($menu_id, 0,  array(
                'menu-item-title' => $nav_item['title'],
                'menu-item-object' => 'page',
                'menu-item-parent-id' => $new_menu_obj[ $nav_item['parent'] ]['id'],
                'menu-item-object-id' => get_page_by_path( $nav_item['path'] )->ID,
                'menu-item-type' => 'post_type',
                'menu-item-status' => 'publish')
        );
    }

    }

2

プログラムでメニュー項目を追加するには。フックしてwp_nav_menu_itemsフィルタリングできます。テーマfunctions.phpのコードの下に配置して、メインメニューにログイン/ログアウトメニュー項目を追加します。「プライマリ」は、登録されているメニューの名前/ IDです。

/**
 * Add login logout menu item in the main menu.
 * ===========================================
 */

add_filter( 'wp_nav_menu_items', 'lunchbox_add_loginout_link', 10, 2 );
function lunchbox_add_loginout_link( $items, $args ) {
    /**
     * If menu primary menu is set & user is logged in.
     */
    if ( is_user_logged_in() && $args->theme_location == 'primary' ) {
        $items .= '<li><a href="'. wp_logout_url() .'">Log Out</a></li>';
    }
    /**
     * Else display login menu item.
     */
    elseif ( !is_user_logged_in() && $args->theme_location == 'primary' ) {
        $items .= '<li><a href="'. site_url('wp-login.php') .'">Log In</a></li>';
    }
    return $items;
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.