Розширення контексту пошуку на екрані публікації списку адміністраторів


34

Я створив спеціальний тип публікації та додав до нього кілька спеціальних полів. Тепер я хотів би шукати, що автори можуть здійснювати на користувальницькому екрані списку публікацій (у адміністративному сервісному вікні), які також можна виконувати на метаполях, а не лише шукати заголовок та вміст, як зазвичай.

Де я можу підключитися і який код я повинен використовувати?

Приклад зображення введіть тут опис зображення

Стефано


1
Старе запитання, але ... я б хотів запропонувати приховати адреси електронної пошти та імена від скріншотів ...
Erenor Paz

Відповіді:


37

Я вирішив фільтрувати запит, додавши приєднання до таблиці postmeta і змінивши пункт де. поради щодо фільтрації пункту 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
Super Model

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' );

Додайте код вище, оновіть його, і він буде працювати без дублікатів.


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");

1
Попросіть належним чином відступити свій код та включити пояснення, чому і як це буде працювати .
tfrommen

Хоча я вперше це схвалив, я зрозумів, що це, на жаль, буде працювати при кожному пошуку цього ж, це може порушити пошук на передньому кінці.
Мацей Папроцький

Якщо додати чек на if ( $query->query['post_type'] != 'your_custom_post_type' ){ return; }початок верхньої частини функції, це не призведе до запуску інших пошукових запитів. Зауважте, що техніка у цій відповіді більше не шукає post_title і додає, що назад у не є тривіальним.
jwinn

Ще одна проблема - індикатор Результати пошуку для дзвінків "ключ">,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

Це не пошук, а якийсь "вибір" за виразним значенням.

У файлі function-iworks-posts-filter.zip У вас є приклад, як додати фільтр для звичайної публікації деяким мета-ключем. Я думаю, що це легко перетворити.


дякую за допомогу ... Я зараз погляну на вашу додаток. Я повідомляю вам результати мого розслідування ;-) Стефано
Стефано

Марчін, я думаю, що ви звертаєтесь до фільтрів на кшталт "за датою" тощо, але мені потрібно зачепитись у поле "безкоштовний пошук" вище. У будь-якому випадку я щойно опублікував своє рішення, можливо, це допомагає Дякую в будь-якому випадку!
Стефано

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