595 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			595 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| if ( ! defined( 'ABSPATH' ) ) {
 | |
| 	exit; // Exit if accessed directly
 | |
| }
 | |
| 
 | |
| if ( ! class_exists( 'ACF_Admin_Tool_Export' ) ) :
 | |
| 
 | |
| 	class ACF_Admin_Tool_Export extends ACF_Admin_Tool {
 | |
| 
 | |
| 		/** @var string View context */
 | |
| 		var $view = '';
 | |
| 
 | |
| 
 | |
| 		/** @var array Export data */
 | |
| 		var $json = '';
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  initialize
 | |
| 		 *
 | |
| 		 *  This function will initialize the admin tool
 | |
| 		 *
 | |
| 		 *  @date    10/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function initialize() {
 | |
| 
 | |
| 			// vars
 | |
| 			$this->name  = 'export';
 | |
| 			$this->title = __( 'Export Field Groups', 'acf' );
 | |
| 
 | |
| 			// active
 | |
| 			if ( $this->is_active() ) {
 | |
| 				$this->title .= ' - ' . __( 'Generate PHP', 'acf' );
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  submit
 | |
| 		 *
 | |
| 		 *  This function will run when the tool's form has been submit
 | |
| 		 *
 | |
| 		 *  @date    10/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function submit() {
 | |
| 
 | |
| 			// vars
 | |
| 			$action = acf_maybe_get_POST( 'action' );
 | |
| 
 | |
| 			// download
 | |
| 			if ( $action === 'download' ) {
 | |
| 
 | |
| 				$this->submit_download();
 | |
| 
 | |
| 				// generate
 | |
| 			} elseif ( $action === 'generate' ) {
 | |
| 
 | |
| 				$this->submit_generate();
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  submit_download
 | |
| 		 *
 | |
| 		 *  description
 | |
| 		 *
 | |
| 		 *  @date    17/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function submit_download() {
 | |
| 
 | |
| 			// vars
 | |
| 			$json = $this->get_selected();
 | |
| 
 | |
| 			// validate
 | |
| 			if ( $json === false ) {
 | |
| 				return acf_add_admin_notice( __( 'No field groups selected', 'acf' ), 'warning' );
 | |
| 			}
 | |
| 
 | |
| 			// headers
 | |
| 			$file_name = 'acf-export-' . date( 'Y-m-d' ) . '.json';
 | |
| 			header( 'Content-Description: File Transfer' );
 | |
| 			header( "Content-Disposition: attachment; filename={$file_name}" );
 | |
| 			header( 'Content-Type: application/json; charset=utf-8' );
 | |
| 
 | |
| 			// return
 | |
| 			echo acf_json_encode( $json ) . "\r\n";
 | |
| 			die;
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  submit_generate
 | |
| 		 *
 | |
| 		 *  description
 | |
| 		 *
 | |
| 		 *  @date    17/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function submit_generate() {
 | |
| 
 | |
| 			// vars
 | |
| 			$keys = $this->get_selected_keys();
 | |
| 
 | |
| 			// validate
 | |
| 			if ( ! $keys ) {
 | |
| 				return acf_add_admin_notice( __( 'No field groups selected', 'acf' ), 'warning' );
 | |
| 			}
 | |
| 
 | |
| 			// url
 | |
| 			$url = add_query_arg( 'keys', implode( '+', $keys ), $this->get_url() );
 | |
| 
 | |
| 			// redirect
 | |
| 			wp_redirect( $url );
 | |
| 			exit;
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  load
 | |
| 		 *
 | |
| 		 *  description
 | |
| 		 *
 | |
| 		 *  @date    21/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function load() {
 | |
| 
 | |
| 			// active
 | |
| 			if ( $this->is_active() ) {
 | |
| 
 | |
| 				// get selected keys
 | |
| 				$selected = $this->get_selected_keys();
 | |
| 
 | |
| 				// add notice
 | |
| 				if ( $selected ) {
 | |
| 					$count = count( $selected );
 | |
| 					$text  = sprintf( _n( 'Exported 1 item.', 'Exported %s items.', $count, 'acf' ), $count );
 | |
| 					acf_add_admin_notice( $text, 'success' );
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  html
 | |
| 		 *
 | |
| 		 *  This function will output the metabox HTML
 | |
| 		 *
 | |
| 		 *  @date    10/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function html() {
 | |
| 
 | |
| 			// single (generate PHP)
 | |
| 			if ( $this->is_active() ) {
 | |
| 
 | |
| 				$this->html_single();
 | |
| 
 | |
| 				// archive
 | |
| 			} else {
 | |
| 
 | |
| 				$this->html_archive();
 | |
| 
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 * Renders the checkboxes to select items to export.
 | |
| 		 *
 | |
| 		 * @date 24/10/17
 | |
| 		 * @since 5.6.3
 | |
| 		 *
 | |
| 		 * @return void
 | |
| 		 */
 | |
| 		public function html_field_selection() {
 | |
| 			// Ensure `l10n_var_export` is always false at the point we're outputting the options.
 | |
| 			acf_update_setting( 'l10n_var_export', false );
 | |
| 			// Reset the field-groups store which may have been corrupted by export.
 | |
| 			$store = acf_get_store( 'field-groups' );
 | |
| 			if ( $store ) {
 | |
| 				$store->reset();
 | |
| 			}
 | |
| 
 | |
| 			$choices      = array();
 | |
| 			$selected     = $this->get_selected_keys();
 | |
| 			$field_groups = acf_get_internal_post_type_posts( 'acf-field-group' );
 | |
| 
 | |
| 			if ( $field_groups ) {
 | |
| 				foreach ( $field_groups as $field_group ) {
 | |
| 					$choices[ $field_group['key'] ] = esc_html( $field_group['title'] );
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			acf_render_field_wrap(
 | |
| 				array(
 | |
| 					'label'   => __( 'Select Field Groups', 'acf' ),
 | |
| 					'type'    => 'checkbox',
 | |
| 					'name'    => 'keys',
 | |
| 					'prefix'  => false,
 | |
| 					'value'   => $selected,
 | |
| 					'toggle'  => true,
 | |
| 					'choices' => $choices,
 | |
| 				)
 | |
| 			);
 | |
| 
 | |
| 			$choices    = array();
 | |
| 			$selected   = $this->get_selected_keys();
 | |
| 			$post_types = acf_get_internal_post_type_posts( 'acf-post-type' );
 | |
| 
 | |
| 			if ( $post_types ) {
 | |
| 				foreach ( $post_types as $post_type ) {
 | |
| 					$choices[ $post_type['key'] ] = esc_html( $post_type['title'] );
 | |
| 				}
 | |
| 
 | |
| 				acf_render_field_wrap(
 | |
| 					array(
 | |
| 						'label'   => __( 'Select Post Types', 'acf' ),
 | |
| 						'type'    => 'checkbox',
 | |
| 						'name'    => 'post_type_keys',
 | |
| 						'prefix'  => false,
 | |
| 						'value'   => $selected,
 | |
| 						'toggle'  => true,
 | |
| 						'choices' => $choices,
 | |
| 					)
 | |
| 				);
 | |
| 			}
 | |
| 
 | |
| 			$choices    = array();
 | |
| 			$selected   = $this->get_selected_keys();
 | |
| 			$taxonomies = acf_get_internal_post_type_posts( 'acf-taxonomy' );
 | |
| 
 | |
| 			if ( $taxonomies ) {
 | |
| 				foreach ( $taxonomies as $taxonomy ) {
 | |
| 					$choices[ $taxonomy['key'] ] = esc_html( $taxonomy['title'] );
 | |
| 				}
 | |
| 
 | |
| 				acf_render_field_wrap(
 | |
| 					array(
 | |
| 						'label'   => __( 'Select Taxonomies', 'acf' ),
 | |
| 						'type'    => 'checkbox',
 | |
| 						'name'    => 'taxonomy_keys',
 | |
| 						'prefix'  => false,
 | |
| 						'value'   => $selected,
 | |
| 						'toggle'  => true,
 | |
| 						'choices' => $choices,
 | |
| 					)
 | |
| 				);
 | |
| 			}
 | |
| 
 | |
| 			$choices       = array();
 | |
| 			$selected      = $this->get_selected_keys();
 | |
| 			$options_pages = acf_get_internal_post_type_posts( 'acf-ui-options-page' );
 | |
| 
 | |
| 			if ( $options_pages ) {
 | |
| 				foreach ( $options_pages as $options_page ) {
 | |
| 					$choices[ $options_page['key'] ] = esc_html( $options_page['title'] );
 | |
| 				}
 | |
| 
 | |
| 				acf_render_field_wrap(
 | |
| 					array(
 | |
| 						'label'   => __( 'Select Options Pages', 'acf' ),
 | |
| 						'type'    => 'checkbox',
 | |
| 						'name'    => 'ui_options_page_keys',
 | |
| 						'prefix'  => false,
 | |
| 						'value'   => $selected,
 | |
| 						'toggle'  => true,
 | |
| 						'choices' => $choices,
 | |
| 					)
 | |
| 				);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 * Renders the side panel for selecting ACF items to export via PHP.
 | |
| 		 *
 | |
| 		 * @date 21/10/17
 | |
| 		 * @since 5.6.3
 | |
| 		 *
 | |
| 		 * @return void
 | |
| 		 */
 | |
| 		public function html_panel_selection() {
 | |
| 			?>
 | |
| 			<div class="acf-panel acf-panel-selection">
 | |
| 				<?php $this->html_field_selection(); ?>
 | |
| 			</div>
 | |
| 			<?php
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 *  html_panel_settings
 | |
| 		 *
 | |
| 		 *  description
 | |
| 		 *
 | |
| 		 *  @date    21/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function html_panel_settings() {
 | |
| 
 | |
| 			?>
 | |
| 		<div class="acf-panel acf-panel-settings">
 | |
| 			<h3 class="acf-panel-title"><?php _e( 'Settings', 'acf' ); ?> <i class="dashicons dashicons-arrow-right"></i></h3>
 | |
| 			<div class="acf-panel-inside">
 | |
| 				<?php
 | |
| 
 | |
| 				/*
 | |
| 				acf_render_field_wrap(array(
 | |
| 					'label'     => __('Empty settings', 'acf'),
 | |
| 					'type'      => 'select',
 | |
| 					'name'      => 'minimal',
 | |
| 					'prefix'    => false,
 | |
| 					'value'     => '',
 | |
| 					'choices'   => array(
 | |
| 						'all'       => __('Include all settings', 'acf'),
 | |
| 						'minimal'   => __('Ignore empty settings', 'acf'),
 | |
| 					)
 | |
| 				));
 | |
| 				*/
 | |
| 
 | |
| 				?>
 | |
| 			</div>
 | |
| 		</div>
 | |
| 			<?php
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		/**
 | |
| 		 *  html_archive
 | |
| 		 *
 | |
| 		 *  description
 | |
| 		 *
 | |
| 		 *  @date    20/10/17
 | |
| 		 *  @since   5.6.3
 | |
| 		 *
 | |
| 		 *  @param   n/a
 | |
| 		 *  @return  n/a
 | |
| 		 */
 | |
| 
 | |
| 		function html_archive() {
 | |
| 
 | |
| 			?>
 | |
| 		<div class="acf-postbox-header">
 | |
| 			<h2 class="acf-postbox-title"><?php esc_html_e( 'Export', 'acf' ); ?></h2>
 | |
| 			<div class="acf-tip"><i tabindex="0" class="acf-icon acf-icon-help acf-js-tooltip" title="<?php esc_attr_e( 'Select the items you would like to export and then select your export method. Export As JSON to export to a .json file which you can then import to another ACF installation. Generate PHP to export to PHP code which you can place in your theme.', 'acf' ); ?>">?</i></div>
 | |
| 		</div>
 | |
| 		<div class="acf-postbox-inner">
 | |
| 			<div class="acf-fields">
 | |
| 				<?php $this->html_field_selection(); ?>
 | |
| 			</div>
 | |
| 			<p class="acf-submit acf-actions-strip">
 | |
| 				<button type="submit" name="action" class="acf-btn acf-button-primary" value="download"><?php _e( 'Export As JSON', 'acf' ); ?></button>
 | |
| 				<button type="submit" name="action" class="acf-btn acf-btn-secondary" value="generate"><?php _e( 'Generate PHP', 'acf' ); ?></button>
 | |
| 			</p>
 | |
| 		</div>
 | |
| 			<?php
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 * Renders the PHP export screen.
 | |
| 		 *
 | |
| 		 * @date 20/10/17
 | |
| 		 * @since 5.6.3
 | |
| 		 *
 | |
| 		 * @return void
 | |
| 		 */
 | |
| 		public function html_single() {
 | |
| 			?>
 | |
| 			<div class="acf-postbox-header">
 | |
| 				<h2 class="acf-postbox-title"><?php esc_html_e( 'Export - Generate PHP', 'acf' ); ?></h2>
 | |
| 				<i tabindex="0" class="acf-icon acf-icon-help acf-js-tooltip" title="<?php esc_attr_e( "The following code can be used to register a local version of the selected items. Storing field groups, post types, or taxonomies locally can provide many benefits such as faster load times, version control & dynamic fields/settings. Simply copy and paste the following code to your theme's functions.php file or include it within an external file, then deactivate or delete the items from the ACF admin.", 'acf' ); ?>">?</i>
 | |
| 			</div>
 | |
| 			<div class="acf-postbox-columns">
 | |
| 				<div class="acf-postbox-main">
 | |
| 					<?php $this->html_generate(); ?>
 | |
| 				</div>
 | |
| 				<div class="acf-postbox-side">
 | |
| 					<?php $this->html_panel_selection(); ?>
 | |
| 					<p class="acf-submit">
 | |
| 						<button type="submit" name="action" class="acf-btn" value="generate"><?php esc_html_e( 'Generate PHP', 'acf' ); ?></button>
 | |
| 					</p>
 | |
| 				</div>
 | |
| 			</div>
 | |
| 			<?php
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 * Generates the HTML for the PHP export functionality.
 | |
| 		 *
 | |
| 		 * @date    17/10/17
 | |
| 		 * @since   5.6.3
 | |
| 		 *
 | |
| 		 * @return void
 | |
| 		 */
 | |
| 		public function html_generate() {
 | |
| 			// Prevent default translation and fake __() within string.
 | |
| 			acf_update_setting( 'l10n_var_export', true );
 | |
| 
 | |
| 			$json      = $this->get_selected();
 | |
| 			$to_export = array();
 | |
| 
 | |
| 			// Sort by ACF post type first so we can wrap them in related functions.
 | |
| 			foreach ( $json as $post ) {
 | |
| 				$post_type = acf_determine_internal_post_type( $post['key'] );
 | |
| 
 | |
| 				if ( $post_type ) {
 | |
| 					$to_export[ $post_type ][] = $post;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			echo '<textarea id="acf-export-textarea" readonly="readonly">';
 | |
| 
 | |
| 			foreach ( $to_export as $post_type => $posts ) {
 | |
| 				if ( 'acf-field-group' === $post_type ) {
 | |
| 					echo "add_action( 'acf/include_fields', function() {\r\n";
 | |
| 					echo "\tif ( ! function_exists( 'acf_add_local_field_group' ) ) {\r\n\t\treturn;\r\n\t}\r\n\r\n";
 | |
| 				} elseif ( 'acf-post-type' === $post_type || 'acf-taxonomy' === $post_type ) {
 | |
| 					echo "add_action( 'init', function() {\r\n";
 | |
| 				} elseif ( 'acf-ui-options-page' === $post_type ) {
 | |
| 					echo "add_action( 'acf/init', function() {\r\n";
 | |
| 				}
 | |
| 
 | |
| 				$count = 0;
 | |
| 				foreach ( $posts as $post ) {
 | |
| 					if ( $count !== 0 ) {
 | |
| 						echo "\r\n";
 | |
| 					}
 | |
| 
 | |
| 					echo "\t" . acf_export_internal_post_type_as_php( $post, $post_type ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- esc_textarea() used earlier.
 | |
| 					$count++;
 | |
| 				}
 | |
| 
 | |
| 				if ( in_array( $post_type, array( 'acf-post-type', 'acf-taxonomy', 'acf-field-group', 'acf-ui-options-page' ), true ) ) {
 | |
| 					echo "} );\r\n\r\n";
 | |
| 				}
 | |
| 
 | |
| 				if ( 'acf-post-type' === $post_type ) {
 | |
| 					echo acf_export_enter_title_here( $posts ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- esc_textarea() used earlier.
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			echo '</textarea>';
 | |
| 			?>
 | |
| 			<p class="acf-submit">
 | |
| 				<a class="button" id="acf-export-copy"><?php _e( 'Copy to clipboard', 'acf' ); ?></a>
 | |
| 			</p>
 | |
| 			<script type="text/javascript">
 | |
| 			(function($){
 | |
| 				const $a = $('#acf-export-copy');
 | |
| 				const $textarea = $('#acf-export-textarea');
 | |
| 
 | |
| 				// Remove $a if 'copy' is not supported.
 | |
| 				if( !document.queryCommandSupported('copy') ) {
 | |
| 					return $a.remove();
 | |
| 				}
 | |
| 
 | |
| 				$a.on('click', function( e ){
 | |
| 					e.preventDefault();
 | |
| 
 | |
| 					$textarea.get(0).select();
 | |
| 
 | |
| 					try {
 | |
| 						var copy = document.execCommand('copy');
 | |
| 						if ( ! copy ) {
 | |
| 							return;
 | |
| 						}
 | |
| 
 | |
| 						acf.newTooltip({
 | |
| 							text: 		"<?php esc_html_e( 'Copied', 'acf' ); ?>",
 | |
| 							timeout:	250,
 | |
| 							target: 	$(this),
 | |
| 						});
 | |
| 					} catch (err) {
 | |
| 						// Do nothing.
 | |
| 					}
 | |
| 				});
 | |
| 			})(jQuery);
 | |
| 			</script>
 | |
| 			<?php
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 * Return an array of keys that have been selected in the export tool.
 | |
| 		 *
 | |
| 		 * @date 20/10/17
 | |
| 		 * @since 5.6.3
 | |
| 		 *
 | |
| 		 * @return array|bool
 | |
| 		 */
 | |
| 		public function get_selected_keys() {
 | |
| 			$key_names = array( 'keys', 'post_type_keys', 'taxonomy_keys', 'ui_options_page_keys' );
 | |
| 			$all_keys  = array();
 | |
| 
 | |
| 			foreach ( $key_names as $key_name ) {
 | |
| 				if ( $keys = acf_maybe_get_POST( $key_name ) ) {
 | |
| 					$all_keys = array_merge( $all_keys, (array) $keys );
 | |
| 				} elseif ( $keys = acf_maybe_get_GET( $key_name ) ) {
 | |
| 					$keys     = str_replace( ' ', '+', $keys );
 | |
| 					$keys     = explode( '+', $keys );
 | |
| 					$all_keys = array_merge( $all_keys, (array) $keys );
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if ( ! empty( $all_keys ) ) {
 | |
| 				return $all_keys;
 | |
| 			}
 | |
| 
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		/**
 | |
| 		 * Returns the JSON data for given $_POST args.
 | |
| 		 *
 | |
| 		 * @date  17/10/17
 | |
| 		 * @since 5.6.3
 | |
| 		 *
 | |
| 		 * @return array|bool
 | |
| 		 */
 | |
| 		public function get_selected() {
 | |
| 			$selected = $this->get_selected_keys();
 | |
| 			$json     = array();
 | |
| 
 | |
| 			if ( ! $selected ) {
 | |
| 				return false;
 | |
| 			}
 | |
| 
 | |
| 			foreach ( $selected as $key ) {
 | |
| 				$post_type = acf_determine_internal_post_type( $key );
 | |
| 				$post      = acf_get_internal_post_type( $key, $post_type );
 | |
| 
 | |
| 				if ( empty( $post ) ) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if ( 'acf-field-group' === $post_type ) {
 | |
| 					$post['fields'] = acf_get_fields( $post );
 | |
| 				}
 | |
| 
 | |
| 				$post   = acf_prepare_internal_post_type_for_export( $post, $post_type );
 | |
| 				$json[] = $post;
 | |
| 			}
 | |
| 
 | |
| 			return $json;
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	// initialize
 | |
| 	acf_register_admin_tool( 'ACF_Admin_Tool_Export' );
 | |
| 
 | |
| endif; // class_exists check
 | |
| 
 | |
| ?>
 |