query = &$query; $this->model = &$model; } /** * Check if translated taxonomy is queried * Compatible with nested queries introduced in WP 4.1 * * @see https://wordpress.org/support/topic/tax_query-bug * * @since 1.7 * * @param array $tax_queries * @return bool */ protected function have_translated_taxonomy( $tax_queries ) { if ( is_array( $tax_queries ) ) { foreach ( $tax_queries as $tax_query ) { if ( isset( $tax_query['taxonomy'] ) && $this->model->is_translated_taxonomy( $tax_query['taxonomy'] ) && ! ( isset( $tax_query['operator'] ) && 'NOT IN' === $tax_query['operator'] ) ) { return true; } // Nested queries elseif ( is_array( $tax_query ) && $this->have_translated_taxonomy( $tax_query ) ) { return true; } } } return false; } /** * Get queried taxonomies * * @since 2.2 * * @return array queried taxonomies */ public function get_queried_taxonomies() { return isset( $this->query->tax_query->queried_terms ) ? array_keys( wp_list_filter( $this->query->tax_query->queried_terms, array( 'operator' => 'NOT IN' ), 'NOT' ) ) : array(); } /** * Sets the language in query * Optimized for ( needs ) WP 3.5+ * * @since 2.2 * * @param object $lang */ public function set_language( $lang ) { // Defining directly the tax_query ( rather than setting 'lang' avoids transforming the query by WP ) $lang_query = array( 'taxonomy' => 'language', 'field' => 'term_taxonomy_id', // Since WP 3.5 'terms' => $lang->term_taxonomy_id, 'operator' => 'IN', ); $tax_query = &$this->query->query_vars['tax_query']; if ( isset( $tax_query['relation'] ) && 'OR' === $tax_query['relation'] ) { $tax_query = array( $lang_query, array( $tax_query ), 'relation' => 'AND', ); } elseif ( is_array( $tax_query ) ) { // The tax query is expected to be *always* an array, but it seems that 3rd parties fill it with a string // Causing a fatal error if we don't check it. // See https://wordpress.org/support/topic/fatal-error-2947/ $tax_query[] = $lang_query; } elseif ( empty( $tax_query ) ) { // Supposing the tax query has been wrongly filled with an empty string $tax_query = array( $lang_query ); } } /** * Add the language in query after it has checked that it won't conflict with other query vars * * @since 2.2 * * @param object $lang Language */ public function filter_query( $lang ) { $qvars = &$this->query->query_vars; if ( ! isset( $qvars['lang'] ) ) { /** * Filter the query vars which disable the language filter in a query * * @since 2.3.5 * * @param array $excludes Query vars excluded from the language filter * @param object $query WP Query * @param object $lang Language */ $excludes = apply_filters( 'pll_filter_query_excluded_query_vars', self::$excludes, $this->query, $lang ); // Do not filter the query if the language is already specified in another way foreach ( $excludes as $k ) { if ( ! empty( $qvars[ $k ] ) ) { // Specific case for 'cat' as it can contain negative values if ( 'cat' === $k ) { foreach ( explode( ',', $qvars['cat'] ) as $cat ) { if ( $cat > 0 ) { return; } } } else { return; } } } $taxonomies = array_intersect( $this->model->get_translated_taxonomies(), get_taxonomies( array( '_builtin' => false ) ) ); foreach ( $taxonomies as $tax ) { $tax = get_taxonomy( $tax ); if ( ! empty( $qvars[ $tax->query_var ] ) ) { return; } } if ( ! empty( $qvars['tax_query'] ) && $this->have_translated_taxonomy( $qvars['tax_query'] ) ) { return; } // Filter queries according to the requested language if ( ! empty( $lang ) ) { $taxonomies = $this->get_queried_taxonomies(); if ( $taxonomies && ( empty( $qvars['post_type'] ) || 'any' === $qvars['post_type'] ) ) { foreach ( $taxonomies as $taxonomy ) { $tax_object = get_taxonomy( $taxonomy ); if ( $this->model->is_translated_post_type( $tax_object->object_type ) ) { $this->set_language( $lang ); break; } } } elseif ( empty( $qvars['post_type'] ) || $this->model->is_translated_post_type( $qvars['post_type'] ) ) { $this->set_language( $lang ); } } } else { // Do not filter untranslatable post types such as nav_menu_item if ( isset( $qvars['post_type'] ) && ! $this->model->is_translated_post_type( $qvars['post_type'] ) && ( empty( $qvars['tax_query'] ) || ! $this->have_translated_taxonomy( $qvars['tax_query'] ) ) ) { unset( $qvars['lang'] ); } // Unset 'all' query var (mainly for admin language filter). if ( isset( $qvars['lang'] ) && 'all' === $qvars['lang'] ) { unset( $qvars['lang'] ); } } } }