Лучший способ обнаружить, если вы находитесь на странице одного сообщения


9

Так что это может показаться довольно мрачной вещью, но следуйте за мной здесь.

Я пытаюсь добавить немного логики через pre_get_postsдействие. Это весь объект WP_Query, данный мне. (см. конец)

Вещи, которые я рассмотрел, используя:

  • is_single() - слишком широк.
  • is_singular()- слишком рано, чтобы использовать это, так как get_queried_object()еще не установлено.
  • $query->single свойство - опять же слишком широк.
  • $query->get('post_type')- не установлено, так как использует nameсвойство.

это nameдействительно единственный показатель здесь?

WP_Query Object
(
    [query] => Array
        (
            [page] => 
            [name] => abcs-of-mental-health
        )

    [query_vars] => Array
        (
            [page] => 
            [name] => abcs-of-mental-health
            [error] => 
            [m] => 0
            [p] => 0
            [post_parent] => 
            [subpost] => 
            [subpost_id] => 
            [attachment] => 
            [attachment_id] => 0
            [static] => 
            [pagename] => 
            [page_id] => 0
            [second] => 
            [minute] => 
            [hour] => 
            [day] => 0
            [monthnum] => 0
            [year] => 0
            [w] => 0
            [category_name] => 
            [tag] => 
            [cat] => 
            [tag_id] => 
            [author_name] => 
            [feed] => 
            [tb] => 
            [paged] => 0
            [comments_popup] => 
            [meta_key] => 
            [meta_value] => 
            [preview] => 
            [s] => 
            [sentence] => 
            [fields] => 
            [menu_order] => 
            [category__in] => Array
                (
                )

            [category__not_in] => Array
                (
                )

            [category__and] => Array
                (
                )

            [post__in] => Array
                (
                )

            [post__not_in] => Array
                (
                )

            [tag__in] => Array
                (
                )

            [tag__not_in] => Array
                (
                )

            [tag__and] => Array
                (
                )

            [tag_slug__in] => Array
                (
                )

            [tag_slug__and] => Array
                (
                )

            [post_parent__in] => Array
                (
                )

            [post_parent__not_in] => Array
                (
                )

        )

    [tax_query] => 
    [meta_query] => 
    [queried_object] => 
    [queried_object_id] => 0
    [post_count] => 0
    [current_post] => -1
    [in_the_loop] => 
    [comment_count] => 0
    [current_comment] => -1
    [found_posts] => 0
    [max_num_pages] => 0
    [max_num_comment_pages] => 0
    [is_single] => 1
    [is_preview] => 
    [is_page] => 
    [is_archive] => 
    [is_date] => 
    [is_year] => 
    [is_month] => 
    [is_day] => 
    [is_time] => 
    [is_author] => 
    [is_category] => 
    [is_tag] => 
    [is_tax] => 
    [is_search] => 
    [is_feed] => 
    [is_comment_feed] => 
    [is_trackback] => 
    [is_home] => 
    [is_404] => 
    [is_comments_popup] => 
    [is_paged] => 
    [is_admin] => 
    [is_attachment] => 
    [is_singular] => 1
    [is_robots] => 
    [is_posts_page] => 
    [is_post_type_archive] => 
    [query_vars_hash] => f473ebf7f725c2627dc5fd9a1429f626
    [query_vars_changed] => 
    [thumbnails_cached] => 
)

Ответы:


10

Я пытался разобраться в этом раньше для моих собственных целей. Насколько я могу сказать ...

  • post_typeна самом деле нигде не установлен postтип сообщения.
  • Для pageтипа сообщения я вижу только тип сообщения queried_object.
  • Для типов CPT есть post_typeключ query_varsи также query.
  • В этом отношении навигационные меню ведут себя как другие CPT.

Данные очень противоречивы, но если вы удалите страницы и CPT, я думаю, вы можете принять postтип.

Изменить: Рабочий код из @EricHolmes:

add_action( 'pre_get_posts', 'something_for_single_posts_only' ) ; 
function something_for_single_posts_only( $query ) { 
  if( $query->is_main_query() 
    && $query->is_singular() 
    && ! $query->get( 'post_type' ) 
    && ! $query->is_page() 
    && ! $query->is_attachment() 
  ) { 
      // do something for single posts only. 
  } 
} 

Мы проверяем is_singular, не после типа (CPTS иметь post_typeв query_vars), а не страницы или приложения.


И это работает внутри pre_get_posts(до запуска запроса)?
gmazzap

Интересно, is_page()установлен ли pre_get_postsуровень. Если это так, и я могу проверить, post_typeне установлен ли он в query_vars, я думаю, это так же хорошо, как и получается? Так сломан.
Эрик Холмс

1
is_pageкажется, установлен.
s_ha_dum

@s_ha_dum Я удалил комментарий, потому что он, кажется, не работает для CPT ...
gmazzap

