' . $title . '
' // phpcs:ignore WordPress.Security.EscapeOutput -- Reason: escaped properly above.
. Newsletter::newsletter_signup_form() // phpcs:ignore WordPress.Security.EscapeOutput -- Reason: escaped in newsletter.php.
. '
';
echo "";
}
/**
* 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[] = '