Полное руководство по пользовательским типам постов в WordPress. Произвольный тип записей в WordPress — register_post_types, register_taxonomy В каких случаях стоит создавать произвольный тип записей

$post_type (строка ) то, как вы хотите назвать свой тип поста, это название в дальнейшем будет использоваться в параметрах многих функций, учтите это. Кроме того, нельзя использовать в качестве названий следующие зарезервированные вордпрессом слова:

  • attachment
  • revision
  • nav_menu_item
  • custom_css
  • customize_changeset
  • action
  • order
  • theme
  • author
$args (массив ) включает в себя следующие параметры: labels (массив ) позволяет подписать элементы интерфейса данного типа записи. Подробнее в примере в конце поста. description (строка ) описание регистрируемого типа записей. Насколько мне известно, этот параметр не используется где-либо в движке WordPress или в стандартных плагинах и темах. Но вполне может использоваться в сторонних плагинах или темах. public (логическое ) обобщает следующие несколько параметров, устанавливая для них соответствующие значения по умолчанию publicly_queryable (логическое ) нужно ли элементы данного типа записей сделать доступными на сайте. Подробнее про задачу этого параметра .
По умолчанию: значение аргумента public . exclude_from_search (логическое )
  • true — исключить записи данного типа из результатов поиска на сайте,
  • false — не исключать.

По умолчанию: противоположные значения параметра public . show_in_nav_menus (логическое ) нужно ли элементы данного типа записей сделать доступными для добавления в меню сайта.
По умолчанию: значение аргумента public . show_ui (логическое ) нужно ли добавлять стандартный интерфейс в админке для редактирования и добавления записей данного типа.
По умолчанию: значение аргумента public .

Show_in_menu (логическое |строка ) нужно ли добавлять пункты в меню админки.

По умолчанию: значение аргумента show_ui . show_in_admin_bar (логическое ) нужно ли добавлять ссылку на создание новой записи данного типа в админ панель.

По умолчанию: значение аргумента show_in_menu . menu_position (целое число ) порядок расположения в меню в админке.

menu_icon (строка ) абсолютный URL иконки для меню. Оптимальное разрешение изображения 16×16 пикселей.

Кроме того, в WordPress 3.8 появился встроенный пакет иконок — вы можете использовать любую из этих иконок, просто указав её название в качестве значения параметра, например dashicons-cart .

По умолчанию используется иконка обычных записей. delete_with_user (логическое ) при удалении пользователя на блоге, нужно ли автоматически удалять все записи данного типа, которые он опубликовал.
По умолчанию: false . hierarchical (логическое ) должен ли данный тип постов иметь иерархию.
По умолчанию: false . has_archive (логическое |строка ) должен ли данный тип постов иметь собственную страницу архивов. В качестве значения параметра можно указать ярлык страницы архивов.
По умолчанию: false . rewrite (массив |логическое ) устанавливает правила для постоянных ссылок в URL. Если в качестве значения данного параметра указать false , то правила для постоянных ссылок создаваться не будут.

