Расширение контекста поиска на экране публикации списка администраторов

34

Я создал собственный тип записи и прикрепил к нему несколько пользовательских полей. Теперь я хотел бы, чтобы поиск, который авторы могли выполнять на настраиваемом экране списка сообщений (в бэкэнде администратора), также выполнялся в мета-полях, а не только просматривался в заголовке и содержимом как обычно.

Где я могу подключиться и какой код я должен использовать?

Пример изображения введите описание изображения здесь

Стефано

Стефано
источник
1
Старый вопрос, но ... я хотел бы предложить скрыть адреса электронной почты и имена на скриншотах ...
Эренор Пас

Ответы:

37

Я решил отфильтровать запрос, добавив объединение в таблицу postmeta и изменив предложение where. Советы по фильтрации предложения WHERE (часто требуется поиск и замена регулярных выражений) приведены здесь :

add_filter( 'posts_join', 'segnalazioni_search_join' );
function segnalazioni_search_join ( $join ) {
    global $pagenow, $wpdb;

    // I want the filter only when performing a search on edit page of Custom Post Type named "segnalazioni".
    if ( is_admin() && 'edit.php' === $pagenow && 'segnalazioni' === $_GET['post_type'] && ! empty( $_GET['s'] ) ) {    
        $join .= 'LEFT JOIN ' . $wpdb->postmeta . ' ON ' . $wpdb->posts . '.ID = ' . $wpdb->postmeta . '.post_id ';
    }
    return $join;
}

add_filter( 'posts_where', 'segnalazioni_search_where' );
function segnalazioni_search_where( $where ) {
    global $pagenow, $wpdb;

    // I want the filter only when performing a search on edit page of Custom Post Type named "segnalazioni".
    if ( is_admin() && 'edit.php' === $pagenow && 'segnalazioni' === $_GET['post_type'] && ! empty( $_GET['s'] ) ) {
        $where = preg_replace(
            "/\(\s*" . $wpdb->posts . ".post_title\s+LIKE\s*(\'[^\']+\')\s*\)/",
            "(" . $wpdb->posts . ".post_title LIKE $1) OR (" . $wpdb->postmeta . ".meta_value LIKE $1)", $where );
    }
    return $where;
}
Стефано
источник
1
Вот Это Да! Как раз то, что я искал. Тем не менее, я думаю, что, возможно, обнаружил ошибку, при поиске по названию поста я получаю совпадение, которое затем дублируется в результатах 5 раз!?! imgur.com/eE52gIA
jnthnclrk
Вот еще один пример с напечатанным SQL- кодом : tinypic.com/view.php?pic=124tqb6&s=5 Не могу понять, почему я получаю 5 предметов!?!
jnthnclrk
1
Размещен отдельный вопрос по исправлению ошибки,
связанной с дублированием
Это и пост ниже были полезны для меня. Теперь, чтобы найти способ включить поиск автора сообщения и показать сообщения, сделанные ими.
Шон Ребело
@ Stefano, результат поиска работает. Есть проблема, поле по умолчанию "Заголовок сообщения", поиск повторяется много раз и на стороне администратора. См .: imgur.com/a/W4wmXhO
супер модель
10

Ответ Стефано великолепен, но в нем нет четкого предложения:

function segnalazioni_search_distinct( $where ){
    global $pagenow, $wpdb;

    if ( is_admin() && $pagenow=='edit.php' && $_GET['post_type']=='segnalazioni' && $_GET['s'] != '') {
    return "DISTINCT";

    }
    return $where;
}
add_filter( 'posts_distinct', 'segnalazioni_search_distinct' );

Добавьте приведенный выше код, обновите его, и он будет работать без дубликатов.

Calara Ionut
источник
7

Это будет работать,

