From c5a753c8d20228b1f998138357e438b7647bef3b Mon Sep 17 00:00:00 2001 From: JC Date: Thu, 20 Jun 2024 05:11:11 +0800 Subject: [PATCH] WDSBT-23 - Override WP Query to respect perPage count even with sticky posts --- inc/hooks/query-block-sticky-override.php | 87 +++++++++++++++++++++++ 1 file changed, 87 insertions(+) create mode 100644 inc/hooks/query-block-sticky-override.php diff --git a/inc/hooks/query-block-sticky-override.php b/inc/hooks/query-block-sticky-override.php new file mode 100644 index 0000000..00b55ba --- /dev/null +++ b/inc/hooks/query-block-sticky-override.php @@ -0,0 +1,87 @@ +query_vars['ignore_sticky_posts'] = isset( $query->query_vars['ignore_sticky_posts'] ) && $query->query_vars['ignore_sticky_posts']; + $query->query_vars['ignore_sticky_posts'] = true; +} +add_filter( 'parse_query', __NAMESPACE__ . '\parse_query_var' ); + +/** + * Modifies the query clauses to handle sticky posts without including them in the main results. + * + * This function alters the WHERE and ORDER BY clauses of the query to ensure that + * sticky posts are included in a specific order without affecting the main result set. + * + * @param array $clauses An associative array of the query clauses. + * @param \WP_Query $query The WP_Query instance. + * @return array The modified query clauses. + */ +function modify_posts_clauses( $clauses, $query ) { + global $wpdb; + + if ( ! $query->query_vars['post__in'] && $query->is_home && ! $query->query_vars['ignore_sticky_posts'] ) { + $sticky_posts = get_option( 'sticky_posts' ); + + // Retrieve published sticky posts of the desired post type(s). + $stickies_query = new \WP_Query( + array( + 'post__in' => $sticky_posts, + 'post_type' => $query->query_vars['post_type'], + 'post_status' => 'publish', + 'fields' => 'ids', + 'posts_per_page' => count( $sticky_posts ), + ) + ); + + $queried_stickies = $stickies_query->posts; + + if ( ! empty( $queried_stickies ) ) { + // Generate the WHERE clause to exclude sticky posts. + $post__in_sticky = implode( ',', array_map( 'absint', $queried_stickies ) ); + $clauses['where'] = 'AND ((1=1 ' . $clauses['where'] . ") OR {$wpdb->posts}.ID IN ($post__in_sticky))"; + $sticky_orderby = "FIELD( {$wpdb->posts}.ID, $post__in_sticky ) DESC"; + + // Add the sticky posts handling to the ORDER BY clause. + if ( $clauses['orderby'] ) { + $clauses['orderby'] = $sticky_orderby . ', ' . $clauses['orderby']; + } else { + $clauses['orderby'] = $sticky_orderby; + } + } + } + + return $clauses; +} +add_filter( 'posts_clauses', __NAMESPACE__ . '\modify_posts_clauses', 10, 2 );