FileMaster
Search
Toggle Dark Mode
Home
/
.
/
wp-content
/
plugins
/
sureforms
/
inc
Edit File: duplicate-form.php
<?php /** * Sureforms form duplication. * * @package sureforms. * @since 2.3.0 */ namespace SRFM\Inc; use SRFM\Inc\Traits\Get_Instance; if ( ! defined( 'ABSPATH' ) ) { exit; // Exit if accessed directly. } /** * Duplicate Form Class. * * @since 2.3.0 */ class Duplicate_Form { use Get_Instance; /** * Duplicate a form with all its metadata * * @param int $form_id Form ID to duplicate. * @param string $title_suffix Suffix to append to title. Default ' (Copy)'. * @return array<string, mixed>|\WP_Error Result with new form ID or error. * @since 2.3.0 */ public function duplicate_form( $form_id, $title_suffix = ' (Copy)' ) { // Validate form ID. $form_id = intval( $form_id ); if ( $form_id <= 0 ) { return new \WP_Error( 'invalid_form_id', __( 'Invalid form ID provided.', 'sureforms' ), [ 'status' => 400 ] ); } // Get source form. $source_form = get_post( $form_id ); if ( ! $source_form ) { return new \WP_Error( 'form_not_found', __( 'Source form not found.', 'sureforms' ), [ 'status' => 404 ] ); } // Verify it's a sureforms_form post type. if ( SRFM_FORMS_POST_TYPE !== $source_form->post_type ) { return new \WP_Error( 'invalid_post_type', __( 'The specified post is not a SureForms form.', 'sureforms' ), [ 'status' => 400 ] ); } // Get all post meta. $post_meta = get_post_meta( $form_id ); // Create new form title with suffix. $new_title = $this->generate_unique_title( $source_form->post_title, $title_suffix ); // Prepare new post data. // Note: wp_insert_post() internally calls wp_unslash() which removes backslashes. // This corrupts unicode escapes like \u003c (used for < in JSON block attributes). // We must use wp_slash() to pre-escape the content so wp_unslash() results in correct content. $new_post_args = [ 'post_title' => $new_title, 'post_content' => wp_slash( $source_form->post_content ), 'post_status' => 'draft', // Always create as draft for safety. 'post_type' => SRFM_FORMS_POST_TYPE, 'post_author' => get_current_user_id(), // Use current user as author. ]; // Create the new post. $new_form_id_or_error = wp_insert_post( $new_post_args ); // Check for WP_Error or invalid post ID. if ( ! is_int( $new_form_id_or_error ) || $new_form_id_or_error <= 0 ) { return new \WP_Error( 'duplication_failed', __( 'Failed to create duplicate form.', 'sureforms' ), [ 'status' => 500 ] ); } // At this point, we're certain $new_form_id_or_error is a valid post ID (int). $new_form_id = Helper::get_integer_value( $new_form_id_or_error ); // Update formId in Gutenberg blocks. $updated_content = $this->update_block_form_ids( $source_form->post_content, $form_id, $new_form_id ); // Update the post content with new formId. // Use wp_slash() for the same reason as above - to preserve unicode escapes. wp_update_post( [ 'ID' => $new_form_id, 'post_content' => wp_slash( $updated_content ), ] ); // Get list of unserialized meta keys. $unserialized_metas = $this->get_unserialized_post_metas(); // Copy all post meta. // Ensure $post_meta is an array before iterating. if ( is_array( $post_meta ) ) { foreach ( $post_meta as $meta_key => $meta_values ) { // Ensure meta_key is a string. if ( ! is_string( $meta_key ) ) { continue; } // Skip WordPress internal meta keys. if ( '_edit_lock' === $meta_key || '_edit_last' === $meta_key ) { continue; } // Handle unserialized metas (these are already arrays/objects). if ( in_array( $meta_key, $unserialized_metas, true ) ) { if ( is_array( $meta_values ) && isset( $meta_values[0] ) ) { // Ensure the value is a string before unserializing. $first_value = $meta_values[0]; if ( is_string( $first_value ) ) { $meta_value = maybe_unserialize( $first_value ); add_post_meta( $new_form_id, $meta_key, $meta_value ); } } } else { // Handle serialized metas (get first value). if ( is_array( $meta_values ) && isset( $meta_values[0] ) ) { add_post_meta( $new_form_id, $meta_key, $meta_values[0] ); } } } } // Allow other plugins to hook after duplication. do_action( 'srfm_after_form_duplicated', $new_form_id, $form_id ); // Get edit URL for the new form. $edit_url = admin_url( 'admin.php?page=sureforms_form_editor&post=' . $new_form_id ); // Return success response. return [ 'success' => true, 'original_form_id' => $form_id, 'new_form_id' => $new_form_id, 'new_form_title' => $new_title, 'edit_url' => $edit_url, ]; } /** * Handle duplicate form REST API request * * Infrastructure through the permission_callback. * * @param \WP_REST_Request $request Full details about the request. * @return \WP_REST_Response|\WP_Error Response object on success, or WP_Error object on failure. * @since 2.3.0 */ public function handle_duplicate_form_rest( $request ) { $nonce = Helper::get_string_value( $request->get_header( 'X-WP-Nonce' ) ); if ( ! wp_verify_nonce( sanitize_text_field( $nonce ), 'wp_rest' ) ) { return new \WP_Error( 'invalid_nonce', __( 'Nonce verification failed.', 'sureforms' ), [ 'status' => 403 ] ); } $form_id = absint( $request->get_param( 'form_id' ) ); $title_suffix = sanitize_text_field( $request->get_param( 'title_suffix' ) ); // Duplicate the form. $result = $this->duplicate_form( $form_id, $title_suffix ); if ( is_wp_error( $result ) ) { return $result; } return new \WP_REST_Response( $result, 200 ); } /** * Generate unique title by appending suffix * * If a form with the same title already exists, append a number. * * @param string $base_title Original form title. * @param string $suffix Suffix to append. Default ' (Copy)'. * @return string Unique title. * @since 2.3.0 */ private function generate_unique_title( $base_title, $suffix = ' (Copy)' ) { $new_title = $base_title . $suffix; $counter = 2; // Check if a form with this title exists. while ( $this->title_exists( $new_title ) ) { $new_title = $base_title . $suffix . ' ' . $counter; ++$counter; } return $new_title; } /** * Check if a form title already exists * * @param string $title Title to check. * @return bool True if title exists, false otherwise. * @since 2.3.0 */ private function title_exists( $title ) { global $wpdb; $query = $wpdb->prepare( "SELECT ID FROM {$wpdb->posts} WHERE post_title = %s AND post_type = 'sureforms_form' AND post_status != 'trash' LIMIT 1", $title ); $existing = $wpdb->get_var( $query ); // phpcs:ignore WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching, WordPress.DB.PreparedSQL.NotPrepared return ! empty( $existing ); } /** * Update formId in Gutenberg blocks * * Replaces the old form ID with new form ID in the block markup. * This function performs a direct string replacement on the post content * to update formId references in Gutenberg block attributes. * * @param string $content Post content with blocks. * @param int $old_id Original form ID. * @param int $new_id New form ID. * @return string Updated content with new form ID references. * @since 2.3.0 */ private function update_block_form_ids( $content, $old_id, $new_id ) { // Direct string replacement - no escaping needed for this operation. return str_replace( '"formId":' . intval( $old_id ), '"formId":' . intval( $new_id ), $content ); } /** * Get list of unserialized post meta keys * * These meta keys are already arrays/objects and don't need double unserializing. * * @return array<string> Array of meta keys. * @since 2.3.0 */ private function get_unserialized_post_metas() { $export_instance = Export::get_instance(); return $export_instance->get_unserialized_post_metas(); } }
Save
Back