Дублирование постов — это довольно полезный функционал, особенно когда вы работаете с большим количеством похожих материалов (например товары одной категории в интернет-магазине). Это бывает особенно полезно, если записи имеют большое количество произвольных полей с одинаковыми значениями.
Быстрое дублирование поста позволит вам не делать одну и ту же рутинную работу много раз и определённо сэкономит вагон времени.
Вот пример:
При нажатии на ссылку «Дублировать» пост будет клонирован, но не опубликован (сохранится как черновик). Я покажу вам, как добавить такую же ссылку в админку на вашем сайте.
Итак, полностью готовый код ниже. Если вы не знаете, куда его вставлять, то рекомендую ознакомиться с моим руководством по вставке кода.
/** * @snippet Добавляет ссылку на дублирование поста в админку * @author Миша Рудрастых * @url https://misha.agency/wordpress/duplicate-post-and-pages.html */ // Функция создает дубликат поста в виде черновика и редиректит на его страницу редактирования function true_duplicate_post_as_draft(){ if( empty( $_GET[ 'post' ] ) ) { wp_die( 'Нечего дублировать!' ); } // проверка одноразовых чисел, куда без неё if ( ! isset( $_GET[ 'true_duplicate_nonce' ] ) || ! wp_verify_nonce( $_GET[ 'true_duplicate_nonce' ], basename( __FILE__ ) ) ) { return; } // получаем ID оригинального поста $post_id = absint( $_GET[ 'post' ] ); // затем получили объект поста $post = get_post( $post_id ); /* * если вы не хотите, чтобы текущий автор был автором нового поста * тогда замените следующие две строчки на: $new_post_author = $post->post_author; * при замене этих строк автор будет копироваться из оригинального поста */ $current_user = wp_get_current_user(); $new_post_author = $current_user->ID; /* * если пост существует, создаем его дубликат */ if ( $post ) { // массив данных нового поста $args = array( 'comment_status' => $post->comment_status, 'ping_status' => $post->ping_status, 'post_author' => $new_post_author, 'post_content' => $post->post_content, 'post_excerpt' => $post->post_excerpt, 'post_parent' => $post->post_parent, 'post_name' => $post->post_name, 'post_password' => $post->post_password, 'post_status' => 'draft', // черновик, если хотите сразу публиковать - замените на publish 'post_title' => $post->post_title, 'post_type' => $post->post_type, 'to_ping' => $post->to_ping, 'menu_order' => $post->menu_order ); // создаем пост при помощи функции wp_insert_post() $new_post_id = wp_insert_post( $args ); // присваиваем новому посту все элементы таксономий (рубрики, метки и т.д.) старого $taxonomies = get_object_taxonomies( $post->post_type ); // возвращает массив названий таксономий, используемых для указанного типа поста, например array("category", "post_tag"); foreach ( $taxonomies as $taxonomy ) { $post_terms = wp_get_object_terms( $post_id, $taxonomy, array( 'fields' => 'slugs' ) ); wp_set_object_terms( $new_post_id, $post_terms, $taxonomy, false ); } // дублируем все произвольные поля $post_meta = get_post_meta( $post_id ); if( $post_meta ) { foreach ( $post_meta as $meta_key => $meta_values ) { if( '_wp_old_slug' == $meta_key ) { // это лучше не трогать continue; } foreach ( $meta_values as $meta_value ) { add_post_meta( $new_post_id, $meta_key, $meta_value ); } } } // и наконец, перенаправляем пользователя на страницу редактирования нового поста wp_safe_redirect( add_query_arg( array( 'action' => 'edit', 'post' => $new_post_id ), admin_url( 'post.php' ) ) ); exit; } else { wp_die( 'Ошибка создания поста, не могу найти оригинальный пост с ID=: ' . $post_id ); } } add_action( 'admin_action_true_duplicate_post_as_draft', 'true_duplicate_post_as_draft' ); // Добавляем ссылку дублирования поста для post_row_actions add_filter( 'post_row_actions', 'true_duplicate_post_link', 10, 2 ); function true_duplicate_post_link( $actions, $post ) { if ( current_user_can( 'edit_posts' ) ) { $actions[ 'duplicate' ] = '<a href="' . wp_nonce_url( add_query_arg( array( 'action' => 'true_duplicate_post_as_draft', 'post' => $post->ID ), 'admin.php' ), basename(__FILE__), 'true_duplicate_nonce' ) . '">Дублировать</a>'; } return $actions; }
Но что делать, если этот код работает только для записей и не работает для страниц и других зарегистрированных типов постов?
Это можно легко исправить. Если вам нужно дублировать страницы, то добавляем фильтр:
add_filter( 'page_row_actions', 'true_duplicate_post_link', 10, 2);
Если нужно дублировать какой-нибудь произвольный тип поста, то фильтр будет выглядеть следующим образом:
add_filter( '{название типа поста}_row_actions', 'true_duplicate_post_link', 10, 2);
Конечно, вы можете использовать и несколько фильтров одновременно.
Чтобы оставить комментарий, пожалуйста, зарегистрируйтесь или войдите.
Спасибо! Очень помогли, у меня как раз много похожего контента (товары), так что воспользовалась Вашим кодом - все отлично работает!
Рад, что смог помочь. :)
Не подскажите какое-нить готовое решение, как сделать, чтобы посты в админке отображались от первого к последнему (т.е. по дате добавления - сверху более ранние, снизу более поздние), а не как сейчас (вверху самые поздние).
Легко — просто нажмите на заголовок колонки «Дата» в таблице с постами и они отсортируются.
Ах, да,точно! Че-то сразу не сообразила, спасибо)
Пожалуйста)
Дублирование постов - это неплохо, но все же даже так тяжеловато вводить большой ассортимент товаров (а у меня их около 500). Установила плагин Wp All import, научилась импортировать через файл xml, получается быстрее, но все же хотелось бы работать с exell. таблицы exell структурно удобней файлов xml. Может, вы знакомы с каким-нибудь таким вариантом - чтобы создавать посты в exell и импортировать их в вордпресс?
К сожалению не знаком.
Спасибо за материалы, отличный сайт!
Странно, что наткнулся на вас только сегодня, наверное вы больше ориентированы на англоязычный сегмент, раньше инфу, например по функциям брал преимущественно у камы, теперь и у вас.
Пожалуйста!
Просто я в SEO не силён :)
Очень удобно!! Спасибо!
Код работает только для стандартных страниц и записей, если создать кастомный тип записи, то для него, к сожалению, кнопка "дублировать" не появляется :( Может что-то не правильно сделал.
Кастомный тип запсии:
А я вот кстати не знаю, почему. Потому что действительно многие спрашивают про это, но я сколько раз ни тестировал – всегда всё было ок!
Могу предположить только – если код дублирования вставить после вставки регистрации типа и поставить приоритет 9999 🤷♂️
Добрый день,
у меня в начале тоже такая проблема была.
Помогло банальное добавление фильтров для записей и страниц,
а после для кастомного типа записи вставил.
Тут есть уязвимость
Используй
esc_sql
для ключей и значенийПруф -
Add 2', '1' UNION ALL SELECT 1,2,(SELECT CASE WHEN (SELECT 1)='1' THEN SLEEP(50) else SLEEP(0) END);
попробуй добавить, скрипт зависнет на 50секИсправь, а то люди тянут себе этот код не зная об уязвимости.
Спасибо большое, что обратил моё внимание на этот супер-давний пост!
Там не только это было к сожалению, но сейчас всё исправил.