Добавление категорий в пользовательский тип записи в постоянную ссылку


22

Я знаю, что люди спрашивали об этом раньше и дошли до того, что добавили пользовательский тип сообщения и переписали для постоянной ссылки.

Проблема в том, что у меня есть 340 существующих категорий, которые я хотел бы продолжить использовать. Я имел обыкновение видеть / category / subcategory / postname

Теперь у меня есть кусок customposttype / postname. Выбор категории больше не отображается в постоянной ссылке ... Я не изменил настройку постоянной ссылки в админке на что-то другое.

Что-то мне не хватает или нужно добавить к этому коду?

function jcj_club_post_types() {
    register_post_type( 'jcj_club', array(
        'labels' => array(
            'name' => __( 'Jazz Clubs' ),
            'singular_name' => __( 'Jazz Club' ),
            'add_new' => __( 'Add New' ),
            'add_new_item' => __( 'Add New Jazz Club' ),
            'edit' => __( 'Edit' ),
            'edit_item' => __( 'Edit Jazz Clubs' ),
            'new_item' => __( 'New Jazz Club' ),
            'view' => __( 'View Jazz Club' ),
            'view_item' => __( 'View Jazz Club' ),
            'search_items' => __( 'Search Jazz Clubs' ),
            'not_found' => __( 'No jazz clubs found' ),
            'not_found_in_trash' => __( 'No jazz clubs found in Trash' ),
            'parent' => __( 'Parent Jazz Club' ),
        ),
        'public' => true,
        'show_ui' => true,
        'publicly_queryable' => true,
        'exclude_from_search' => false,
        'menu_position' => 5,
        'query_var' => true,
        'supports' => array( 
            'title',
            'editor',
            'comments',
            'revisions',
            'trackbacks',
            'author',
            'excerpt',
            'thumbnail',
            'custom-fields',
        ),
        'rewrite' => array( 'slug' => 'jazz-clubs-in', 'with_front' => true ),
        'taxonomies' => array( 'category','post_tag'),
        'can_export' => true,
    )
);

2
это может быть глупый вопрос, но вы переписали свои переписывания?
Кристина Чайлдс

В последнее время я сталкиваюсь с этой проблемой. Решено! [# 188834] [1] [1]: wordpress.stackexchange.com/questions/94817/…
maheshwaghmare

Ответы:


16

При добавлении пользовательских правил перезаписи типов записей необходимо учитывать две точки атаки:

Переписать правила

Это происходит, когда правила перезаписи генерируются wp-includes/rewrite.phpв WP_Rewrite::rewrite_rules(). WordPress позволяет фильтровать правила перезаписи для определенных элементов, таких как записи, страницы и различные типы архивов. Где вы видите часть должна быть именем вашего пользовательского почтового типа. В качестве альтернативы вы можете использовать фильтр, если вы не стираете стандартные правила публикации.posttype_rewrite_rulesposttypepost_rewrite_rules

Далее нам нужна функция для генерации правил перезаписи:

// add our new permastruct to the rewrite rules
add_filter( 'posttype_rewrite_rules', 'add_permastruct' );

function add_permastruct( $rules ) {
    global $wp_rewrite;

    // set your desired permalink structure here
    $struct = '/%category%/%year%/%monthnum%/%postname%/';

    // use the WP rewrite rule generating function
    $rules = $wp_rewrite->generate_rewrite_rules(
        $struct,       // the permalink structure
        EP_PERMALINK,  // Endpoint mask: adds rewrite rules for single post endpoints like comments pages etc...
        false,         // Paged: add rewrite rules for paging eg. for archives (not needed here)
        true,          // Feed: add rewrite rules for feed endpoints
        true,          // For comments: whether the feed rules should be for post comments - on a singular page adds endpoints for comments feed
        false,         // Walk directories: whether to generate rules for each segment of the permastruct delimited by '/'. Always set to false otherwise custom rewrite rules will be too greedy, they appear at the top of the rules
        true           // Add custom endpoints
    );

    return $rules;
}

Главное, на что следует обратить внимание, если вы решите поиграть, это логическое значение «Walk каталоги». Он генерирует правила перезаписи для каждого сегмента permastruct и может вызвать несоответствия правил перезаписи. При запросе URL-адреса WordPress массив правил перезаписи проверяется сверху вниз. Как только совпадение найдено, оно будет загружать все, что встречалось, например, если у вашей permastruct есть жадное совпадение, например. /%category%/%postname%/каталоги for и walk включены и выведут правила перезаписи для /%category%/%postname%/AND, /%category%/которые будут соответствовать чему угодно. Если это произойдет слишком рано, вы облажались.

Permalinks

Это функция, которая анализирует постоянные ссылки типа записей и преобразует постоянную структуру (например, '/% year% /% monthnum% /% postname% /') в фактический URL.

Следующая часть представляет собой простой пример того, что в идеале должно быть версией get_permalink()функции, найденной в wp-includes/link-template.php. Пользовательские постоянные ссылки на посты генерируются по get_post_permalink()более размытой версии get_permalink(). get_post_permalink()фильтруется, post_type_linkпоэтому мы используем это для создания пользовательской пермаструктуры.

// parse the generated links
add_filter( 'post_type_link', 'custom_post_permalink', 10, 4 );

function custom_post_permalink( $permalink, $post, $leavename, $sample ) {

    // only do our stuff if we're using pretty permalinks
    // and if it's our target post type
    if ( $post->post_type == 'posttype' && get_option( 'permalink_structure' ) ) {

        // remember our desired permalink structure here
        // we need to generate the equivalent with real data
        // to match the rewrite rules set up from before

        $struct = '/%category%/%year%/%monthnum%/%postname%/';

        $rewritecodes = array(
            '%category%',
            '%year%',
            '%monthnum%',
            '%postname%'
        );

        // setup data
        $terms = get_the_terms($post->ID, 'category');
        $unixtime = strtotime( $post->post_date );

        // this code is from get_permalink()
        $category = '';
        if ( strpos($permalink, '%category%') !== false ) {
            $cats = get_the_category($post->ID);
            if ( $cats ) {
                usort($cats, '_usort_terms_by_ID'); // order by ID
                $category = $cats[0]->slug;
                if ( $parent = $cats[0]->parent )
                    $category = get_category_parents($parent, false, '/', true) . $category;
            }
            // show default category in permalinks, without
            // having to assign it explicitly
            if ( empty($category) ) {
                $default_category = get_category( get_option( 'default_category' ) );
                $category = is_wp_error( $default_category ) ? '' : $default_category->slug;
            }
        }

        $replacements = array(
            $category,
            date( 'Y', $unixtime ),
            date( 'm', $unixtime ),
            $post->post_name
        );

        // finish off the permalink
        $permalink = home_url( str_replace( $rewritecodes, $replacements, $struct ) );
        $permalink = user_trailingslashit($permalink, 'single');
    }

    return $permalink;
}

Как уже упоминалось, это очень упрощенный пример создания настраиваемого набора правил перезаписи и постоянных ссылок, и он не особенно гибок, но этого должно быть достаточно для начала работы.

Мошенничество

Я написал плагин, который позволяет вам определять пермаструктуры для любых пользовательских типов постов, но, как вы можете использовать %category%в структуре постоянных ссылок для постов, мой плагин поддерживает %custom_taxonomy_name%любые custom_taxonomy_nameваши собственные таксономии, например, где указано название вашей таксономии. %club%,

Это будет работать так, как вы ожидаете, с иерархической / неиерархической таксономией.

http://wordpress.org/extend/plugins/wp-permastructure/


1
плагин отличный, но не могли бы вы объяснить, как исправить проблему в вопросе без вашего плагина?
Евгений Мануйлов

Я согласен, что замечательно, что есть плагин для решения этой проблемы (я поставил его в закладки, и это первое, что пришло мне в голову по этому вопросу), но ответ выиграл бы от краткого изложения того, что является проблемой и как плагин победил ее. :)
Rarst

@EugeneManuilov Хорошо, извините, это длинный ответ. Вот и я, раздели его до основ!
sanchothefat

Похоже, что первое $permalink = home_url(...переопределяется $permalink = user_trailingslashit(...и никогда не используется. Или я что-то упустил? $post_linkдаже не определен. Это должно было быть $permalink = user_trailingslashit( $permalink, 'single' );?
Ян Данн

Хороший улов, он должен быть $permalinkне $post_link. Ура :)
sanchothefat

1

Получил решение!

Чтобы иметь иерархические постоянные ссылки для пользовательского типа записи, установите плагин « Пользовательские типы сообщений» ( https://wordpress.org/plugins/custom-post-type-permalinks/ ).

Обновить зарегистрированный тип сообщения. У меня есть имя типа сообщения в качестве справочного центра

function help_centre_post_type(){
    register_post_type('helpcentre', array( 
        'labels'            =>  array(
            'name'          =>      __('Help Center'),
            'singular_name' =>      __('Help Center'),
            'all_items'     =>      __('View Posts'),
            'add_new'       =>      __('New Post'),
            'add_new_item'  =>      __('New Help Center'),
            'edit_item'     =>      __('Edit Help Center'),
            'view_item'     =>      __('View Help Center'),
            'search_items'  =>      __('Search Help Center'),
            'no_found'      =>      __('No Help Center Post Found'),
            'not_found_in_trash' => __('No Help Center Post in Trash')
                                ),
        'public'            =>  true,
        'publicly_queryable'=>  true,
        'show_ui'           =>  true, 
        'query_var'         =>  true,
        'show_in_nav_menus' =>  false,
        'capability_type'   =>  'page',
        'hierarchical'      =>  true,
        'rewrite'=> [
            'slug' => 'help-center',
            "with_front" => false
        ],
        "cptp_permalink_structure" => "/%help_centre_category%/%post_id%-%postname%/",
        'menu_position'     =>  21,
        'supports'          =>  array('title','editor', 'thumbnail'),
        'has_archive'       =>  true
    ));
    flush_rewrite_rules();
}
add_action('init', 'help_centre_post_type');

А вот и зарегистрированная таксономия

function themes_taxonomy() {  
    register_taxonomy(  
        'help_centre_category',  
        'helpcentre',        
        array(
            'label' => __( 'Categories' ),
            'rewrite'=> [
                'slug' => 'help-center',
                "with_front" => false
            ],
            "cptp_permalink_structure" => "/%help_centre_category%/",
            'hierarchical'               => true,
            'public'                     => true,
            'show_ui'                    => true,
            'show_admin_column'          => true,
            'show_in_nav_menus'          => true,
            'query_var' => true
        ) 
    );  
}  
add_action( 'init', 'themes_taxonomy');

Эта строка делает вашу постоянную работу

"cptp_permalink_structure" => "/%help_centre_category%/%post_id%-%postname%/",

вы можете удалить %post_id% и можете сохранить/%help_centre_category%/%postname%/"

Не забудьте удалить постоянные ссылки с приборной панели.


+1 самое простое решение , чтобы просто использовать этот плагин: wordpress.org/plugins/custom-post-type-permalinks работает отлично
Жюль

Да, но это для случая, когда у вас есть один пользовательский тип сообщения, но если у вас есть несколько пользовательских типов сообщений в одной теме, то решение выше - это решение. Более того, он также изменяет вашу категорию слагов так же, как ваш тип сообщений.
Варша Дхадж

1

Я нашел РЕШЕНИЕ !!!

(После того, как бесконечные исследования .. Я могу иметь ТАМОЖЕННЫЙ ПОСТ ТИП пермалинки как:
example.com/category/sub_category/my-post-name

вот код (в functions.php или плагине):

//===STEP 1 (affect only these CUSTOM POST TYPES)
$GLOBALS['my_post_typesss__MLSS'] = array('my_product1','....');

//===STEP 2  (create desired PERMALINKS)
add_filter('post_type_link', 'my_func88888', 6, 4 );

function my_func88888( $post_link, $post, $sdsd){
    if (!empty($post->post_type) && in_array($post->post_type, $GLOBALS['my_post_typesss']) ) {  
        $SLUGG = $post->post_name;
        $post_cats = get_the_category($id);     
        if (!empty($post_cats[0])){ $target_CAT= $post_cats[0];
            while(!empty($target_CAT->slug)){
                $SLUGG =  $target_CAT->slug .'/'.$SLUGG; 
                if  (!empty($target_CAT->parent)) {$target_CAT = get_term( $target_CAT->parent, 'category');}   else {break;}
            }
            $post_link= get_option('home').'/'. urldecode($SLUGG);
        }
    }
    return  $post_link;
}

// STEP 3  (by default, while accessing:  "EXAMPLE.COM/category/postname"
// WP thinks, that a standard post is requested. So, we are adding CUSTOM POST
// TYPE into that query.
add_action('pre_get_posts', 'my_func4444',  12); 

function my_func4444($q){     
    if ($q->is_main_query() && !is_admin() && $q->is_single){
        $q->set( 'post_type',  array_merge(array('post'), $GLOBALS['my_post_typesss'] )   );
    }
    return $q;
}

-2

У вас есть несколько ошибок с вашим кодом. Я очистил ваш существующий код:

<?php
function jcj_club_post_types() {
  $labels = array(
    'name' => __( 'Jazz Clubs' ),
    'singular_name' => __( 'Jazz Club' ),
    'add_new' => __( 'Add New' ),
    'add_new_item' => __( 'Add New Jazz Club' ),
    'edit' => __( 'Edit' ),
    'edit_item' => __( 'Edit Jazz Clubs' ),
    'new_item' => __( 'New Jazz Club' ),
    'view' => __( 'View Jazz Club' ),
    'view_item' => __( 'View Jazz Club' ),
    'search_items' => __( 'Search Jazz Clubs' ),
    'not_found' => __( 'No jazz clubs found' ),
    'not_found_in_trash' => __( 'No jazz clubs found in Trash' ),
    'parent' => __( 'Parent Jazz Club' ),
    );
  $args = array(
    'public' => true,
    'show_ui' => true,
    'publicly_queryable' => true,
    'exclude_from_search' => false,
    'menu_position' => 5,
    'query_var' => true,
    'supports' => array( 'title','editor','comments','revisions','trackbacks','author','excerpt','thumbnail','custom-fields' ),
    'rewrite' => array( 'slug' => 'jazz-clubs-in', 'with_front' => true ),
    'has_archive' => true
    );
  register_post_type( 'jcj_club', $args );
  }
add_action( 'init','jcj_club_post_types' );
?>

Замените свой код на код выше и посмотрите, работает ли он. Ответьте, если у вас есть дополнительные вопросы, и я постараюсь помочь.

РЕДАКТИРОВАТЬ:

Я заметил, что я не учел 'has_archive' => true.

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.