Slug (строка ) ярлык, используемый для записей данного типа (по умолчанию — название типа поста) with_front (логическое ) нужно ли добавлять в постоянные ссылки значение $wp_rewite->front (по умолчанию — true) feeds (логическое ) нужно ли создавать RSS ленту для данного типа поста (по умолчанию — значение параметра has_archive) pages (логическое ) нужно ли разрешить постраничную навигацию в постах регистрируемого типа, используя тег (по умолчанию — true) supports (массив ) какие нужно добавить на страницы редактирования записей данного типа.

  • title — поле для ввода заголовка поста
  • editor — текстовый редактор
  • excerpt — метабокс «Цитата»
  • author — метабокс «Автор»
  • thumbnail — метабокс «Миниатюра записи» (кроме того, ваша тема должна их поддерживать)
  • comments — метабокс «Комментарии» (если указано, то разрешены комментарии к постам регистрируемого типа)
  • trackbacks — метабокс «Отправить обратные ссылки»
  • custom-fields — метабокс «Произвольные поля» (произвольные поля будут поддерживаться в любом случае)
  • revisions — метабокс «Редакции» (если указано, то в базе данных будут создаваться редакции постов данного типа)
  • page-attributes — метабокс «Атрибуты страницы» с возможностью выбора родительского эоемента и установления порядка menu_order
  • post-formats — метабокс «Формат», про форматы постов читайте подробнее .
  • taxonomies (массив ) массив таксономий, например category или post_tag , которые будут использоваться для данного типа записей. Присвоить таксономии можно также при помощи функции .

    Пример регистрации типа поста

    Если не знаете, куда вставлять код — вставляйте в functions.php вашей текущей темы.

    add_action ( "init" , "true_register_products" ) ; // Использовать функцию только внутри хука init function true_register_products() { $labels = array ( "name" => "Товары" , "singular_name" => "Товар" , // админ панель Добавить->Функцию "add_new" => "Добавить товар" , "add_new_item" => "Добавить новый товар" , // заголовок тега </span> "edit_item" => <span>"Редактировать товар" </span>, "new_item" => "Новый товар" , "all_items" => "Все товары" , "view_item" => <span>"Просмотр товаров на сайте" </span>, "search_items" => "Искать товары" , "not_found" => <span>"Товаров не найдено." </span>, "not_found_in_trash" => <span>"В корзине нет товаров." </span>, "menu_name" => "Товары" <span>// ссылка в меню в админке </span> ) ; $args = array ( "labels" => $labels , "public" => true , <span>// благодаря этому некоторые параметры можно пропустить </span> "menu_icon" => "dashicons-cart" , <span>// иконка корзины </span> "menu_position" => 5 , "has_archive" => true , "supports" => array ( "title" , "editor" , "excerpt" , "thumbnail" , "comments" ) , "taxonomies" => array ("post_tag" ) ) ; register_post_type("product" ,$args ) ; } </td> </tr></table><p><span class="koXPsjWcBis"></span></p></ul> <p>Создаем кастомный тип записи (Custom Post Type) <b>Articles </b> с кастомными категориями (Custom Taxonomy) <b>Articles Category </b>.</p><p>В моем случае все стандартные записи – это Товары, поэтому Статьи выведем через кастомные записи.</p><p>В файле функций functions.php регистрируем кастомный тип записи Articles:</p><p>Function wptp_create_post_type() { $labels = array("name" => __("Articles"), "singular_name" => __("Articles"), "add_new" => __("New Article"), "add_new_item" => __("Add New Article"), "edit_item" => __("Edit Article"), "new_item" => __("New Article"), "view_item" => __("View Article"), "search_items" => __("Search Articles"), "not_found" => __("No Articles Found"), "not_found_in_trash" => __("No Articles found in Trash"),); $args = array("labels" => $labels, "has_archive" => true, "public" => true, "hierarchical" => false, "menu_position" => 5, "supports" => array("title", "editor", "excerpt", "custom-fields", "thumbnail"),); register_post_type("articles", $args); } add_action("init", "wptp_create_post_type"); </p><p>Если вы хотите, чтобы в кастомных постах выводились стандартные категории, тогда привяжите их к таксономии Категорий:</p><p> "taxonomies" => array("category"), </p><p>Если для кастомных постов вы создаете еще и кастомные таксономии, то связывать их надо с созданными таксономиями, если брать пример ниже, то это будут articles_category</p><p> "taxonomies" => array("articles_category"), </p><p>Затем для кастомного типа записи Articles регистрируем таксономии ‘Article Category’, чтобы разные записи могли принадлежать разным категориям.</p><p>Function wptp_register_taxonomy() { register_taxonomy("articles_category", "articles", array("labels" => array("name" => "Article Categories", "singular_name" => "Article Category", "search_items" => "Search Article Categories", "all_items" => "All Article Categories", "edit_item" => "Edit Article Categories", "update_item" => "Update Article Category", "add_new_item" => "Add New Article Category", "new_item_name" => "New Article Category Name", "menu_name" => "Article Category",), "hierarchical" => true, "sort" => true, "args" => array("orderby" => "term_order"), "show_admin_column" => true)); } add_action("init", "wptp_register_taxonomy"); </p><p>Для кастомной таксономии можно изменить слаг на более красивый, чтобы он не был articles_category , но нужно учесть, чтобы таких слдагов больше не было на других страницах или постах, иначе будут проблемы:</p><p> "rewrite" => array("slug" => "blog"), </p><p>При изменении слагов необходимо обновить настройки “Постоянных ссылок” в админке, иначе будете получать 404 ошибку вместо нужной страницы.</p><p>Больше примеров можно найти и . Но привести весь этот код в порядок помогла , за что автору большая благодарность.</p><p>За внешний вид кастомной записи отвечает файл single.php , но чтобы изменить вид кастомной записи можно создать файл single-{post_type}.php – в моем случае будет single-articles.php со своим содержимым.</p><p>Теперь нужно вывести кастомные записи на странице. По умолчанию за отображение архива кастомных записей отвечает файл index.php . Но можно создать файл, который будет отвечать за вывод кастомных записей в своей таксономии.<br></p><h2>Вариант 1 – самый правильный. Выводим записи в таксономии так же как и в обычной категории.</h2><p>Для этого создаем файл taxonomy-{taxonomy}.php – в моем случае будет taxonomy-articles_category.php и в нем выводим обычный цикл, как и для стандартных записей в категории:</p><p> <div class="container"> <div class="row"> <div class="col-md-12"> <div class="cat-title"><?php single_cat_title(); ?></div> <div class="tax-wrap"> <?php if (have_posts()) : while (have_posts()) : the_post(); get_template_part("include/tax-item"); endwhile; get_template_part("include/pagination"); endif; ?> </div><!-- /tax-wrap --> </div><!-- /col-md-12 --> </div><!-- /row --> </div><!-- /container --> </p><p>В файле tax-item.php выводим данные, которые нам нужно получить из каждой записи, например, заголовок, ссылку на запись, миниатюру и excerpt.</p><p>В файле pagination.php выводим пагинацию вот такого формата .</p><p>Цикл не изменяем, пагинация отлично работает. Это самый оптимальный вариант для отображения Кастомных Таксономий.<br></p><h2>Вариант 2 – если нет кастомных таксономий, то можно просто получить все кастомные записи в виде Архива</h2><p>Для этого в шаблоне создаем файл archive-{post_type}.php – в моем случае будет archive-articles.php , в котором точно так же как и в таксономии выводим обычный цикл, только вместо заголовка Таксономии выводим имя кастомного типа записи <?php post_type_archive_title(); ??> :</p><p> <div class="container"> <div class="row"> <div class="col-md-12"> <div class="cat-title"><?php post_type_archive_title(); ?></div> <div class="tax-wrap"> <?php if (have_posts()) : while (have_posts()) : the_post(); get_template_part("include/tax-item"); endwhile; get_template_part("include/pagination"); endif; ?> </div><!-- /tax-wrap --> </div><!-- /col-md-12 --> </div><!-- /row --> </div><!-- /container --> </p><p>При этом варианте, если не создана страница архива для кастомной записи, тогда получить список всех кастомных записей можно по прямой ссылке BLOG_URL?post_type={post_type} или в моем случае http://site.com/articles/ .<br></p><h2>Вариант 3. Просто выводим все кастомные записи Articles на странице с заданным шаблоном</h2> <?php $args = array("post_type" => "articles", "posts_per_page" => -1); $loop = new WP_Query($args); while ($loop->have_posts()) : $loop->the_post(); get_template_part("include/tax-item"); endwhile; ?> <p>get_template_part("include/tax-item"); – в файле tax-item.php я вывожу содержимое записи, которое нужно мне для отображения записей внутри цикла (заголовок, миниатюру, дату, цитату и т.д.)</p><p>Этот вариант выводит все записи Articles на странице, не зависимо от таксономий (категорий).</p><p>А если нам нужно вывести каждую категорию отдельно со своими новостями, тогда используем первый вариант, описанный выше.</p><p>При этом, если вы используете плагин Yoast SEO и используете его хлебные крошки:</p><p> <?php if (function_exists("yoast_breadcrumb")) { yoast_breadcrumb(" "); } ?> </p><p>Тогда при выборе в настройках плагина таксономии “Articles Category” в “Taxonomy to show in breadcrumbs for post types”, в хлебных крошках вы получите ссылку на категорию, к которой принадлежит Новость, при других вариантах вывода кастомных записей это не получалось сделать.</p> <p>О произвольных типах записей в Вордпрессе (CPT, Custom Post Types in WordPress), как создать и настроить, добавить категории и теги, как сделать вложенные URL /тип записи/рубрика/запись или http://example.com/post_type/category/post/ .<br> Тут также описывается, как расширить стандартный функционал WordPress, создав новые типы записей, которые можно использовать в различных целях: добавить собственное порфтолио, картотеку фильмов и музыки, каталог продукции, календарь событий, даже сделать свой небольшой интернет-магазин (потому что для большого лучше пользоваться готовыми решениями навроде WooCommerce) или социальную сеть, и много чего ещё в этом духе.<br> Вначале общая информация, дальше — частности.</p> <h2>О записях в WordPress: где хранятся в базе данных, и как их получить</h2> <p>Абсолютно все записи в WordPress, про которые чуть ниже, хранятся в одной таблице: wp_posts . Метаданные постов, например, данные из метабоксов , хранятся в таблице wp_postmeta .</p> <p>Приведу пример, как получить 10 записей типа post (стандартные <b>Записи </b> в панели администратора).<br> В общих чертах, данные особо запрашивать не надо. В шаблонах они уже предустановлены, например, в single.php данные по записи уже доступны для обработки в цикле</p><p>If (have_posts()) { while (have_posts()) { the_post(); // Тут используем данные из цикла, такие как the_title() и т.п. } } </p><p>Если данные нужно получить где-то вне цикла или шаблона, используется запрос WP_Query</p><p> $args = array("post_type" => "post", // Тип поста: page, attachment, ... "posts_per_page" => 10, // 10 записей за раз); $p = get_posts($args); // Данные можно раскрыть в цикле if (!empty($p)) { foreach ($p as $post) { setup_postdata($post); ?> "><?= the_title() ?> <?php } wp_reset_postdata(); } </p><p>Данные можно получить посредством SQL запроса:</p><p>Global $wpdb; $query = "SELECT * FROM {$wpdb -> posts} WHERE post_type = "post" LIMIT 10"; $p = $wpdb -> get_results($query); exit(print_r($p)); // в $p теперь массив с данными о постах </p><p>Все три способа выше дадут одинаковый результат на выходе.<br></p> <h2>Предустановленные типы записей в WordPress: post, page, attachment, revision, nav_menu_item</h2> <h3>Записи или посты (post)</h3> <p>Самая используемая единица из всех типов, что есть в WordPress — Записи (они же посты, post). Используется в роли постов блога и тому подобного. Имеет 2 предустановленных таксономии: рубрики, они же категории (category) и метки, они же теги (post_tag).<br> Таксономии служат для сортировки и упорядочивания записей.<br> Рубрики отличаются от меток тем, что имеют древовидную структуру (могут быть вложенными друг в друга).<br> Метки являются независимыми друг от друга единицами и этим чем-то похожи на Записи.<br> Также, по умолчанию, из записей формируется RSS лента сайта на Вордпрессе.</p> <p>Для Записей используются следующие файлы шаблонов (в порядке приоритета):</p> <ol><li>single-post.php</li> <li>single.php</li> <li>singular.php</li> <li>index.php</li> </ol><blockquote class="note"><p>Файлы шаблонов ищутся сверху вниз по списку в порядке приоритета. Если файл шаблона найден в теме, используется он, а поиск прекращается.</p> </blockquote> <h3>Страницы (page)</h3> <p>Страницы используются, в основном, в роли служебных страниц, посадочных страниц — лендингов, сборника энциклопедии и тому подобного. Имеют древовидную иерархию, то есть могут быть вложенными друг в друга, что отразится в адресе конечной страницы (ярлыки родительских будут присутствовать в цепочке URL), и в этой роли имеется возможность выставить им приоритет в сортировке.<br> Чтобы сделать лендинг, вы можете пойти двумя путями:<br></p> <h4>Создаём специальный шаблон страницы (page)</h4> <p>Где-нибудь внутри темы в корне или её поддиректории создать файл с произвольным названием и расширением php , например, landing.php . Внутри вы можете разместить совершенно любой шаблон, который может быть абсолютно не похож на другие страницы сайта. Но главное, это поместить вот такой код в начало файла:</p><p> <?php /* Template Name: Наш уникальный лендинг */ </p><p>Template Name — это специальная метка, которая говорит WordPress о том, что этот файл — специальный шаблон.<br> Теперь при создании и редактировании любой страницы (page) посмотрите в блоке справа с названием Атрибуты страницы, в нём в разделе Шаблон вы можете выбрать наш уникальный лендинг.</p> <p>Для Страниц (Page) используется следующая иерархия шаблонов. Как и с Записями, указываю в порядке приоритета:</p> <ol><li>{шаблон}.php</li> <li>page-{ярлык_страницы}.php</li> <li>page-{ID_страницы}</li> <li>page.php</li> <li>singular.php</li> <li>index.php</li> </ol><h3>Прикрепления, вложения или аттачменты (attachment)</h3> <p>Служебный тип, предназначенный для хранения информации о файлах (изображениях, аудио, видео и тому подобных), которые загружались через загрузчик в стандартном редакторе WordPress при редактировании Записи, Страницы или подобного: размер, вес, описание, к какому посту или странице прикреплены, и тому подобное.</p> <p>Получить вложения для последующей манипуляции над ними можно с помощью следующего кода:</p><p> $args = array("post_type" => "attachment", // Тип поста: attachment "post_status" => "inherit", // По умолчанию "publish", а с ним вложения не получить, поэтому указываем специальный статус вложения "inherit"); $p = get_posts($args); exit(print_r($p)); // На выходе будем иметь массив с вложениями </p><p>Иерархия шаблонов для аттачментов:</p> <ol><li>{mime-тип}.php</li> <li>{mime-подтип}.php</li> <li>{mime-тип-подтип}.php</li> <li>attachment.php</li> <li>single.php</li> <li>singular.php</li> <li>index.php</li> </ol><h3>Редакции, черновики или ревизии (revision)</h3> <p>Редакции, они же Ревизии — это версии черновиков записей, которые создаются автоматически, пока вы пишете статью в административной панели Вордпресса или сохраняете статью без фактической её публикации.<br> В процессе написания, для каждой статьи по умолчанию сохраняется каждая версия черновика. Можно сравнить две разные версии, и к нужной можно откатиться назад.<br> Хранить помногу версий одной и той же статьи — довольно накладно и часто бессмысленно (хотя, наверное, правильнее было бы оставлять всё по умолчанию, регулярно подчищая старые редакции с помощью плагина, например WP Optimize), поэтому число сохраняемых версий Ревизий можно изменить с помощью 2 вариантов:</p> <ol><li>Использовать фильтр wp_revisions_to_keep</li> <li>Прописать в wp-config.php //Отключаем ревизии до минимально возможного значения define("WP_POST_REVISIONS", 0); <p>Возможные значения:</p> <ul><li>true или -1: сохраняет каждую версию черновика. Вариант по умолчанию</li> <li>false или 0: отключает сохранение черновиков, кроме 1 автосохранения</li> <li>Целое число больше нуля: сохраняется указанное число версий черновиков + 1 автосохранение. Старые версии, не укладывающиеся в указанное число, автоматически удаляются</li> </ul></li> </ol><h3>Элементы навигационного меню (nav_menu_item)</h3> <p>Навигационное меню (nav_menu_item) — это тип записей, который хранит информацию об единице навигации в WordPress. Первый, и пока единственный тип записи, который используется не как остальные типы записей, данные для работы и отображения на сайте получают свои отдельные функции.<br> Также, навигационные меню по умолчанию не включены. Чтобы их включить, нужно объявить об их поддержке в :</p> <ol><li>Прописать add_theme_support("menus");</li> <li>Или зарегистрировать место под меню с помощью register_nav_menu() , тогда поддержка меню включится автоматически</li> </ol><p>Для получения данных пользуйтесь wp_nav_menu() , потому что WP_Query не будет работать, и это отличительная особенность типа постов nav_menu_item</p><p> // Этот код сработает wp_nav_menu(); // Выведет первое зарегистрированное непустое меню // А код ниже работать не будет $args = array("post_type" => "nav_menu_item", // Тип поста: page, attachment, ...); $p = get_posts($args); exit(print_r($p)); // На выходе будем иметь пустой массив </p><h2>Пользовательский Произвольный тип записи (Custom Post Type, CPT WordPress)</h2> <p>Вот мы и подошли к главному — тому инструменту, который позволяет расширить стандартный блоговый функционал WordPress до бесконечных возможностей: пользовательский произвольный тип записи.<br> Самый простой пример, как можно зарегистрировать наш новый тип записи sheensay_product</p><p>Add_action("init", "sheensay_post_type"); function sheensay_post_type() { register_post_type("sheensay_product", array("labels" => array("name" => "Продукция", "singular_name" => "Продукцию",), "public" => true, // тип записи открыт для поиска и тому подобного "has_archive" => true, // Включаем страницы архивов "supports" => array("title", "editor", "thumbnail", "comments"), // Включаем поддержку заголовка, редактора, миниатюры, комментариев)); } </p><p>Здесь sheensay_product — это название нового типа записей. Оно не должно конфликтовать с другими в системе, об этом подробнее ниже.<br> Также, в этом варианте оно служит ярлыком этого типа записей, то есть присутствует в URL. Если же хотите указать другой ярлык, например <b>продукция </b>, делайте как на примере ниже</p><p>Add_action("init", "sheensay_post_type"); function sheensay_post_type() { register_post_type("sheensay_product", array("labels" => array("name" => "Продукция", "singular_name" => "Продукцию",), "public" => true, "rewrite" => array("slug" => "продукция"), // Тут определяется ярлык Custom Post Type "has_archive" => true, "supports" => array("title", "editor", "thumbnail"),)); } </p><h3>Как правильно подобрать название нового типа записи</h3> <blockquote class="attention"><p>В WordPress зарезервированы следующие названия, которые нельзя использовать в качестве имени нового типа записи:</p> <ul><li>attachment</li> <li>revision</li> <li>nav_menu_item</li> <li>action</li> <li>theme</li> <li>order</li> </ul><p>Также, стоит воздержаться от использования префикса wp_ в начале названия, так как, возможно, это вызовет конфликты с будущими версиями ядра WordPress.<br> Лучше всего, если вы будете предварять названия произвольными префиксами, связанными с названием вашего сайта, продукта или бренда, например sheensay_product , тогда гарантированно избежите потенциальных конфликтов</p> </blockquote> <h3>Как сделать произвольную таксономию</h3> <p>Произвольные типы записей могут использовать таксономии из записей, например, рубрики (категории) или метки (теги), но можно, чтобы таксономии были свои.</p><p>Add_action("init", "sheensay_post_type"); function sheensay_post_type() { // Регистрируем таксономию register_taxonomy("sheensay_product_type", "sheensay_product", array("label" => "Типы", "hierarchical" => true, // Если TRUE, таксономия будет аналогом рубрик (категорий). Если FALSE (по умолчанию), то таксономия станет аналогом меток (тегов). "rewrite" => array("slug" => "тип-продукции"),)); // Регистрируем произвольный тип записи (Custom Post Type) register_post_type("sheensay_product", array("labels" => array("name" => "Продукция", "singular_name" => "Продукцию",), "public" => true, "rewrite" => array("slug" => "продукция"), // Тут определяется ярлык CPT "has_archive" => true, "supports" => array("title", "editor", "thumbnail"), // Включаем поддержку заголовка, редактора, миниатюры)); } </p><h3>Шаблоны для произвольных типов постов</h3> <p>Всё зависит от того, какого рода информация отображается. Вариантов может быть 3: шаблон конкретной записи, шаблон архивов записей и шаблон таксономий<br></p> <h4>Шаблон страницы записи</h4> <p>Перечисляются в порядке приоритета</p> <ol><li>single-{тип_поста}.php</li> <li>single.php</li> <li>index.php</li> </ol><h4>Шаблон архива записей</h4> <ol><li>archive-{тип_поста}.php</li> <li>archive.php</li> <li>index.php</li> </ol><h4>Шаблон произвольной таксономии</h4> <ol><li>taxonomy-{имя_таксономии}-{имя_термина}.php</li> <li>taxonomy-{имя_таксономии}.php</li> <li>taxonomy.php</li> <li>archive.php</li> <li>index.php</li> </ol><p>Здесь имя_таксономии — это sheensay_product_type , а имя_термина — это ярлык той таксономии, что вы создадите в админке.</p> <h3>Как получить данные произвольного типа записей (Custom Post Type WordPress) и отобразить на сайте</h3> <p>Получить данные произвольного типа записей (Custom Post Type) в WordPress для отображения на сайте можно теми же способами, что и обычные Записи и Страницы</p><p> $args = array("post_type" => "sheensay_product", // Указываем наш новый тип записи "posts_per_page" => 10,); $p = get_posts($args); foreach ($p as $post) { setup_postdata($post); ?> "><?= the_title() ?><br /> <?php } wp_reset_postdata(); ?> </p><p>Если же говорить про главный запрос, по которому данные, например, предзагружаются для вывода на главной странице или стандартных страницах архивов, то произвольные типы записей там отключены.<br> Поэтому, если вы хотите, чтобы какая-либо созданный тип записей отображался, скажем, на страницах архивов наряду с обычными записями, вам нужно изменить фильтр предзагрузки постов</p><p> // Подключаем к стандартным "post" и "page" наш "sheensay_product" add_action("pre_get_posts", "add_sheensay_product_in_main_query"); function add_sheensay_product_in_main_query($query) { if (is_archive() && $query -> is_main_query()) $query -> set("post_type", array("post", "page", "sheensay_product")); return $query; } </p><h3>Готовый класс для создания произвольного типа записей с произвольной таксономией</h3> <p>Ниже представлен класс, с помощью которого вы сможете зарегистрировать любой произвольный тип записей с собственной таксономией, а URL будет иметь вид<br> http://example.com/продукция/тип/продукт</p> <blockquote class="note"><p>Чтобы URL формировался, вы должны перейти в настройки постоянных ссылок /wp-admin/options-permalink.php и установить любой отличный от простого вид общих настроек</p> <p><img src='https://i0.wp.com/sheensay.ru/wp-content/uploads/2016/08/options-permalink.png' width="100%" loading=lazy></p> </blockquote> <p>Ниже сам код класса. Вы можете не вносить в него никаких изменений, а поменять ярлык типа записи можно в самом конце в строке new Sheensay_Product("продукция"); <br> Сам код пишется в или в . Первый вариант предпочтительнее, так как в этом случае, вы не потеряете доступа к контенту нового типа записей при смене активной темы.</p><p> <??p=1713 * */ final class Sheensay_Product { // Ярлык произвольного типа записи по умолчанию private $post_type = "продукция"; function __construct($post_type = "") { // Переопределяем значение ярлыка по умолчанию if ($post_type) $this -> post_type = $post_type; /* * Регистрируем Custom Post Type */ add_action("init", array($this, "sheensay_cpt_product")); /* * Фильтруем URL */ add_filter("post_type_link", array($this, "product_permalink_structure"), 10, 2); /* * Чтобы работала пагинация */ add_action("generate_rewrite_rules", array($this, "fix_product_category_pagination")); } function sheensay_cpt_product() { /* * Регистрируем произвольную таксономию к новому типу записей */ register_taxonomy("sheensay_product_type", "sheensay_product", array("label" => "Типы", "hierarchical" => true, "query_var" => true, "rewrite" => array("slug" => $this -> post_type),)); /* * Регистрируем новый тип записи */ $labels = array("name" => "Продукция", // Основное название "singular_name" => "Продукцию", // Добавить "add_new" => "Добавить новую", // Имя ссылки на новую запись в сайдбаре "add_new_item" => "Добавить новую продукцию", // Заголовок в редакторе при добавлении новой записи); $args = array("labels" => $labels, "public" => true, "publicly_queryable" => true, "show_ui" => true, "query_var" => true, "capability_type" => "post", "hierarchical" => false, "menu_position" => null, "supports" => array("title", "editor", "thumbnail", "excerpt"), "rewrite" => array("slug" => $this -> post_type . "/%sheensay_product_type%", "with_front" => false,), "has_archive" => $this -> post_type,); register_post_type("sheensay_product", $args); if (current_user_can("manage_options")) // Вот с этой функцией осторожней. Она сбрасывает все правила определения URL. Лучше её закомментировать после завершения всех работ flush_rewrite_rules(); } function product_permalink_structure($post_link, $post) { if (FALSE !== strpos($post_link, "%sheensay_product_type%")) { $product_type_term = get_the_terms($post -> ID, "sheensay_product_type"); if (!empty($product_type_term)) $post_link = str_replace("%sheensay_product_type%", $product_type_term -> slug, $post_link); } return $post_link; } function fix_product_category_pagination($wp_rewrite) { unset($wp_rewrite -> rules[ $this -> post_type . "/([^/]+)/page/?({1,})/?$"]); $wp_rewrite -> rules = array($this -> post_type . "/?$" => $wp_rewrite -> index . "?post_type=sheensay_product", $this -> post_type . "/page/?({1,})/?$" => $wp_rewrite -> index . "?post_type=sheensay_product&paged=" . $wp_rewrite -> preg_index(1), $this -> post_type. "/([^/]+)/page/?({1,})/?$" => $wp_rewrite -> index . "?sheensay_product_type=" . $wp_rewrite -> preg_index(1) . "&paged=" . $wp_rewrite -> preg_index(2),) + $wp_rewrite -> rules; } } /* * Запускаем класс * В скобках можно определить название ярлыка типа записи */ new Sheensay_Product("продукция"); </p><p>Теперь в админке вы можете зарегистрировать новую продукцию и их типы</p> <p>a! Система управления сайтом <b>WordPress </b> завоевала признание в течении нескольких лет, но настоящим прорывом стала реализация возможности разделять записи на типы. В этом уроке подробно рассмотрим пользовательские типы записей их создание и использование.</p> <h3>Немного истории</h3> <p>На практике, пользовательские типы записей появились достаточно давно, а точнее с 17 февраля 2005 года, когда в <b>WordPress 1.5 </b> была добавлена ​​поддержка пользовательских типов для статических страниц, посредством post_type поля в базе данных. Функция Wp_insert_post() была примерно с <b>WordPress 1.0 </b> так что, когда поле post_type было реализовано в 1.5, можно было достаточно просто заполнить его с помощью этой функции.</p> <p>И только в версии 2.8 появилась функция register_post_type() для создания пользовательских типов и некоторые другие полезные вещи были доступны в «ночных сборках», а уже с 2.9 функции стали доступны всем.</p> <h3>А что сейчас?!</h3> <p>Пользовательский тип записи это не более чем обычная запись (статья) с определенным значением поля post_type в базе данных. В обычной записи поле post_type имеет значение <b>post </b>, страница имеет значение <b>page </b> и так далее. Однако теперь мы можем создавать свои собственные типы, чтобы указать особенности контента, содержащегося в записи. Можно создать пользовательские типы записей для книг, фильмов, анекдотов, продуктов и всего чего угодно.<br> Если все сделано правильно, то с помощью всего лишь нескольких строк кода можно достичь следующих результатов:</p> <ul><li>Вынести пользовательский тип на основную панель админ зоны с пунктами подменю: список всех записей с таким типов, а также создать новую запись с таким типом;</li> <li>Создать архив записей подобного типа, то есть сделать что-то на подобии главной страницы для пользовательского типа;</li> <li>Создать категории и теги, которые могут быть доступными для пользовательских типов записей, а также пользовательские таксономии.</li> </ul><p>Различные типы контента предъявляют различные требования к данным. Для обычных записей вы хотите, чтобы был указан автор, категория и дата. В то время как для записи с типом «книги», хотелось бы иметь возможность указать автора книги, количество страниц, жанр, издательство и другие конкретные данные. Этого легко добиться используя пользовательские (meta boxes) области для ввода данных.</p> <p> — области для ввода дополнительных данных прямо на странице создания записи. Такие области упрощают работу с пользовательскими типами записей.</p><p><img src='https://i0.wp.com/xozblog.ru/wp-content/uploads/2012/11/meta-boxes.jpg' align="center" height="202" width="299" loading=lazy><br></p> <h3>Работа с пользовательскими типами записей</h3> <p>Чтобы эффективно создавать и использовать пользовательские типы записей, Вы должны быть знакомы со следующими составляющими:</p> <ul><li>Создание пользовательских типов записей;</li> <li>Создание пользовательской таксономии;</li> <li>Создание пользовательских областей данных.</li> </ul><h4>Создание пользовательских типов записей</h4> <p>Первое что необходимо сделать, так это создать непосредственно сам пользовательский тип записи. В идеале нужно создать плагин для работы с пользовательскими типами записей, но Вы также можете использовать файл functions.php вашей темы.</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br></p> </td><td><p><br>$args = array () ; <br>} <br></p> </td> </tr></tbody></table><p>Это простейшая форма создания типа, который практически не имеет настроек. Для разработки нового типа наших записей, будем использовать некоторые из наиболее часто используемых опций и добавим их к ранее пустому массиву $args .</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br>12<br>13<br>14<br>15<br>16<br>17<br>18<br>19<br>20<br>21<br>22<br>23<br>24<br>25<br>26<br>27<br></p> </td><td><p>function my_custom_post_product() { <br>$labels = array (<br>"name" => _x( "Продукция" , "post type general name" ) , <br>"singular_name" => _x( "Продукт" , "post type singular name" ) , <br>"add_new" => _x( "Добавить новый" , "product" ) , <br>"add_new_item" => __( <span>"Добавить новый продукт" </span> ) , <br>"edit_item" => __( <span>"Редактировать продукт" </span> ) , <br>"new_item" => __( "Новый продукт" ) , <br>"all_items" => __( "Вся продукция" ) , <br>"view_item" => __( <span>"Смотреть продукт" </span> ) , <br>"search_items" => __( "Найти продукт" ) , <br>"not_found" => __( <span>"Продукты не найдены" </span> ) , <br>"not_found_in_trash" => __( <span>"Нет удаленной продукции" </span> ) , <br>"parent_item_colon" => "" , <br>"menu_name" => "Продукция" <br>) ; <br>$args = array (<br>"labels" => $labels , <br>"description" => <span>"Пользовательский тип записей продукции" </span>, <br>"public" => true , <br>"menu_position" => 5 , <br>"supports" => array ( "title" , "editor" , "thumbnail" , "excerpt" , "comments" , "product_category" ) , <br>"has_archive" => true , <br>) ; <br> register_post_type( "product" , $args ) ; <br>} <br> add_action( "init" , "my_custom_post_product" ) ; </p> </td> </tr></tbody></table> <ul><li><b>labels </b> — данный массив меток используется для описания создаваемого пользовательского типа записи в теме./li></li><li><b>description </b> — краткая информация создаваемого пользовательского типа записи, что он делает и почему мы его используем.</li> <li><b>public </b> — использовать ли пользовательский тип публично и показывать ли его в административной зоне. В данном случае установлено истина.</li> <li><b>menu_position </b> — позиция пункта меню нашего типа на основной панели администратора. Значение 5 значит пункт установиться сразу после пункта меню «Записи», если 10 значит после пункта «Медиафайлы» и тд.</li> <li><b>supports </b> — данная опция содержит массив, в котором описаны те поля которые мы можем редактировать на странице создания записи. То есть title — появится поле для ввода названия записи, editor — будет отображена текстовая область для ввода текста записи и тд. А также указана используемая пользовательская таксономия product_category .</li> <li><b>has_archive </b> — если установлено true, будет создано правило rewrite, позволяя получить список записей нашего типа по адресу http://mysite.com/product/</li> </ul><p><br><img src='https://i0.wp.com/xozblog.ru/wp-content/uploads/2012/11/custom-post-types-1.png' align="center" width="100%" loading=lazy><br></p> <p>После установки этого кода в файл function.php , вы должны увидеть меню для пользовательского типа записей. Появилась возможность добавлять запись и просмотреть список записей.</p> <p>Однако, это далеко не все опции для массива аргументов, полный список опций и все возможности пользовательских типов Вы найдете .</p> <h4>Интерактивные оповещения</h4> <p>WordPress генерирует некоторые сообщения, вызванные действиями пользователя. Мы также можем создать подобные сообщения, чтобы оповестить пользователя при работе с типами. Делается это post_updated_messages .</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br>12<br>13<br>14<br>15<br>16<br>17<br>18<br></p> </td><td><p>function my_updated_messages( $messages ) { <br>global $post , $post_ID ; <br>$messages [ "product" ] = array (<br>0 => "" , <br>1 => sprintf ( __(<span>"Продукт обновлен. Посмотреть" </span><br>2 => __() , <br>3 => __(<span>"Пользовательские поля обновлены." </span>) , <br>4 => __(<span>"Продукт обновлен." </span>) , <br>5 => isset ($_GET [ "revision" ] ) ? sprintf ( __(<span>"Product restored to revision from %s" </span>) , wp_post_revision_title( (int) $_GET [ "revision" ] , false ) ) : false , <br>6 => sprintf ( __(<span>"Продукт опубликован. Посмотреть" </span>) , esc_url( get_permalink($post_ID ) ) ) , <br>7 => __(<span>"Продукт сохранен." </span>) , <br>8 => sprintf ( __(<span>"Продукт отправлен. Посмотреть" </span><br>9 => sprintf ( __(<span>"Продукт запланирован на: <strong>%1$s</strong>. Посмотреть" </span>) , date_i18n( __( "M j, Y @ G:i" ) , strtotime ( $post -> post_date ) ) , esc_url( get_permalink($post_ID ) ) ) , <br>10 => sprintf ( __(<span>"Product draft updated. Посмотреть" </span>) , esc_url( add_query_arg( "preview" , "true" , get_permalink($post_ID ) ) ) ) , <br>) ; <br>return $messages ; <br>} <br> add_filter( "post_updated_messages" , "my_updated_messages" ) ; </p> </td> </tr></tbody></table><p>Обратите внимание, что вы можете задать оповещения для всех пользовательских типов записей с помощью этой одной функции. Если не указывать название типа то оповещения будут использоваться для всех пользовательских типов.</p><p><img src='https://i2.wp.com/xozblog.ru/wp-content/uploads/2012/11/custom-post-types-2.png' align="center" width="100%" loading=lazy><br></p> <p>Контекстные подсказки достаточно редко реализуются и используются. Как пользователь, я никогда не использовал эту возможность, но я уверен, что все же кто-то ими пользуется. Поэтому давайте создадим такую контекстную помощь.<br> Контекстные подсказки находятся в правом верхнем углу, возле вкладки настройки экрана.</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br>12<br>13<br>14<br>15<br>16<br></p> </td><td><p>function my_contextual_help( $contextual_help , $screen_id , $screen ) { <br>if ( "edit-product" == $screen -> id ) { </p><p>$contextual_help = <span>"<h2>Продукция</h2><br> <p>На этой странице находится список всей продукции, которая продается на сайте. Записи расположены в обратном хронологическом порядке, последними в списке являются товары, которые мы добавили первыми.</p> <br> <p>Вы можете просматривать / редактировать сведения о каждом продукте, нажав на его название, или можете выполнить массовые действие с помощью выпадающего меню, выбрав несколько элементов.</p>" </span>; </p><p>} elseif ( "product" == $screen -> id ) { </p><p>$contextual_help = <span>"<h2>Создание/редактирование продукта</h2><br> <p>Эта страница позволяет создать продукт или отредактировать уже имеющиеся данные о нем. Пожалуйста, не забудьте заполнить дополнительные поля.</p>" </span>; </p><p>} <br>return $contextual_help ; <br>} <br> add_action( "contextual_help" , "my_contextual_help" , 10 , 3 ) ; </p> </td> </tr></tbody></table><p>Для того чтобы показать такую подсказку нам необходимо знать идентификатор экрана. Если при создании понадобится узнать ID экрана просто делаем так:</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"> </td><td><p>echo $screen -> id ; </p> </td> </tr></tbody></table><p><br><img src='https://i1.wp.com/xozblog.ru/wp-content/uploads/2012/11/custom-post-types-3.png' align="center" width="100%" loading=lazy><br></p> <h3>Пользовательская таксономия</h3> <p>На обычного блога для создания организационной структуры вполне хватает категорий и тегов. В то время как для пользовательских типов постов этого не всегда достаточно. Такую проблему может решить пользовательская таксономия. В нашем случае таксономия с названием «Категории продуктов», которая используется только для записей с пользовательским типов «Продукция».</p> <p>Процесс создания пользовательской таксономии практически идентичен созданию пользовательских типов записей. Давайте посмотрим на нашем примере:</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br>12<br>13<br>14<br>15<br>16<br>17<br>18<br>19<br>20<br>21<br></p> </td><td><p>function my_taxonomies_product() { <br>$labels = array (<br>"name" => _x( <span>"Категории продуктов" </span>, "taxonomy general name" ) , <br>"singular_name" => _x( <span>"Категория продуктов" </span>, "taxonomy singular name" ) , <br>"search_items" => __( <span>"Найти категорию продуктов" </span> ) , <br>"all_items" => __( <span>"Все категории продуктов" </span> ) , <br>"parent_item" => __( <span>"Родительская категория продуктов" </span> ) , <br>"parent_item_colon" => __( <span>"Родительская категория продуктов:" </span> ) , <br>"edit_item" => __( <span>"Редактировать категорию продуктов" </span> ) , <br>"update_item" => __( <span>"Обновить категорию продуктов" </span> ) , <br>"add_new_item" => __( <span>"Добавить новую категорию продуктов" </span> ) , <br>"new_item_name" => __( <span>"Новая категория продуктов" </span> ) , <br>"menu_name" => __( <span>"Категории продуктов" </span> ) , <br>) ; <br>$args = array (<br>"labels" => $labels , <br>"hierarchical" => true , <br>) ; <br> register_taxonomy( "product_category" , "product" , $args ) ; <br>} <br> add_action( "init" , "my_taxonomies_product" , 0 ) ; </p> </td> </tr></tbody></table><p>Как и при создании пользовательского типа мы сформировали массив label, и указали что для создаваемой таксономии актуальна иерархическую структуру (т.е. могут быть родительский и дочерний элементы) — это свойственно рубрикам в обычных записях. В противном случае если структура не иерархическая — созданы обычные теги. Более подробно о таксономии Вы можете почитать .</p><p><img src='https://i2.wp.com/xozblog.ru/wp-content/uploads/2012/11/custom-post-types-4.png' align="center" width="100%" loading=lazy><br></p> <h3>Дополнительные области данных</h3> <p>Дополнительные области или блоки для ввода данных (meta boxes) вы могли видеть на странице редактирования записи. Все знают стандартные , такие как выбор рубрики или тегов. Также в некоторых темах встречаются позволяющие прикрепить картинку к записи и тд.</p> <p>Так как мы создаем пользовательский тип «Продукция», то нам явно понадобится цена продукта, давайте рассмотрим процесс создания пользовательских .</p> <p>Процесс создания можно разделить на 3 этапа:</p> <ul><li>Определение самого блока;</li> <li>Определение содержимого (какие поля присутствуют в блоке);</li> <li>Описание алгоритмов обработки введенных данных.</li> </ul><h4>Определение meta boxes</h4> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br></p> </td><td><p>add_action( "add_meta_boxes" , "product_price_box" ) ; <br>function product_price_box() { <br> add_meta_box( <br>"product_price_box" , <br> __( "Цена продукта" , "myplugin_textdomain" ) , <br>"product_price_box_content" , <br>"product" , <br>"side" , <br>"high" <br>) ; <br>} </p> </td> </tr></tbody></table><p>Приведенный выше код создает блок со следующими параметрами:</p> <ul><li><b>product_price_box </b> — уникальный идентификатор для meta box (он не обязательно должен совпадать с названием функции);</li> <li><b>Цена продукта </b> — название meta box, которое видит админ на странице;</li> <li><b>product_price_box_content </b> — функция, которая будет отображать содержимое окна;</li> <li><b>product </b> — название пользовательского типа записи, к которому принадлежит meta boxes;</li> <li><b>side </b> — положение блока на странице (side , normal или advanced — по-умолчанию);</li> <li><b>high </b> — приоритет meta boxes (в данном случае «высокий», блок находится в самом верху сайдбара. Варианты: high , core , low или default — по-умолчанию).</li> </ul><h4>Определение содержимого</h4> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br></p> </td><td><p>function product_price_box_content( $post ) { <br> wp_nonce_field( plugin_basename( __FILE__ ) , ) ; <br>echo <span>"<label for="product_price"></label>" </span>; <br>echo <span>"<input type="text" id="product_price" name="product_price" placeholder="введите цену">" </span>; <br>} </p> </td> </tr></tbody></table><p>Добавляем всего лишь одно поле, для ввода цены продукта. Заметьте название функции совпадает со значением третьего параметра при объявлении (код выше).</p> <h4>Обработка введенных данных</h4> <p>Последний шаг — это сохранение введенной цены на продукт в базу данных.</p> <table cellspacing="0" cellpadding="0"><tbody><tr><td class="line-numbers"><p>1<br>2<br>3<br>4<br>5<br>6<br>7<br>8<br>9<br>10<br>11<br>12<br>13<br>14<br>15<br>16<br>17<br>18<br>19<br></p> </td><td><p>add_action( "save_post" , "product_price_box_save" ) ; <br>function product_price_box_save( $post_id ) { </p><p>if ( defined ( "DOING_AUTOSAVE" ) && DOING_AUTOSAVE ) <br>return ; </p><p>if ( ! wp_verify_nonce( $_POST [ <span>"product_price_box_content_nonce" </span>] , plugin_basename( __FILE__ ) ) ) <br>return ; </p><p>if ( "page" == $_POST [ "post_type" ] ) { <br>if ( ! current_user_can( "edit_page" , $post_id ) ) <br>return ; <br>} else { <br>if ( ! current_user_can( "edit_post" , $post_id ) ) <br>return ; <br>} <br>$product_price = $_POST [ "product_price" ] ; <br> update_post_meta( $post_id , "product_price" , $product_price ) ; <br>} </p> </td> </tr></tbody></table><p>Большая часть функции это проверки. Прежде всего проверяется автосохранение, цена не будет сохранена до тех пор пока не будет нажата кнопка сохранить или обновить. После проверяются права доступа на редактирование данных этой записи и только после этого с помощью функции update_post_meta цена заносится в базу.</p> <h3>Отображение записей созданного типа на блоге</h3> <p>Все этапы создания пользовательского типа записей мы прошли, осталось воспользоваться плодами наших трудов, и сформировать отображение записей на блоге в соответствии с установленной нами таксономией.</p> <p>Так как в процессе создании пользовательского типа мы указали истину для параметра has_archive то список записей типа product доступны по адресу http://mysite.com/product/ .</p> <p>Для отображения используется файл archive-.php (в нашем случае archive-product.php) если такой существует. Иначе, для отображения будет использован archive.php и если такой файл отсутствует в теме, то будет использовать ) <br>) ; <br>$products = new WP_Query( $args ) ; <br>if ( $products -> have_posts () ) { <br>while ( $products -> have_posts () ) { <br>$products -> the_post () ; <br>?> <br>< h1> <?php the_title() ?> </ h1> <br>< div class = "content" > <br><?php the_content() ?> <br></ div> <br><?php <br>} <br>} <br>else { <br>echo <span>"О нет, продукты не обнаружены!" </span>; <br>} <br>?> </p> <h4>Отображение цены</h4> <p>Введенные дополнительные данные, в нашем случае цена продукта, могут быть получены с помощью функции get_post_meta () . Так как мы используем дополнительно поле product_price , то чтобы получить значение цены:</p> <h3>Плагин для создания пользовательских типов записей</h3> <p>Если Вы не уверены в своих силах в области программирования, то всегда можно найти уже готовое решение (плагин) и воспользоваться им. Пользовательские типы не исключение. Плагин <b>WCK Custom Post Type Creator </b> позволяет вам легко создавать пользовательские типы записей для WordPress без знания программирования. </p> <p>Категории по умолчанию используются только для постов. Однако в некоторых сценариях у вас может возникнуть желание использовать их с пользовательским типом постов. В этой статье мы расскажем вам как это можно сделать.</p> <h4>Способ с плагином</h4> <p>Для начинающих пользователей мы рекомендуем использовать Custom Post Type UI плагин, чтобы создать пользовательский тип постов. Используя этот плагин, у вас есть возможность ассоциировать пользовательский тип постов с любой встроенной или пользовательской таксономией, включая категории. После установки плагина зайдите в CPT UI » Add/Edit Post Types для того, чтобы создать новый пользовательский тип постов или отредактировать существующий.</p> <p>Прокрутите вниз до Advanced Options и там вы увидите параметр Built in Taxnomies. Отметьте ячейку напротив категорий и сохраните свой тип постов.</p> <p><img src='https://i0.wp.com/coma.lv/wp-content/uploads/2016/11/enablecategoriesforcpt.jpg' align="center" width="100%" loading=lazy></p> <h4>Ручное добавление категорий для пользовательского типа постов</h4> <p>Если вы создали пользовательский тип постов, добавив код в файл functions.php, то вам придется изменить этот код, чтобы добавить категорию как поддерживаемую таксономию. Все, что от вас требуется, это добавить эту строчку в аргументы вашего пользовательского типа постов:</p> <p><i>‘taxonomies’ => array(‘category’), </i></p> <p>Не исключено, что у вас уже есть эта строка в коде с какой-нибудь другой пользовательской таксономией. Если это так, то вам надо просто добавить запятую после нее и добавить категорию:</p> <p><i>‘taxonomies’ => array(‘topics’, ‘category’), </i></p> <p>Вот пример целого кода, где мы создали пользовательский тип постов под названием «фильмы» с поддержкой всех встроенных категорий.</p><p>Function custom_post_type() { // Set UI labels for Custom Post Type $labels = array("name" => _x("Movies", "Post Type General Name", "twentythirteen"), "singular_name" => _x("Movie", "Post Type Singular Name", "twentythirteen"), "menu_name" => __("Movies", "twentythirteen"), "parent_item_colon" => __("Parent Movie", "twentythirteen"), "all_items" => __("All Movies", "twentythirteen"), "view_item" => __("View Movie", "twentythirteen"), "add_new_item" => __("Add New Movie", "twentythirteen"), "add_new" => __("Add New", "twentythirteen"), "edit_item" => __("Edit Movie", "twentythirteen"), "update_item" => __("Update Movie", "twentythirteen"), "search_items" => __("Search Movie", "twentythirteen"), "not_found" => __("Not Found", "twentythirteen"), "not_found_in_trash" => __("Not found in Trash", "twentythirteen"),); // Set other options for Custom Post Type $args = array("label" => __("movies", "twentythirteen"), "description" => __("Movie news and reviews", "twentythirteen"), "labels" => $labels, "supports" => array("title", "editor", "excerpt", "author", "thumbnail", "comments", "revisions", "custom-fields",), "hierarchical" => false, "public" => true, "show_ui" => true, "show_in_menu" => true, "show_in_nav_menus" => true, "show_in_admin_bar" => true, "menu_position" => 5, "can_export" => true, "has_archive" => true, "exclude_from_search" => false, "publicly_queryable" => true, "capability_type" => "page", // This is where we add taxonomies to our CPT "taxonomies" => array("category"),); // Registering your Custom Post Type register_post_type("movies", $args); } /* Hook into the "init" action so that the function * Containing our post type registration is not * unnecessarily executed. */ add_action("init", "custom_post_type", 0); </p><h4>Отображение нескольких типов постов на странице категории</h4> <p>По умолчанию страницы категории на сайте WordPress отображают стандартный тип постов. Если хотите, чтобы ваш тип постов отображался бы на той же странице категорий, что и посты по умолчанию, то вам надо добавить следующий код в файл functions.php:</p><p>Add_filter("pre_get_posts", "query_post_type"); function query_post_type($query) { if(is_category()) { $post_type = get_query_var("post_type"); if($post_type) $post_type = $post_type; else $post_type = array("nav_menu_item", "post", "movies"); // don"t forget nav_menu_item to allow menus to work! $query->set("post_type",$post_type); return $query; } } </p><p>Не забудьте поменять movies на название своего пользовательского типа постов.</p> <p><i>Наша специальность - разработка и поддержка сайтов на WordPress. Контакты для бесплатной консультации - , </i></p> <script type="text/javascript"> <!-- var _acic={dataProvider:10};(function(){var e=document.createElement("script");e.type="text/javascript";e.async=true;e.src="https://www.acint.net/aci.js";var t=document.getElementsByTagName("script")[0];t.parentNode.insertBefore(e,t)})() //--> </script><br> <br> <script>document.write("<img style='display:none;' src='//counter.yadro.ru/hit;artfast_after?t44.1;r"+ escape(document.referrer)+((typeof(screen)=="undefined")?"": ";s"+screen.width+"*"+screen.height+"*"+(screen.colorDepth? screen.colorDepth:screen.pixelDepth))+";u"+escape(document.URL)+";h"+escape(document.title.substring(0,150))+ ";"+Math.random()+ "border='0' width='1' height='1' loading=lazy>");</script> </div> <div class="clear"></div> </div> </article> <div class='yarpp-related'> <section id="related_posts"> <div class="block-head"> <span>Больше информации по теме</span> <div class="stripe-line"></div> </div> <div class="post-listing"> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/47ec25d591cd094f5abc899417bd868c.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Внутреннее устройство ноутбука" / loading=lazy></div> <a class="post_7839" href="/vnutrennee-ustroistvo-noutbuka-iz-chego-sostoit-noutbuk-chto/">Внутреннее устройство ноутбука</a></div> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/af04abdcf7234c8c3e89dfa268f35291.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Передача показаний, счет за газ" / loading=lazy></div> <a class="post_7839" href="/pokazaniya-schetchika-gaza-nizhegorodskaya-peredacha-pokazanii/">Передача показаний, счет за газ</a></div> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/0e0d357be8de6482eddfab74638d57df.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Как конвертировать файлы FLAC в MP3, AAC, WMA, WAV и т" / loading=lazy></div> <a class="post_7839" href="/iz-wav-v-flac-onlain-kak-konvertirovat-faily-flac-v-mp3-aac-wma-wav-i-t-d/">Как конвертировать файлы FLAC в MP3, AAC, WMA, WAV и т</a></div> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/0ad2a4ee8e58154eb6ed58fbb05e699c.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Cкачать приложение Yota Ready для Windows на компьютер или ноутбук" / loading=lazy></div> <a class="post_7839" href="/iota-prilozhenie-dlya-pk-ckachat-prilozhenie-yota-ready-dlya-windows-na-kompyuter/">Cкачать приложение Yota Ready для Windows на компьютер или ноутбук</a></div> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/5bd09a3d5e798134f8879b4ed427a2b0.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Простые способы установки ipa файлов на гаджеты Apple Открыть ipa файл на ipad" / loading=lazy></div> <a class="post_7839" href="/kak-ustanovit-neoficialnye-prilozheniya-na-ios-bez-dzheilbreika-prostye/">Простые способы установки ipa файлов на гаджеты Apple Открыть ipa файл на ipad</a></div> <div class="related-item"> <div class="post-thumbnail tie-appear replacemy" data="post_7839"> <img width="195" height="110" src="/uploads/d79ff37aeb17ff4d73244884cfe7a1c1.jpg" class="attachment-tie-related size-tie-related wp-post-image" alt="Учимся выкладывать в YouTube и "Инстаграм"" / loading=lazy></div> <a class="post_7839" href="/kak-vylozhit-video-v-internet-uchimsya-vykladyvat-v-youtube-i/">Учимся выкладывать в YouTube и "Инстаграм"</a></div> </div> </section> </div> <div id="comments"> <ol class="commentlist"> <div class='rreekk' rel='15'> <div id="smartrotator_ad_4" onclick="yaCounter40492595.reachGoal ('tizercommentbefore1'); return true;"></div> </div> </ol> <div class="clear"></div> <div id="respond" class="comment-respond"> </div> </div> </div> <aside id="sidebar"> <div class="theiaStickySidebar"> <div class="execphpwidget"> <div id="text-4" class="widget widget_text"> <div class="widget-top"> <span>Полезное</span> <div class="stripe-line"></div> </div> <div class="widget-container"> <div class="textwidget"> <style> .wpp-list li img { -webkit-border-radius: 65px; -moz-border-radius: 65px; border-radius: 65px; } .wpp-list li .post-views { display: block; margin-left: 70px; color: #828283; font-size: 12px; } .wpp-list li .post-views i { font-size: 13px; } .wpp-list li { line-height: 22px !important; } .replacemy { cursor: pointer; } } </style> <ul class="wpp-list"> <li> <div class="replacemy"><img src="/uploads/6de910abbef3c8b9f40ae4b49b080f72.jpg" width="65" height="65" title="Как найти потерянный телефон андроид: все способы и рекомендации" alt="Как найти потерянный телефон андроид: все способы и рекомендации" class="wpp-thumbnail wpp_cached_thumb wpp_featured" / loading=lazy></div><a href="/prilozhenie-dlya-otslezhivaniya-mestonahozhdeniya-telefona-kak/" class=""><br>Как найти потерянный телефон андроид: все способы и рекомендации</a></li> <li> <div class="replacemy"><img src="/uploads/e01a8d75522fc41460b904b2ede30a34.jpg" width="65" height="65" title="Настроить интернет на андроиде vertex оператор мтс" alt="Настроить интернет на андроиде vertex оператор мтс" class="wpp-thumbnail wpp_cached_thumb wpp_featured" / loading=lazy></div><a href="/mobilnyi-telefon-vertex-instrukciya-po-ekspluatacii-nastroit-internet-na/" class=""><br>Настроить интернет на андроиде vertex оператор мтс</a></li> <li> <div class="replacemy"><img src="/uploads/6a39a62b938cc8f6a726d7a723ebfe20.jpg" width="65" height="65" title="LG G Flex - Технические характеристики Lg вогнутый" alt="LG G Flex - Технические характеристики Lg вогнутый" class="wpp-thumbnail wpp_cached_thumb wpp_featured" / loading=lazy></div><a href="/obzor-lg-g-flex-vot-tak-zagnuli-lg-g-flex---tehnicheskie-harakteristiki-lg-vognutyi/" class=""><br>LG G Flex - Технические характеристики Lg вогнутый</a></li> <li> <div class="replacemy"><img src="/uploads/2bb7a7d49f5d45a79cc42a6e2e1fbec5.jpg" width="65" height="65" title="Открытки и поздравления по email Как отправить открытку по эл почте" alt="Открытки и поздравления по email Как отправить открытку по эл почте" class="wpp-thumbnail wpp_cached_thumb wpp_featured" / loading=lazy></div><a href="/kak-otpravit-otkrytku-po-elektronnoi-pochte-otkrytki-i/" class=""><br>Открытки и поздравления по email Как отправить открытку по эл почте</a></li> <li> <div class="replacemy"><img src="/uploads/13499baceef818e8f97fdfc337b01da6.jpg" width="65" height="65" title="Какая Винда лучше для игр?" alt="Какая Винда лучше для игр?" class="wpp-thumbnail wpp_cached_thumb wpp_featured" / loading=lazy></div><a href="/kakuyu-vindovs-7-luchshe-ustanovit-dlya-igr-kakaya-vinda-luchshe-dlya-igr/" class=""><br>Какая Винда лучше для игр?</a></li> </ul> </div> </div> </div> </div> </div> <div> </div> </aside> <div class="clear"></div> </div> <footer id="theme-footer"> <div id="footer-widget-area" class="wide-narrow-2c"><div id="footer-first" class="footer-widgets-box"><div id="text-2" class="footer-widget widget_text"><div class="footer-widget-top"></div><div class="footer-widget-container"><div class="textwidget"> <p>Все о современной технике. Ошибки. Игры. Поломки. Техника. Интернет <p></p> </div></div></div></div><div id="footer-second" class="footer-widgets-box"><div id="text-3" class="footer-widget widget_text"> <div class="footer-widget-top"></div> <div class="footer-widget-container"> <div class="textwidget"><a href="/sitemap.xml">Карта сайта</a></div> </div> </div><div id="text-7" class="footer-widget widget_text"><div class="footer-widget-top"></div><div class="footer-widget-container"><div class="textwidget"><br> <a href="/feedback/">Обратная связь</a> </div></div></div></div></div><div class="clear"></div> </footer> <div class="clear"></div> <div class="footer-bottom"> <div class="container"> <div class="alignright"></div> <div class="alignleft"> © Copyright 2017, https://qzoreteam.ru</div> <div class="clear"></div> </div> </div> </div> </div> </div> <div id="topcontrol" class="fa fa-angle-up" title="Перейти к началу страницы"></div> <div id="fb-root"></div> <script type="text/javascript"> jQuery(function($) { $(document).on("click", ".pseudo-link", function() { window.open($(this).data("uri")); }); }); </script> <script type='text/javascript'> /* */ var tocplus = { "smooth_scroll": "1", "visibility_show": "\u043f\u043e\u043a\u0430\u0437\u0430\u0442\u044c", "visibility_hide": "\u0441\u043a\u0440\u044b\u0442\u044c", "width": "310px" }; /* */ </script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/plugins/table-of-contents-plus/front.min.js'></script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/plugins/wp-polls/polls-js.js'></script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/plugins/wp-postratings/js/postratings-js.js'></script> <script type='text/javascript'> /* */ var tie = { "mobile_menu_active": "true", "mobile_menu_top": "", "lightbox_all": "", "lightbox_gallery": "", "woocommerce_lightbox": "", "lightbox_skin": "dark", "lightbox_thumb": "vertical", "lightbox_arrows": "true", "sticky_sidebar": "", "is_singular": "1", "SmothScroll": "true", "reading_indicator": "", "lang_no_results": "\u041d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u043d\u0430\u0448\u043b\u043e\u0441\u044c", "lang_results_found": "\u041d\u0430\u0439\u0434\u0435\u043d\u043d\u044b\u0435 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b" }; /* */ </script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/themes/pochk/js/tie-scripts.js'></script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/themes/pochk/js/ilightbox.packed.js'></script> <script type='text/javascript' src='/wp-includes/js/comment-reply.min.js'></script> <script type='text/javascript' src='/wp-includes/js/wp-embed.min.js'></script> <script type='text/javascript'> /* */ var gglcptch_pre = { "messages": { "in_progress": "\u041f\u043e\u0436\u0430\u043b\u0443\u0439\u0441\u0442\u0430, \u0434\u043e\u0436\u0434\u0438\u0442\u0435\u0441\u044c \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u044f \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 Google reCAPTCHA.", "timeout": "\u041d\u0435 \u0443\u0434\u0430\u043b\u043e\u0441\u044c \u0437\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044c Google reCAPTCHA. \u041f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u0412\u0430\u0448\u0435 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 \u043a \u0441\u0435\u0442\u0438 \u0418\u043d\u0442\u0435\u0440\u043d\u0435\u0442 \u0438 \u043f\u0435\u0440\u0435\u0437\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u0435 \u044d\u0442\u0443 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0443." } }; /* */ </script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/plugins/google-captcha/js/pre-api-script.js'></script> <script type='text/javascript' data-cfasync="false" async="async" defer="defer" src='https://www.google.com/recaptcha/api.js?onload=gglcptch_onload_callback&render=explicit'></script> <script type='text/javascript' src='https://qzoreteam.ru/wp-content/plugins/google-captcha/js/script.js'></script> <script type="text/javascript" src="https://qzoreteam.ru/wp-content/themes/pochk/my/my.js"></script> <script type="text/javascript" src="https://qzoreteam.ru/wp-content/themes/pochk/my/etimer.js"></script> </body> </html>