<?php /** * Class for the customizer importer. * * Code is mostly from the Customizer Export/Import plugin. * * @see https://wordpress.org/plugins/customizer-export-import/ * * @package Merlin WP */ class Merlin_Customizer_Importer { /** * Import customizer from a DAT file, generated by the Customizer Export/Import plugin. * * @param string $customizer_import_file_path path to the customizer import file. */ public static function import( $customizer_import_file_path ) { // Try to import the customizer settings. $results = self::import_customizer_options( $customizer_import_file_path ); // Check for errors, else write the results to the log file. if ( is_wp_error( $results ) ) { Merlin_Logger::get_instance()->error( $results->get_error_message() ); return false; } Merlin_Logger::get_instance()->info( __( 'The customizer import has finished successfully', 'merlin-wp' ) ); return true; } /** * Imports uploaded mods and calls WordPress core customize_save actions so * themes that hook into them can act before mods are saved to the database. * * Update: WP core customize_save actions were removed, because of some errors. * * @since 1.1.1 * @param string $import_file_path Path to the import file. * @return WP_Error */ public static function import_customizer_options( $import_file_path ) { // Setup global vars. global $wp_customize; // Setup internal vars. $template = get_template(); // Make sure we have an import file. if ( ! file_exists( $import_file_path ) ) { return new \WP_Error( 'missing_cutomizer_import_file', sprintf( esc_html__( 'Error: The customizer import file is missing! File path: %s', 'merlin-wp' ), $import_file_path ) ); } // Get the upload data. $raw = file_get_contents( $import_file_path ); // Make sure we got the data. if ( empty( $raw ) ) { return new \WP_Error( 'customizer_import_data_missing_content', esc_html__( 'Error: The customizer import file does not have any content in it. Please make sure to use the correct customizer import file.', 'merlin-wp' ) ); } $data = unserialize( $raw ); // Data checks. if ( ! is_array( $data ) && ( ! isset( $data['template'] ) || ! isset( $data['mods'] ) ) ) { return new \WP_Error( 'customizer_import_data_error', esc_html__( 'Error: The customizer import file is not in a correct format. Please make sure to use the correct customizer import file.', 'merlin-wp' ) ); } if ( $data['template'] !== $template ) { return new \WP_Error( 'customizer_import_wrong_theme', esc_html__( 'Error: The customizer import file is not suitable for current theme. You can only import customizer settings for the same theme or a child theme.', 'merlin-wp' ) ); } // Import images. if ( apply_filters( 'merlin_customizer_import_images', true ) ) { $data['mods'] = self::import_customizer_images( $data['mods'] ); } // Import custom options. if ( isset( $data['options'] ) ) { // Require modified customizer options class. if ( ! class_exists( '\WP_Customize_Setting' ) ) { require_once ABSPATH . 'wp-includes/class-wp-customize-setting.php'; } foreach ( $data['options'] as $option_key => $option_value ) { $option = new Merlin_Customizer_Option( $wp_customize, $option_key, array( 'default' => '', 'type' => 'option', 'capability' => 'edit_theme_options', ) ); $option->import( $option_value ); } } // Should the customizer import use the WP customize_save* hooks? $use_wp_customize_save_hooks = apply_filters( 'merlin_enable_wp_customize_save_hooks', false ); if ( $use_wp_customize_save_hooks ) { do_action( 'customize_save', $wp_customize ); } // Loop through the mods and save the mods. foreach ( $data['mods'] as $key => $val ) { if ( $use_wp_customize_save_hooks ) { do_action( 'customize_save_' . $key, $wp_customize ); } set_theme_mod( $key, $val ); } if ( $use_wp_customize_save_hooks ) { do_action( 'customize_save_after', $wp_customize ); } return true; } /** * Helper function: Customizer import - imports images for settings saved as mods. * * @param array $mods An array of customizer mods. * @return array The mods array with any new import data. */ private static function import_customizer_images( $mods ) { foreach ( $mods as $key => $val ) { if ( self::customizer_is_image_url( $val ) ) { $data = self::customizer_sideload_image( $val ); if ( ! is_wp_error( $data ) ) { $mods[ $key ] = $data->url; // Handle header image controls. if ( isset( $mods[ $key . '_data' ] ) ) { $mods[ $key . '_data' ] = $data; update_post_meta( $data->attachment_id, '_wp_attachment_is_custom_header', get_stylesheet() ); } } } } return $mods; } /** * Helper function: Customizer import * Taken from the core media_sideload_image function and * modified to return an array of data instead of html. * * @param string $file The image file path. * @return array An array of image data. */ private static function customizer_sideload_image( $file ) { $data = new \stdClass(); if ( ! function_exists( 'media_handle_sideload' ) ) { require_once( ABSPATH . 'wp-admin/includes/media.php' ); require_once( ABSPATH . 'wp-admin/includes/file.php' ); require_once( ABSPATH . 'wp-admin/includes/image.php' ); } if ( ! empty( $file ) ) { // Set variables for storage, fix file filename for query strings. preg_match( '/[^\?]+\.(jpe?g|jpe|gif|png)\b/i', $file, $matches ); $file_array = array(); $file_array['name'] = basename( $matches[0] ); // Download file to temp location. $file_array['tmp_name'] = download_url( $file ); // If error storing temporarily, return the error. if ( is_wp_error( $file_array['tmp_name'] ) ) { return $file_array['tmp_name']; } // Do the validation and storage stuff. $id = media_handle_sideload( $file_array, 0 ); // If error storing permanently, unlink. if ( is_wp_error( $id ) ) { unlink( $file_array['tmp_name'] ); return $id; } // Build the object to return. $meta = wp_get_attachment_metadata( $id ); $data->attachment_id = $id; $data->url = wp_get_attachment_url( $id ); $data->thumbnail_url = wp_get_attachment_thumb_url( $id ); $data->height = $meta['height']; $data->width = $meta['width']; } return $data; } /** * Checks to see whether a string is an image url or not. * * @param string $string The string to check. * @return bool Whether the string is an image url or not. */ private static function customizer_is_image_url( $string = '' ) { if ( is_string( $string ) ) { if ( preg_match( '/\.(jpg|jpeg|png|gif)/i', $string ) ) { return true; } } return false; } }