如何使用自定义分类法和自定义帖子类型(例如基本名称/父税/子税收/自定义帖子类型名称)创建永久链接结构


53

我一直在梳理这个网站和google的答案,而我完全是空的。基本上我想完全按照这篇文章的要求做,但是我需要一个层次分类法。该帖子中给出的答案很好,但仅适用于单级分类法。可以做我想做的吗?我已经尝试了上百万种方法,但没有任何效果,充其量我可以得到正确的永久链接,但它们可以出现404。

为了直观地说明我想要什么:

/basename/ - ideally a page, but I think this will cause a permalink collision    
/basename/top-cat/ - top parent custom taxonomy archive    
/basename/top-cat/child-cat/ - child cat custom taxonomy archive     
/basename/top-cat/child-cat/grandchild-cat/ - grandchild cat custom taxonomy archive    
/basename/top-cat/child-cat/grandchild-cat/post-name/ - my custom post type post

您可以使用内置的帖子和类别来做到这一点,如何使用自定义分类法和自定义帖子类型呢?我知道您需要使用'rewrite' => array( 'slug' => 'tax-name', 'with_front' => true, 'hierarchical' => true ),获取层次结构的标签,该标签在存档页面上可以正常工作,但是自定义帖子类型的帖子出现404。如果我删除了该'hierarchical' => true部分,则帖子可以工作,但是我丢失了层次结构网址(仅/ basename / grandchild-cat / post-name /作品)。

那么,有什么解决方案吗?非常感谢,这已经让我发疯了大约三个星期。

Answers:


68

在结合了许多其他答案之后,我开始工作了!因此,这也是为那些同样为此而苦苦挣扎的人提供的解决方案:

这篇文章这篇文章帮助了我一些,所以感谢那些家伙。

请注意,所有这些代码,加上您的初始自定义帖子类型和分类注册代码都在您的functions.php文件中。

在定义自定义帖子类型和分类法时,首先要使子标签正确:对于自定义帖子类型,分类应该是basename/%taxonomy_name%,分类标准的子标签应该是just basename。别忘了还添加'hierarchical' => true到分类法重写数组中,以在URL中获得嵌套术语。还要确保在两种情况下query_var都将其设置为true

您需要添加一个新的重写规则,以便WordPress知道如何解释您的url结构。在我的情况下,uri的自定义帖子类型部分将始终是uri的第5个细分,因此我相应地定义了匹配规则。请注意,如果使用更多或更少的uri细分,则可能必须更改此设置。如果您要使用不同级别的嵌套术语,则需要编写一个函数来检查最后的uri段是自定义帖子类型还是分类术语,以了解要添加的规则(如果需要帮助,请咨询我那)。

add_filter('rewrite_rules_array', 'mmp_rewrite_rules');
function mmp_rewrite_rules($rules) {
    $newRules  = array();
    $newRules['basename/(.+)/(.+)/(.+)/(.+)/?$'] = 'index.php?custom_post_type_name=$matches[4]'; // my custom structure will always have the post name as the 5th uri segment
    $newRules['basename/(.+)/?$']                = 'index.php?taxonomy_name=$matches[1]'; 

    return array_merge($newRules, $rules);
}

然后,您需要添加以下代码,以使workpress如何处理%taxonomy_name%自定义帖子类型中的重写段结构:

function filter_post_type_link($link, $post)
{
    if ($post->post_type != 'custom_post_type_name')
        return $link;

    if ($cats = get_the_terms($post->ID, 'taxonomy_name'))
    {
        $link = str_replace('%taxonomy_name%', get_taxonomy_parents(array_pop($cats)->term_id, 'taxonomy_name', false, '/', true), $link); // see custom function defined below
    }
    return $link;
}
add_filter('post_type_link', 'filter_post_type_link', 10, 2);

我基于Wordpress自己创建了一个自定义函数get_category_parents

// my own function to do what get_category_parents does for other taxonomies
function get_taxonomy_parents($id, $taxonomy, $link = false, $separator = '/', $nicename = false, $visited = array()) {    
    $chain = '';   
    $parent = &get_term($id, $taxonomy);

    if (is_wp_error($parent)) {
        return $parent;
    }

    if ($nicename)    
        $name = $parent -> slug;        
else    
        $name = $parent -> name;

    if ($parent -> parent && ($parent -> parent != $parent -> term_id) && !in_array($parent -> parent, $visited)) {    
        $visited[] = $parent -> parent;    
        $chain .= get_taxonomy_parents($parent -> parent, $taxonomy, $link, $separator, $nicename, $visited);

    }

    if ($link) {
        // nothing, can't get this working :(
    } else    
        $chain .= $name . $separator;    
    return $chain;    
}

然后,您需要刷新永久链接(只需加载永久链接设置页面)。

现在一切都“应该”有望运行!开始制作一堆分类术语并将其正确嵌套,然后制作一些自定义帖子类型的帖子并将其正确分类。您还可以使用slug制作页面basename,所有内容都应按照我在问题中指定的方式工作。您可能需要创建一些自定义分类存档页面来控制它们的外观,并添加某种分类小部件插件以在侧边栏中显示嵌套类别。

希望对您有帮助!


固定-请参阅问题。杰夫,谢谢您的解释!但是,您是否有可能发布自定义的posttype /分类法构建?这样我就能知道我做错了什么吗?真的会很感激!
Reitze Bos 2012年

嘿杰夫,谢谢你的回答!经过4个小时的战斗,我快到了。我唯一的问题是,我在帖子名称前加了一个双斜杠(例如:Portfolio / diseno-grafico-2 / logo // alquimia-sonora /),您能帮我吗?
Cmorales'4

1
@Cmorales,不确定是否看不到您的代码,而是在帖子名称之前寻找一个手动定义斜杠的地方,例如cpt slug注册还是在filter_post_type函数中?如果您无法找到要在其中添加额外斜杠的位置,则可以从filter_post_type_link中调用的get_taxonomy_parents函数的返回值中删除最后一个字符,因为这将为您留下一个斜杠,这是第一个双。祝好运。
杰夫

1
“如果嵌套术语的级别不同,那么您需要编写一个函数来检查最后一个uri段是自定义帖子类型还是分类术语,以了解要添加的规则(如果需要帮助,请咨询我”)“好吧,能帮我吗?:)
Cmorales

2
^这正是我所需要的。有人知道怎么做吗?看到我的问题在这里:wordpress.stackexchange.com/questions/102246/...
reikyoushin

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.