WordPress如何按照文章类型排序文章?

4

我正在尝试创建一个系统,从2个文章类型中获取文章,并希望按照它们的文章类型显示文章顺序。我想先显示一个文章类型的文章,然后再显示另一个文章类型的文章,但我使用的查询混合了所有文章类型。有人可以为此提供一个好的解决方案吗?这是我的代码。

    $paged = get_query_var( 'paged' ) ? get_query_var( 'paged' ) : 1;
 $args = array(

     'post_type' => array('review','directory'),
     'orderby' => 'name',
     'order' => 'ASC',
     'posts_per_page'  => '4',
     'paged'          => $paged ,
     'tax_query' => array( array(
        'taxonomy' => 'Reviews',
        'field' => 'id',
        'terms' => $cat_id
    ), ),
     );

 query_posts($args);
        if (have_posts()) :
       while (have_posts()) : the_post();
      $product_terms = wp_get_post_terms(get_the_ID(), 'Reviews', array("fields" => "all", "order" => "DESC"));
      $postype=get_post_type( get_the_ID() );
      if($postype=='review')
      {
     ?>   

        <div class="review-post">
          <a href="http://<?php the_permalink(); ?>" target="_blank"><h3><?php the_title(); ?></h3></a>
          <div class="review-cat-new">
            <?php echo get_the_term_list( $post->ID, 'Reviews', 'Category: ', ', ', '' ); ?>
          </div>

          <?php
          if(get_field('see_it'))
              $seeit_text= get_field('see_it');
         if(get_field('skip_it'))
              $skipit_text= get_field('skip_it');
         ?>
          <div class="see-skip">
            <p class="see-it"><span>See it:</span>
                <?php echo $seeit_text; ?>
            </p>
            <p class="skip-it"><span>Skip it:</span>
                <?php echo $skipit_text;  ?>
            </p>
          </div>
          <?php echo custom_field_excerpt(); ?>
        </div>
          <?php }
          else
          {
              ?>
                <div class="review-post">
          <a href="h<?php the_permalink(); ?>" target="_blank"><h3><?php the_title(); ?></h3></a>
          <div class="review-cat-new">
            <?php echo get_the_term_list( get_the_ID(), 'Reviews', 'Category: ', ', ', '' ); ?>
          </div>


          <?php echo the_field('enter_content_direc'); ?>
          <?php
          if(get_field('enter_textdirec'))
              $text= get_field('enter_textdirec');
         if(get_field('enter_linkdirec'))
              $textlink= get_field('enter_linkdirec');
         ?>
          <div class="see-skip">
            <p class="see-it direc"><span><a  target="_blank" style="color:#5D6D71; text-transform: lowercase;" href="<?php echo $textlink;?>"><?php echo $textlink;?>  </a></span>

            </p>
          </div>
        </div>
          <?php }
          endwhile;
                  echo '<div class="paging">';
                 wp_pagenavi();
                 echo '</div>';
                endif;

有没有一种方法可以先显示评论的文章,再显示目录的文章?
5个回答

5

如文档所示https://developer.wordpress.org/reference/classes/wp_query/#order-orderby-parameters,您可以通过向 'orderby' 传递一个数组来使用常规的 wp 查询:

$wp_query = new WP_Query(
  array(
      's' => $search,
      'orderby'=> array('type'=>'DESC', 'title'=>'ASC'),
      'paged'=>$paged
  )
);
if ( $wp_query->have_posts() ) {
    while ( $wp_query->have_posts() ) {
        $wp_query->the_post();
        // do your loop stufff here
    }
}

4
自WordPress 4.0以来,您只需在WP_Query对象中将选项类型(post_type)传递给orderby参数即可。
orderby(字符串|数组)-按参数排序检索的文章。要按文章类型排序,只需传递:
'type'-按文章类型排序(自版本4.0起可用)。 (也接受'post_type'。)

能否按特定顺序获取 post_types?例如:数组的顺序。 - waterschaats
开发者文档没有提到这一点,但是它是真实存在的,这里是代码:https://github.com/WordPress/wordpress-develop/blob/5.8.1/src/wp-includes/class-wp-query.php#L1574 - Design.Garden

3

最终我找到了解决方案,可以使用过滤器来完成。

解决方案如下所示。

add_filter( 'posts_request' , 'modify_request' );
function modify_request( $query) {
    global $wpdb;
    if(strstr($query,"post_type IN ('review', 'directory')")){
        $where = str_replace("ORDER BY {$wpdb->posts}.post_date","ORDER BY {$wpdb->posts}.post_type",$query);
    }
    return $where;
}

你把这段代码放在哪里?如果我尝试将其放在archive-name.php中,会出现错误。 - LTech
把它放在 functions.php 文件中。 - Nirmal Ram
1
更多信息。如果你只想改变 "order by",WordPress 有 posts_orderby_request 过滤器。 - EThaiZone
嗨@NirmalRam,这里似乎有些问题...当if条件不满足时,$query未返回,实际上没有产生任何结果。此外,$where是否应该以这种方式实现?它难道不是$query对象的属性吗? - Theunis

2

使用自定义文章类型顺序按文章类型排序:

add_filter('pre_get_posts', function ($query) {

    if ($query->is_search && !is_admin()) :
        $query->set('post_type', [ 'cpt-1', 'cpt-2', 'post', 'cpt-3']);
    endif;

    return $query;
});



add_filter('posts_orderby', function ( $order ) {

    if ( ! is_admin() ) :

        if ( is_search() && is_main_query() ) :

            global $wpdb;
            $order = "FIELD( post_type, 'cpt-1', 'cpt-2', 'post' ), {$wpdb->posts}.post_modified DESC";      
        endif;

    // Disable this filter for future queries!
    remove_filter( current_filter(), __FUNCTION__);
    endif;

    return $order;
});

0

我脑海中首先想到的解决方案是仅获取第一个类别的文章。重置查询(wp_reset_query()),然后再为第二个类别重新获取它们。当然,你需要复制你的代码,但我没有看到其他解决方案 - 这是WordPress - 你的选择很有限。

第二种解决方案是直接查询数据库,像这样:

$result = mysql_query('SELECT * 
FROM  `wp_posts` 
WHERE post_type =  "page"
OR post_type =  "post"
ORDER BY post_type ASC , post_date DESC ');

$posts = array();

if($result) {
    while ($row = mysql_fetch_assoc($result)) {
        $posts[] = $row;
    }
}

mysql_free_result($result);

echo('<pre>');
var_dump($posts);
echo('</pre>');
die();

请注意,我使用了 post page 类型,因为这些是默认值,并存在于我的WP安装中。但你可以使用不同的类型。你可以随意调整此查询。

谢谢,但我不想要那个。我想要的是按post_type对帖子进行排序。 - Nirmal Ram
那么你可能需要自己编写查询 PHP 函数,直接从数据库中读取数据。这对你来说是一个选项吗? - Karol
不,这不是正确的。我已经想出来了,它可以工作。只需要一个过滤器就可以了。 - Nirmal Ram

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接