我目前正在使用query_posts来修改我网站上的自定义页面上的主要查询,据我所知,尽管我已经阅读过,使用query_posts是不好的做法,原因有很多.
那么,为什么我使用query_posts而不是创建一个你可能会问的WP_Query对象?
这是因为我正在使用无限滚动插件,无限滚动不会很好用WP_query,但是当您使用query_posts修改主查询时,它的工作效果非常好.例如,使用无限卷动WP_query(主要关注)分页不起作用.
<?PHP $paged = get_query_var( 'paged' ) ? get_query_var( 'paged' ) : 1; ?> <?PHP query_posts( array( 'Meta_key' => 'wpb_post_views_count','orderby' => 'Meta_value_num','order' => 'DESC','paged' => $paged,) ); ?> <?PHP if (have_posts()) : ?> <?PHP while ( have_posts() ) : the_post() ?> <?PHP if ( has_post_format( 'video' )) { get_template_part( 'video-post' ); }elseif ( has_post_format( 'image' )) { get_template_part( 'image-post' ); } else { get_template_part( 'standard-post' ); } ?> <?PHP endwhile;?> <?PHP endif; ?>
所以经过很多阅读,我收集到我的其他选项修改主要查询是使用pre_get_posts,虽然我有点不确定如何去做这个.
以此为例: –
function textdomain_exclude_category( $query ) { if ( $query->is_home() && $query->is_main_query() ) { $query->set( 'cat','-1,-2' ); } } add_action( 'pre_get_posts','textdomain_exclude_category' );
我感到困惑,无法想像的是:
>自定义页面模板的用例场景.使用我的query_posts修改我可以放在数组之前if(have_posts()),选择我的页面模板,发布它,我走了.
使用pre_get_posts,我无法弄清楚如何查询$query->最多浏览等等
> array(‘Meta_key’=>’wpb_post_views_count’,’orderby’=>’Meta_value_num’,’order’=>’DESC’,’paged’=> $paged,));
我如何用pre_get_posts做这件事,并确保它是分页的,即.与无限滚动一起工作?在所有使用pre_get_posts的示例中,没有数组.
我一直在玩pre_get_posts钩子,这里有一个想法
步骤1:
example.com/show
第2步:
tpl_show.PHP
位于当前的主题目录.
步骤3:
我们构建以下pre_get_posts动作回调:
function b2e_pre_get_posts( $query ) { $target_page = 'show'; // EDIT to your needs if ( ! is_admin() // front-end only && $query->is_main_query() // main query only && $target_page === $query->get( 'pagename' ) // matching pagename only ) { // modify query_vars: $query->set( 'post_type','post' ); // override 'post_type' $query->set( 'pagename',null ); // override 'pagename' $query->set( 'posts_per_page',10 ); $query->set( 'Meta_key','wpb_post_views_count' ); $query->set( 'orderby','Meta_value_num' ); $query->set( 'order','DESC' ); // Support for paging $query->is_singular = 0; // custom page template add_filter( 'template_include','b2e_template_include',99 ); } } add_action( 'pre_get_posts','b2e_pre_get_posts' );
哪里
function b2e_template_include( $template ) { $target_tpl = 'tpl_show.PHP'; // EDIT to your needs remove_filter( 'template_include',99 ); $new_template = locate_template( array( $target_tpl ) ); if ( ! empty( $new_template ) ) $template = $new_template; ; return $template; }
这也应该给我们分页:
example.com/show/page/2 example.com/show/page/3
等等
笔记
我更新了答案,并根据@PieterGoosen的建议删除了查询对象部分修改,因为它可以例如打破他的设置中的面包屑.
还删除了pre_get_posts钩子中的is_page()检查,因为在某些情况下它可能仍然会出现不规则.原因是查询对象并不总是可用.正在进行中. #27015.如果要使用is_page()或is_front_page(),则有workarounds possible.
我构建了下表,只是为了更好地概述主WP_Query对象的一些属性和查询变量,对于给定的块:
这是interesting to note,WP_Query中的分页依赖于nopaging未设置,当前页面不是单数(从4.4 source):
// Paging if ( empty($q['nopaging']) && !$this->is_singular ) { $page = absint($q['paged']); if ( !$page ) $page = 1; // If 'offset' is provided,it takes precedence over 'paged'. if ( isset( $q['offset'] ) && is_numeric( $q['offset'] ) ) { $q['offset'] = absint( $q['offset'] ); $pgstrt = $q['offset'] . ','; } else { $pgstrt = absint( ( $page - 1 ) * $q['posts_per_page'] ) . ','; } $limits = 'LIMIT ' . $pgstrt . $q['posts_per_page']; }
我们可以看到生成的SQL查询的LIMIT部分在条件检查中.这就解释了为什么我们修改上面的is_singular属性.
我们可以使用其他过滤器/钩子,但是在这里我们使用了OP提到的pre_get_posts.
希望这个帮助.