2
Добавление рабочего решения к вашему ответу.
Эрик Холмс

0

Я не знаю, будет ли это полезно:

function hwl_home_pagesize( $query ) {
    global $wp_query;
    if (is_main_query() && count($wp_query->posts) < 2) {

    }

}
add_action( 'pre_get_posts', 'hwl_home_pagesize', 1 );

Используйте $ wp_query-> posts (array), чтобы проверить тип записи.


Еще нет. Предварительное получение сообщений происходит до $wp_query->postsфактического заполнения переменной. Спасибо хоть!
Эрик Холмс

0

После некоторых тестов я вижу, что, к сожалению, невозможно получить тип сообщения cpt внутри pre_get_postsхука. Только is_pageработает, но не стандартный тип записи, ни CPT могут быть восстановлены там.

Если у вас есть только страница и пост (без CPT) чек на is_single()с trueкак средством реагирования на этот пост типа пост, потому что возвращать ложь страниц.

Если у вас есть CPT, боюсь, вам придется выполнить дополнительный запрос . Самое простое, что я могу подумать, это просто получить post_typeстолбец, в котором публикуется статус публикации, а post_name является обязательным (пропуская ревизии):

function test( $q ) {
  if ( is_single() ) {
    global $wpdb;
    $type = $wpdb->get_var( $wpdb->prepare(
      "SELECT post_type FROM $wpdb->posts WHERE post_name = %s AND post_status = 'publish' AND post_type <> 'revision'",
      $q->query['name']
    ) );
    var_dump($type);
  }
}
add_action( 'pre_get_posts', 'test', 1); 

Если вы хотите проверить определенный тип записи, вы можете написать пользовательский условный тег, который будет просто считать строки с заданным post_type и заданным именем:

function is_single_post_type( $type = 'post' ) {
  if ( is_single() ) {
    global $wpdb, $wp_query;
    $is = $wpdb->get_var( $wpdb->prepare(
      "SELECT count(ID) FROM $wpdb->posts WHERE post_name = %s AND post_status = 'publish' AND post_type = %s",
      $wp_query->query['name'], $type
    ) );
    return $is > 0;
  }
  return false;
}

Конечно, это необходимо pre_get_postв любом последующем хуке, который вы можете использовать get_post_type()...


1
Это определенно сработает, но посмотрите в принятом ответе, это гораздо менее сложное условие, без дополнительного запроса к базе данных. :)
Эрик Холмс

Да. Я понимаю, что вы хотели точно знать, какой тип поста вы показываете. Не только если вы находитесь в одном посте просмотра. @EricHolmes
gmazzap

-1

Это то, что я использую, хотя это специализировано для моей собственной структуры каталогов.

/**
 * Function to list all templates used in a page
 * @author Imperative Ideas with thanks to Rarst
 * @uri http://wordpress.stackexchange.com/a/89005
 */

function thelist() {
    $included_files = get_included_files();
    $stylesheet_dir = str_replace( '\\', '/', get_stylesheet_directory() );
    $template_dir   = str_replace( '\\', '/', get_template_directory() );
    echo '<h3 class="debugtitle">Theme file templates used in this page</h3>';
    foreach ( $included_files as $key => $path ) {

        $path   = str_replace( '\\', '/', $path );

        if ( false === strpos( $path, $stylesheet_dir ) && false === strpos( $path, $template_dir ) )
            unset( $included_files[$key] );

        if(!strpos($path, '/wp-content/themes/') === false) { // Files IN this directory
            if(strpos($path, '/library/') === false) { // Ignore this subdir
                if(strpos($path, '/hybrid-core/') === false) { // Ignore this subdir
                    echo $key." = ". $path."</br>"; // Finally, output the list
                }
            }
        }
    }
}

Пробег может отличаться. Биты strpos, где я проверяю, что файлы находятся в одном каталоге, но не в другом, должны быть изменены для вашей сборки и, вероятно, могут быть пересмотрены более эффективно. Они существуют, чтобы вырезать результаты выше и ниже определенной структуры каталогов.

Запуск thelist () в нижнем колонтитуле даст вам нумерованный список каждого файла шаблона .php, использованного для компиляции текущего представления. Это особенно полезно при работе с дочерней темой, которая выводит загадочный компонент.


Кажется излишним. Классная идея, хотя?
Эрик Холмс

Существует не так много хороших способов определить, какие шаблоны управляют страницей, кроме их анализа. Я думаю, что если вы ищете что-то такое же простое, как одиночная страница против страницы, да, это излишне. Однако попробуйте запустить функцию некоторое время, за исключением дополнительных условий игнорирования, и вы поймете, почему я использую это. Существует множество файлов, которые используются для управления WP-страницей, и очень полезно знать, что они все из себя представляют. Даже это не идентифицирует подключенные функции.
Императивные идеи
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.