WordPress多站点-全球类别


21

设置WP多站点实例-客户端具有现有的本体/类别集,他们希望对整个博客集中的所有内容进行分类。同样,人们希望在“网络博客”级别添加任何新类别,并将其同步到其他博客。

最好的方法是什么?


我想使类别分配给全局变量,然后在主题初始化上导入。
kaiser

4
我认为这个问题与在3.0中跨多个博客共享一个分类法相同。但是,这个问题没有得到很好的答案。这是一个有趣的问题,我将提供赏金。
1月Fabry

Answers:


14
function __add_global_categories( $term_id )
{
    if ( get_current_blog_id() !== BLOG_ID_CURRENT_SITE || ( !$term = get_term( $term_id, 'category' ) ) )
        return $term_id; // bail

    if ( !$term->parent || ( !$parent = get_term( $term->parent, 'category' ) ) )
        $parent = null;

    global $wpdb;

    $blogs = $wpdb->get_col( "SELECT blog_id FROM {$wpdb->blogs} WHERE site_id = '{$wpdb->siteid}'" );
    foreach ( $blogs as $blog ) {
        $wpdb->set_blog_id( $blog );

        if ( $parent && ( $_parent = get_term_by( 'slug', $parent->slug, 'category' ) ) )
            $_parent_ID = $_parent->term_id;
        else
            $_parent_ID = 0;

        wp_insert_term( $term->name, 'category',  array(
            'slug' => $term->slug,
            'parent' => $_parent_ID,
            'description' => $term->description
        ));
    }

    $wpdb->set_blog_id( BLOG_ID_CURRENT_SITE );
}
add_action( 'created_category', '__add_global_categories' );

只要在主站点上添加类别,此命令就将运行。一些注意事项/要点;

  • 如果您有很多博客,则此功能可能会非常密集。
  • 平均而言,我们每个博客运行5到8个查询(可能更多)的查询-根据您数据库的速度,此功能可能需要分块。
  • 只有新添加的类别是“同步的”。更新和删除类​​别不是(代码将需要修改)。
  • 如果新添加的类别有一个父类别,并且在相关的多站点博客中找不到该父类别,则将在没有父类别的情况下创建该类别(只有在安装此功能之前创建了父类别的情况下,才应如此)。

1
是否有-或可能有-一个这样做的插件?以及编辑和删除?还有一个设置页面,可以选择将其应用于哪些分类法和哪些子站点?
Marcus Downing

实际上,如果我使用您的代码作为编写插件的起点,您会反对吗?
Marcus Downing

没问题-我的回答属于堆栈交换的授权,必须
注明出处

11

哦,周日拖延...

https://github.com/maugly/Network-Terminator

  • 允许跨网络批量添加条款
  • 您可以选择哪些站点将受到影响
  • 与自定义分类法一起使用
  • 不删除
  • 不同步

这是我在最近几个小时内完成的工作,现在没有时间进行更多测试。无论如何-它对我有用!)

试试看。还实现了“测试运行”功能,因此您可以在实际执行操作之前检查结果。

更新->屏幕截图:

行动前:

行动前

试运行后:

试运行后

上面链接的插件添加了用户界面,但是几乎所有重要的事情都在此函数中发生:

        <?php function mau_add_network_terms($terms_to_add, $siteids, $testrun = false) {

        // check if this is multisite install
        if ( !is_multisite() )
            return 'This is not a multisite WordPress installation.';

        // very basic input check
        if ( empty($terms_to_add) || empty($siteids) || !is_array($terms_to_add) || !is_array($siteids) )
            return 'Nah, I eat only arrays!';

        if ($testrun) $log = '<p><em>No need to get excited. This is just a test run.</em></p>';
        else $log = '';

        // loop thru blogs
        foreach ($siteids as $blog_id) :

            switch_to_blog( absint($blog_id) );

            $log .= '<h4>'.get_blog_details(  $blog_id  )->blogname.':</h4>';
            $log .= '<ul id="ntlog">';

            // loop thru taxonomies
            foreach ( $terms_to_add as $taxonomy => $terms ) {

                // check if taxonomy exists
                if ( taxonomy_exists($taxonomy) ) {
                    // get taxonomy name
                    $tax_name = get_taxonomy($taxonomy);
                    $tax_name = $tax_name->labels->name;

                    //loop thru terms   
                    foreach ( $terms as $term ) {

                        // check if term exists
                        if ( term_exists($term, $taxonomy) ) {
                            $log .= "<li class='notice' ><em>$term already exists in the $tax_name taxonomy - not added!</em></li>";

                        } else {

                            // if it doesn't exist insert the $term to $taxonomy
                            $term = strip_tags($term);
                            $taxonomy = strip_tags($taxonomy);
                            if (!$testrun)
                                wp_insert_term( $term, $taxonomy );
                            $log .= "<li><b>$term</b> successfully added to the <b>$tax_name</b> taxonomy</li>"; 
                        }
                    }
                } else {
                    // tell our log that taxonomy doesn't exists
                    $log .= "<li class='notice'><em>The $tax_name taxonomy doesn't exist! Skipping...</em></li>"; 
                }
            }

            $log .= '</ul>';    

            // we're done here
            restore_current_blog();

        endforeach;
        if ($testrun) $log .= '<p><em>No need to get excited. This was just the test run.</em></p>';
        return $log;
    } ?>

稍后,我将返回并使用更多信息对其进行编辑(如果需要)。

这远非完美(请阅读插件头中的已知问题)。
任何反馈表示赞赏!


3
当人们响应问题创建插件时,我喜欢它!您应该得到赏金!
Jan Fabry

感谢您的支持@Jan Fabry。如果我旁边有人真的觉得这件事有用,我会很高兴。
Michal Mau


5

TheDeadMedic的答案看起来不错,但我最终还是采用了另一种方法来解决该问题。我没有在许多站点上复制相同的术语,而是让其他站点使用主站点的表作为术语。

add_action('init', 'central_taxonomies');

function central_taxonomies () {
  global $wpdb;

  $wpdb->terms = "wp_terms";
  $wpdb->term_taxonomy = "wp_term_taxonomy";
}

这将替换表名wp_2_termswp_terms,等你的数据库课程检查你应该做的确认表的确切名称,如果你改变你的前缀,这可能是不同的。

您可以从插件或主题中运行它(尽管我建议使用插件)。我可能会在某个时候发布插件来执行此操作。这种方法有两个缺点:

  • 它仅在激活了插件的子站点上有效。无法从父站点强制执行此操作。
  • 它适用于所有分类法,而不仅仅是选定的分类法。

这种方法非常灵活-可以适应从任何博客中提取类别,而不仅仅是中心博客。


更新:我已经将其制作成插件,可以在以下站点范围内激活:MU Central Taxonomies


这种方法存在一个大问题:帖子和术语之间的关系可能不正确。表格term_relationships包含基于帖子ID和术语ID的关系。但是总有可能子站点中的帖子具有相同的ID。更改一个帖子的术语可能会对另一个博客中的另一个帖子产生不可预测的影响。
Anh Tran

正确,该term_relationships表不应包括在内。我很早就在插件中发现并修复了该问题,但从未更新此答案以使其匹配。
马库斯·唐宁

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.