This repository has been archived on 2022-06-23. You can view files and clone it, but cannot push or open issues or pull requests.
divi/includes/builder/module/helpers/OptionTemplate.php

732 lines
21 KiB
PHP
Raw Normal View History

2021-12-07 11:08:05 +00:00
<?php
/**
* Option Templates helper methods.
*
* @since 3.28
*
* Class ET_Builder_Module_Helper_OptionTemplate
*/
class ET_Builder_Module_Helper_OptionTemplate {
private $map = array();
private $templates = array();
private $data = array();
private $cache = array();
private $tab_slug_map = array();
public $template_prefix = '%t';
protected static $_ = null;
public static function instance() {
static $instance;
return $instance ? $instance : $instance = new self();
}
private function __construct() {
self::$_ = ET_Core_Data_Utils::instance();
}
private function uniq( $prefix, $content ) {
$key = md5( $prefix . serialize( $content ) );
if ( isset( $this->map[ $key ] ) ) {
return $this->map[ $key ];
}
return ( $this->map[ $key ] = $this->template_prefix . $key );
}
/**
* Determine whether option template is enabled on current request or not
*
* @since 3.28
*
* @return bool
*/
public function is_enabled() {
// Option template tends to be enabled on most request to speed up performance
$status = true;
// Option template is disabled on:
// 1. AJAX request for fetching classic builder (BB)'s module data. BB data is shipped as
// optimized template markup which is rendered on server then sent as string. Hence
// Option Template's sent-config-rebuild-on-js won't be usable for BB
// 2. BB's editing page. BB edit page scans for field dependency and generates visibility
// setting on `window.et_pb_module_field_dependencies` variable for field depency thus
// actual field should be rendered here instead of templateId
if ( et_builder_is_loading_bb_data() || et_builder_is_bb_page() ) {
$status = false;
}
/**
* Filters option template status
*
* @since 3.28
*
* @param bool $status
*/
return apply_filters( 'et_builder_option_template_is_active', $status );
}
/**
* Determine whether given field name is option template field based on its first two characters
*
* @since 3.28
*
* @return bool
*/
public function is_option_template_field( $field_name = '' ) {
return $this->template_prefix === substr( $field_name, 0, 2 );
}
public function has( $key ) {
return isset( $this->templates[ $key ] );
}
public function add( $key, $template ) {
$fields_template = array_merge( $template, et_pb_responsive_options()->create( $template ) );
// Populate tab_slug of given template because advance fields can be rendered on any tab
foreach ( $fields_template as $field_name => $field ) {
if ( isset( $field['tab_slug'] ) ) {
$tab_slug = '' === $field['tab_slug'] ? 'advanced' : $field['tab_slug'];
if ( ! isset( $this->tab_slug_map[ $tab_slug ] ) ) {
$this->tab_slug_map[ $tab_slug ] = array( $key );
continue;
}
if ( ! in_array( $key, $this->tab_slug_map[ $tab_slug ] ) ) {
$this->tab_slug_map[ $tab_slug ][] = $key;
}
}
}
$this->templates[ $key ] = $fields_template;
}
public function create( $key, $config, $return_template_id = false ) {
$data = array( $key, $config );
$id = $this->uniq( $key, $data );
// Alternative, this will save the values directly in the Module $this->unprocessed_fields
// instead of this Calls $this->data and hence require a simpler logic.
// Theoretically it should require more memory but blackfire begs to differ.
$this->data[ $id ] = $data;
// Return as template id instead of id => key if needed
if ( $return_template_id ) {
return $id;
}
return array( $id => $key );
}
/**
* Create placeholders for template's params
*
* @return string[]
*/
public function placeholders( $config, $idx = 1, $path = array() ) {
$placeholders = array();
foreach ( $config as $key => $value ) {
if ( is_array( $value ) ) {
// Prepend current key as path so placeholder later can correctly fetch correct
// value from template data using dot notation path (both lodash get() or utils's
// array_get() support this).
$path[] = $key;
$value = $this->placeholders( $value, $idx, $path );
} else {
// Prepend dot notation path as prefix if needed
$prefix = empty( $path ) ? '' : implode( '.', $path ) . '.';
$value = "%%{$prefix}{$key}%%";
}
$placeholders[ $key ] = $value;
$idx++;
}
return $placeholders;
}
/**
* Get module's data
*
* @return array[]
*/
public function all() {
return $this->data;
}
/**
* Get templates
*
* @since 3.28
*
* @return array
*/
public function templates() {
return $this->templates;
}
/**
* Set `$this->data` property from external source (ie: static field definition cache).
*
* @since 3.28
*
* @param array $cached_data
*/
public function set_data( $cached_data = array() ) {
$this->data = wp_parse_args(
$cached_data,
$this->data
);
}
/**
* Set `$this->templates` property from external source (ie: static field definition cache).
*
* @since 3.28
*
* @param array $cached_template
*/
public function set_templates( $cached_templates = array() ) {
$this->templates = wp_parse_args(
$cached_templates,
$this->templates
);
}
/**
* Set `$this->tab_slug_map` from external source (ie: static field definition cache).
*
* @since 3.29
*
* @param array $cached_tab_slug_map
*/
public function set_tab_slug_map( $cached_tab_slug_map = array() ) {
$this->tab_slug_map = wp_parse_args(
$cached_tab_slug_map,
$this->tab_slug_map
);
}
/**
* Get template data based on given template id
*
* @since 3.28
*
* @param string $template_id
*
* @return array
*/
public function get_data( $template_id = '' ) {
return isset( $this->data[ $template_id ] ) ? $this->data[ $template_id ] : array();
}
/**
* Get template based on given template type
*
* @since 3.28
*
* @param string $type
*
* @return array
*/
public function get_template( $type = '' ) {
return isset( $this->templates[ $type ] ) ? $this->templates[ $type ] : array();
}
/**
* Get hashed cache key based on params given
*
* @since 3.28
*
* @param mixed $params
*
* @return string
*/
public function get_cache_key( $params ) {
$params = is_string( $params ) ? $params : serialize( $params );
return md5( $params );
}
/**
* Get cached value
* Return null if no cached value found
*
* @since 3.28
*
* @param string $name
* @param string $key
*
* @return mixed
*/
public function get_cache( $name, $key ) {
if (
! empty( $this->cache )
&& ! empty( $this->cache[ $name ] )
&& ! empty( $this->cache[ $name ][ $key ] )
) {
return $this->cache[ $name ][ $key ];
}
return null;
}
/**
* Set value to be cached
*
* @since 3.28
*
* @param string $name
* @param string $key
* @param mixed $value
*
* @return void
*/
public function set_cache( $name, $key, $value ) {
self::$_->array_set( $this->cache, "{$name}.{$key}", $value );
}
/**
* Get placeholder of given template
*
* @since 3.28
*
* @param string $template
*
* @return array|bool
*/
public function get_template_placeholder( $template ) {
// Check for cached result first for faster performance
$cache_name = 'template_placeholder';
$cache_key = $this->get_cache_key( $template );
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
preg_match( '/(?<=%%).*(?=%%)/', $template, $placeholder );
// Cache result
$this->set_cache( $cache_name, $cache_key, $placeholder );
return $placeholder;
}
/**
* Get tab slug maps
*
* @since 3.29
*
* @return array
*/
public function get_tab_slug_map() {
return $this->tab_slug_map;
}
public function is_template_inside_tab( $tab_name, $template_type ) {
// Template which has `%%tab_slug%%` tab_slug can exist on any tab
if ( in_array( $template_type, self::$_->array_get( $this->tab_slug_map, '%%tab_slug%%', array() ) ) ) {
return true;
}
if ( in_array( $template_type, self::$_->array_get( $this->tab_slug_map, $tab_name, array() ) ) ) {
return true;
}
return false;
}
public function rebuild_string_placeholder( $template, $data = array(), $settings = array() ) {
// Placeholder settings
$default_settings = array(
'suffix' => '',
'remove_suffix_if_empty' => false,
);
$placeholder_settings = wp_parse_args( $settings, $default_settings );
// Check for cached result first for faster performance
$cache_name = 'string_placeholder';
$cache_key = $this->get_cache_key( array( $template, $data, $placeholder_settings ) );
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
// Get placeholder
$placeholder = is_string( $template ) ? $this->get_template_placeholder( $template ) : false;
// If found, replace placeholder with correct value from data
if ( is_array( $placeholder ) && isset( $placeholder[0] ) ) {
// Get placeholder replacement
$replacement = ! empty( $data[1] ) && ! empty( $data[1][ $placeholder[0] ] ) ? $data[1][ $placeholder[0] ] : '';
// Pass null as empty string; null as attribute affect builder differently.
// Attribute with empty string will be omitted later.
if ( is_null( $replacement ) ) {
$replacement = '';
}
// If placeholder is identical to template, return replacement early. This also
// handles the case where replacement as array type
if ( "%%{$placeholder[0]}%%" === $template ) {
// Cache result
$this->set_cache( $cache_name, $cache_key, $replacement );
return $replacement;
}
// Get placeholder suffix
$has_suffix = '' === $replacement && $placeholder_settings['remove_suffix_if_empty'];
$suffix = $has_suffix ? $placeholder_settings['suffix'] : '';
// Make sure replacement is string before proceed;
if ( is_string( $replacement ) ) {
$rebuilt_string = str_replace( "%%{$placeholder[0]}%%{$suffix}", $replacement, $template );
// Cache result
$this->set_cache( $cache_name, $cache_key, $rebuilt_string );
return $rebuilt_string;
}
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $template );
return $template;
}
public function rebuild_preset_placeholder( $template, $data = array(), $settings = array() ) {
// Check for cached result first for faster performance
$cache_name = 'preset_placeholder';
$cache_key = $this->get_cache_key( array( $template, $data, $settings ) );
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
$rebuild_attr = array();
foreach ( $template as $preset_attr_key => $preset_attr_value ) {
// Object inside preset array mostly contains fields attribute which its object key
// contains placeholder while its object value contains actual value without placeholder.
if ( is_array( $preset_attr_value ) ) {
$rebuilt_preset_attr_object = array();
foreach ( $preset_attr_value as $name => $value ) {
$object_item_name = $this->rebuild_string_placeholder( $name, $data, $settings );
$rebuilt_preset_attr_object[ $object_item_name ] = $value;
}
$rebuild_attr[ $preset_attr_key ] = $rebuilt_preset_attr_object;
continue;
}
$rebuild_attr[ $preset_attr_key ] = $preset_attr_value;
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $rebuild_attr );
return $rebuild_attr;
}
public function rebuild_composite_structure_placeholder( $template_type, $template, $data = array() ) {
// Check for cached result first for faster performance
$cache_name = 'composite_structure_placeholder';
$cache_key = $this->get_cache_key( array( $template_type, $template, $data ) );
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
$rebuilt_composite_structure_field = $template;
// Replaces placeholder with actual value on border's nested composite structure fields
if ( 'border' === $template_type ) {
// Reset `controls` attribute output
$rebuilt_composite_structure_field['controls'] = array();
// Loop composite structure's original `controls` from template
foreach ( $template['controls'] as $field_name => $field ) {
$rebuilt_field_name = $this->rebuild_string_placeholder( $field_name, $data );
$rebuilt_field = $field;
// Loop field on composite structure controls
foreach ( $rebuilt_field as $attr_name => $attr_value ) {
$settings = array(
'suffix' => 'label' === $attr_name ? ' ' : '',
'remove_suffix_if_empty' => 'label' === $attr_name,
);
$rebuilt_field[ $attr_name ] = $this->rebuild_string_placeholder( $attr_value, $data, $settings );
}
$rebuilt_composite_structure_field['controls'][ $rebuilt_field_name ] = $rebuilt_field;
}
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $rebuilt_composite_structure_field );
return $rebuilt_composite_structure_field;
}
public function rebuild_field_attr_value( $attr_name, $attr_value, $template_data ) {
// Check for cached result first for faster performance
$cache_name = 'field_attr_value';
$cache_key = $this->get_cache_key( array( $attr_name, $attr_value, $template_data ) );
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
$template_type = ! empty( $template_data[0] ) ? $template_data[0] : '';
$prefix = ! empty( $template_data[1]['prefix'] ) && ! empty( $template_data[1]['prefix'] ) ? $template_data[1]['prefix'] : '';
// Certain advanced field (ie. Text Shadow) automatically adds underscore
$auto_add_prefix_underscore = isset( $template_data[0] ) && 'text_shadow' === $template_data[0] && '' === $prefix;
// 1. Field attribute value's type is string
if ( is_string( $attr_value ) ) {
$placeholder_has_space_suffix = 'label' === $attr_name && in_array( $template_type, array( 'border', 'text_shadow' ) );
$settings = array(
'suffix' => $placeholder_has_space_suffix ? ' ' : '',
'remove_suffix_if_empty' => $placeholder_has_space_suffix ? true : false,
);
$rebuilt_placeholder = $this->rebuild_string_placeholder( $attr_value, $template_data, $settings );
// Cache result
$this->set_cache( $cache_name, $cache_key, $rebuilt_placeholder );
return $rebuilt_placeholder;
}
// 2. Field attribute value's type is array (sequential)
if ( is_array( $attr_value ) && isset( $attr_value[0] ) ) {
$rebuild_attr_value = array();
foreach ( $attr_value as $array_value ) {
// Array consists of string is most likely used for defining field relationship
// such as `show_if` attribute; Replace prefix and suffix placeholder with
// placeholder replacement also consider that text_shadow advanced field
// automatically adds underscore after prefix so it needs to be adjusted as well
if ( is_string( $array_value ) ) {
$settings = array(
'suffix' => '_',
'remove_suffix_if_empty' => $auto_add_prefix_underscore,
);
$rebuild_attr_value[] = $this->rebuild_string_placeholder( $array_value, $template_data, $settings );
} elseif ( 'presets' === $attr_name ) {
// Handle preset attribute specifically due to how it is structured
$settings = array(
'suffix' => '_',
'remove_suffix_if_empty' => $auto_add_prefix_underscore,
);
$rebuild_attr_value[] = $this->rebuild_preset_placeholder( $array_value, $template_data, $settings );
} else {
// Non string and `presets` attribute less likely contains placeholder
$rebuild_attr_value[] = $array_value;
}
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $rebuild_attr_value );
return $rebuild_attr_value;
}
// 3. Field attribute value's type is array (associative)
if ( is_array( $attr_value ) && ! isset( $attr_value[0] ) ) {
$attr_object = array();
// Loop existing attrValue and populate the rebuilt result on `attrObject`.
foreach ( $attr_value as $item_key => $item_value ) {
$attr_object_key = $this->rebuild_string_placeholder( $item_key, $template_data );
// Replaces placeholder with actual value on border's nested composite structure fields
if ( 'composite_structure' === $attr_name ) {
$item_value = $this->rebuild_composite_structure_placeholder( $template_type, $item_value, $template_data );
}
$attr_object[ $attr_object_key ] = $item_value;
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $attr_object );
return $attr_object;
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $attr_value );
// 4. Unknown attribute value type; directly pass it
return $attr_value;
}
public function rebuild_field_template( $template_id, $parent_template_id = false ) {
// Check for cached result first for faster performance
$cache_name = 'field_template';
$cache_key = $parent_template_id ? "{$template_id}-inherits-{$parent_template_id}" : $template_id;
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
$fields = array();
$template_data = $this->get_data( $template_id );
// No fields will be found without template data. Return early;
if ( empty( $template_data ) ) {
return $fields;
}
$template_type = ! empty( $template_data[0] ) ? $template_data[0] : '';
$template_settings = ! empty( $template_data[1] ) ? $template_data[1] : array();
$prefix = ! empty( $template_settings['prefix'] ) ? $template_settings['prefix'] : '';
$parent_template_data = false;
// If rebuilt parent template is inside another templateId (ie. Text Shadow inside Font)
// its placeholder is passed for data; The expected structure becomes
// `[templateType, parentTemplateSettings]` instead of `[templateType, templateSettings]`;
// Thus get parent template's settings and use it
if ( $parent_template_id ) {
$parent_template_data = $this->get_data( $parent_template_id );
$parent_template_settings = ! empty( $parent_template_data[1] ) ? $parent_template_data[1] : true;
$template_settings_inherits_from_parant = array();
foreach ( $template_settings as $name => $value ) {
$placeholder = $this->get_template_placeholder( $value );
if ( is_array( $placeholder ) && isset( $placeholder[0] ) ) {
$template_settings_inherits_from_parant[ $name ] = self::$_->array_get(
$parent_template_settings,
$placeholder[0],
$value
);
}
}
$parent_template_data = array(
$template_type,
$template_settings_inherits_from_parant,
);
}
// Get fields template for given template type
$fields_template = $this->get_template( $template_type );
// Loop fields template and replace placeholder with actual value
foreach ( $fields_template as $field_name_template => $field_template ) {
// Certain advanced field (ie. Text Shadow) automatically adds underscore
// Related template type needs to be adjusted
$remove_suffix_if_empty = 'text_shadow' === $template_type && '' === $prefix;
// Replace field attribute name's placeholder
$field_template_data = $parent_template_id ? $parent_template_data : $template_data;
$field_name = $this->rebuild_string_placeholder(
$field_name_template,
$field_template_data,
array(
'remove_suffix_if_empty' => $remove_suffix_if_empty,
// placeholder's suffix, not placeholder named %%suffix%%
'suffix' => '_',
)
);
// Replace field attribute value's placeholder
$field = array();
if ( is_array( $field_template ) ) {
foreach ( $field_template as $attr_name => $attr_value ) {
$rebuilt_attr_value = $this->rebuild_field_attr_value(
$attr_name,
$attr_value,
$field_template_data
);
// Omit attribute with empty value: existance of attribute even with empty
// string value is handled differently in many field (ie, `show_if`)
if ( '' === $rebuilt_attr_value ) {
continue;
}
$field[ $attr_name ] = $rebuilt_attr_value;
}
} else {
$fields = array_merge(
$fields,
$this->rebuild_field_template( $field_name_template, $template_id )
);
}
// `name` attribute is dynamically added based on field's array key
$field['name'] = $field_name;
// Populate rebuilt field
$fields[ $field_name ] = $field;
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $fields );
return $fields;
}
public function rebuild_default_props( $template_id ) {
// Check for cached result first for faster performance
$cache_name = 'default_props';
$cache_key = $template_id;
$cache = $this->get_cache( $cache_name, $cache_key );
if ( ! is_null( $cache ) ) {
return $cache;
}
$default_props = array();
$rebuilt_fields = $this->rebuild_field_template( $template_id );
foreach ( $rebuilt_fields as $field_name => $field ) {
$value = '';
if ( isset( $field['composite_type'], $field['composite_structure'] ) ) {
require_once ET_BUILDER_DIR . 'module/field/attribute/composite/Parser.php';
$composite_atts = ET_Builder_Module_Field_Attribute_Composite_Parser::parse( $field['composite_type'], $field['composite_structure'] );
$default_props = array_merge( $default_props, $composite_atts );
} else {
if ( isset( $field['default_on_front'] ) ) {
$value = $field['default_on_front'];
} elseif ( isset( $field['default'] ) ) {
$value = $field['default'];
}
$default_props[ $field_name ] = $value;
}
}
// Cache result
$this->set_cache( $cache_name, $cache_key, $default_props );
return $default_props;
}
}