问题描述
我需要排序 (自定义)2 个自定义字段值的帖子…
自定义字段名称 1:is_sponsored
[值可以是 1
或 0
]
自定义字段名称 2:sfp_date
[timestamp
aka 当前发布日期以秒为单位]
「is_sponsored
」 值为 1 的帖子需要在顶部,按 DESC
的 「sfp_date
」 的顺序排列。 「is_sponsored
」 值为 0 的所有其他帖子都应按照降序 (由 「sfp_date
」) 列出。
我有这样的东西:
$sfp_query_args = array(
'tax_query' => array(
array(
'taxonomy' => 'sfp_posts',
'terms' => array( 1, 5, 8 )
)
),
'post_type' => 'sfpposts',
'post_status' => 'publish',
'showposts' => 15,
'paged' => $paged,
'meta_key' => 'sfp_date',
'orderby' => 'meta_value_num',
'order' => 'DESC',
'meta_query' => array(
'key' => 'is_sponsored',
'value' => 2,
'type' => 'NUMERIC',
'compare' => '<='
)
);
$wp_q = new WP_Query( $sfp_query_args );
… 但不工作有任何想法吗?
编辑注意:这是一个小插件,应该显示查询的外观,因为我们可能没有可用的数据集来测试。
<?php
/** Plugin Name: (#67600) Dump Query parts */
function wpse67600_dump_query_parts( $pieces )
{
echo '<pre>'.var_export( $pieces, true ).'</pre>';
return $pieces;
}
add_filter( 'posts_clauses', 'wpse67600_dump_query_parts' );
OP 请添加此处的插入输出 – 使用”edit” 链接。
Dameer 编辑
好的,跟踪请求和许多解决方法后,我已经提出了以下内容:
如果我简化了”$sfp_query_args”,结果是接近所需要的,但是,无法排序帖子保持原样。这里是:
$sfp_query_args1 = array(
'tax_query' => array( array( 'taxonomy' => 'sfp_post_category', 'terms' => $cat_id_arr ) ),
'post_type' => 'sfpposts',
'post_status' => 'publish',
'showposts' => (int)$per_page,
'paged' => $paged,
'meta_key' => 'is_sponsored',
'orderby' => 'meta_value date'
);
-
* orderby 有两个属性:meta_value 和 date *
所以 $ wpdb-> 请求与查询中的上述参数如下所示:
SELECT SQL_CALC_FOUND_ROWS $wpdb->posts.ID
FROM $wpdb->posts
INNER JOIN $wpdb->term_relationships
ON ($wpdb->posts.ID = $wpdb->term_relationships.object_id)
INNER JOIN $wpdb->postmeta
ON ($wpdb->posts.ID = $wpdb->postmeta.post_id)
WHERE 1=1
AND $wpdb->posts.post_type = 'sfpposts'
AND ($wpdb->posts.post_status = 'publish')
AND ($wpdb->postmeta.meta_key = 'is_sponsored' )
GROUP BY $wpdb->posts.ID
ORDER BY $wpdb->postmeta.meta_value, $wpdb->posts.post_date DESC
LIMIT 0, $per_page
最后,为了能够通过 meta_value 进行排序,查询应该只设置一个小的区别:
SELECT SQL_CALC_FOUND_ROWS $wpdb->posts.ID
FROM $wpdb->posts
INNER JOIN $wpdb->term_relationships
ON ($wpdb->posts.ID = $wpdb->term_relationships.object_id)
INNER JOIN $wpdb->postmeta
ON ($wpdb->posts.ID = $wpdb->postmeta.post_id)
WHERE 1=1
AND $wpdb->posts.post_type = 'sfpposts'
AND ($wpdb->posts.post_status = 'publish')
AND ($wpdb->postmeta.meta_key = 'is_sponsored' )
GROUP BY $wpdb->posts.ID
ORDER BY $wpdb->postmeta.meta_value [!ORDER MISSING!], $wpdb->posts.post_date DESC
LIMIT 0, $per_page
请发现 [!ORDER MISSING!] 占位符。我想上面应该解释问题出在哪里。
最佳解决方案
好的,最后的解决方法是分割查询:
$sfp_query_args = array(
'tax_query' => array( array( 'taxonomy' => 'sfp_post_category', 'terms' => $cat_id_arr ) ),
'meta_key' => 'is_sponsored',
'post_type' => 'sfpposts',
'post_status' => 'publish',
'showposts' => (int)$per_page,
'paged' => $paged
);
… 并使用”posts_orderby” 过滤器修改 ORDER 部分:
add_filter( 'posts_orderby', 'sfp_modify_orderby' );
function sfp_modify_orderby( $orderby ) {
if( !is_admin() && is_tax( 'sfp_post_category' ) ) {
global $wpdb;
$orderby = " $wpdb->postmeta.meta_value DESC, $wpdb->posts.post_date DESC ";
}
return $orderby;
}
为了防止’posts_orderby’ 影响任何其他查询 (侧边栏或页脚),最可能需要在页面上的循环之后删除过滤器。所以这里有另一个功能放在”functions.php” 中:
function sfp_remove_orderby_filter() {
remove_filter( 'posts_orderby', 'sfp_modify_orderby' );
}
… 并在页面上使用我们的查询丢弃过滤器:
if( have_posts() ) : while( have_posts() ) : the_post();
// code
endwhile;
else :
// code
endif;
sfp_remove_orderby_filter();
希望是有道理的!
参考文献
注:本文内容整合自 Google/Baidu/Bing 辅助翻译的英文资料结果。如果您对结果不满意,可以加入我们改善翻译效果:薇晓朵技术论坛。