Дублирование постов и страниц в WordPress без использования плагинов

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

Быстрое дублирование поста позволит вам не одну и ту же рутинную работу по многу раз и определённо сэкономит вам время.

Вот пример:

Ссылка дублирования поста

При нажатии на ссылку «Дублировать» пост будет клонирован, но не опубликован (сохранится как черновик). Я покажу вам, как добавить такую же ссылку в админку на вашем сайте.

Итак, всё, что вам требуется сделать — это вставить следующий код в ваш functions.php (можно и в другой файл, если вы знаете, что делаете).

/*
 * Функция создает дубликат поста в виде черновика и редиректит на его страницу редактирования
 */
function true_duplicate_post_as_draft(){
	global $wpdb;
	if (! ( isset( $_GET['post']) || isset( $_POST['post'])  || ( isset($_REQUEST['action']) && 'true_duplicate_post_as_draft' == $_REQUEST['action'] ) ) ) {
		wp_die('Нечего дублировать!');
	}
 
	/*
	 * получаем ID оригинального поста
	 */
	$post_id = (isset($_GET['post']) ? $_GET['post'] : $_POST['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 (isset( $post ) && $post != null) {
 
		/*
		 * массив данных нового поста
		 */
		$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_name'      => $post->post_name,
			'post_parent'    => $post->post_parent,
			'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_infos = $wpdb->get_results("SELECT meta_key, meta_value FROM $wpdb->postmeta WHERE post_id=$post_id");
		if (count($post_meta_infos)!=0) {
			$sql_query = "INSERT INTO $wpdb->postmeta (post_id, meta_key, meta_value) ";
			foreach ($post_meta_infos as $meta_info) {
				$meta_key = $meta_info->meta_key;
				$meta_value = addslashes($meta_info->meta_value);
				$sql_query_sel[]= "SELECT $new_post_id, '$meta_key', '$meta_value'";
			}
			$sql_query.= implode(" UNION ALL ", $sql_query_sel);
			$wpdb->query($sql_query);
		}
 
 
		/*
		 * и наконец, перенаправляем пользователя на страницу редактирования нового поста
		 */
		wp_redirect( admin_url( 'post.php?action=edit&post=' . $new_post_id ) );
		exit;
	} else {
		wp_die('Ошибка создания поста, не могу найти оригинальный пост с ID=: ' . $post_id);
	}
}
add_action( 'admin_action_true_duplicate_post_as_draft', 'true_duplicate_post_as_draft' );
 
/*
 * Добавляем ссылку дублирования поста для post_row_actions
 */
function true_duplicate_post_link( $actions, $post ) {
	if (current_user_can('edit_posts')) {
		$actions['duplicate'] = '<a href="admin.php?action=true_duplicate_post_as_draft&amp;post=' . $post->ID . '" title="Дублировать этот пост" rel="permalink">Дублировать</a>';
	}
	return $actions;
}
 
add_filter( 'post_row_actions', 'true_duplicate_post_link', 10, 2 );

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

Это можно легко исправить. Если вам нужно дублировать страницы, то добавляем фильтр:

add_filter( 'page_row_actions', 'true_duplicate_post_link', 10, 2);

Если нужно дублировать какой-нибудь произвольный тип поста, то фильтр будет выглядеть следующим образом:

add_filter( '{название типа поста}_row_actions', 'true_duplicate_post_link', 10, 2);

Конечно, вы можете использовать и несколько фильтров одновременно.

Источник: http://rudrastyh.com/wordpress/duplicate-post.html (мой англоязычный блог)

Подпишитесь, чтобы раз в неделю получать свежие статьи с блога по email.

Смотрите также

Комментарии 11

  • Ольга18 ноября 2014 в 19:11 #

    Спасибо! Очень помогли, у меня как раз много похожего контента (товары), так что воспользовалась Вашим кодом - все отлично работает!

    • Миша18 ноября 2014 в 21:11 #

      Рад, что смог помочь. :)

      • Ольга24 ноября 2014 в 22:11 #

        Не подскажите какое-нить готовое решение, как сделать, чтобы посты в админке отображались от первого к последнему (т.е. по дате добавления - сверху более ранние, снизу более поздние), а не как сейчас (вверху самые поздние).

        • Миша25 ноября 2014 в 07:11 #

          Легко — просто нажмите на заголовок колонки «Дата» в таблице с постами и они отсортируются.

          • Ольга25 ноября 2014 в 07:11 #

            Ах, да,точно! Че-то сразу не сообразила, спасибо)

            • Миша25 ноября 2014 в 10:11 #

              Пожалуйста)

              • Ольга26 ноября 2014 в 08:11 #

                Дублирование постов - это неплохо, но все же даже так тяжеловато вводить большой ассортимент товаров (а у меня их около 500). Установила плагин Wp All import, научилась импортировать через файл xml, получается быстрее, но все же хотелось бы работать с exell. таблицы exell структурно удобней файлов xml. Может, вы знакомы с каким-нибудь таким вариантом - чтобы создавать посты в exell и импортировать их в вордпресс?

  • Роман llgr19 марта 2015 в 12:03 #

    Спасибо за материалы, отличный сайт!
    Странно, что наткнулся на вас только сегодня, наверное вы больше ориентированы на англоязычный сегмент, раньше инфу, например по функциям брал преимущественно у камы, теперь и у вас.

  • Наталья21 октября 2015 в 21:10 #

    Очень удобно!! Спасибо!

Оставить комментарий / вопрос

phpjsHTMLCSSSQLПросто код
  Для того, чтобы оставить комментарий, пожалуйста, зарегистрируйтесь или авторизуйтесь на сайте.
Получайте новости блога по email или следите за мной в социальных сетях.
  • Миша: Да, точно)

  • Владимир: и на третьей строке не хватает скобки ) закрывающей if :)

  • Миша: Добрый вечер! Рекомендую получить ID пользователя через функцию get_current_user_id().

  • Дмитрий: Миша, подскажите пожалуйста, я использую такой редирект, но он не срабатывает: add_action( 'template_redirect', funct...

  • Миша: Ну как не определена - это аргумент функции. Или я просто вас не до конца понимаю.