function custom_search_query( $query ) {
    $custom_fields = array(
        // put all the meta fields you want to search for here
        "rg_first_name",
        "rg_1job_designation"
    );
    $searchterm = $query->query_vars['s'];

    // we have to remove the "s" parameter from the query, because it will prevent the posts from being found
    $query->query_vars['s'] = "";

    if ($searchterm != "") {
        $meta_query = array('relation' => 'OR');
        foreach($custom_fields as $cf) {
            array_push($meta_query, array(
                'key' => $cf,
                'value' => $searchterm,
                'compare' => 'LIKE'
            ));
        }
        $query->set("meta_query", $meta_query);
    };
}
add_filter( "pre_get_posts", "custom_search_query");
Vin_fugen
источник
1
Пожалуйста, сделайте отступ в своем коде и включите объяснение того, почему и как это будет работать .
tfrommen
Хотя я впервые проголосовал за него, я понял, что, к сожалению, это сработает при каждом поиске, и это может нарушить внешний поиск.
Мацей Папроцки
Добавление проверки для if ( $query->query['post_type'] != 'your_custom_post_type' ){ return; }верхней части функции не позволит запускать ее при других поисках. Обратите внимание, что техника в этом ответе больше не ищет post_title и добавление этого обратно не является тривиальным.
Джвинн
Еще одна проблема - индикатор Результаты поиска для вызовов «<ключевое слово>»,get_search_query() которые выполняют дополнительные вызовы get_query_var( 's' ). Так как для «s» задана пустая строка, в результатах поиска «» всегда будет пустое значение между кавычками. Есть ли у этого решения твик, который обходит это?
jschrab
1

Ответ 1: Добавьте этот код в файл функции, измените и добавьте имя столбца, которое вы использовали в своем типе записи

function extend_admin_search( $query ) {

    // use your post type
    $post_type = 'document';
    // Use your Custom fields/column name to search for
    $custom_fields = array(
        "_file_name",
    );

    if( ! is_admin() )
        return;

    if ( $query->query['post_type'] != $post_type )
        return;

    $search_term = $query->query_vars['s'];

    // Set to empty, otherwise it won't find anything
    $query->query_vars['s'] = '';

    if ( $search_term != '' ) {
        $meta_query = array( 'relation' => 'OR' );

        foreach( $custom_fields as $custom_field ) {
            array_push( $meta_query, array(
                'key' => $custom_field,
                'value' => $search_term,
                'compare' => 'LIKE'
            ));
        }

        $query->set( 'meta_query', $meta_query );
    };
}

add_action( 'pre_get_posts', 'extend_admin_search' );

Ответ 2: рекомендуется использовать этот код в функциональном файле без каких-либо изменений

function cf_search_join( $join ) {
    global $wpdb;

    if ( is_search() ) {    
        $join .=' LEFT JOIN '.$wpdb->postmeta. ' ON '. $wpdb->posts . '.ID = ' . $wpdb->postmeta . '.post_id ';
    }

    return $join;
}
add_filter('posts_join', 'cf_search_join' );
function cf_search_where( $where ) {
    global $pagenow, $wpdb;

    if ( is_search() ) {
        $where = preg_replace(
            "/\(\s*".$wpdb->posts.".post_title\s+LIKE\s*(\'[^\']+\')\s*\)/",
            "(".$wpdb->posts.".post_title LIKE $1) OR (".$wpdb->postmeta.".meta_value LIKE $1)", $where );
    }

    return $where;
}
add_filter( 'posts_where', 'cf_search_where' );

function cf_search_distinct( $where ) {
    global $wpdb;

    if ( is_search() ) {
        return "DISTINCT";
    }

    return $where;
}
add_filter( 'posts_distinct', 'cf_search_distinct' );
Супер модель
источник
0

Это не поиск, а какой-то «выбор» по отдельному значению.

В файле functions-iworks-posts-filter.zip у вас есть пример того, как добавить фильтр для обычной записи с помощью некоторого meta_key. Я думаю, что это легко конвертировать.

