Rozszerzanie kontekstu wyszukiwania na ekranie postów z listy administratorów

34

Utworzyłem niestandardowy typ posta i dołączyłem do niego niektóre niestandardowe pola. Teraz chciałbym, aby wyszukiwanie, które autorzy mogą przeprowadzić na niestandardowym ekranie listy postów (w zapleczu administratora), zostało przeprowadzone również na polach meta, a nie tylko jak zwykle w tytule i treści.

Gdzie mogę się podłączyć i jakiego kodu muszę użyć?

Przykładowy obraz wprowadź opis zdjęcia tutaj

Stefano

Stefano
źródło
1
Stare pytanie, ale ... chciałbym zasugerować, aby ukryć adresy e-mail i nazwiska przed zrzutami ekranu ...
Erenor Paz

Odpowiedzi:

37

Rozwiązałem filtrowanie zapytania, dodając złączenie w tabeli postmeta i zmieniając klauzulę where. wskazówki dotyczące filtrowania klauzuli WHERE (często wymagają wyszukiwania i zamiany wyrażeń regularnych) znajdują się w kodeksie :

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;
}
Stefano
źródło
1
Łał! Właśnie tego szukałem. Myślę jednak, że mogłem znaleźć błąd, podczas wyszukiwania tytułu postu dostaję dopasowanie, które jest następnie duplikowane w wynikach 5 razy!?! imgur.com/eE52gIA
jnthnclrk
Oto kolejna chwila z wydrukowanym SQL: tinypic.com/view.php?pic=124tqb6&s=5 Nie możesz zrozumieć, dlaczego dostaję 5 przedmiotów!?!
jnthnclrk
1
Wysłano osobne pytanie na temat naprawy błędu dupe: wordpress.stackexchange.com/questions/111185/…
jnthnclrk
To i poniższy post były dla mnie przydatne. Teraz, aby znaleźć sposób na włączenie wyszukiwania autora postu i pokazanie zrobionych przez niego postów.
Shawn Rebelo
@Stefano, wynik wyszukiwania działa. Masz problem, domyślnie pole „Tytuł postu”, rekord wyszukiwania jest wielokrotnie powtarzany po stronie administratora. Zobacz: imgur.com/a/W4wmXhO
Super Model
10

Odpowiedź Stefano jest świetna, ale brakuje w niej wyraźnej klauzuli:

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

Dodaj powyższy kod, zaktualizuj go i będzie działać bez duplikatów.

Calara Ionut
źródło
7

To zadziała,

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
źródło
1
Prośby odpowiednio wcięcia kodu i wyjaśnienie, dlaczego i jak to będzie działać .
tfrommen
Chociaż po raz pierwszy go głosowałem, zdałem sobie sprawę, że to niestety zadziała przy każdym wyszukiwaniu w ten sam sposób, może przerwać wyszukiwanie frontonu.
Maciej Paprocki,
Dodanie zaznaczenia do if ( $query->query['post_type'] != 'your_custom_post_type' ){ return; }góry u góry funkcji spowoduje, że nie będzie można go uruchomić podczas innych wyszukiwań. Zauważ, że technika opisana w tej odpowiedzi nie przeszukuje już post_title, a dodanie tego z powrotem nie jest trywialne.
jwinn
Kolejny problem - Wskaźnik Wyniki wyszukiwania dla połączeń „<słowo kluczowe>”,get_search_query() które wywołują kolejne get_query_var( 's' ). Ponieważ „s” jest ustawiony na pusty ciąg, wyniki wyszukiwania dla „” zawsze będą miały pustą wartość między cudzysłowami. Czy istnieje poprawka do tego rozwiązania, która obejdzie ten problem?
jschrab
1

Odpowiedź 1: Dodaj ten kod do pliku funkcji oraz zmień i dodaj więcej kolumn, których użyłeś w swoim niestandardowym typie postu

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

Odpowiedź 2: Zalecane Użyj tego kodu w pliku funkcji bez żadnych zmian

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' );
Super Model
źródło
0

Nie jest to wyszukiwanie, ale niektóre „wybierają” według odrębnej wartości.

W funkcjach plików- iworks-posts-filter.zip Masz przykład, jak dodać filtr do zwykłego postu przez jakiś klucz meta_key. Myślę, że łatwo to przekonwertować.

Marcin
źródło
dziękuję za pomoc ... Przejdę teraz do twojego załącznika. Dam ci znać wyniki mojego dochodzenia ;-) Stefano
Stefano
Marcin Myślę, że odnosisz się do filtrów typu „według daty” itp., Ale muszę podpiąć pole „bezpłatne wyszukiwanie” powyżej. W każdym razie właśnie opublikowałem swoje rozwiązanie, może i tak dziękuję!
Stefano
0

Wersja kodu tutaj w kilku odpowiedziach, która modyfikuje parametr meta_query WP_Query wyszukiwania w pre_get_posts, nie szukała już post_title. Dodanie możliwości przeszukiwania tytułu postu LUB meta wartości nie może być wykonane bezpośrednio w WP_Query bez modyfikacji SQL niestety, ponieważ to pytanie rozwija się w następujący sposób: Używanie kwerendy meta („meta_query”) z zapytaniem („s”)

Połączyłem tutaj niektóre techniki, aby uzyskać działającą wersję, która unika preg_replaces i zbyt dużej modyfikacji SQL (chciałbym, żeby można było tego całkowicie uniknąć). Jedynym minusem jest to, że po wyszukiwaniu tekst napisów u góry strony mówi „Wyniki wyszukiwania dla„ ”. Właśnie ukryłem to w CSS dla niestandardowego typu postu mojej wtyczki.

/**
 * 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
źródło