2016-05-26 06:33:30 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Taxonomy API: WP_Term_Query class.
|
|
|
|
*
|
|
|
|
* @package WordPress
|
|
|
|
* @subpackage Taxonomy
|
|
|
|
* @since 4.6.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class used for querying terms.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
|
|
|
* @see WP_Term_Query::__construct() for accepted arguments.
|
|
|
|
*/
|
|
|
|
class WP_Term_Query {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SQL string used to perform database query.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public $request;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metadata query container.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
2020-07-23 09:39:02 +02:00
|
|
|
* @var WP_Meta_Query A meta query instance.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
public $meta_query = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metadata query clauses.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $meta_query_clauses;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SQL query clauses.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $sql_clauses = array(
|
|
|
|
'select' => '',
|
|
|
|
'from' => '',
|
|
|
|
'where' => array(),
|
|
|
|
'orderby' => '',
|
|
|
|
'limits' => '',
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Query vars set by the user.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
public $query_vars;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default values for query vars.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
public $query_var_defaults;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of terms located by the query.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
public $terms;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor.
|
|
|
|
*
|
|
|
|
* Sets up the term query, based on the query vars passed.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
2016-06-11 05:49:27 +02:00
|
|
|
* @since 4.6.0 Introduced 'term_taxonomy_id' parameter.
|
2016-09-28 05:55:33 +02:00
|
|
|
* @since 4.7.0 Introduced 'object_ids' parameter.
|
2017-10-05 02:37:47 +02:00
|
|
|
* @since 4.9.0 Added 'slug__in' support for 'orderby'.
|
2021-11-20 01:07:59 +01:00
|
|
|
* @since 5.1.0 Introduced the 'meta_compare_key' parameter.
|
|
|
|
* @since 5.3.0 Introduced the 'meta_type_key' parameter.
|
2016-05-26 06:33:30 +02:00
|
|
|
*
|
|
|
|
* @param string|array $query {
|
|
|
|
* Optional. Array or query string of term query parameters. Default empty.
|
|
|
|
*
|
2021-12-07 13:20:02 +01:00
|
|
|
* @type string|string[] $taxonomy Taxonomy name, or array of taxonomy names, to which results
|
|
|
|
* should be limited.
|
2021-11-20 01:07:59 +01:00
|
|
|
* @type int|int[] $object_ids Object ID, or array of object IDs. Results will be
|
|
|
|
* limited to terms associated with these objects.
|
|
|
|
* @type string $orderby Field(s) to order terms by. Accepts:
|
|
|
|
* - Term fields ('name', 'slug', 'term_group', 'term_id', 'id',
|
|
|
|
* 'description', 'parent', 'term_order'). Unless `$object_ids`
|
|
|
|
* is not empty, 'term_order' is treated the same as 'term_id'.
|
|
|
|
* - 'count' to use the number of objects associated with the term.
|
|
|
|
* - 'include' to match the 'order' of the `$include` param.
|
|
|
|
* - 'slug__in' to match the 'order' of the `$slug` param.
|
|
|
|
* - 'meta_value'
|
|
|
|
* - 'meta_value_num'.
|
|
|
|
* - The value of `$meta_key`.
|
|
|
|
* - The array keys of `$meta_query`.
|
|
|
|
* - 'none' to omit the ORDER BY clause.
|
|
|
|
* Default 'name'.
|
|
|
|
* @type string $order Whether to order terms in ascending or descending order.
|
|
|
|
* Accepts 'ASC' (ascending) or 'DESC' (descending).
|
|
|
|
* Default 'ASC'.
|
|
|
|
* @type bool|int $hide_empty Whether to hide terms not assigned to any posts. Accepts
|
|
|
|
* 1|true or 0|false. Default 1|true.
|
|
|
|
* @type int[]|string $include Array or comma/space-separated string of term IDs to include.
|
|
|
|
* Default empty array.
|
|
|
|
* @type int[]|string $exclude Array or comma/space-separated string of term IDs to exclude.
|
|
|
|
* If `$include` is non-empty, `$exclude` is ignored.
|
|
|
|
* Default empty array.
|
|
|
|
* @type int[]|string $exclude_tree Array or comma/space-separated string of term IDs to exclude
|
|
|
|
* along with all of their descendant terms. If `$include` is
|
|
|
|
* non-empty, `$exclude_tree` is ignored. Default empty array.
|
|
|
|
* @type int|string $number Maximum number of terms to return. Accepts ''|0 (all) or any
|
|
|
|
* positive number. Default ''|0 (all). Note that `$number` may
|
|
|
|
* not return accurate results when coupled with `$object_ids`.
|
|
|
|
* See #41796 for details.
|
|
|
|
* @type int $offset The number by which to offset the terms query. Default empty.
|
|
|
|
* @type string $fields Term fields to query for. Accepts:
|
|
|
|
* - 'all' Returns an array of complete term objects (`WP_Term[]`).
|
|
|
|
* - 'all_with_object_id' Returns an array of term objects
|
|
|
|
* with the 'object_id' param (`WP_Term[]`). Works only
|
|
|
|
* when the `$object_ids` parameter is populated.
|
|
|
|
* - 'ids' Returns an array of term IDs (`int[]`).
|
|
|
|
* - 'tt_ids' Returns an array of term taxonomy IDs (`int[]`).
|
|
|
|
* - 'names' Returns an array of term names (`string[]`).
|
|
|
|
* - 'slugs' Returns an array of term slugs (`string[]`).
|
|
|
|
* - 'count' Returns the number of matching terms (`int`).
|
|
|
|
* - 'id=>parent' Returns an associative array of parent term IDs,
|
|
|
|
* keyed by term ID (`int[]`).
|
|
|
|
* - 'id=>name' Returns an associative array of term names,
|
|
|
|
* keyed by term ID (`string[]`).
|
|
|
|
* - 'id=>slug' Returns an associative array of term slugs,
|
|
|
|
* keyed by term ID (`string[]`).
|
|
|
|
* Default 'all'.
|
|
|
|
* @type bool $count Whether to return a term count. If true, will take precedence
|
|
|
|
* over `$fields`. Default false.
|
2021-12-07 13:20:02 +01:00
|
|
|
* @type string|string[] $name Name or array of names to return term(s) for.
|
2021-11-20 01:07:59 +01:00
|
|
|
* Default empty.
|
2021-12-07 13:20:02 +01:00
|
|
|
* @type string|string[] $slug Slug or array of slugs to return term(s) for.
|
2021-11-20 01:07:59 +01:00
|
|
|
* Default empty.
|
|
|
|
* @type int|int[] $term_taxonomy_id Term taxonomy ID, or array of term taxonomy IDs,
|
|
|
|
* to match when querying terms.
|
|
|
|
* @type bool $hierarchical Whether to include terms that have non-empty descendants
|
|
|
|
* (even if `$hide_empty` is set to true). Default true.
|
|
|
|
* @type string $search Search criteria to match terms. Will be SQL-formatted with
|
|
|
|
* wildcards before and after. Default empty.
|
|
|
|
* @type string $name__like Retrieve terms with criteria by which a term is LIKE
|
|
|
|
* `$name__like`. Default empty.
|
|
|
|
* @type string $description__like Retrieve terms where the description is LIKE
|
|
|
|
* `$description__like`. Default empty.
|
|
|
|
* @type bool $pad_counts Whether to pad the quantity of a term's children in the
|
|
|
|
* quantity of each term's "count" object variable.
|
|
|
|
* Default false.
|
|
|
|
* @type string $get Whether to return terms regardless of ancestry or whether the
|
|
|
|
* terms are empty. Accepts 'all' or '' (disabled).
|
|
|
|
* Default ''.
|
|
|
|
* @type int $child_of Term ID to retrieve child terms of. If multiple taxonomies
|
|
|
|
* are passed, `$child_of` is ignored. Default 0.
|
|
|
|
* @type int $parent Parent term ID to retrieve direct-child terms of.
|
|
|
|
* Default empty.
|
|
|
|
* @type bool $childless True to limit results to terms that have no children.
|
|
|
|
* This parameter has no effect on non-hierarchical taxonomies.
|
|
|
|
* Default false.
|
|
|
|
* @type string $cache_domain Unique cache key to be produced when this query is stored in
|
|
|
|
* an object cache. Default 'core'.
|
|
|
|
* @type bool $update_term_meta_cache Whether to prime meta caches for matched terms. Default true.
|
|
|
|
* @type string|string[] $meta_key Meta key or keys to filter by.
|
|
|
|
* @type string|string[] $meta_value Meta value or values to filter by.
|
|
|
|
* @type string $meta_compare MySQL operator used for comparing the meta value.
|
|
|
|
* See WP_Meta_Query::__construct for accepted values and default value.
|
|
|
|
* @type string $meta_compare_key MySQL operator used for comparing the meta key.
|
|
|
|
* See WP_Meta_Query::__construct for accepted values and default value.
|
|
|
|
* @type string $meta_type MySQL data type that the meta_value column will be CAST to for comparisons.
|
|
|
|
* See WP_Meta_Query::__construct for accepted values and default value.
|
|
|
|
* @type string $meta_type_key MySQL data type that the meta_key column will be CAST to for comparisons.
|
|
|
|
* See WP_Meta_Query::__construct for accepted values and default value.
|
|
|
|
* @type array $meta_query An associative array of WP_Meta_Query arguments.
|
|
|
|
* See WP_Meta_Query::__construct for accepted values.
|
2016-05-26 06:33:30 +02:00
|
|
|
* }
|
|
|
|
*/
|
|
|
|
public function __construct( $query = '' ) {
|
|
|
|
$this->query_var_defaults = array(
|
|
|
|
'taxonomy' => null,
|
2016-09-28 05:55:33 +02:00
|
|
|
'object_ids' => null,
|
2016-05-26 06:33:30 +02:00
|
|
|
'orderby' => 'name',
|
|
|
|
'order' => 'ASC',
|
|
|
|
'hide_empty' => true,
|
|
|
|
'include' => array(),
|
|
|
|
'exclude' => array(),
|
|
|
|
'exclude_tree' => array(),
|
|
|
|
'number' => '',
|
|
|
|
'offset' => '',
|
|
|
|
'fields' => 'all',
|
|
|
|
'count' => false,
|
|
|
|
'name' => '',
|
|
|
|
'slug' => '',
|
2016-06-11 05:49:27 +02:00
|
|
|
'term_taxonomy_id' => '',
|
2016-05-26 06:33:30 +02:00
|
|
|
'hierarchical' => true,
|
|
|
|
'search' => '',
|
|
|
|
'name__like' => '',
|
|
|
|
'description__like' => '',
|
|
|
|
'pad_counts' => false,
|
|
|
|
'get' => '',
|
|
|
|
'child_of' => 0,
|
|
|
|
'parent' => '',
|
|
|
|
'childless' => false,
|
|
|
|
'cache_domain' => 'core',
|
|
|
|
'update_term_meta_cache' => true,
|
|
|
|
'meta_query' => '',
|
2016-08-23 16:45:28 +02:00
|
|
|
'meta_key' => '',
|
|
|
|
'meta_value' => '',
|
|
|
|
'meta_type' => '',
|
|
|
|
'meta_compare' => '',
|
2016-05-26 06:33:30 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
if ( ! empty( $query ) ) {
|
|
|
|
$this->query( $query );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse arguments passed to the term query with default query parameters.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
|
|
|
* @param string|array $query WP_Term_Query arguments. See WP_Term_Query::__construct()
|
|
|
|
*/
|
|
|
|
public function parse_query( $query = '' ) {
|
|
|
|
if ( empty( $query ) ) {
|
|
|
|
$query = $this->query_vars;
|
|
|
|
}
|
|
|
|
|
2016-08-03 15:51:30 +02:00
|
|
|
$taxonomies = isset( $query['taxonomy'] ) ? (array) $query['taxonomy'] : null;
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Filters the terms query default arguments.
|
|
|
|
*
|
2016-05-26 17:17:27 +02:00
|
|
|
* Use {@see 'get_terms_args'} to filter the passed arguments.
|
2016-05-26 06:33:30 +02:00
|
|
|
*
|
|
|
|
* @since 4.4.0
|
|
|
|
*
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param array $defaults An array of default get_terms() arguments.
|
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
$this->query_var_defaults = apply_filters( 'get_terms_defaults', $this->query_var_defaults, $taxonomies );
|
|
|
|
|
|
|
|
$query = wp_parse_args( $query, $this->query_var_defaults );
|
|
|
|
|
|
|
|
$query['number'] = absint( $query['number'] );
|
|
|
|
$query['offset'] = absint( $query['offset'] );
|
|
|
|
|
|
|
|
// 'parent' overrides 'child_of'.
|
2020-10-08 23:15:13 +02:00
|
|
|
if ( 0 < (int) $query['parent'] ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$query['child_of'] = false;
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( 'all' === $query['get'] ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$query['childless'] = false;
|
|
|
|
$query['child_of'] = 0;
|
|
|
|
$query['hide_empty'] = 0;
|
2016-05-26 06:33:30 +02:00
|
|
|
$query['hierarchical'] = false;
|
2017-12-01 00:11:00 +01:00
|
|
|
$query['pad_counts'] = false;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$query['taxonomy'] = $taxonomies;
|
|
|
|
|
2016-10-31 16:47:33 +01:00
|
|
|
$this->query_vars = $query;
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Fires after term query vars have been parsed.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
2021-07-30 21:35:58 +02:00
|
|
|
* @param WP_Term_Query $query Current instance of WP_Term_Query.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
do_action( 'parse_term_query', $this );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-08 16:24:07 +01:00
|
|
|
* Sets up the query and retrieves the results.
|
|
|
|
*
|
|
|
|
* The return type varies depending on the value passed to `$args['fields']`. See
|
|
|
|
* WP_Term_Query::get_terms() for details.
|
2016-05-26 06:33:30 +02:00
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
|
|
|
* @param string|array $query Array or URL query string of parameters.
|
2021-01-08 16:24:07 +01:00
|
|
|
* @return WP_Term[]|int[]|string[]|string Array of terms, or number of terms as numeric string
|
|
|
|
* when 'count' is passed as a query var.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
public function query( $query ) {
|
|
|
|
$this->query_vars = wp_parse_args( $query );
|
|
|
|
return $this->get_terms();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-08 16:24:07 +01:00
|
|
|
* Retrieves the query results.
|
|
|
|
*
|
|
|
|
* The return type varies depending on the value passed to `$args['fields']`.
|
|
|
|
*
|
|
|
|
* The following will result in an array of `WP_Term` objects being returned:
|
|
|
|
*
|
|
|
|
* - 'all'
|
|
|
|
* - 'all_with_object_id'
|
|
|
|
*
|
|
|
|
* The following will result in a numeric string being returned:
|
|
|
|
*
|
|
|
|
* - 'count'
|
|
|
|
*
|
|
|
|
* The following will result in an array of text strings being returned:
|
|
|
|
*
|
|
|
|
* - 'id=>name'
|
|
|
|
* - 'id=>slug'
|
|
|
|
* - 'names'
|
|
|
|
* - 'slugs'
|
|
|
|
*
|
|
|
|
* The following will result in an array of numeric strings being returned:
|
|
|
|
*
|
|
|
|
* - 'id=>parent'
|
|
|
|
*
|
|
|
|
* The following will result in an array of integers being returned:
|
|
|
|
*
|
|
|
|
* - 'ids'
|
|
|
|
* - 'tt_ids'
|
|
|
|
*
|
2017-01-04 14:39:43 +01:00
|
|
|
* @since 4.6.0
|
2016-05-26 06:33:30 +02:00
|
|
|
*
|
2016-10-10 08:38:31 +02:00
|
|
|
* @global wpdb $wpdb WordPress database abstraction object.
|
|
|
|
*
|
2021-01-08 16:24:07 +01:00
|
|
|
* @return WP_Term[]|int[]|string[]|string Array of terms, or number of terms as numeric string
|
|
|
|
* when 'count' is passed as a query var.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
public function get_terms() {
|
2016-10-10 08:38:31 +02:00
|
|
|
global $wpdb;
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->parse_query( $this->query_vars );
|
2016-12-21 04:02:40 +01:00
|
|
|
$args = &$this->query_vars;
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
// Set up meta_query so it's available to 'pre_get_terms'.
|
|
|
|
$this->meta_query = new WP_Meta_Query();
|
|
|
|
$this->meta_query->parse_query_vars( $args );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fires before terms are retrieved.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
2021-07-30 21:35:58 +02:00
|
|
|
* @param WP_Term_Query $query Current instance of WP_Term_Query (passed by reference).
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
2020-11-17 21:09:05 +01:00
|
|
|
do_action_ref_array( 'pre_get_terms', array( &$this ) );
|
2016-05-26 06:33:30 +02:00
|
|
|
|
2017-03-03 04:02:43 +01:00
|
|
|
$taxonomies = (array) $args['taxonomy'];
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
// Save queries by not crawling the tree in the case of multiple taxes or a flat tax.
|
|
|
|
$has_hierarchical_tax = false;
|
|
|
|
if ( $taxonomies ) {
|
|
|
|
foreach ( $taxonomies as $_tax ) {
|
|
|
|
if ( is_taxonomy_hierarchical( $_tax ) ) {
|
|
|
|
$has_hierarchical_tax = true;
|
|
|
|
}
|
|
|
|
}
|
2019-08-23 18:05:56 +02:00
|
|
|
} else {
|
|
|
|
// When no taxonomies are provided, assume we have to descend the tree.
|
|
|
|
$has_hierarchical_tax = true;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $has_hierarchical_tax ) {
|
|
|
|
$args['hierarchical'] = false;
|
2017-12-01 00:11:00 +01:00
|
|
|
$args['pad_counts'] = false;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// 'parent' overrides 'child_of'.
|
2020-10-08 23:15:13 +02:00
|
|
|
if ( 0 < (int) $args['parent'] ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$args['child_of'] = false;
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( 'all' === $args['get'] ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$args['childless'] = false;
|
|
|
|
$args['child_of'] = 0;
|
|
|
|
$args['hide_empty'] = 0;
|
2016-05-26 06:33:30 +02:00
|
|
|
$args['hierarchical'] = false;
|
2017-12-01 00:11:00 +01:00
|
|
|
$args['pad_counts'] = false;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filters the terms query arguments.
|
|
|
|
*
|
|
|
|
* @since 3.1.0
|
|
|
|
*
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param array $args An array of get_terms() arguments.
|
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
$args = apply_filters( 'get_terms_args', $args, $taxonomies );
|
|
|
|
|
|
|
|
// Avoid the query if the queried parent/child_of term has no descendants.
|
|
|
|
$child_of = $args['child_of'];
|
|
|
|
$parent = $args['parent'];
|
|
|
|
|
|
|
|
if ( $child_of ) {
|
|
|
|
$_parent = $child_of;
|
|
|
|
} elseif ( $parent ) {
|
|
|
|
$_parent = $parent;
|
|
|
|
} else {
|
|
|
|
$_parent = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( $_parent ) {
|
|
|
|
$in_hierarchy = false;
|
|
|
|
foreach ( $taxonomies as $_tax ) {
|
|
|
|
$hierarchy = _get_term_hierarchy( $_tax );
|
|
|
|
|
|
|
|
if ( isset( $hierarchy[ $_parent ] ) ) {
|
|
|
|
$in_hierarchy = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $in_hierarchy ) {
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( 'count' === $args['fields'] ) {
|
2017-11-20 23:46:47 +01:00
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
$this->terms = array();
|
|
|
|
return $this->terms;
|
|
|
|
}
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-28 05:55:33 +02:00
|
|
|
// 'term_order' is a legal sort order only when joining the relationship table.
|
|
|
|
$_orderby = $this->query_vars['orderby'];
|
|
|
|
if ( 'term_order' === $_orderby && empty( $this->query_vars['object_ids'] ) ) {
|
|
|
|
$_orderby = 'term_id';
|
|
|
|
}
|
2020-06-03 16:37:09 +02:00
|
|
|
|
2016-09-28 05:55:33 +02:00
|
|
|
$orderby = $this->parse_orderby( $_orderby );
|
|
|
|
|
2016-07-19 04:13:28 +02:00
|
|
|
if ( $orderby ) {
|
|
|
|
$orderby = "ORDER BY $orderby";
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
$order = $this->parse_order( $this->query_vars['order'] );
|
|
|
|
|
|
|
|
if ( $taxonomies ) {
|
2022-03-21 12:56:04 +01:00
|
|
|
$this->sql_clauses['where']['taxonomy'] =
|
|
|
|
"tt.taxonomy IN ('" . implode( "', '", array_map( 'esc_sql', $taxonomies ) ) . "')";
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
2022-04-29 14:36:13 +02:00
|
|
|
if ( empty( $args['exclude'] ) ) {
|
|
|
|
$args['exclude'] = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( empty( $args['include'] ) ) {
|
|
|
|
$args['include'] = array();
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
$exclude = $args['exclude'];
|
|
|
|
$exclude_tree = $args['exclude_tree'];
|
|
|
|
$include = $args['include'];
|
|
|
|
|
|
|
|
$inclusions = '';
|
|
|
|
if ( ! empty( $include ) ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$exclude = '';
|
2016-05-26 06:33:30 +02:00
|
|
|
$exclude_tree = '';
|
2017-12-01 00:11:00 +01:00
|
|
|
$inclusions = implode( ',', wp_parse_id_list( $include ) );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $inclusions ) ) {
|
|
|
|
$this->sql_clauses['where']['inclusions'] = 't.term_id IN ( ' . $inclusions . ' )';
|
|
|
|
}
|
|
|
|
|
|
|
|
$exclusions = array();
|
|
|
|
if ( ! empty( $exclude_tree ) ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$exclude_tree = wp_parse_id_list( $exclude_tree );
|
2016-05-26 06:33:30 +02:00
|
|
|
$excluded_children = $exclude_tree;
|
|
|
|
foreach ( $exclude_tree as $extrunk ) {
|
|
|
|
$excluded_children = array_merge(
|
|
|
|
$excluded_children,
|
2017-12-01 00:11:00 +01:00
|
|
|
(array) get_terms(
|
2018-08-17 03:51:36 +02:00
|
|
|
array(
|
2019-08-03 05:35:56 +02:00
|
|
|
'taxonomy' => reset( $taxonomies ),
|
2020-10-08 23:15:13 +02:00
|
|
|
'child_of' => (int) $extrunk,
|
2017-12-01 00:11:00 +01:00
|
|
|
'fields' => 'ids',
|
|
|
|
'hide_empty' => 0,
|
|
|
|
)
|
|
|
|
)
|
2016-05-26 06:33:30 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
$exclusions = array_merge( $excluded_children, $exclusions );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $exclude ) ) {
|
|
|
|
$exclusions = array_merge( wp_parse_id_list( $exclude ), $exclusions );
|
|
|
|
}
|
|
|
|
|
|
|
|
// 'childless' terms are those without an entry in the flattened term hierarchy.
|
|
|
|
$childless = (bool) $args['childless'];
|
|
|
|
if ( $childless ) {
|
|
|
|
foreach ( $taxonomies as $_tax ) {
|
|
|
|
$term_hierarchy = _get_term_hierarchy( $_tax );
|
2017-12-01 00:11:00 +01:00
|
|
|
$exclusions = array_merge( array_keys( $term_hierarchy ), $exclusions );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $exclusions ) ) {
|
|
|
|
$exclusions = 't.term_id NOT IN (' . implode( ',', array_map( 'intval', $exclusions ) ) . ')';
|
|
|
|
} else {
|
|
|
|
$exclusions = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filters the terms to exclude from the terms query.
|
|
|
|
*
|
|
|
|
* @since 2.3.0
|
|
|
|
*
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param string $exclusions `NOT IN` clause of the terms query.
|
|
|
|
* @param array $args An array of terms query arguments.
|
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
$exclusions = apply_filters( 'list_terms_exclusions', $exclusions, $args, $taxonomies );
|
|
|
|
|
|
|
|
if ( ! empty( $exclusions ) ) {
|
|
|
|
// Must do string manipulation here for backward compatibility with filter.
|
|
|
|
$this->sql_clauses['where']['exclusions'] = preg_replace( '/^\s*AND\s*/', '', $exclusions );
|
|
|
|
}
|
|
|
|
|
2022-03-21 12:56:04 +01:00
|
|
|
if ( '' === $args['name'] ) {
|
|
|
|
$args['name'] = array();
|
|
|
|
} else {
|
|
|
|
$args['name'] = (array) $args['name'];
|
|
|
|
}
|
|
|
|
|
2022-03-21 04:55:03 +01:00
|
|
|
if ( ! empty( $args['name'] ) ) {
|
|
|
|
$names = $args['name'];
|
2016-05-26 06:33:30 +02:00
|
|
|
foreach ( $names as &$_name ) {
|
|
|
|
// `sanitize_term_field()` returns slashed data.
|
|
|
|
$_name = stripslashes( sanitize_term_field( 'name', $_name, 0, reset( $taxonomies ), 'db' ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->sql_clauses['where']['name'] = "t.name IN ('" . implode( "', '", array_map( 'esc_sql', $names ) ) . "')";
|
|
|
|
}
|
|
|
|
|
2022-03-21 12:56:04 +01:00
|
|
|
if ( '' === $args['slug'] ) {
|
|
|
|
$args['slug'] = array();
|
|
|
|
} else {
|
|
|
|
$args['slug'] = array_map( 'sanitize_title', (array) $args['slug'] );
|
|
|
|
}
|
|
|
|
|
2022-03-21 04:55:03 +01:00
|
|
|
if ( ! empty( $args['slug'] ) ) {
|
|
|
|
$slug = implode( "', '", $args['slug'] );
|
|
|
|
|
|
|
|
$this->sql_clauses['where']['slug'] = "t.slug IN ('" . $slug . "')";
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
2022-03-21 12:56:04 +01:00
|
|
|
if ( '' === $args['term_taxonomy_id'] ) {
|
|
|
|
$args['term_taxonomy_id'] = array();
|
|
|
|
} else {
|
|
|
|
$args['term_taxonomy_id'] = array_map( 'intval', (array) $args['term_taxonomy_id'] );
|
|
|
|
}
|
|
|
|
|
2016-06-11 05:49:27 +02:00
|
|
|
if ( ! empty( $args['term_taxonomy_id'] ) ) {
|
2022-03-21 04:55:03 +01:00
|
|
|
$tt_ids = implode( ',', $args['term_taxonomy_id'] );
|
|
|
|
|
|
|
|
$this->sql_clauses['where']['term_taxonomy_id'] = "tt.term_taxonomy_id IN ({$tt_ids})";
|
2016-06-11 05:49:27 +02:00
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
if ( ! empty( $args['name__like'] ) ) {
|
2022-03-21 12:56:04 +01:00
|
|
|
$this->sql_clauses['where']['name__like'] = $wpdb->prepare(
|
|
|
|
't.name LIKE %s',
|
|
|
|
'%' . $wpdb->esc_like( $args['name__like'] ) . '%'
|
|
|
|
);
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $args['description__like'] ) ) {
|
2022-03-21 12:56:04 +01:00
|
|
|
$this->sql_clauses['where']['description__like'] = $wpdb->prepare(
|
|
|
|
'tt.description LIKE %s',
|
|
|
|
'%' . $wpdb->esc_like( $args['description__like'] ) . '%'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( '' === $args['object_ids'] ) {
|
|
|
|
$args['object_ids'] = array();
|
|
|
|
} else {
|
|
|
|
$args['object_ids'] = array_map( 'intval', (array) $args['object_ids'] );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
2016-09-28 05:55:33 +02:00
|
|
|
if ( ! empty( $args['object_ids'] ) ) {
|
2022-03-21 04:55:03 +01:00
|
|
|
$object_ids = implode( ', ', $args['object_ids'] );
|
2016-09-28 05:55:33 +02:00
|
|
|
|
|
|
|
$this->sql_clauses['where']['object_ids'] = "tr.object_id IN ($object_ids)";
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When querying for object relationships, the 'count > 0' check
|
|
|
|
* added by 'hide_empty' is superfluous.
|
|
|
|
*/
|
|
|
|
if ( ! empty( $args['object_ids'] ) ) {
|
|
|
|
$args['hide_empty'] = false;
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
if ( '' !== $parent ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$parent = (int) $parent;
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->sql_clauses['where']['parent'] = "tt.parent = '$parent'";
|
|
|
|
}
|
|
|
|
|
|
|
|
$hierarchical = $args['hierarchical'];
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( 'count' === $args['fields'] ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$hierarchical = false;
|
|
|
|
}
|
2017-12-01 00:11:00 +01:00
|
|
|
if ( $args['hide_empty'] && ! $hierarchical ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->sql_clauses['where']['count'] = 'tt.count > 0';
|
|
|
|
}
|
|
|
|
|
|
|
|
$number = $args['number'];
|
|
|
|
$offset = $args['offset'];
|
|
|
|
|
|
|
|
// Don't limit the query results when we have to descend the family tree.
|
|
|
|
if ( $number && ! $hierarchical && ! $child_of && '' === $parent ) {
|
|
|
|
if ( $offset ) {
|
|
|
|
$limits = 'LIMIT ' . $offset . ',' . $number;
|
|
|
|
} else {
|
|
|
|
$limits = 'LIMIT ' . $number;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$limits = '';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! empty( $args['search'] ) ) {
|
|
|
|
$this->sql_clauses['where']['search'] = $this->get_search_sql( $args['search'] );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Meta query support.
|
2017-12-01 00:11:00 +01:00
|
|
|
$join = '';
|
2016-05-26 06:33:30 +02:00
|
|
|
$distinct = '';
|
|
|
|
|
|
|
|
// Reparse meta_query query_vars, in case they were modified in a 'pre_get_terms' callback.
|
|
|
|
$this->meta_query->parse_query_vars( $this->query_vars );
|
2017-12-01 00:11:00 +01:00
|
|
|
$mq_sql = $this->meta_query->get_sql( 'term', 't', 'term_id' );
|
2016-05-26 06:33:30 +02:00
|
|
|
$meta_clauses = $this->meta_query->get_clauses();
|
|
|
|
|
|
|
|
if ( ! empty( $meta_clauses ) ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$join .= $mq_sql['join'];
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->sql_clauses['where']['meta_query'] = preg_replace( '/^\s*AND\s*/', '', $mq_sql['where'] );
|
2017-12-01 00:11:00 +01:00
|
|
|
$distinct .= 'DISTINCT';
|
2017-10-16 20:35:47 +02:00
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$selects = array();
|
|
|
|
switch ( $args['fields'] ) {
|
|
|
|
case 'count':
|
|
|
|
$orderby = '';
|
2017-12-01 00:11:00 +01:00
|
|
|
$order = '';
|
2016-05-26 06:33:30 +02:00
|
|
|
$selects = array( 'COUNT(*)' );
|
|
|
|
break;
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
default:
|
|
|
|
$selects = array( 't.term_id' );
|
|
|
|
if ( 'all_with_object_id' === $args['fields'] && ! empty( $args['object_ids'] ) ) {
|
|
|
|
$selects[] = 'tr.object_id';
|
|
|
|
}
|
2016-05-26 06:33:30 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
$_fields = $args['fields'];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filters the fields to select in the terms query.
|
|
|
|
*
|
|
|
|
* Field lists modified using this filter will only modify the term fields returned
|
|
|
|
* by the function when the `$fields` parameter set to 'count' or 'all'. In all other
|
|
|
|
* cases, the term fields in the results array will be determined by the `$fields`
|
|
|
|
* parameter alone.
|
|
|
|
*
|
|
|
|
* Use of this filter can result in unpredictable behavior, and is not recommended.
|
|
|
|
*
|
|
|
|
* @since 2.8.0
|
|
|
|
*
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param string[] $selects An array of fields to select for the terms query.
|
|
|
|
* @param array $args An array of term query arguments.
|
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
$fields = implode( ', ', apply_filters( 'get_terms_fields', $selects, $args, $taxonomies ) );
|
|
|
|
|
2016-10-10 08:38:31 +02:00
|
|
|
$join .= " INNER JOIN $wpdb->term_taxonomy AS tt ON t.term_id = tt.term_id";
|
2016-05-26 06:33:30 +02:00
|
|
|
|
2016-09-28 05:55:33 +02:00
|
|
|
if ( ! empty( $this->query_vars['object_ids'] ) ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$join .= " INNER JOIN {$wpdb->term_relationships} AS tr ON tr.term_taxonomy_id = tt.term_taxonomy_id";
|
|
|
|
$distinct = 'DISTINCT';
|
2016-09-28 05:55:33 +02:00
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
$where = implode( ' AND ', $this->sql_clauses['where'] );
|
|
|
|
|
2022-04-14 02:04:11 +02:00
|
|
|
$clauses = array( 'fields', 'join', 'where', 'distinct', 'orderby', 'order', 'limits' );
|
2022-03-21 12:56:04 +01:00
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
/**
|
|
|
|
* Filters the terms query SQL clauses.
|
|
|
|
*
|
|
|
|
* @since 3.1.0
|
|
|
|
*
|
2022-04-28 11:46:21 +02:00
|
|
|
* @param string[] $clauses {
|
|
|
|
* Associative array of the clauses for the query.
|
|
|
|
*
|
|
|
|
* @type string $fields The SELECT clause of the query.
|
|
|
|
* @type string $join The JOIN clause of the query.
|
|
|
|
* @type string $where The WHERE clause of the query.
|
|
|
|
* @type string $distinct The DISTINCT clause of the query.
|
|
|
|
* @type string $orderby The ORDER BY clause of the query.
|
|
|
|
* @type string $order The ORDER clause of the query.
|
|
|
|
* @type string $limits The LIMIT clause of the query.
|
|
|
|
* }
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
|
|
|
* @param array $args An array of term query arguments.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
2022-04-14 02:04:11 +02:00
|
|
|
$clauses = apply_filters( 'terms_clauses', compact( $clauses ), $taxonomies, $args );
|
2016-05-26 06:33:30 +02:00
|
|
|
|
2017-12-01 00:11:00 +01:00
|
|
|
$fields = isset( $clauses['fields'] ) ? $clauses['fields'] : '';
|
|
|
|
$join = isset( $clauses['join'] ) ? $clauses['join'] : '';
|
|
|
|
$where = isset( $clauses['where'] ) ? $clauses['where'] : '';
|
|
|
|
$distinct = isset( $clauses['distinct'] ) ? $clauses['distinct'] : '';
|
|
|
|
$orderby = isset( $clauses['orderby'] ) ? $clauses['orderby'] : '';
|
|
|
|
$order = isset( $clauses['order'] ) ? $clauses['order'] : '';
|
|
|
|
$limits = isset( $clauses['limits'] ) ? $clauses['limits'] : '';
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
if ( $where ) {
|
|
|
|
$where = "WHERE $where";
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->sql_clauses['select'] = "SELECT $distinct $fields";
|
2016-10-10 08:38:31 +02:00
|
|
|
$this->sql_clauses['from'] = "FROM $wpdb->terms AS t $join";
|
2016-07-19 04:13:28 +02:00
|
|
|
$this->sql_clauses['orderby'] = $orderby ? "$orderby $order" : '';
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->sql_clauses['limits'] = $limits;
|
|
|
|
|
2022-03-22 15:56:05 +01:00
|
|
|
$this->request = "
|
|
|
|
{$this->sql_clauses['select']}
|
|
|
|
{$this->sql_clauses['from']}
|
|
|
|
{$where}
|
|
|
|
{$this->sql_clauses['orderby']}
|
|
|
|
{$this->sql_clauses['limits']}
|
|
|
|
";
|
2016-05-26 06:33:30 +02:00
|
|
|
|
2019-07-01 22:48:58 +02:00
|
|
|
$this->terms = null;
|
|
|
|
|
|
|
|
/**
|
2020-08-11 02:34:08 +02:00
|
|
|
* Filters the terms array before the query takes place.
|
2019-07-01 22:48:58 +02:00
|
|
|
*
|
2020-09-17 12:45:03 +02:00
|
|
|
* Return a non-null value to bypass WordPress' default term queries.
|
2019-07-01 22:48:58 +02:00
|
|
|
*
|
|
|
|
* @since 5.3.0
|
|
|
|
*
|
|
|
|
* @param array|null $terms Return an array of term data to short-circuit WP's term query,
|
|
|
|
* or null to allow WP queries to run normally.
|
2021-01-08 15:30:14 +01:00
|
|
|
* @param WP_Term_Query $query The WP_Term_Query instance, passed by reference.
|
2019-07-01 22:48:58 +02:00
|
|
|
*/
|
|
|
|
$this->terms = apply_filters_ref_array( 'terms_pre_query', array( $this->terms, &$this ) );
|
|
|
|
|
|
|
|
if ( null !== $this->terms ) {
|
|
|
|
return $this->terms;
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
// $args can be anything. Only use the args defined in defaults to compute the key.
|
2022-03-21 04:55:03 +01:00
|
|
|
$cache_args = wp_array_slice_assoc( $args, array_keys( $this->query_var_defaults ) );
|
2022-03-21 12:56:04 +01:00
|
|
|
|
2022-03-21 04:55:03 +01:00
|
|
|
unset( $cache_args['pad_counts'], $cache_args['update_term_meta_cache'] );
|
2022-03-21 12:56:04 +01:00
|
|
|
|
2022-03-21 04:55:03 +01:00
|
|
|
if ( 'count' !== $_fields && 'all_with_object_id' !== $_fields ) {
|
|
|
|
$cache_args['fields'] = 'all';
|
|
|
|
}
|
2022-03-21 12:56:04 +01:00
|
|
|
|
2022-03-21 04:55:03 +01:00
|
|
|
$key = md5( serialize( $cache_args ) . serialize( $taxonomies ) . $this->request );
|
2016-10-21 04:54:34 +02:00
|
|
|
$last_changed = wp_cache_get_last_changed( 'terms' );
|
2017-12-01 00:11:00 +01:00
|
|
|
$cache_key = "get_terms:$key:$last_changed";
|
|
|
|
$cache = wp_cache_get( $cache_key, 'terms' );
|
2022-03-21 12:56:04 +01:00
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
if ( false !== $cache ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
if ( 'ids' === $_fields ) {
|
|
|
|
$term_ids = wp_list_pluck( $cache, 'term_id' );
|
|
|
|
$cache = array_map( 'intval', $term_ids );
|
|
|
|
} elseif ( 'count' !== $_fields ) {
|
|
|
|
$term_ids = wp_list_pluck( $cache, 'term_id' );
|
|
|
|
_prime_term_caches( $term_ids, $args['update_term_meta_cache'] );
|
|
|
|
$term_objects = $this->populate_terms( $cache );
|
|
|
|
$cache = $this->format_terms( $term_objects, $_fields );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
2016-08-07 06:48:29 +02:00
|
|
|
$this->terms = $cache;
|
|
|
|
return $this->terms;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( 'count' === $_fields ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$count = $wpdb->get_var( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
|
2016-10-12 17:30:31 +02:00
|
|
|
wp_cache_set( $cache_key, $count, 'terms' );
|
|
|
|
return $count;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$terms = $wpdb->get_results( $this->request ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
|
2016-05-26 06:33:30 +02:00
|
|
|
|
|
|
|
if ( empty( $terms ) ) {
|
2022-02-03 18:43:06 +01:00
|
|
|
wp_cache_add( $cache_key, array(), 'terms' );
|
2016-05-26 06:33:30 +02:00
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$term_ids = wp_list_pluck( $terms, 'term_id' );
|
|
|
|
_prime_term_caches( $term_ids, false );
|
|
|
|
$term_objects = $this->populate_terms( $terms );
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
if ( $child_of ) {
|
|
|
|
foreach ( $taxonomies as $_tax ) {
|
|
|
|
$children = _get_term_hierarchy( $_tax );
|
|
|
|
if ( ! empty( $children ) ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$term_objects = _get_term_children( $child_of, $term_objects, $_tax );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update term counts to include children.
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( $args['pad_counts'] && 'all' === $_fields ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
foreach ( $taxonomies as $_tax ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
_pad_term_counts( $term_objects, $_tax );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we show empty categories that have children.
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
if ( $hierarchical && $args['hide_empty'] && is_array( $term_objects ) ) {
|
|
|
|
foreach ( $term_objects as $k => $term ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
if ( ! $term->count ) {
|
|
|
|
$children = get_term_children( $term->term_id, $term->taxonomy );
|
|
|
|
if ( is_array( $children ) ) {
|
|
|
|
foreach ( $children as $child_id ) {
|
|
|
|
$child = get_term( $child_id, $term->taxonomy );
|
|
|
|
if ( $child->count ) {
|
|
|
|
continue 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// It really is empty.
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
unset( $term_objects[ $k ] );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-28 05:55:33 +02:00
|
|
|
/*
|
|
|
|
* When querying for terms connected to objects, we may get
|
|
|
|
* duplicate results. The duplicates should be preserved if
|
|
|
|
* `$fields` is 'all_with_object_id', but should otherwise be
|
|
|
|
* removed.
|
|
|
|
*/
|
2020-05-16 20:42:12 +02:00
|
|
|
if ( ! empty( $args['object_ids'] ) && 'all_with_object_id' !== $_fields ) {
|
2019-07-03 01:42:58 +02:00
|
|
|
$_tt_ids = array();
|
|
|
|
$_terms = array();
|
2016-09-28 05:55:33 +02:00
|
|
|
foreach ( $terms as $term ) {
|
|
|
|
if ( isset( $_tt_ids[ $term->term_id ] ) ) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$_tt_ids[ $term->term_id ] = 1;
|
2017-12-01 00:11:00 +01:00
|
|
|
$_terms[] = $term;
|
2016-09-28 05:55:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$terms = $_terms;
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
// Hierarchical queries are not limited, so 'offset' and 'number' must be handled now.
|
|
|
|
if ( $hierarchical && $number && is_array( $terms ) ) {
|
|
|
|
if ( $offset >= count( $terms ) ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$terms = array();
|
|
|
|
$term_objects = array();
|
2016-05-26 06:33:30 +02:00
|
|
|
} else {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$terms = array_slice( $terms, $offset, $number, true );
|
|
|
|
$term_objects = array_slice( $term_objects, $offset, $number, true );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
// Prime termmeta cache.
|
|
|
|
if ( $args['update_term_meta_cache'] ) {
|
|
|
|
$term_ids = wp_list_pluck( $term_objects, 'term_id' );
|
|
|
|
update_termmeta_cache( $term_ids );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
wp_cache_add( $cache_key, $terms, 'terms' );
|
|
|
|
$terms = $this->format_terms( $term_objects, $_fields );
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
$this->terms = $terms;
|
|
|
|
return $this->terms;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse and sanitize 'orderby' keys passed to the term query.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
2016-10-10 08:38:31 +02:00
|
|
|
* @global wpdb $wpdb WordPress database abstraction object.
|
|
|
|
*
|
2016-05-26 06:33:30 +02:00
|
|
|
* @param string $orderby_raw Alias for the field to order by.
|
|
|
|
* @return string|false Value to used in the ORDER clause. False otherwise.
|
|
|
|
*/
|
|
|
|
protected function parse_orderby( $orderby_raw ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$_orderby = strtolower( $orderby_raw );
|
2016-05-26 06:33:30 +02:00
|
|
|
$maybe_orderby_meta = false;
|
2016-09-28 05:55:33 +02:00
|
|
|
|
|
|
|
if ( in_array( $_orderby, array( 'term_id', 'name', 'slug', 'term_group' ), true ) ) {
|
|
|
|
$orderby = "t.$_orderby";
|
|
|
|
} elseif ( in_array( $_orderby, array( 'count', 'parent', 'taxonomy', 'term_taxonomy_id', 'description' ), true ) ) {
|
|
|
|
$orderby = "tt.$_orderby";
|
|
|
|
} elseif ( 'term_order' === $_orderby ) {
|
|
|
|
$orderby = 'tr.term_order';
|
2020-05-16 20:42:12 +02:00
|
|
|
} elseif ( 'include' === $_orderby && ! empty( $this->query_vars['include'] ) ) {
|
2016-09-01 18:51:30 +02:00
|
|
|
$include = implode( ',', wp_parse_id_list( $this->query_vars['include'] ) );
|
2016-05-26 06:33:30 +02:00
|
|
|
$orderby = "FIELD( t.term_id, $include )";
|
2020-05-16 20:42:12 +02:00
|
|
|
} elseif ( 'slug__in' === $_orderby && ! empty( $this->query_vars['slug'] ) && is_array( $this->query_vars['slug'] ) ) {
|
2017-12-01 00:11:00 +01:00
|
|
|
$slugs = implode( "', '", array_map( 'sanitize_title_for_query', $this->query_vars['slug'] ) );
|
2017-10-05 02:37:47 +02:00
|
|
|
$orderby = "FIELD( t.slug, '" . $slugs . "')";
|
2020-05-16 20:42:12 +02:00
|
|
|
} elseif ( 'none' === $_orderby ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$orderby = '';
|
2020-05-16 20:42:12 +02:00
|
|
|
} elseif ( empty( $_orderby ) || 'id' === $_orderby || 'term_id' === $_orderby ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
$orderby = 't.term_id';
|
|
|
|
} else {
|
|
|
|
$orderby = 't.name';
|
|
|
|
|
|
|
|
// This may be a value of orderby related to meta.
|
|
|
|
$maybe_orderby_meta = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filters the ORDERBY clause of the terms query.
|
|
|
|
*
|
|
|
|
* @since 2.8.0
|
|
|
|
*
|
2018-03-25 21:33:31 +02:00
|
|
|
* @param string $orderby `ORDERBY` clause of the terms query.
|
|
|
|
* @param array $args An array of term query arguments.
|
|
|
|
* @param string[] $taxonomies An array of taxonomy names.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
$orderby = apply_filters( 'get_terms_orderby', $orderby, $this->query_vars, $this->query_vars['taxonomy'] );
|
|
|
|
|
|
|
|
// Run after the 'get_terms_orderby' filter for backward compatibility.
|
|
|
|
if ( $maybe_orderby_meta ) {
|
|
|
|
$maybe_orderby_meta = $this->parse_orderby_meta( $_orderby );
|
|
|
|
if ( $maybe_orderby_meta ) {
|
|
|
|
$orderby = $maybe_orderby_meta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $orderby;
|
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
/**
|
|
|
|
* Format response depending on field requested.
|
|
|
|
*
|
|
|
|
* @since 6.0.0
|
|
|
|
*
|
|
|
|
* @param WP_Term[] $term_objects Array of term objects.
|
2022-04-28 11:59:13 +02:00
|
|
|
* @param string $_fields Field to format.
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
*
|
|
|
|
* @return WP_Term[]|int[]|string[] Array of terms / strings / ints depending on field requested.
|
|
|
|
*/
|
|
|
|
protected function format_terms( $term_objects, $_fields ) {
|
|
|
|
$_terms = array();
|
|
|
|
if ( 'id=>parent' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[ $term->term_id ] = $term->parent;
|
|
|
|
}
|
|
|
|
} elseif ( 'ids' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[] = (int) $term->term_id;
|
|
|
|
}
|
|
|
|
} elseif ( 'tt_ids' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[] = (int) $term->term_taxonomy_id;
|
|
|
|
}
|
|
|
|
} elseif ( 'names' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[] = $term->name;
|
|
|
|
}
|
|
|
|
} elseif ( 'slugs' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[] = $term->slug;
|
|
|
|
}
|
|
|
|
} elseif ( 'id=>name' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[ $term->term_id ] = $term->name;
|
|
|
|
}
|
|
|
|
} elseif ( 'id=>slug' === $_fields ) {
|
|
|
|
foreach ( $term_objects as $term ) {
|
|
|
|
$_terms[ $term->term_id ] = $term->slug;
|
|
|
|
}
|
|
|
|
} elseif ( 'all' === $_fields || 'all_with_object_id' === $_fields ) {
|
|
|
|
$_terms = $term_objects;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $_terms;
|
|
|
|
}
|
|
|
|
|
2016-05-26 06:33:30 +02:00
|
|
|
/**
|
|
|
|
* Generate the ORDER BY clause for an 'orderby' param that is potentially related to a meta query.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
|
|
|
* @param string $orderby_raw Raw 'orderby' value passed to WP_Term_Query.
|
2017-01-04 14:39:43 +01:00
|
|
|
* @return string ORDER BY clause.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
|
|
|
protected function parse_orderby_meta( $orderby_raw ) {
|
|
|
|
$orderby = '';
|
|
|
|
|
2016-06-26 16:26:29 +02:00
|
|
|
// Tell the meta query to generate its SQL, so we have access to table aliases.
|
|
|
|
$this->meta_query->get_sql( 'term', 't', 'term_id' );
|
2016-05-26 06:33:30 +02:00
|
|
|
$meta_clauses = $this->meta_query->get_clauses();
|
|
|
|
if ( ! $meta_clauses || ! $orderby_raw ) {
|
|
|
|
return $orderby;
|
|
|
|
}
|
|
|
|
|
2017-12-01 00:11:00 +01:00
|
|
|
$allowed_keys = array();
|
|
|
|
$primary_meta_key = null;
|
2016-05-26 06:33:30 +02:00
|
|
|
$primary_meta_query = reset( $meta_clauses );
|
|
|
|
if ( ! empty( $primary_meta_query['key'] ) ) {
|
|
|
|
$primary_meta_key = $primary_meta_query['key'];
|
2017-12-01 00:11:00 +01:00
|
|
|
$allowed_keys[] = $primary_meta_key;
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
$allowed_keys[] = 'meta_value';
|
|
|
|
$allowed_keys[] = 'meta_value_num';
|
|
|
|
$allowed_keys = array_merge( $allowed_keys, array_keys( $meta_clauses ) );
|
|
|
|
|
|
|
|
if ( ! in_array( $orderby_raw, $allowed_keys, true ) ) {
|
|
|
|
return $orderby;
|
|
|
|
}
|
|
|
|
|
2017-12-01 00:11:00 +01:00
|
|
|
switch ( $orderby_raw ) {
|
2016-05-26 06:33:30 +02:00
|
|
|
case $primary_meta_key:
|
|
|
|
case 'meta_value':
|
|
|
|
if ( ! empty( $primary_meta_query['type'] ) ) {
|
|
|
|
$orderby = "CAST({$primary_meta_query['alias']}.meta_value AS {$primary_meta_query['cast']})";
|
|
|
|
} else {
|
|
|
|
$orderby = "{$primary_meta_query['alias']}.meta_value";
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'meta_value_num':
|
|
|
|
$orderby = "{$primary_meta_query['alias']}.meta_value+0";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if ( array_key_exists( $orderby_raw, $meta_clauses ) ) {
|
|
|
|
// $orderby corresponds to a meta_query clause.
|
|
|
|
$meta_clause = $meta_clauses[ $orderby_raw ];
|
2017-12-01 00:11:00 +01:00
|
|
|
$orderby = "CAST({$meta_clause['alias']}.meta_value AS {$meta_clause['cast']})";
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $orderby;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse an 'order' query variable and cast it to ASC or DESC as necessary.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
|
|
|
* @param string $order The 'order' query variable.
|
|
|
|
* @return string The sanitized 'order' query variable.
|
|
|
|
*/
|
|
|
|
protected function parse_order( $order ) {
|
|
|
|
if ( ! is_string( $order ) || empty( $order ) ) {
|
|
|
|
return 'DESC';
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( 'ASC' === strtoupper( $order ) ) {
|
|
|
|
return 'ASC';
|
|
|
|
} else {
|
|
|
|
return 'DESC';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used internally to generate a SQL string related to the 'search' parameter.
|
|
|
|
*
|
|
|
|
* @since 4.6.0
|
|
|
|
*
|
2016-10-10 08:38:31 +02:00
|
|
|
* @global wpdb $wpdb WordPress database abstraction object.
|
|
|
|
*
|
2022-04-26 16:04:10 +02:00
|
|
|
* @param string $search Search string.
|
|
|
|
* @return string Search SQL.
|
2016-05-26 06:33:30 +02:00
|
|
|
*/
|
Code Modernization: Rename parameters that use reserved keywords in `wp-includes/class-wp-term-query.php`.
While using reserved PHP keywords as parameter name labels is allowed, in the context of function calls using named parameters in PHP 8.0+, this will easily lead to confusion. To avoid that, it is recommended not to use reserved keywords as function parameter names.
This commit renames the `$string` parameter to `$search` in `WP_Term_Query::get_search_sql()`.
Follow-up to [52946], [52996], [52997], [52998], [53003], [53014], [53029], [53039], [53116], [53117], [53137], [53174], [53184], [53185], [53192], [53193], [53198], [53203], [53207], [53215], [53216], [53220], [53230], [53232], [53236], [53239], [53240], [53242], [53243], [53245], [53246], [53257], [53269], [53270], [53271], [53272], [53273], [53274].
Props jrf, aristath, poena, justinahinon, SergeyBiryukov.
See #55327.
Built from https://develop.svn.wordpress.org/trunk@53275
git-svn-id: http://core.svn.wordpress.org/trunk@52864 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-04-26 13:46:18 +02:00
|
|
|
protected function get_search_sql( $search ) {
|
2016-10-10 08:38:31 +02:00
|
|
|
global $wpdb;
|
|
|
|
|
Code Modernization: Rename parameters that use reserved keywords in `wp-includes/class-wp-term-query.php`.
While using reserved PHP keywords as parameter name labels is allowed, in the context of function calls using named parameters in PHP 8.0+, this will easily lead to confusion. To avoid that, it is recommended not to use reserved keywords as function parameter names.
This commit renames the `$string` parameter to `$search` in `WP_Term_Query::get_search_sql()`.
Follow-up to [52946], [52996], [52997], [52998], [53003], [53014], [53029], [53039], [53116], [53117], [53137], [53174], [53184], [53185], [53192], [53193], [53198], [53203], [53207], [53215], [53216], [53220], [53230], [53232], [53236], [53239], [53240], [53242], [53243], [53245], [53246], [53257], [53269], [53270], [53271], [53272], [53273], [53274].
Props jrf, aristath, poena, justinahinon, SergeyBiryukov.
See #55327.
Built from https://develop.svn.wordpress.org/trunk@53275
git-svn-id: http://core.svn.wordpress.org/trunk@52864 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-04-26 13:46:18 +02:00
|
|
|
$like = '%' . $wpdb->esc_like( $search ) . '%';
|
2016-05-26 06:33:30 +02:00
|
|
|
|
2016-10-10 08:38:31 +02:00
|
|
|
return $wpdb->prepare( '((t.name LIKE %s) OR (t.slug LIKE %s))', $like, $like );
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|
2018-04-30 23:08:22 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates an array of term objects from an array of term IDs.
|
|
|
|
*
|
|
|
|
* Also discards invalid term objects.
|
|
|
|
*
|
2018-07-17 18:25:26 +02:00
|
|
|
* @since 4.9.8
|
2018-04-30 23:08:22 +02:00
|
|
|
*
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
* @param Object[]|int[] $terms List of objects or term ids.
|
|
|
|
* @return WP_Term[] Array of `WP_Term` objects.
|
2018-04-30 23:08:22 +02:00
|
|
|
*/
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
protected function populate_terms( $terms ) {
|
|
|
|
$term_objects = array();
|
|
|
|
if ( ! is_array( $terms ) ) {
|
|
|
|
return $term_objects;
|
2018-04-30 23:08:22 +02:00
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
foreach ( $terms as $key => $term_data ) {
|
|
|
|
if ( is_object( $term_data ) && property_exists( $term_data, 'term_id' ) ) {
|
|
|
|
$term = get_term( $term_data->term_id );
|
|
|
|
if ( property_exists( $term_data, 'object_id' ) ) {
|
|
|
|
$term->object_id = (int) $term_data->object_id;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$term = get_term( $term_data );
|
|
|
|
}
|
|
|
|
|
2018-04-30 23:08:22 +02:00
|
|
|
if ( $term instanceof WP_Term ) {
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
$term_objects[ $key ] = $term;
|
2018-04-30 23:08:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Taxonomy: Only store term_ids and object_ids in `WP_Term_Query` query caches.
The query cache currently implemented in `WP_Term_Query` caches the final output of the query, depending on what fields are requested. This is wasteful, as if a user requests `fields` => `all`, then an unlimited array of `WP_Term` objects could be stored in the object cache. Instead of storing the whole WP_Term object, this change only the term_id is stored. To get an array the full WP_Term objects, the `_prime_term_caches` function is called with an array of ids. In instances where a persistent object cache is not in use, then this will result in another SQL query to be run. After `_prime_term_caches` is called if this term is requested again in the same page load, then it will already be loaded into memory. If a user runs `WP_Term_Query` with the fields param set to `all_with_object_id`, an array of objects containing both the term_id and object_ids are stored in cache.
This change also improves the logic to load term meta caches. This change ensures that term meta is always primed for all terms loaded in the term query.
Props Spacedmonkey, boonebgorges, jbpaul17, peterwilsoncc, flixos90, pbearne.
Fixes #37189.
Built from https://develop.svn.wordpress.org/trunk@52836
git-svn-id: http://core.svn.wordpress.org/trunk@52425 1a063a9b-81f0-0310-95a4-ce76da25c4cd
2022-03-10 11:58:09 +01:00
|
|
|
return $term_objects;
|
2018-04-30 23:08:22 +02:00
|
|
|
}
|
2016-05-26 06:33:30 +02:00
|
|
|
}
|