Marcin
источник
спасибо за помощь ... я сейчас посмотрю на ваше приложение. Я сообщу вам результаты моего расследования ;-) Стефано
Стефано
Марсин Я думаю, что вы ссылаетесь на фильтры типа «по дате» и т. Д., Но мне нужно подключиться к полю «бесплатный поиск» выше. Во всяком случае, я только что опубликовал свое решение, может быть, оно помогает Спасибо в любом случае!
Стефано
0

Версия кода в паре ответов, которая изменяет параметр meta_query поискового WP_Query в pre_get_posts, больше не выполняет поиск в post_title. Добавление возможности поиска либо по заголовку записи, либо по мета-значениям ИЛИ не может быть выполнено напрямую в WP_Query без изменения SQL, к сожалению, так как этот вопрос детализируется: Использование мета-запроса ('meta_query') с поисковым запросом ('s')

Я объединил некоторые из методов здесь, чтобы получить рабочую версию, которая избегает preg_replaces и слишком много модификаций SQL (я хотел бы, чтобы этого можно было избежать полностью). Единственным недостатком является то, что после поиска текст субтитров в верхней части страницы гласит «Результаты поиска для« ». Я только что скрыл это с помощью CSS для пользовательского типа поста моего плагина.

/**
 * Extend custom post type search to also search meta fields
 * @param  WP_Query $query
 */
function extend_cpt_admin_search( $query ) {
  // Make sure we're in the admin area and that this is our custom post type
  if ( !is_admin() || $query->query['post_type'] != 'your_custom_post_type' ){
    return;
  }

  // Put all the meta fields you want to search for here
  $custom_fields = array(
    "your_custom_meta_field",
    "your_custom_meta_field2",
    "your_custom_meta_field3"
  );
  // The string submitted via the search form
  $searchterm = $query->query_vars['s'];

  // Set to empty, otherwise no results will be returned.
  // The one downside is that the displayed search text is empty at the top of the page.
  $query->query_vars['s'] = '';

  if ($searchterm != ""){
    // Add additional meta_query parameter to the WP_Query object.
    // Reference: https://codex.wordpress.org/Class_Reference/WP_Query#Custom_Field_Parameters
    $meta_query = array();
    foreach($custom_fields as $cf) {
      array_push($meta_query, array(
        'key' => $cf,
        'value' => $searchterm,
        'compare' => 'LIKE'
      ));
    }
    // Use an 'OR' comparison for each additional custom meta field.
    if (count($meta_query) > 1){
      $meta_query['relation'] = 'OR';
    }
    // Set the meta_query parameter
    $query->set('meta_query', $meta_query);


    // To allow the search to also return "OR" results on the post_title
    $query->set('_meta_or_title', $searchterm);
  }
}
add_action('pre_get_posts', 'extend_cpt_admin_search');



/**
 * WP_Query parameter _meta_or_title to allow searching post_title when also
 * checking searching custom meta values
 * https://wordpress.stackexchange.com/questions/78649/using-meta-query-meta-query-with-a-search-query-s
 * https://wordpress.stackexchange.com/a/178492
 * This looks a little scary, but basically it's modifying the WHERE clause in the 
 * SQL to say "[like the post_title] OR [the existing WHERE clause]"
 * @param  WP_Query $q
 */
function meta_or_title_search( $q ){
  if( $title = $q->get( '_meta_or_title' ) ){
    add_filter( 'get_meta_sql', function( $sql ) use ( $title ){
      global $wpdb;

      // Only run once:
      static $nr = 0;
      if( 0 != $nr++ ) return $sql;

      // Modified WHERE
      $sql['where'] = sprintf(
          " AND ( (%s) OR (%s) ) ",
          $wpdb->prepare( "{$wpdb->posts}.post_title LIKE '%%%s%%'", $title),
          mb_substr( $sql['where'], 5, mb_strlen( $sql['where'] ) )
      );

      return $sql;
    });
  }
}
add_action('pre_get_posts', 'meta_or_title_search');
jwinn
источник