<?php
/**
* Backend functions.
*
* @package Yoast\WP\Duplicate_Post
* @since 2.0
*/
if ( ! is_admin() ) {
return;
}
use Yoast\WP\Duplicate_Post\UI\Newsletter;
require_once DUPLICATE_POST_PATH . 'options.php';
require_once DUPLICATE_POST_PATH . 'compat/wpml-functions.php';
require_once DUPLICATE_POST_PATH . 'compat/jetpack-functions.php';
/**
* Wrapper for the option 'duplicate_post_version'.
*/
function duplicate_post_get_installed_version() {
return get_option( 'duplicate_post_version' );
}
/**
* Wrapper for the defined constant DUPLICATE_POST_CURRENT_VERSION.
*/
function duplicate_post_get_current_version() {
return DUPLICATE_POST_CURRENT_VERSION;
}
add_action( 'admin_init', 'duplicate_post_admin_init' );
/**
* Adds handlers depending on the options.
*/
function duplicate_post_admin_init() {
duplicate_post_plugin_upgrade();
if ( intval( get_site_option( 'duplicate_post_show_notice' ) ) === 1 ) {
if ( is_multisite() ) {
add_action( 'network_admin_notices', 'duplicate_post_show_update_notice' );
}
else {
add_action( 'admin_notices', 'duplicate_post_show_update_notice' );
}
add_action( 'wp_ajax_duplicate_post_dismiss_notice', 'duplicate_post_dismiss_notice' );
}
add_action( 'dp_duplicate_post', 'duplicate_post_copy_post_meta_info', 10, 2 );
add_action( 'dp_duplicate_page', 'duplicate_post_copy_post_meta_info', 10, 2 );
if ( intval( get_option( 'duplicate_post_copychildren' ) ) === 1 ) {
add_action( 'dp_duplicate_post', 'duplicate_post_copy_children', 20, 3 );
add_action( 'dp_duplicate_page', 'duplicate_post_copy_children', 20, 3 );
}
if ( intval( get_option( 'duplicate_post_copyattachments' ) ) === 1 ) {
add_action( 'dp_duplicate_post', 'duplicate_post_copy_attachments', 30, 2 );
add_action( 'dp_duplicate_page', 'duplicate_post_copy_attachments', 30, 2 );
}
if ( intval( get_option( 'duplicate_post_copycomments' ) ) === 1 ) {
add_action( 'dp_duplicate_post', 'duplicate_post_copy_comments', 40, 2 );
add_action( 'dp_duplicate_page', 'duplicate_post_copy_comments', 40, 2 );
}
add_action( 'dp_duplicate_post', 'duplicate_post_copy_post_taxonomies', 50, 2 );
add_action( 'dp_duplicate_page', 'duplicate_post_copy_post_taxonomies', 50, 2 );
add_filter( 'plugin_row_meta', 'duplicate_post_add_plugin_links', 10, 2 );
}
/**
* Plugin upgrade.
*/
function duplicate_post_plugin_upgrade() {
$installed_version = duplicate_post_get_installed_version();
if ( duplicate_post_get_current_version() === $installed_version ) {
return;
}
if ( empty( $installed_version ) ) {
// Get default roles.
$default_roles = [
'editor',
'administrator',
'wpseo_manager',
'wpseo_editor',
];
foreach ( $default_roles as $name ) {
$role = get_role( $name );
if ( ! empty( $role ) ) {
$role->add_cap( 'copy_posts' );
}
}
add_option( 'duplicate_post_show_notice', 1 );
}
else {
update_option( 'duplicate_post_show_notice', 0 );
}
$show_links_in_defaults = [
'row' => '1',
'adminbar' => '1',
'submitbox' => '1',
'bulkactions' => '1',
];
add_option( 'duplicate_post_copytitle', '1' );
add_option( 'duplicate_post_copydate', '0' );
add_option( 'duplicate_post_copystatus', '0' );
add_option( 'duplicate_post_copyslug', '0' );
add_option( 'duplicate_post_copyexcerpt', '1' );
add_option( 'duplicate_post_copycontent', '1' );
add_option( 'duplicate_post_copythumbnail', '1' );
add_option( 'duplicate_post_copytemplate', '1' );
add_option( 'duplicate_post_copyformat', '1' );
add_option( 'duplicate_post_copyauthor', '0' );
add_option( 'duplicate_post_copypassword', '0' );
add_option( 'duplicate_post_copyattachments', '0' );
add_option( 'duplicate_post_copychildren', '0' );
add_option( 'duplicate_post_copycomments', '0' );
add_option( 'duplicate_post_copymenuorder', '1' );
add_option( 'duplicate_post_taxonomies_blacklist', [] );
add_option( 'duplicate_post_blacklist', '' );
add_option( 'duplicate_post_types_enabled', [ 'post', 'page' ] );
add_option( 'duplicate_post_show_original_column', '0' );
add_option( 'duplicate_post_show_original_in_post_states', '0' );
add_option( 'duplicate_post_show_original_meta_box', '0' );
add_option(
'duplicate_post_show_link',
[
'new_draft' => '1',
'clone' => '1',
'rewrite_republish' => '1',
]
);
add_option( 'duplicate_post_show_link_in', $show_links_in_defaults );
$taxonomies_blacklist = get_option( 'duplicate_post_taxonomies_blacklist' );
if ( $taxonomies_blacklist === '' ) {
$taxonomies_blacklist = [];
}
if ( in_array( 'post_format', $taxonomies_blacklist, true ) ) {
update_option( 'duplicate_post_copyformat', 0 );
$taxonomies_blacklist = array_diff( $taxonomies_blacklist, [ 'post_format' ] );
update_option( 'duplicate_post_taxonomies_blacklist', $taxonomies_blacklist );
}
$meta_blacklist = explode( ',', get_option( 'duplicate_post_blacklist' ) );
if ( $meta_blacklist === '' ) {
$meta_blacklist = [];
}
$meta_blacklist = array_map( 'trim', $meta_blacklist );
if ( in_array( '_wp_page_template', $meta_blacklist, true ) ) {
update_option( 'duplicate_post_copytemplate', 0 );
$meta_blacklist = array_diff( $meta_blacklist, [ '_wp_page_template' ] );
}
if ( in_array( '_thumbnail_id', $meta_blacklist, true ) ) {
update_option( 'duplicate_post_copythumbnail', 0 );
$meta_blacklist = array_diff( $meta_blacklist, [ '_thumbnail_id' ] );
}
update_option( 'duplicate_post_blacklist', implode( ',', $meta_blacklist ) );
if ( version_compare( $installed_version, '4.0.0' ) < 0 ) {
// Migrate the 'Show links in' options to the new array-based structure.
duplicate_post_migrate_show_links_in_options( $show_links_in_defaults );
}
delete_site_option( 'duplicate_post_version' );
update_option( 'duplicate_post_version', duplicate_post_get_current_version() );
}
/**
* Runs the upgrade routine for version 4.0 to update the options in the database.
*
* @param array $defaults The default options to fall back on.
*
* @return void
*/
function duplicate_post_migrate_show_links_in_options( $defaults ) {
$options_to_migrate = [
'duplicate_post_show_row' => 'row',
'duplicate_post_show_adminbar' => 'adminbar',
'duplicate_post_show_submitbox' => 'submitbox',
'duplicate_post_show_bulkactions' => 'bulkactions',
];
$new_options = [];
foreach ( $options_to_migrate as $old => $new ) {
$new_options[ $new ] = get_option( $old, $defaults[ $new ] );
delete_option( $old );
}
update_option( 'duplicate_post_show_link_in', $new_options );
}
/**
* Shows the welcome notice.
*
* @global string $wp_version The WordPress version string.
*/
function duplicate_post_show_update_notice() {
if ( ! current_user_can( 'manage_options' ) ) {
return;
}
$current_screen = get_current_screen();
if ( empty( $current_screen )
|| empty( $current_screen->base )
|| ( $current_screen->base !== 'dashboard' && $current_screen->base !== 'plugins' )
) {
return;
}
$title = sprintf(
/* translators: %s: Yoast Duplicate Post. */
esc_html__( 'You\'ve successfully installed %s!', 'duplicate-post' ),
'Yoast Duplicate Post'
);
$img_path = plugins_url( '/duplicate_post_yoast_icon-125x125.png', __FILE__ );
echo '<div id="duplicate-post-notice" class="notice is-dismissible" style="display: flex; align-items: flex-start;">
<img src="' . esc_url( $img_path ) . '" alt="" style="margin: 1em 1em 1em 0; width: 130px; align-self: center;"/>
<div stle="margin: 0.5em">
<h1 style="font-size: 14px; color: #a4286a; font-weight: 600; margin-top: 8px;">' . $title . '</h1>' // phpcs:ignore WordPress.Security.EscapeOutput -- Reason: escaped properly above.
. Newsletter::newsletter_signup_form() // phpcs:ignore WordPress.Security.EscapeOutput -- Reason: escaped in newsletter.php.
. '</div>
</div>';
echo "<script>
function duplicate_post_dismiss_notice(){
var data = {
'action': 'duplicate_post_dismiss_notice',
};
jQuery.post(ajaxurl, data, function(response) {
jQuery('#duplicate-post-notice').hide();
});
}
jQuery(document).ready(function(){
jQuery('body').on('click', '.notice-dismiss', function(){
duplicate_post_dismiss_notice();
});
});
</script>";
}
/**
* Dismisses the notice.
*
* @return bool
*/
function duplicate_post_dismiss_notice() {
return update_site_option( 'duplicate_post_show_notice', 0 );
}
/**
* Copies the taxonomies of a post to another post.
*
* @global wpdb $wpdb WordPress database abstraction object.
*
* @param int $new_id New post ID.
* @param WP_Post $post The original post object.
*/
function duplicate_post_copy_post_taxonomies( $new_id, $post ) {
global $wpdb;
if ( isset( $wpdb->terms ) ) {
// Clear default category (added by wp_insert_post).
wp_set_object_terms( $new_id, null, 'category' );
$post_taxonomies = get_object_taxonomies( $post->post_type );
// Several plugins just add support to post-formats but don't register post_format taxonomy.
if ( post_type_supports( $post->post_type, 'post-formats' ) && ! in_array( 'post_format', $post_taxonomies, true ) ) {
$post_taxonomies[] = 'post_format';
}
$taxonomies_blacklist = get_option( 'duplicate_post_taxonomies_blacklist' );
if ( $taxonomies_blacklist === '' ) {
$taxonomies_blacklist = [];
}
if ( intval( get_option( 'duplicate_post_copyformat' ) ) === 0 ) {
$taxonomies_blacklist[] = 'post_format';
}
/**
* Filters the taxonomy excludelist when copying a post.
*
* @param array $taxonomies_blacklist The taxonomy excludelist from the options.
*
* @return array
*/
$taxonomies_blacklist = apply_filters( 'duplicate_post_taxonomies_excludelist_filter', $taxonomies_blacklist );
$taxonomies = array_diff( $post_taxonomies, $taxonomies_blacklist );
foreach ( $taxonomies as $taxonomy ) {
$post_terms = wp_get_object_terms( $post->ID, $taxonomy, [ 'orderby' => 'term_order' ] );
$terms = [];
$num_terms = count( $post_terms );
for ( $i = 0; $i < $num_terms; $i++ ) {
$terms[] = $post_terms[ $i ]->slug;
}
wp_set_object_terms( $new_id, $terms, $taxonomy );
}
}
}
/**
* Copies the meta information of a post to another post
*
* @param int $new_id The new post ID.
* @param WP_Post $post The original post object.
*/
function duplicate_post_copy_post_meta_info( $new_id, $post ) {
$post_meta_keys = get_post_custom_keys( $post->ID );
if ( empty( $post_meta_keys ) ) {
return;
}
$meta_blacklist = get_option( 'duplicate_post_blacklist' );
if ( $meta_blacklist === '' ) {
$meta_blacklist = [];
}
else {
$meta_blacklist = explode( ',', $meta_blacklist );
$meta_blacklist = array_filter( $meta_blacklist );
$meta_blacklist = array_map( 'trim', $meta_blacklist );
}
$meta_blacklist[] = '_edit_lock'; // Edit lock.
$meta_blacklist[] = '_edit_last'; // Edit lock.
$meta_blacklist[] = '_dp_is_rewrite_republish_copy';
$meta_blacklist[] = '_dp_has_rewrite_republish_copy';
if ( intval( get_option( 'duplicate_post_copytemplate' ) ) === 0 ) {
$meta_blacklist[] = '_wp_page_template';
}
if ( intval( get_option( 'duplicate_post_copythumbnail' ) ) === 0 ) {
$meta_blacklist[] = '_thumbnail_id';
}
$meta_blacklist = apply_filters_deprecated( 'duplicate_post_blacklist_filter', [ $meta_blacklist ], '3.2.5', 'duplicate_post_excludelist_filter' );
/**
* Filters the meta fields excludelist when copying a post.
*
* @param array $meta_blacklist The meta fields excludelist from the options.
*
* @return array
*/
$meta_blacklist = apply_filters( 'duplicate_post_excludelist_filter', $meta_blacklist );
$meta_blacklist_string = '(' . implode( ')|(', $meta_blacklist ) . ')';
if ( strpos( $meta_blacklist_string, '*' ) !== false ) {
$meta_blacklist_string = str_replace( [ '*' ], [ '[a-zA-Z0-9_]*' ], $meta_blacklist_string );
$meta_keys = [];
foreach ( $post_meta_keys as $meta_key ) {
if ( ! preg_match( '#^' . $meta_blacklist_string . '$#', $meta_key ) ) {
$meta_keys[] = $meta_key;
}
}
}
else {
$meta_keys = array_diff( $post_meta_keys, $meta_blacklist );
}
/**
* Filters the list of meta fields names when copying a post.
*
* @param array $meta_keys The list of meta fields name, with the ones in the excludelist already removed.
*
* @return array
*/
$meta_keys = apply_filters( 'duplicate_post_meta_keys_filter', $meta_keys );
foreach ( $meta_keys as $meta_key ) {
$meta_values = get_post_custom_values( $meta_key, $post->ID );
foreach ( $meta_values as $meta_value ) {
$meta_value = maybe_unserialize( $meta_value );
add_post_meta( $new_id, $meta_key, duplicate_post_wp_slash( $meta_value ) );
}
}
}
/**
* Workaround for inconsistent wp_slash.
* Works only with WP 4.4+ (map_deep)
*
* @param mixed $value Array or object to be recursively slashed.
* @return string|mixed
*/
function duplicate_post_addslashes_deep( $value ) {
if ( function_exists( 'map_deep' ) ) {
return map_deep( $value, 'duplicate_post_addslashes_to_strings_only' );
}
else {
return wp_slash( $value );
}
}
/**
* Adds slashes only to strings.
*
* @param mixed $value Value to slash only if string.
* @return string|mixed
*/
function duplicate_post_addslashes_to_strings_only( $value ) {
return Yoast\WP\Duplicate_Post\Utils::addslashes_to_strings_only( $value );
}
/**
* Replacement function for faulty core wp_slash().
*
* @param mixed $value What to add slash to.
* @return mixed
*/
function duplicate_post_wp_slash( $value ) {
return duplicate_post_addslashes_deep( $value );
}
/**
* Copies attachments, including physical files.
*
* @param int $new_id The new post ID.
* @param WP_Post $post The original post object.
*/
function duplicate_post_copy_attachments( $new_id, $post ) {
// Get thumbnail ID.
$old_thumbnail_id = get_post_thumbnail_id( $post->ID );
// Get children.
$children = get_posts(
[
'post_type' => 'any',
'numberposts' => -1,
'post_status' => 'any',
'post_parent' => $post->ID,
]
);
// Clone old attachments.
foreach ( $children as $child ) {
if ( $child->post_type !== 'attachment' ) {
continue;
}
$url = wp_get_attachment_url( $child->ID );
// Let's copy the actual file.
$tmp = download_url( $url );
if ( is_wp_error( $tmp ) ) {
continue;
}
$desc = wp_slash( $child->post_content );
$file_array = [];
$file_array['name'] = basename( $url );
$file_array['tmp_name'] = $tmp;
// "Upload" to the media collection
$new_attachment_id = media_handle_sideload( $file_array, $new_id, $desc );
if ( is_wp_error( $new_attachment_id ) ) {
unlink( $file_array['tmp_name'] );
continue;
}
$new_post_author = wp_get_current_user();
$cloned_child = [
'ID' => $new_attachment_id,
'post_title' => $child->post_title,
'post_exceprt' => $child->post_title,
'post_author' => $new_post_author->ID,
];
wp_update_post( wp_slash( $cloned_child ) );
$alt_title = get_post_meta( $child->ID, '_wp_attachment_image_alt', true );
if ( $alt_title ) {
update_post_meta( $new_attachment_id, '_wp_attachment_image_alt', wp_slash( $alt_title ) );
}
// If we have cloned the post thumbnail, set the copy as the thumbnail for the new post.
if ( intval( get_option( 'duplicate_post_copythumbnail' ) ) === 1 && $old_thumbnail_id === $child->ID ) {
set_post_thumbnail( $new_id, $new_attachment_id );
}
}
}
/**
* Copies child posts.
*
* @param int $new_id The new post ID.
* @param WP_Post $post The original post object.
* @param string $status Optional. The destination status.
*/
function duplicate_post_copy_children( $new_id, $post, $status = '' ) {
// Get children.
$children = get_posts(
[
'post_type' => 'any',
'numberposts' => -1,
'post_status' => 'any',
'post_parent' => $post->ID,
]
);
foreach ( $children as $child ) {
if ( $child->post_type === 'attachment' ) {
continue;
}
duplicate_post_create_duplicate( $child, $status, $new_id );
}
}
/**
* Copies comments.
*
* @param int $new_id The new post ID.
* @param WP_Post $post The original post object.
*/
function duplicate_post_copy_comments( $new_id, $post ) {
$comments = get_comments(
[
'post_id' => $post->ID,
'order' => 'ASC',
'orderby' => 'comment_date_gmt',
]
);
$old_id_to_new = [];
foreach ( $comments as $comment ) {
// Do not copy pingbacks or trackbacks.
if ( $comment->comment_type === 'pingback' || $comment->comment_type === 'trackback' ) {
continue;
}
$parent = ( $comment->comment_parent && $old_id_to_new[ $comment->comment_parent ] ) ? $old_id_to_new[ $comment->comment_parent ] : 0;
$commentdata = [
'comment_post_ID' => $new_id,
'comment_author' => $comment->comment_author,
'comment_author_email' => $comment->comment_author_email,
'comment_author_url' => $comment->comment_author_url,
'comment_content' => $comment->comment_content,
'comment_type' => $comment->comment_type,
'comment_parent' => $parent,
'user_id' => $comment->user_id,
'comment_author_IP' => $comment->comment_author_IP,
'comment_agent' => $comment->comment_agent,
'comment_karma' => $comment->comment_karma,
'comment_approved' => $comment->comment_approved,
];
if ( intval( get_option( 'duplicate_post_copydate' ) ) === 1 ) {
$commentdata['comment_date'] = $comment->comment_date;
$commentdata['comment_date_gmt'] = get_gmt_from_date( $comment->comment_date );
}
$new_comment_id = wp_insert_comment( $commentdata );
$commentmeta = get_comment_meta( $new_comment_id );
foreach ( $commentmeta as $meta_key => $meta_value ) {
add_comment_meta( $new_comment_id, $meta_key, duplicate_post_wp_slash( $meta_value ) );
}
$old_id_to_new[ $comment->comment_ID ] = $new_comment_id;
}
}
/**
* Creates a duplicate from a post.
*
* This is the main functions that does the cloning.
*
* @param WP_Post $post The original post object.
* @param string $status Optional. The intended destination status.
* @param string $parent_id Optional. The parent post ID if we are calling this recursively.
* @return int|WP_Error
*/
function duplicate_post_create_duplicate( $post, $status = '', $parent_id = '' ) {
/**
* Fires before duplicating a post.
*
* @param WP_Post $post The original post object.
* @param bool $status The intended destination status.
* @param int $parent_id The parent post ID if we are calling this recursively.
*/
do_action( 'duplicate_post_pre_copy', $post, $status, $parent_id );
/**
* Filter allowing to copy post.
*
* @param bool $can_duplicate Default to `true`.
* @param WP_Post $post The original post object.
* @param bool $status The intended destination status.
* @param int $parent_id The parent post ID if we are calling this recursively.
*
* @return bool
*/
$can_duplicate = apply_filters( 'duplicate_post_allow', true, $post, $status, $parent_id );
if ( ! $can_duplicate ) {
wp_die( esc_html( __( 'You aren\'t allowed to duplicate this post', 'duplicate-post' ) ) );
}
if ( ! duplicate_post_is_post_type_enabled( $post->post_type ) && $post->post_type !== 'attachment' ) {
wp_die(
esc_html(
__( 'Copy features for this post type are not enabled in options page', 'duplicate-post' ) . ': '
. $post->post_type
)
);
}
$new_post_status = ( empty( $status ) ) ? $post->post_status : $status;
$title = ' ';
if ( $post->post_type !== 'attachment' ) {
$prefix = sanitize_text_field( get_option( 'duplicate_post_title_prefix' ) );
$suffix = sanitize_text_field( get_option( 'duplicate_post_title_suffix' ) );
if ( intval( get_option( 'duplicate_post_copytitle' ) ) === 1 ) {
$title = $post->post_title;
if ( ! empty( $prefix ) ) {
$prefix .= ' ';
}
if ( ! empty( $suffix ) ) {
$suffix = ' ' . $suffix;
}
}
else {
$title = ' ';
}
$title = trim( $prefix . $title . $suffix );
/*
* Not sure we should force a title. Instead, we should respect what WP does.
* if ( '' === $title ) {
* // empty title.
* $title = __( 'Untitled', 'default' );
* }
*/
if ( intval( get_option( 'duplicate_post_copystatus' ) ) === 0 ) {
$new_post_status = 'draft';
}
elseif ( $new_post_status === 'publish' || $new_post_status === 'future' ) {
// Check if the user has the right capability.
if ( is_post_type_hierarchical( $post->post_type ) ) {
if ( ! current_user_can( 'publish_pages' ) ) {
$new_post_status = 'pending';
}
}
elseif ( ! current_user_can( 'publish_posts' ) ) {
$new_post_status = 'pending';
}
}
}
$new_post_author = wp_get_current_user();
$new_post_author_id = $new_post_author->ID;
if ( intval( get_option( 'duplicate_post_copyauthor' ) ) === 1 ) {
// Check if the user has the right capability.
if ( is_post_type_hierarchical( $post->post_type ) ) {
if ( current_user_can( 'edit_others_pages' ) ) {
$new_post_author_id = $post->post_author;
}
}
elseif ( current_user_can( 'edit_others_posts' ) ) {
$new_post_author_id = $post->post_author;
}
}
$menu_order = ( intval( get_option( 'duplicate_post_copymenuorder' ) ) === 1 ) ? $post->menu_order : 0;
$increase_menu_order_by = get_option( 'duplicate_post_increase_menu_order_by' );
if ( ! empty( $increase_menu_order_by ) && is_numeric( $increase_menu_order_by ) ) {
$menu_order += intval( $increase_menu_order_by );
}
$post_name = $post->post_name;
if ( intval( get_option( 'duplicate_post_copyslug' ) ) !== 1 ) {
$post_name = '';
}
$new_post_parent = empty( $parent_id ) ? $post->post_parent : $parent_id;
$new_post = [
'menu_order' => $menu_order,
'comment_status' => $post->comment_status,
'ping_status' => $post->ping_status,
'post_author' => $new_post_author_id,
'post_content' => ( intval( get_option( 'duplicate_post_copycontent' ) ) === 1 ) ? $post->post_content : '',
'post_content_filtered' => ( intval( get_option( 'duplicate_post_copycontent' ) ) === 1 ) ? $post->post_content_filtered : '',
'post_excerpt' => ( intval( get_option( 'duplicate_post_copyexcerpt' ) ) === 1 ) ? $post->post_excerpt : '',
'post_mime_type' => $post->post_mime_type,
'post_parent' => $new_post_parent,
'post_password' => ( intval( get_option( 'duplicate_post_copypassword' ) ) === 1 ) ? $post->post_password : '',
'post_status' => $new_post_status,
'post_title' => $title,
'post_type' => $post->post_type,
'post_name' => $post_name,
];
if ( intval( get_option( 'duplicate_post_copydate' ) ) === 1 ) {
$new_post_date = $post->post_date;
$new_post['post_date'] = $new_post_date;
$new_post['post_date_gmt'] = get_gmt_from_date( $new_post_date );
}
/**
* Filter new post values.
*
* @param array $new_post New post values.
* @param WP_Post $post Original post object.
*
* @return array
*/
$new_post = apply_filters( 'duplicate_post_new_post', $new_post, $post );
$new_post_id = wp_insert_post( wp_slash( $new_post ), true );
// If you have written a plugin which uses non-WP database tables to save
// information about a post you can hook this action to dupe that data.
if ( $new_post_id !== 0 && ! is_wp_error( $new_post_id ) ) {
if ( $post->post_type === 'page' || is_post_type_hierarchical( $post->post_type ) ) {
do_action( 'dp_duplicate_page', $new_post_id, $post, $status );
}
else {
do_action( 'dp_duplicate_post', $new_post_id, $post, $status );
}
delete_post_meta( $new_post_id, '_dp_original' );
add_post_meta( $new_post_id, '_dp_original', $post->ID );
}
/**
* Fires after duplicating a post.
*
* @param int|WP_Error $new_post_id The new post id or WP_Error object on error.
* @param WP_Post $post The original post object.
* @param bool $status The intended destination status.
* @param int $parent_id The parent post ID if we are calling this recursively.
*/
do_action( 'duplicate_post_post_copy', $new_post_id, $post, $status, $parent_id );
return $new_post_id;
}
/**
* Adds some links on the plugin page.
*
* @param array $links The links array.
* @param string $file The file name.
* @return array
*/
function duplicate_post_add_plugin_links( $links, $file ) {
if ( plugin_basename( __DIR__ . '/duplicate-post.php' ) === $file ) {
$links[] = '<a href="https://yoast.com/wordpress/plugins/duplicate-post">' . esc_html__( 'Documentation', 'duplicate-post' ) . '</a>';
}
return $links;
}