You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1504 lines
50 KiB
PHTML
1504 lines
50 KiB
PHTML
7 months ago
|
<?php
|
||
|
/**
|
||
|
* Create a new layout.
|
||
|
*
|
||
|
* @since 4.0
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
function et_theme_builder_api_create_layout() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_create_layout', 'nonce' );
|
||
|
|
||
|
$layout_type = isset( $_POST['layout_type'] ) ? sanitize_text_field( $_POST['layout_type'] ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Missing -- No need to use nonce.
|
||
|
$post_type = et_theme_builder_get_valid_layout_post_type( $layout_type );
|
||
|
|
||
|
if ( '' === $post_type ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => 'Invalid layout type: ' . $layout_type,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$post_id = et_theme_builder_insert_layout(
|
||
|
array(
|
||
|
'post_type' => $post_type,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
if ( is_wp_error( $post_id ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => 'Failed to create layout.',
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'id' => $post_id,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_create_layout', 'et_theme_builder_api_create_layout' );
|
||
|
|
||
|
/**
|
||
|
* Duplicate a layout.
|
||
|
*
|
||
|
* @since 4.0
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
function et_theme_builder_api_duplicate_layout() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_duplicate_layout', 'nonce' );
|
||
|
|
||
|
$layout_type = isset( $_POST['layout_type'] ) ? sanitize_text_field( $_POST['layout_type'] ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Missing -- No need to use nonce.
|
||
|
$post_type = et_theme_builder_get_valid_layout_post_type( $layout_type );
|
||
|
$layout_id = isset( $_POST['layout_id'] ) ? (int) $_POST['layout_id'] : 0; // phpcs:ignore WordPress.Security.NonceVerification.Missing -- No need to use nonce.
|
||
|
$layout = get_post( $layout_id );
|
||
|
|
||
|
if ( ! $layout ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => 'Failed to duplicate layout.',
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$post_id = et_theme_builder_insert_layout(
|
||
|
array(
|
||
|
'post_type' => '' !== $post_type ? $post_type : $layout->post_type,
|
||
|
'post_status' => $layout->post_status,
|
||
|
'post_title' => $layout->post_title,
|
||
|
'post_content' => $layout->post_content,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
if ( is_wp_error( $post_id ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => 'Failed to duplicate layout.',
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$meta = et_core_get_post_builder_meta( $layout_id );
|
||
|
|
||
|
foreach ( $meta as $entry ) {
|
||
|
update_post_meta( $post_id, $entry['key'], $entry['value'] );
|
||
|
}
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'id' => $post_id,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_duplicate_layout', 'et_theme_builder_api_duplicate_layout' );
|
||
|
|
||
|
/**
|
||
|
* Get layout url.
|
||
|
*
|
||
|
* @since 4.0
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
function et_theme_builder_api_get_layout_url() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_get_layout_url', 'nonce' );
|
||
|
|
||
|
$layout_id = isset( $_POST['layout_id'] ) ? (int) $_POST['layout_id'] : 0; // phpcs:ignore WordPress.Security.NonceVerification.Missing -- No need to use nonce.
|
||
|
$layout = get_post( $layout_id );
|
||
|
|
||
|
if ( ! $layout ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => 'Failed to load layout.',
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$edit_url = add_query_arg( 'et_tb', '1', et_fb_get_builder_url( get_permalink( $layout_id ) ) );
|
||
|
// If Admin is SSL but FE is not, we need to fix VB url or it won't work
|
||
|
// because trying to load insecure resource.
|
||
|
$edit_url = set_url_scheme( $edit_url, is_ssl() ? 'https' : 'http' );
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'editUrl' => $edit_url,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_get_layout_url', 'et_theme_builder_api_get_layout_url' );
|
||
|
|
||
|
/**
|
||
|
* Save the theme builder post.
|
||
|
*
|
||
|
* The templates upload will be chunked into several POST requests with size 30 templates per request.
|
||
|
* Hence we need to store the uploaded templates data into temporary file in cache directory before
|
||
|
* making changes into database.
|
||
|
*
|
||
|
* @since 4.0
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
function et_theme_builder_api_save() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_save', 'nonce' );
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Nonce is done in `et_builder_security_check`.
|
||
|
$_ = et_();
|
||
|
$live = '1' === $_->array_get( $_POST, 'live', '0' );
|
||
|
$first_request = '1' === $_->array_get( $_POST, 'first_request', '1' );
|
||
|
$last_request = '1' === $_->array_get( $_POST, 'last_request', '1' );
|
||
|
$templates = wp_unslash( $_->array_get( $_POST, 'templates', array() ) );
|
||
|
$processed_templates = wp_unslash( $_->array_get( $_POST, 'processed_templates', array() ) );
|
||
|
$library_tb_id = (int) $_->array_get( $_POST, 'library_theme_builder_id', 0 );
|
||
|
$library_item_id = (int) $_->array_get( $_POST, 'library_item_id', 0 );
|
||
|
$theme_builder_id = $library_tb_id ? $library_tb_id : et_theme_builder_get_theme_builder_post_id( $live, true );
|
||
|
$has_default = '1' === $_->array_get( $_POST, 'hasDefault', '0' );
|
||
|
$updated_ids = array();
|
||
|
// phpcs:enable
|
||
|
|
||
|
// Remove this action as it not necessary when we're saving entire TB.
|
||
|
// save_post_cb is a heavy operation and significanlty slows down the saving of TB.
|
||
|
// We remove static page resources after TB save below in this function.
|
||
|
remove_action( 'save_post', array( 'ET_Core_PageResource', 'save_post_cb' ), 10, 3 );
|
||
|
|
||
|
$templates_to_process = array();
|
||
|
|
||
|
// Populate the templates.
|
||
|
foreach ( $templates as $index => $template ) {
|
||
|
$templates_to_process[ $_->array_get( $template, 'id', 'unsaved_' . $index ) ] = $template;
|
||
|
}
|
||
|
|
||
|
$affected_templates = array();
|
||
|
|
||
|
// Update or insert templates.
|
||
|
foreach ( $templates_to_process as $template ) {
|
||
|
$raw_post_id = $_->array_get( $template, 'id', 0 );
|
||
|
$post_id = is_numeric( $raw_post_id ) ? (int) $raw_post_id : 0;
|
||
|
$new_post_id = et_theme_builder_store_template( $theme_builder_id, $template, ! $has_default );
|
||
|
|
||
|
if ( ! $new_post_id ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$is_default = get_post_meta( $new_post_id, '_et_default', true ) === '1';
|
||
|
|
||
|
if ( $is_default ) {
|
||
|
$has_default = true;
|
||
|
}
|
||
|
|
||
|
// Add template ID into $affected_templates for later use
|
||
|
// to Add mapping template ID to theme builder ID
|
||
|
// and delete existing template mapping.
|
||
|
$affected_templates[] = array(
|
||
|
'raw' => $raw_post_id,
|
||
|
'normalized' => $post_id,
|
||
|
'new_post_id' => $new_post_id,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
foreach ( $affected_templates as $template_pair ) {
|
||
|
if ( $template_pair['normalized'] !== $template_pair['new_post_id'] ) {
|
||
|
$updated_ids[ $template_pair['raw'] ] = $template_pair['new_post_id'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $last_request ) {
|
||
|
$existing_templates = get_post_meta( $theme_builder_id, '_et_template', false );
|
||
|
|
||
|
if ( $existing_templates ) {
|
||
|
// Store existing template mapping as backup to avoid data lost
|
||
|
// when user interrupting the saving process before completed.
|
||
|
update_option( 'et_tb_templates_backup_' . $theme_builder_id, $existing_templates );
|
||
|
}
|
||
|
|
||
|
// Delete existing template mapping.
|
||
|
delete_post_meta( $theme_builder_id, '_et_template' );
|
||
|
|
||
|
$processed_templates = array_merge( $processed_templates, $affected_templates );
|
||
|
|
||
|
// Insert new template mapping.
|
||
|
foreach ( $processed_templates as $template_pair ) {
|
||
|
add_post_meta( $theme_builder_id, '_et_template', $template_pair['new_post_id'] );
|
||
|
}
|
||
|
|
||
|
// Delete existing template mapping backup.
|
||
|
delete_option( 'et_tb_templates_backup_' . $theme_builder_id );
|
||
|
|
||
|
if ( $live ) {
|
||
|
et_theme_builder_trash_draft_and_unused_posts();
|
||
|
}
|
||
|
|
||
|
et_theme_builder_clear_wp_cache( 'all' );
|
||
|
|
||
|
// Remove static resources on save. It's necessary because how we are generating the dynamic assets for the TB.
|
||
|
ET_Core_PageResource::remove_static_resources( 'all', 'all', false, 'dynamic' );
|
||
|
}
|
||
|
|
||
|
// Edit Template and Edit Preset: Save the templates into local library.
|
||
|
if ( $library_tb_id && $library_item_id ) {
|
||
|
et_theme_builder_update_library_item( $library_item_id, $templates );
|
||
|
}
|
||
|
|
||
|
// Add this action back.
|
||
|
add_action( 'save_post', array( 'ET_Core_PageResource', 'save_post_cb' ), 10, 3 );
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'updatedTemplateIds' => (object) $updated_ids,
|
||
|
'processedTemplatesData' => (object) $affected_templates,
|
||
|
'hasDefault' => $has_default ? '1' : '0',
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_save', 'et_theme_builder_api_save' );
|
||
|
|
||
|
/**
|
||
|
* Drop the theme builder post autosave.
|
||
|
*
|
||
|
* @since 4.0
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
function et_theme_builder_api_drop_autosave() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_drop_autosave', 'nonce' );
|
||
|
|
||
|
et_theme_builder_trash_draft_and_unused_posts();
|
||
|
|
||
|
wp_send_json_success();
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_drop_autosave', 'et_theme_builder_api_drop_autosave' );
|
||
|
|
||
|
function et_theme_builder_api_get_template_settings() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_get_template_settings', 'nonce', '_GET' );
|
||
|
|
||
|
$parent = isset( $_GET['parent'] ) ? sanitize_text_field( $_GET['parent'] ) : '';
|
||
|
$search = isset( $_GET['search'] ) ? sanitize_text_field( $_GET['search'] ) : '';
|
||
|
$page = isset( $_GET['page'] ) ? (int) $_GET['page'] : 1;
|
||
|
$page = $page >= 1 ? $page : 1;
|
||
|
$per_page = 30;
|
||
|
$settings = et_theme_builder_get_flat_template_settings_options();
|
||
|
|
||
|
if ( ! isset( $settings[ $parent ] ) || empty( $settings[ $parent ]['options'] ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'message' => __( 'Invalid parent setting specified.', 'et_builder' ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$setting = $settings[ $parent ];
|
||
|
$results = et_theme_builder_get_template_setting_child_options( $setting, array(), $search, $page, $per_page );
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'results' => array_values( $results ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_get_template_settings', 'et_theme_builder_api_get_template_settings' );
|
||
|
|
||
|
function et_theme_builder_api_reset() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_reset', 'nonce' );
|
||
|
|
||
|
$live_id = et_theme_builder_get_theme_builder_post_id( true, false );
|
||
|
|
||
|
if ( $live_id > 0 && current_user_can( 'delete_others_posts' ) ) {
|
||
|
wp_trash_post( $live_id );
|
||
|
// Reset cache when theme builder is reset.
|
||
|
ET_Core_PageResource::remove_static_resources( 'all', 'all', true );
|
||
|
}
|
||
|
|
||
|
et_theme_builder_trash_draft_and_unused_posts();
|
||
|
|
||
|
wp_send_json_success();
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_reset', 'et_theme_builder_api_reset' );
|
||
|
|
||
|
function et_theme_builder_api_export_theme_builder() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check(
|
||
|
'et_theme_builder_portability',
|
||
|
et_core_portability_cap( 'et_theme_builder' ),
|
||
|
'et_theme_builder_api_export_theme_builder',
|
||
|
'nonce'
|
||
|
);
|
||
|
|
||
|
$_ = et_();
|
||
|
$raw_templates = wp_unslash( $_->array_get( $_POST, 'templates', array() ) );
|
||
|
$global_layouts = array(
|
||
|
'header' => (int) $_->array_get( $_POST, 'global_layouts.header', 0 ),
|
||
|
'body' => (int) $_->array_get( $_POST, 'global_layouts.body', 0 ),
|
||
|
'footer' => (int) $_->array_get( $_POST, 'global_layouts.footer', 0 ),
|
||
|
);
|
||
|
$has_default = false;
|
||
|
$steps = array();
|
||
|
|
||
|
foreach ( $raw_templates as $template ) {
|
||
|
$is_default = ! $has_default && '1' === $_->array_get( $template, 'default', '0' );
|
||
|
|
||
|
if ( $is_default ) {
|
||
|
$has_default = true;
|
||
|
}
|
||
|
|
||
|
$sanitized = et_theme_builder_sanitize_template(
|
||
|
array_merge(
|
||
|
$template,
|
||
|
array(
|
||
|
'default' => $is_default ? '1' : '0',
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$template_item_id = isset( $template['item_id'] ) ? absint( $template['item_id'] ) : 0;
|
||
|
if ( $template_item_id > 0 ) {
|
||
|
$sanitized = array_merge(
|
||
|
$sanitized,
|
||
|
[
|
||
|
'description' => et_theme_builder_library_get_item_description( $template['item_id'], $is_default ),
|
||
|
]
|
||
|
);
|
||
|
} else {
|
||
|
$sanitized = array_merge(
|
||
|
$sanitized,
|
||
|
[
|
||
|
'description' => et_theme_builder_library_get_item_description_from_payload( $template ),
|
||
|
]
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$steps[] = array(
|
||
|
'type' => 'template',
|
||
|
'data' => $sanitized,
|
||
|
);
|
||
|
|
||
|
$layout_keys = array( 'header', 'body', 'footer' );
|
||
|
foreach ( $layout_keys as $key ) {
|
||
|
$layout_id = (int) $_->array_get( $sanitized, array( 'layouts', $key, 'id' ), 0 );
|
||
|
|
||
|
if ( 0 === $layout_id ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$steps[] = array(
|
||
|
'type' => 'layout',
|
||
|
'data' => array(
|
||
|
'post_id' => $layout_id,
|
||
|
'is_global' => $layout_id === $global_layouts[ $key ],
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$presets_manager = ET_Builder_Global_Presets_Settings::instance();
|
||
|
$presets = $presets_manager->get_global_presets();
|
||
|
|
||
|
if ( ! empty( $presets ) ) {
|
||
|
$steps[] = array(
|
||
|
'type' => 'presets',
|
||
|
'data' => $presets,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$id = md5( get_current_user_id() . '_' . uniqid( 'et_theme_builder_export_', true ) );
|
||
|
$transient = 'et_theme_builder_export_' . get_current_user_id() . '_' . $id;
|
||
|
|
||
|
set_transient(
|
||
|
$transient,
|
||
|
array(
|
||
|
'ready' => false,
|
||
|
'steps' => $steps,
|
||
|
'temp_file' => '',
|
||
|
'temp_file_id' => '',
|
||
|
),
|
||
|
60 * 60 * 24
|
||
|
);
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'id' => $id,
|
||
|
'steps' => count( $steps ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_export_theme_builder', 'et_theme_builder_api_export_theme_builder' );
|
||
|
|
||
|
function et_theme_builder_api_export_theme_builder_step() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check(
|
||
|
'et_theme_builder_portability',
|
||
|
et_core_portability_cap( 'et_theme_builder' ),
|
||
|
'et_theme_builder_api_export_theme_builder',
|
||
|
'nonce'
|
||
|
);
|
||
|
|
||
|
$_ = et_();
|
||
|
$id = sanitize_text_field( $_->array_get( $_POST, 'id', '' ) );
|
||
|
$step = (int) $_->array_get( $_POST, 'step', 0 );
|
||
|
$chunk = (int) $_->array_get( $_POST, 'chunk', 0 );
|
||
|
$transient = 'et_theme_builder_export_' . get_current_user_id() . '_' . $id;
|
||
|
$export = get_transient( $transient );
|
||
|
|
||
|
if ( false === $export || ! isset( $export['steps'][ $step ] ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$portability = et_core_portability_load( 'et_theme_builder' );
|
||
|
$export_step = isset( $export['steps'][ $step ] ) ? $export['steps'][ $step ] : array();
|
||
|
$result = $portability->export_theme_builder( $id, $export_step, count( $export['steps'] ), $step, $chunk );
|
||
|
|
||
|
if ( false === $result ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
if ( $result['ready'] ) {
|
||
|
set_transient(
|
||
|
$transient,
|
||
|
array_merge(
|
||
|
$export,
|
||
|
array(
|
||
|
'ready' => $result['ready'],
|
||
|
'temp_file' => $result['temp_file'],
|
||
|
'temp_file_id' => $result['temp_file_id'],
|
||
|
)
|
||
|
),
|
||
|
60 * 60 * 24
|
||
|
);
|
||
|
}
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'chunks' => $result['chunks'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_export_theme_builder_step', 'et_theme_builder_api_export_theme_builder_step' );
|
||
|
|
||
|
function et_theme_builder_api_export_theme_builder_download() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check(
|
||
|
'et_theme_builder_portability',
|
||
|
et_core_portability_cap( 'et_theme_builder' ),
|
||
|
'et_theme_builder_api_export_theme_builder',
|
||
|
'nonce',
|
||
|
'_GET'
|
||
|
);
|
||
|
|
||
|
$_ = et_();
|
||
|
$id = sanitize_text_field( $_->array_get( $_GET, 'id', '' ) );
|
||
|
$filename = sanitize_text_field( $_->array_get( $_GET, 'filename', '' ) );
|
||
|
$filename = '' !== $filename ? $filename : 'Divi Theme Builder Templates';
|
||
|
$filename = sanitize_file_name( $filename );
|
||
|
$transient = 'et_theme_builder_export_' . get_current_user_id() . '_' . $id;
|
||
|
$export = get_transient( $transient );
|
||
|
|
||
|
if ( false === $export || ! $export['ready'] ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$portability = et_core_portability_load( 'et_theme_builder' );
|
||
|
$portability->download_file( $filename, $export['temp_file_id'], $export['temp_file'] );
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_export_theme_builder_download', 'et_theme_builder_api_export_theme_builder_download' );
|
||
|
|
||
|
/**
|
||
|
* Save a layout in a temporary file to prepare it for import.
|
||
|
*
|
||
|
* @since 4.1.0
|
||
|
*
|
||
|
* @param ET_Core_Portability $portability Portability object.
|
||
|
* @param string $template_id Template ID.
|
||
|
* @param integer $layout_id Layout ID.
|
||
|
* @param array $layout Layout.
|
||
|
* @param string $temp_id Temporary ID.
|
||
|
* @param string $temp_group Temporary Group.
|
||
|
*/
|
||
|
function et_theme_builder_api_import_theme_builder_save_layout( $portability, $template_id, $layout_id, $layout, $temp_id, $temp_group ) {
|
||
|
if ( ! current_user_can( 'edit_others_posts' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
if ( ! empty( $layout['images'] ) ) {
|
||
|
// Split up images into individual temporary files
|
||
|
// to avoid hitting the memory limit.
|
||
|
foreach ( $layout['images'] as $url => $data ) {
|
||
|
$image_temp_id = $temp_id . '-image-' . md5( $url );
|
||
|
|
||
|
$portability->temp_file( $image_temp_id, $temp_group, false, wp_json_encode( $data ) );
|
||
|
|
||
|
$layout['images'][ $url ] = array(
|
||
|
'id' => $image_temp_id,
|
||
|
'group' => $temp_group,
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$portability->temp_file(
|
||
|
$temp_id,
|
||
|
$temp_group,
|
||
|
false,
|
||
|
wp_json_encode(
|
||
|
array(
|
||
|
'type' => 'layout',
|
||
|
'data' => $layout,
|
||
|
'id' => $layout_id,
|
||
|
'template_id' => $template_id,
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Load a previously saved layout from a temporary file.
|
||
|
*
|
||
|
* @since 4.1.0
|
||
|
*
|
||
|
* @param ET_Core_Portability $portability Portability Object.
|
||
|
* @param string $temp_id Temporary ID.
|
||
|
* @param string $temp_group Temporary Group.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
function et_theme_builder_api_import_theme_builder_load_layout( $portability, $temp_id, $temp_group ) {
|
||
|
if ( ! current_user_can( 'edit_others_posts' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$import = $portability->get_temp_file_contents( $temp_id, $temp_group );
|
||
|
$import = ! empty( $import ) ? json_decode( $import, true ) : array();
|
||
|
$images = et_()->array_get( $import, array( 'data', 'images' ), array() );
|
||
|
|
||
|
// Hydrate images back from their individual temporary files.
|
||
|
foreach ( $images as $url => $file ) {
|
||
|
$import['data']['images'][ $url ] = json_decode( $portability->get_temp_file_contents( $file['id'], $file['group'] ), true );
|
||
|
}
|
||
|
|
||
|
return $import;
|
||
|
}
|
||
|
|
||
|
function et_theme_builder_api_import_theme_builder() {
|
||
|
if ( ! current_user_can( 'edit_others_posts' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$i18n = array_merge(
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/generic.php',
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/portability.php',
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/theme-builder.php'
|
||
|
);
|
||
|
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::UNKNOWN,
|
||
|
'error' => $i18n['An unknown error has occurred. Please try again later.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
et_builder_security_check(
|
||
|
'et_theme_builder_portability',
|
||
|
et_core_portability_cap( 'et_theme_builder' ),
|
||
|
'et_theme_builder_api_import_theme_builder',
|
||
|
'nonce'
|
||
|
);
|
||
|
|
||
|
if ( ! isset( $_FILES['file']['name'] ) || ! et_()->ends_with( sanitize_file_name( $_FILES['file']['name'] ), '.json' ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::PORTABILITY_IMPORT_INVALID_FILE,
|
||
|
'error' => $i18n['$invalid_file'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$_ = et_();
|
||
|
$upload = wp_handle_upload(
|
||
|
$_FILES['file'],
|
||
|
array(
|
||
|
'test_size' => false,
|
||
|
'test_type' => false,
|
||
|
'test_form' => false,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
if ( ! $_->array_get( $upload, 'file', null ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::UNKNOWN,
|
||
|
'error' => $i18n['An unknown error has occurred. Please try again later.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$export = json_decode( et_()->WPFS()->get_contents( $upload['file'] ), true );
|
||
|
|
||
|
if ( null === $export ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::UNKNOWN,
|
||
|
'error' => $i18n['An unknown error has occurred. Please try again later.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$portability = et_core_portability_load( 'et_theme_builder' );
|
||
|
|
||
|
if ( ! $portability->is_valid_theme_builder_export( $export ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::PORTABILITY_INCORRECT_CONTEXT,
|
||
|
'error' => $i18n['This file should not be imported in this context.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Nonce verfied in `et_builder_security_check`.
|
||
|
$override_default_website_template = '1' === $_->array_get( $_POST, 'override_default_website_template', '0' );
|
||
|
$import_presets = '1' === $_->array_get( $_POST, 'import_presets', '0' );
|
||
|
$library_template_import = '1' === $_->array_get( $_POST, 'library_template_import', '0' );
|
||
|
$has_default_template = $_->array_get( $export, 'has_default_template', false );
|
||
|
$has_global_layouts = $_->array_get( $export, 'has_global_layouts', false );
|
||
|
$presets = $_->array_get( $export, 'presets', array() );
|
||
|
$presets_rewrite_map = array();
|
||
|
$incoming_layout_duplicate = false;
|
||
|
$uploaded_file_name = substr( sanitize_file_name( $_FILES['file']['name'] ), 0, -5 );
|
||
|
$cloud_item_editor = $_->array_get( $_POST, 'cloud_item_editor', '' );
|
||
|
$temp_import = '1' === $_->array_get( $_POST, 'temp_import', '0' );
|
||
|
|
||
|
// Maybe ask the user to make a decision on how to deal with global layouts.
|
||
|
if ( ( ! $override_default_website_template || ! $has_default_template ) && $has_global_layouts ) {
|
||
|
$incoming_layout_duplicate_decision = $_->array_get( $_POST, 'incoming_layout_duplicate_decision', '' );
|
||
|
|
||
|
if ( 'duplicate' === $incoming_layout_duplicate_decision || $library_template_import ) {
|
||
|
$incoming_layout_duplicate = true;
|
||
|
} elseif ( 'relink' === $incoming_layout_duplicate_decision ) {
|
||
|
$incoming_layout_duplicate = false;
|
||
|
} else {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::PORTABILITY_REQUIRE_INCOMING_LAYOUT_DUPLICATE_DECISION,
|
||
|
'error' => $i18n['This import contains references to global layouts.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
// phpcs:enable
|
||
|
|
||
|
// Make imported preset overrides to avoid collisions with local presets.
|
||
|
if ( $import_presets && is_array( $presets ) && ! empty( $presets ) ) {
|
||
|
$presets_rewrite_map = $portability->prepare_to_import_layout_presets( $presets );
|
||
|
}
|
||
|
|
||
|
// Prepare import steps.
|
||
|
$layout_id_map = array();
|
||
|
$layout_keys = array( 'header', 'body', 'footer' );
|
||
|
$id = md5( get_current_user_id() . '_' . uniqid( 'et_theme_builder_import_', true ) );
|
||
|
$transient = 'et_theme_builder_import_' . get_current_user_id() . '_' . $id;
|
||
|
$steps_files = array();
|
||
|
|
||
|
foreach ( $export['templates'] as $index => $template ) {
|
||
|
foreach ( $layout_keys as $key ) {
|
||
|
$layout_id = (int) $_->array_get( $template, array( 'layouts', $key, 'id' ), 0 );
|
||
|
|
||
|
if ( 0 === $layout_id ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$layout = $_->array_get( $export, array( 'layouts', $layout_id ), null );
|
||
|
|
||
|
if ( empty( $layout ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Use a temporary string id to avoid numerical keys being reset by various array functions.
|
||
|
$template_id = 'template_' . $index;
|
||
|
$is_global = (bool) $_->array_get( $layout, 'theme_builder.is_global', false );
|
||
|
$create_new = ( $template['default'] && $override_default_website_template ) || ! $is_global || $incoming_layout_duplicate;
|
||
|
|
||
|
if ( $create_new ) {
|
||
|
$temp_id = 'tbi-step-' . count( $steps_files );
|
||
|
|
||
|
et_theme_builder_api_import_theme_builder_save_layout( $portability, $template_id, $layout_id, $layout, $temp_id, $transient );
|
||
|
|
||
|
$steps_files[] = array(
|
||
|
'id' => $temp_id,
|
||
|
'group' => $transient,
|
||
|
);
|
||
|
} else {
|
||
|
if ( ! isset( $layout_id_map[ $layout_id ] ) ) {
|
||
|
$layout_id_map[ $layout_id ] = array();
|
||
|
}
|
||
|
|
||
|
$layout_id_map[ $layout_id ][ $template_id ] = 'use_global';
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
set_transient(
|
||
|
$transient,
|
||
|
array(
|
||
|
'file_name' => $uploaded_file_name,
|
||
|
'ready' => false,
|
||
|
'steps' => $steps_files,
|
||
|
'templates' => $export['templates'],
|
||
|
'override_default_website_template' => $override_default_website_template,
|
||
|
'incoming_layout_duplicate' => $incoming_layout_duplicate,
|
||
|
'layout_id_map' => $layout_id_map,
|
||
|
'presets' => $presets,
|
||
|
'import_presets' => $import_presets,
|
||
|
'library_template_import' => $library_template_import,
|
||
|
'presets_rewrite_map' => $presets_rewrite_map,
|
||
|
'cloud_item_editor' => $cloud_item_editor,
|
||
|
'temp_import' => $temp_import,
|
||
|
),
|
||
|
60 * 60 * 24
|
||
|
);
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'id' => $id,
|
||
|
'steps' => count( $steps_files ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_import_theme_builder', 'et_theme_builder_api_import_theme_builder' );
|
||
|
|
||
|
function et_theme_builder_api_import_theme_builder_step() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check(
|
||
|
'et_theme_builder_portability',
|
||
|
et_core_portability_cap( 'et_theme_builder' ),
|
||
|
'et_theme_builder_api_import_theme_builder',
|
||
|
'nonce'
|
||
|
);
|
||
|
|
||
|
$_ = et_();
|
||
|
$id = sanitize_text_field( $_->array_get( $_POST, 'id', '' ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce is done in `et_builder_security_check`.
|
||
|
$step = (int) $_->array_get( $_POST, 'step', 0 ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce is done in `et_builder_security_check`.
|
||
|
$chunk = (int) $_->array_get( $_POST, 'chunk', 0 ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce is done in `et_builder_security_check`.
|
||
|
$transient = 'et_theme_builder_import_' . get_current_user_id() . '_' . $id;
|
||
|
$export = get_transient( $transient );
|
||
|
|
||
|
if ( false === $export ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$layout_keys = array( 'header', 'body', 'footer' );
|
||
|
$portability = et_core_portability_load( 'et_theme_builder' );
|
||
|
$steps = $export['steps'];
|
||
|
$ready = empty( $steps );
|
||
|
$layout_id_map = $export['layout_id_map'];
|
||
|
$presets = $export['presets'];
|
||
|
$presets_rewrite_map = $export['presets_rewrite_map'];
|
||
|
$import_presets = $export['import_presets'];
|
||
|
$library_template_import = $export['library_template_import'];
|
||
|
$file_name = $export['file_name'];
|
||
|
$cloud_item_editor = $export['cloud_item_editor'];
|
||
|
$temp_import = $export['temp_import'];
|
||
|
$templates = array();
|
||
|
$template_settings = array();
|
||
|
$chunks = 1;
|
||
|
$preset_id = 0;
|
||
|
|
||
|
if ( ! $ready ) {
|
||
|
$import_step = et_theme_builder_api_import_theme_builder_load_layout( $portability, $steps[ $step ]['id'], $steps[ $step ]['group'] );
|
||
|
$import_step = array_merge( $import_step, array( 'presets' => $presets ) );
|
||
|
$import_step = array_merge( $import_step, array( 'presets_rewrite_map' => $presets_rewrite_map ) );
|
||
|
$import_step['import_presets'] = $import_presets;
|
||
|
|
||
|
if ( $temp_import ) {
|
||
|
$import_step['data']['post_status'] = 'draft';
|
||
|
}
|
||
|
|
||
|
$result = $portability->import_theme_builder( $id, $import_step, count( $steps ), $step, $chunk );
|
||
|
|
||
|
if ( false === $result ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$ready = $result['ready'];
|
||
|
$chunks = $result['chunks'];
|
||
|
|
||
|
foreach ( $result['layout_id_map'] as $old_id => $new_ids ) {
|
||
|
$layout_id_map[ $old_id ] = array_merge(
|
||
|
$_->array_get( $layout_id_map, $old_id, array() ),
|
||
|
$new_ids
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $ready ) {
|
||
|
if ( $import_presets && is_array( $presets ) && ! empty( $presets ) ) {
|
||
|
if ( ! $portability->import_global_presets( $presets ) ) {
|
||
|
$presets_error = apply_filters( 'et_core_portability_import_error_message', '' );
|
||
|
|
||
|
if ( $presets_error ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::PORTABILITY_IMPORT_PRESETS_FAILURE,
|
||
|
'error' => $presets_error,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$portability->delete_temp_files( $transient );
|
||
|
|
||
|
$conditions = array();
|
||
|
$global_layouts = array();
|
||
|
|
||
|
foreach ( $export['templates'] as $index => $template ) {
|
||
|
$sanitized = et_theme_builder_sanitize_template( $template );
|
||
|
$is_default = $_->array_get( $sanitized, 'default', false );
|
||
|
|
||
|
foreach ( $layout_keys as $key ) {
|
||
|
$old_layout_id = (int) $_->array_get( $sanitized, array( 'layouts', $key, 'id' ), 0 );
|
||
|
$layout_id = et_()->array_get( $layout_id_map, array( $old_layout_id, 'template_' . $index ), '' );
|
||
|
$layout_id = ! empty( $layout_id ) ? $layout_id : 0;
|
||
|
|
||
|
$_->array_set( $sanitized, array( 'layouts', $key, 'id' ), $layout_id );
|
||
|
|
||
|
if ( $is_default ) {
|
||
|
$global_layouts[ $key ]['id'] = $layout_id;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$conditions = array_merge( $conditions, $sanitized['use_on'], $sanitized['exclude_from'] );
|
||
|
$_->array_set( $sanitized, array( 'global_layouts' ), $global_layouts );
|
||
|
|
||
|
$templates[] = $sanitized;
|
||
|
}
|
||
|
|
||
|
// Load all conditions from templates.
|
||
|
$conditions = array_unique( $conditions );
|
||
|
$template_settings = array_replace(
|
||
|
et_theme_builder_get_flat_template_settings_options(),
|
||
|
et_theme_builder_load_template_setting_options( $conditions )
|
||
|
);
|
||
|
$valid_settings = array_keys( $template_settings );
|
||
|
|
||
|
// Strip all invalid conditions from templates.
|
||
|
foreach ( $templates as $index => $template ) {
|
||
|
$templates[ $index ]['use_on'] = array_values( array_intersect( $template['use_on'], $valid_settings ) );
|
||
|
$templates[ $index ]['exclude_from'] = array_values( array_intersect( $template['exclude_from'], $valid_settings ) );
|
||
|
}
|
||
|
|
||
|
if ( $library_template_import ) {
|
||
|
$is_multi_template = count( $templates ) > 1;
|
||
|
|
||
|
if ( $is_multi_template || 'set' === $cloud_item_editor ) {
|
||
|
$template_settings['set_name'] = $file_name;
|
||
|
|
||
|
foreach ( $templates as $key => $template ) {
|
||
|
foreach ( array( 'body', 'header', 'footer' ) as $layout_type ) {
|
||
|
$layout_id = $_->array_get( $template, array( 'layouts', $layout_type, 'id' ) );
|
||
|
if ( 'use_global' === $layout_id && isset( $global_layouts[ $layout_type ] ) ) {
|
||
|
$global_layout_id = $_->array_get( $global_layouts, array( $layout_type, 'id' ) );
|
||
|
$_->array_set( $templates, array( $key, 'layouts', $layout_type, 'id' ), $global_layout_id );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $temp_import ) {
|
||
|
$template_settings['post_status'] = 'draft';
|
||
|
}
|
||
|
|
||
|
$preset_id = et_theme_builder_save_preset_to_library( $templates, $template_settings );
|
||
|
} else {
|
||
|
$first_template = $templates[0];
|
||
|
if ( 'template' === $cloud_item_editor ) {
|
||
|
$template_settings['template_name'] = $first_template['title'];
|
||
|
} else {
|
||
|
$template_settings['template_name'] = $file_name;
|
||
|
}
|
||
|
|
||
|
if ( $temp_import ) {
|
||
|
$first_template['status'] = 'draft';
|
||
|
}
|
||
|
|
||
|
$templates[0]['template_id'] = et_theme_builder_save_template_to_library( $first_template, $template_settings );
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
set_transient(
|
||
|
$transient,
|
||
|
array_merge(
|
||
|
$export,
|
||
|
array(
|
||
|
'layout_id_map' => $layout_id_map,
|
||
|
)
|
||
|
),
|
||
|
60 * 60 * 24
|
||
|
);
|
||
|
}
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'presetId' => $preset_id,
|
||
|
'chunks' => $chunks,
|
||
|
'templates' => $templates,
|
||
|
'templateSettings' => $template_settings,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_import_theme_builder_step', 'et_theme_builder_api_import_theme_builder_step' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: save template into the local library.
|
||
|
*/
|
||
|
function et_theme_builder_api_save_template_to_library() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_save_template_to_library', 'nonce' );
|
||
|
|
||
|
$_ = et_();
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
$raw_templates = wp_unslash( $_->array_get( $_POST, 'template', array() ) );
|
||
|
$preferences = wp_unslash( $_->array_get( $_POST, 'preferences', array() ) );
|
||
|
// phpcs:enable
|
||
|
|
||
|
$post_id = et_theme_builder_save_template_to_library( $raw_templates, $preferences );
|
||
|
|
||
|
if ( $post_id ) {
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'post_id' => $post_id,
|
||
|
)
|
||
|
);
|
||
|
} else {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_save_template_to_library', 'et_theme_builder_api_save_template_to_library' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: save preset into the local library.
|
||
|
*/
|
||
|
function et_theme_builder_api_save_preset_to_library() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_save_preset_to_library', 'nonce' );
|
||
|
|
||
|
$_ = et_();
|
||
|
$preferences = wp_unslash( $_->array_get( $_POST, 'preferences', [] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
$raw_templates = wp_unslash( $_->array_get( $_POST, 'templates', [] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
|
||
|
$post_id = et_theme_builder_save_preset_to_library( $raw_templates, $preferences );
|
||
|
|
||
|
if ( $post_id ) {
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'post_id' => $post_id,
|
||
|
)
|
||
|
);
|
||
|
} else {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_save_preset_to_library', 'et_theme_builder_api_save_preset_to_library' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Retrieve terms for the given taxonomy.
|
||
|
*/
|
||
|
function et_theme_builder_api_get_terms() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_get_terms', 'nonce' );
|
||
|
|
||
|
$_ = et_();
|
||
|
$tax = sanitize_text_field( $_->array_get( $_POST, 'tax', '' ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
|
||
|
if ( ! in_array( $tax, array( 'layout_category', 'layout_tag' ), true ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$terms_by_id = et_theme_builder_get_terms( $tax );
|
||
|
|
||
|
wp_send_json_success( $terms_by_id );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_get_terms', 'et_theme_builder_api_get_terms' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Use local library item.
|
||
|
*/
|
||
|
function et_theme_builder_api_use_library_item() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_api_use_library_item', 'nonce' );
|
||
|
|
||
|
$i18n = array_merge(
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/generic.php',
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/portability.php',
|
||
|
require ET_BUILDER_DIR . 'frontend-builder/i18n/theme-builder.php'
|
||
|
);
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
$_ = et_();
|
||
|
$item_id = (int) $_->array_get( $_POST, 'item_id', 0 );
|
||
|
$override_default_website_template = '1' === $_->array_get( $_POST, 'override_default_website_template', '0' );
|
||
|
$override_assignments = '1' === $_->array_get( $_POST, 'override_assignments', '0' );
|
||
|
$download_backup = '1' === $_->array_get( $_POST, 'download_backup', '0' );
|
||
|
$incoming_layout_duplicate_decision = $_->array_get( $_POST, 'incoming_layout_duplicate_decision', '' );
|
||
|
$item_id = (int) $_->array_get( $_POST, 'item_id', 0 );
|
||
|
// phpcs:enable
|
||
|
|
||
|
$args = [
|
||
|
'override_default_website_template' => $override_default_website_template,
|
||
|
'override_assignments' => $override_assignments,
|
||
|
'download_backup' => $download_backup,
|
||
|
'incoming_layout_duplicate_decision' => $incoming_layout_duplicate_decision,
|
||
|
];
|
||
|
|
||
|
$return_additional_args = [];
|
||
|
$library_item = new ET_Theme_Builder_Local_Library_Item( $item_id );
|
||
|
|
||
|
if ( ET_THEME_BUILDER_ITEM_SET === $library_item->get_item_type() ) {
|
||
|
$has_global_layouts = $library_item->has_global_layouts();
|
||
|
$has_default_template = $library_item->has_default_template();
|
||
|
$show_layout_decision = ( ! $override_default_website_template || ! $has_default_template ) && $has_global_layouts;
|
||
|
|
||
|
if ( $show_layout_decision && empty( $incoming_layout_duplicate_decision ) ) {
|
||
|
wp_send_json_error(
|
||
|
array(
|
||
|
'code' => ET_Theme_Builder_Api_Errors::PORTABILITY_REQUIRE_INCOMING_LAYOUT_DUPLICATE_DECISION,
|
||
|
'error' => $i18n['This import contains references to global layouts.'],
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$return_additional_args = [
|
||
|
'override_default_website_template' => $override_default_website_template,
|
||
|
'override_assignments' => $override_assignments,
|
||
|
'download_backup' => $download_backup,
|
||
|
'incoming_layout_duplicate_decision' => $incoming_layout_duplicate_decision,
|
||
|
];
|
||
|
}
|
||
|
|
||
|
$item_data = $library_item->use_library_item( $args );
|
||
|
|
||
|
if ( is_wp_error( $item_data ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$data = array_merge(
|
||
|
array(
|
||
|
'item_type' => $library_item->item_type,
|
||
|
'item_data' => $item_data,
|
||
|
),
|
||
|
$return_additional_args
|
||
|
);
|
||
|
|
||
|
wp_send_json_success( $data );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_api_use_library_item', 'et_theme_builder_api_use_library_item' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Trash interim library editor posts.
|
||
|
*/
|
||
|
function et_theme_builder_trash_theme_builder() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_trash_theme_builder', 'nonce' );
|
||
|
|
||
|
// phpcs:disable WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
$_ = et_();
|
||
|
$theme_builder_id = (int) $_->array_get( $_POST, 'theme_builder_id', 0 );
|
||
|
$template_ids = et_theme_builder_get_theme_builder_template_ids( true, $theme_builder_id );
|
||
|
$item_id = (int) $_->array_get( $_POST, 'item_id' );
|
||
|
$cloud_item_id = (int) $_->array_get( $_POST, 'cloud_item_id' );
|
||
|
$used_posts = array();
|
||
|
// phpcs:enable
|
||
|
|
||
|
foreach ( $template_ids as $template_id ) {
|
||
|
foreach ( array( 'header', 'body', 'footer' ) as $layout_type ) {
|
||
|
$layout_id = get_post_meta( $template_id, "_et_{$layout_type}_layout_id", true );
|
||
|
|
||
|
// Clean layouts.
|
||
|
if ( $layout_id ) {
|
||
|
$used_posts[] = $layout_id;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Clean template.
|
||
|
$used_posts[] = $template_id;
|
||
|
}
|
||
|
|
||
|
$used_posts[] = $theme_builder_id;
|
||
|
|
||
|
$used_posts = array_map( 'intval', $used_posts );
|
||
|
foreach ( $used_posts as $used_post ) {
|
||
|
if ( current_user_can( 'delete_others_posts' ) ) {
|
||
|
wp_trash_post( $used_post );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Delete local library item.
|
||
|
if ( $cloud_item_id ) {
|
||
|
if ( current_user_can( 'delete_others_posts' ) ) {
|
||
|
wp_delete_post( $item_id );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_trash_theme_builder', 'et_theme_builder_trash_theme_builder' );
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Gets items data for the theme builder's library UI.
|
||
|
*/
|
||
|
function et_theme_builder_library_get_items_data() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_get_items_data', 'nonce' );
|
||
|
|
||
|
$item_type = isset( $_POST['et_item_type'] ) ? (string) sanitize_text_field( $_POST['et_item_type'] ) : 'template';
|
||
|
|
||
|
if ( ! in_array( $item_type, array( 'set', 'template' ), true ) ) {
|
||
|
wp_send_json_error( 'Error: Wrong item type provided.' );
|
||
|
}
|
||
|
|
||
|
$item_library_local = et_pb_theme_builder_library_local();
|
||
|
$data = $item_library_local->get_library_items( $item_type );
|
||
|
|
||
|
wp_send_json_success(
|
||
|
$data
|
||
|
);
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_get_items_data', 'et_theme_builder_library_get_items_data' );
|
||
|
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Add/Remove/Rename Library terms for taxonomies.
|
||
|
*/
|
||
|
function et_theme_builder_library_update_terms() {
|
||
|
et_builder_security_check( 'theme_builder', 'manage_categories', 'et_theme_builder_library_update_terms', 'nonce' );
|
||
|
|
||
|
$payload = isset( $_POST['payload'] ) ? (array) $_POST['payload'] : array(); // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- $_POST['payload'] is an array, it's value sanitization is done at the time of accessing value.
|
||
|
|
||
|
$et_library_taxonomy = isset( $_POST['et_library_taxonomy'] ) ? (string) $_POST['et_library_taxonomy'] : ''; // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- $_POST['et_library_taxonomy'] is a string, it's value sanitization is done at the time of accessing value.
|
||
|
|
||
|
$response = et_theme_builder_library_update_taxonomy_terms( $payload, $et_library_taxonomy );
|
||
|
|
||
|
if ( ! $response ) {
|
||
|
wp_send_json_error( 'Error: Please provide valid payload and taxonomy' );
|
||
|
}
|
||
|
|
||
|
return wp_send_json_success( $response );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_update_terms', 'et_theme_builder_library_update_terms' );
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Update the theme builder library item.
|
||
|
*/
|
||
|
function et_theme_builder_library_update_item() {
|
||
|
if ( ! et_pb_is_allowed( 'theme_builder' ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_update_item', 'nonce' );
|
||
|
|
||
|
$payload = isset( $_POST['payload'] ) ? (array) $_POST['payload'] : array(); // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- $_POST['payload'] is an array, it's value sanitization is done at the time of accessing value.
|
||
|
|
||
|
if ( empty( $payload ) ) {
|
||
|
wp_send_json_error( 'Error: Payload is empty.' );
|
||
|
}
|
||
|
|
||
|
$item_library_local = et_pb_theme_builder_library_local();
|
||
|
$response = $item_library_local->perform_item_update( $payload );
|
||
|
|
||
|
if ( ! $response ) {
|
||
|
wp_send_json_error( 'Error: Provide valid data.' );
|
||
|
}
|
||
|
|
||
|
return wp_send_json_success( $response );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_update_item', 'et_theme_builder_library_update_item' );
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Save the theme builder library temporary item.
|
||
|
*/
|
||
|
function et_theme_builder_library_save_temp_layout() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_save_temp_layout', 'nonce' );
|
||
|
|
||
|
// phpcs:disable -- Sanitization will be handled inside the method.
|
||
|
$local_content = isset( $_POST['localContent'] ) ? (array) $_POST['localContent'] : array();
|
||
|
$cloud_content = isset( $_POST['cloudContent'] ) ? (array) $_POST['cloudContent'] : array();
|
||
|
// phpcs:enable
|
||
|
|
||
|
$templates = array();
|
||
|
$global_layouts = array();
|
||
|
$is_cloud_item = ! empty( $cloud_content['layouts'] );
|
||
|
$data = $is_cloud_item ? $cloud_content['templates'] : $local_content;
|
||
|
|
||
|
foreach ( $data as $template ) :
|
||
|
if ( $is_cloud_item ) {
|
||
|
$is_default = filter_var( $template['default'], FILTER_VALIDATE_BOOLEAN );
|
||
|
|
||
|
$template = array_merge(
|
||
|
$template,
|
||
|
et_theme_builder_library_save_temp_cloud_layout_data( $template, $cloud_content['layouts'], $global_layouts )
|
||
|
);
|
||
|
} else {
|
||
|
$post = get_post( $template['id'] );
|
||
|
|
||
|
if ( ! $post ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$content = json_decode( $post->post_content );
|
||
|
$is_default = get_post_meta( $post->ID, '_et_default', true );
|
||
|
|
||
|
$template = array_merge(
|
||
|
$template,
|
||
|
et_theme_builder_library_save_temp_local_layout_data( $post->ID, $content, $global_layouts )
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// Set global references.
|
||
|
$layouts = $template['layouts'];
|
||
|
$layout_types = array( 'header', 'body', 'footer' );
|
||
|
|
||
|
foreach ( $layout_types as $layout_type ) {
|
||
|
if ( ! isset( $layouts[ $layout_type ] ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if ( $is_cloud_item ) {
|
||
|
$global_info = et_()->array_get( $template, $layout_type . '_layout_global', false );
|
||
|
$is_global = filter_var( $global_info, FILTER_VALIDATE_BOOLEAN );
|
||
|
} else {
|
||
|
$is_global = get_post_meta( $post->ID, '_et_' . $layout_type . '_layout_global', true );
|
||
|
}
|
||
|
|
||
|
if ( $is_default || $is_global ) {
|
||
|
$global_layouts[ $layout_type ] = $layouts[ $layout_type ]['id'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
array_push( $templates, $template );
|
||
|
endforeach;
|
||
|
|
||
|
$response = array(
|
||
|
'templates' => $templates,
|
||
|
'global_layouts' => $global_layouts,
|
||
|
);
|
||
|
|
||
|
wp_send_json_success( $response );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_save_temp_layout', 'et_theme_builder_library_save_temp_layout' );
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Remove the theme builder library temporary item.
|
||
|
*/
|
||
|
function et_theme_builder_library_remove_temp_layout() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_remove_temp_layout', 'nonce' );
|
||
|
|
||
|
$payload = isset( $_POST['payload'] ) ? (array) $_POST['payload'] : array(); // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- $_POST['payload'] is an array, it's value sanitization is done at the time of accessing value.
|
||
|
|
||
|
foreach ( $payload as $template ) {
|
||
|
et_theme_builder_library_remove_temp_layout_data( $template );
|
||
|
}
|
||
|
|
||
|
wp_send_json_success();
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_remove_temp_layout', 'et_theme_builder_library_remove_temp_layout' );
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Gets an item by ID.
|
||
|
*/
|
||
|
function et_theme_builder_library_get_item() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_get_item', 'nonce' );
|
||
|
|
||
|
$id = isset( $_POST['id'] ) ? intval( $_POST['id'] ) : 0;
|
||
|
$item_type = isset( $_POST['itemType'] ) ? (string) sanitize_text_field( $_POST['itemType'] ) : 'template';
|
||
|
|
||
|
if ( empty( $id ) || ! in_array( $item_type, array( 'set', 'template' ), true ) ) {
|
||
|
wp_send_json_error( 'Error: Please provide an ID and a valid item type.' );
|
||
|
}
|
||
|
|
||
|
$result = array();
|
||
|
$items_id = array( $id );
|
||
|
|
||
|
if ( ET_THEME_BUILDER_ITEM_SET === $item_type ) {
|
||
|
$items_id = array();
|
||
|
$callback = 'et_theme_builder_library_get_' . ET_THEME_BUILDER_ITEM_SET . '_items_data';
|
||
|
$items = $callback( $id );
|
||
|
$default_template_id = (int) get_post_meta( $id, '_et_default_template_id', true );
|
||
|
|
||
|
foreach ( $items as $item ) {
|
||
|
array_push( $items_id, $item->id );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Continue processing for both set and template.
|
||
|
|
||
|
$result['exported'] = et_theme_builder_library_get_exported_content( $items_id );
|
||
|
|
||
|
if (
|
||
|
! isset( $result['exported']['context'] ) ||
|
||
|
! isset( $result['exported']['templates'] ) ||
|
||
|
! isset( $result['exported']['layouts'] )
|
||
|
) {
|
||
|
wp_send_json_error( 'Error: Invalid data.' );
|
||
|
}
|
||
|
|
||
|
$response = wp_json_encode(
|
||
|
array(
|
||
|
'success' => true,
|
||
|
'data' => $result,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
if ( ! $response ) {
|
||
|
wp_send_json_error( 'Error: Invalid response.' );
|
||
|
}
|
||
|
|
||
|
$tmp_dir = function_exists( 'sys_get_temp_dir' ) ? sys_get_temp_dir() : '/tmp';
|
||
|
|
||
|
$tmp_file = tempnam( $tmp_dir, 'et' );
|
||
|
|
||
|
et_()->WPFS()->put_contents( $tmp_file, $response );
|
||
|
|
||
|
// Remove any previous buffered content since we're setting `Content-Length` header
|
||
|
// based on $response value only.
|
||
|
while ( ob_get_level() ) {
|
||
|
ob_end_clean();
|
||
|
}
|
||
|
|
||
|
header( 'Content-Length: ' . @filesize( $tmp_file ) ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged -- `filesize` may fail due to the permissions denied error.
|
||
|
|
||
|
@unlink( $tmp_file ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged -- `unlink` may fail due to the permissions denied error.
|
||
|
|
||
|
// Charset has to be explicitly mentioned when it is other than UTF-8.
|
||
|
header( 'Content-Type: application/json; charset=' . esc_attr( get_option( 'blog_charset' ) ) );
|
||
|
|
||
|
die( et_core_intentionally_unescaped( $response, 'html' ) );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_get_item', 'et_theme_builder_library_get_item' );
|
||
|
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Get the theme builder library preset items.
|
||
|
*/
|
||
|
function et_theme_builder_library_get_set_items() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_get_set_items', 'nonce' );
|
||
|
|
||
|
$item_id = isset( $_POST['itemId'] ) ? intval( $_POST['itemId'] ) : 0;
|
||
|
|
||
|
if ( empty( $item_id ) ) {
|
||
|
wp_send_json_error();
|
||
|
}
|
||
|
|
||
|
$items = et_theme_builder_library_get_set_items_data( $item_id );
|
||
|
|
||
|
wp_send_json_success( $items );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_get_set_items', 'et_theme_builder_library_get_set_items' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Get default template id of the preset.
|
||
|
*/
|
||
|
function et_theme_builder_get_preset_default_template_id() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_get_preset_default_template_id', 'nonce' );
|
||
|
|
||
|
$_ = et_();
|
||
|
$item_id = (int) $_->array_get( $_POST, 'item_id', 0 ); // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
$default_template_id = (int) get_post_meta( $item_id, '_et_default_template_id', true );
|
||
|
|
||
|
if ( $default_template_id > 0 ) {
|
||
|
wp_send_json_success( array( 'default_template_id' => $default_template_id ) );
|
||
|
} else {
|
||
|
wp_send_json_error( array( 'default_template_id' => 0 ) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_get_preset_default_template_id', 'et_theme_builder_get_preset_default_template_id' );
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Remove the Library item after it is moved to the Cloud.
|
||
|
*/
|
||
|
function et_theme_builder_library_toggle_cloud_status() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_toggle_cloud_status', 'nonce' );
|
||
|
|
||
|
$post_id = isset( $_POST['id'] ) ? intval( $_POST['id'] ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Missing -- Nonce verified in `et_builder_security_check`.
|
||
|
|
||
|
if ( ! $post_id ) {
|
||
|
wp_send_json_error( 'Error: ID is required.' );
|
||
|
}
|
||
|
|
||
|
$post_type = get_post_type( $post_id );
|
||
|
|
||
|
if ( ! current_user_can( 'edit_post', $post_id ) || ET_TB_ITEM_POST_TYPE !== $post_type ) {
|
||
|
wp_send_json_error( 'You do not have permission.' );
|
||
|
}
|
||
|
|
||
|
wp_send_json_success( wp_delete_post( $post_id, true ) );
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_toggle_cloud_status', 'et_theme_builder_library_toggle_cloud_status' );
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Ajax action: Remove temp layouts, templates theme builder.
|
||
|
*/
|
||
|
function et_theme_builder_library_clear_temp_data() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_clear_temp_data', 'nonce' );
|
||
|
|
||
|
$args = array(
|
||
|
'post_status' => 'draft',
|
||
|
'post_type' => array( 'et_tb_item', 'et_header_layout', 'et_body_layout', 'et_footer_layout' ),
|
||
|
'author' => get_current_user_id(),
|
||
|
'fields' => 'ids',
|
||
|
'no_found_rows' => true,
|
||
|
'nopaging' => true,
|
||
|
);
|
||
|
|
||
|
$draft_query = new WP_Query( $args );
|
||
|
|
||
|
foreach ( $draft_query->posts as $draft_post_id ) {
|
||
|
$post_type = get_post_type( $draft_post_id );
|
||
|
|
||
|
if ( current_user_can( 'edit_post', $draft_post_id ) && ET_TB_ITEM_POST_TYPE === $post_type ) {
|
||
|
wp_delete_post( $draft_post_id );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
wp_send_json_success();
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_clear_temp_data', 'et_theme_builder_library_clear_temp_data' );
|
||
|
|
||
|
|
||
|
/**
|
||
|
* AJAX action: Gets Cloud access token from DB and send it to client.
|
||
|
*/
|
||
|
function et_theme_builder_library_get_cloud_token() {
|
||
|
et_builder_security_check( 'theme_builder', 'edit_others_posts', 'et_theme_builder_library_get_cloud_token', 'nonce' );
|
||
|
|
||
|
wp_send_json_success(
|
||
|
array(
|
||
|
'accessToken' => get_transient( 'et_cloud_access_token' ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
add_action( 'wp_ajax_et_theme_builder_library_get_cloud_token', 'et_theme_builder_library_get_cloud_token' );
|