Повідомлення, що містять щонайменше 3 теги списку тегів


13

Наприклад, є теги { foo, bar, chocolate, mango, hammock, leaf}

Я хотів би знайти всі повідомлення, принаймні 3 з цих тегів .

Стовп з тегами { foo, mango, vannilla, nuts, leaf} буде відповідати його , тому що він має { foo, mango, leaf} - так , по крайней мере 3 тегів необхідного набору тегів.

Тому це було б у списку відповідних постів.

Чи є простий спосіб зробити це, не роблячи декількох циклів через усі пости?

Відповіді:


8

Відповідь нижче спрощена та може бути розширена, щоб перевірити, чи є у будь-яких публікацій 3 відповідні теги перед виведенням списку. Використовуючи один запит і припустивши, що у вас є принаймні одна публікація з трьома відповідними тегами:

//List of tag slugs
$tags = array('foo', 'bar', 'chocolate', 'mango', 'hammock', 'leaf');

$args = array(
    'tag_slug__in' => $tags
    //Add other arguments here
);

// This query contains posts with at least one matching tag
$tagged_posts = new WP_Query($args);

echo '<ul>';
while ( $tagged_posts->have_posts() ) : $tagged_posts->the_post();
   // Check each single post for up to 3 matching tags and output <li>
   $tag_count = 0;
   $tag_min_match = 3;
   foreach ( $tags as $tag ) {
      if ( has_tag( $tag ) && $tag_count < $tag_min_match ) {
         $tag_count ++;
      }
   }
   if ($tag_count == $tag_min_match) {
      //Echo list style here
      echo '<li><a href="'. get_permalink() .'" title="'. get_the_title() .'">'. get_the_title() .'</a></li>';
   }
endwhile;
wp_reset_query();
echo '</ul>';

РЕДАКТУВАННЯ: Регулювання змінної $tag_min_matchвстановить кількість збігів


2

Ось один із способів зробити це:

Враховуючи набір з 5 тегів {a, b, c, d, e}:

1) У PHP генеруйте всі можливі підмножини, що містять 3 елементи, без повторення:

{a, b, c}
{a, b, d}
{a, b, e}
{a, c, d}
{a, c, e}
{b, c, d}
{b, c, e}
{c, d, e}

2) Перетворити ці підмножини в масивний запит таксономії:

$q = new WP_Query( array(
  'tax_query' => array(
    'relation' => 'OR',
    array(
      'terms' => array( 'a', 'b', 'c' ),
      'field' => 'slug',
      'operator' => 'AND'
    ),
    array(
      'terms' => array( 'a', 'b', 'd' ),
      'field' => 'slug',
      'operator' => 'AND'
    ),
    ...
  )
) );

1

Я використовував підхід sprclldr . Що стосується циклу while, то ось що я використовував замість цього:

$relatedPosts = $tagged_posts->posts;
$indexForSort = array();

for ($i = count($relatedPosts) - 1; $i >= 0; $i--) {
  $relatedPostTags = get_tags($relatedPosts[$i]->ID);
  //get the ids of each related post
  $relatedPostTags = $this->my_array_column($relatedPostTags, 'term_id');
  $relatedPostTagsInPostTag = array_intersect($tags, $relatedPostTags);
  $indexForSort[$i] = count($relatedPostTagsInPostTag);
}

//sort by popularity, using indexForSort
array_multisort($indexForSort, $relatedPosts, SORT_DESC);

Потім я приймаю головні пости:

$a_relatedPosts = array_slice($relatedPosts, 0, $this->numberRelatedPosts);

my_array_column є аналогічною функцією, ніж масив_колонці PHP 5,5:

  protected function my_array_column($array, $column) {
    if (is_array($array) && !empty($array)) {
      foreach ($array as &$value) {
        //it also get the object's attribute, not only array's values
        $value = is_object($value) ? $value->$column : $value[$column];
      }
      return $array;
    }
    else
      return array();
  }

Він не відповідає на початкове запитання (але це вирішує мою кореневу проблему ), як: якщо немає пов’язаних публікацій з 3 загальними тегами, то це все одно дасть деякі повідомлення.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.