'layout',
'slug' => 'layout',
'domain' => 'layout_type',
);
$post['terms'][] = array(
'name' => 'not_global',
'slug' => 'not_global',
'domain' => 'scope',
);
}
$update_built_for_post_type = true;
// check whether _et_pb_built_for_post_type custom field exists.
if ( ! empty( $post['postmeta'] ) ) {
foreach ( $post['postmeta'] as $index => $value ) {
if ( '_et_pb_built_for_post_type' === $value['key'] ) {
$update_built_for_post_type = false;
}
}
}
}
// set _et_pb_built_for_post_type value to 'page' if not exists.
if ( $update_built_for_post_type ) {
$post['postmeta'][] = array(
'key' => '_et_pb_built_for_post_type',
'value' => 'page',
);
}
$processed_posts[] = $post;
}
}
return $processed_posts;
}
add_filter( 'wp_import_posts', 'et_update_old_layouts_taxonomy', 10 );
if ( ! function_exists( 'et_pb_add_layout_filters' ) ) :
/**
* Add custom filters for posts in the Divi Library.
*/
function et_pb_add_layout_filters() {
// phpcs:disable WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
if ( isset( $_GET['post_type'] ) && 'et_pb_layout' === $_GET['post_type'] ) {
$layout_categories = get_terms( 'layout_category' );
$filter_category = array();
$filter_category[''] = esc_html__( 'All Categories', 'et_builder' );
if ( is_array( $layout_categories ) && ! empty( $layout_categories ) ) {
foreach ( $layout_categories as $category ) {
$filter_category[ $category->slug ] = $category->name;
}
}
$layout_packs = get_terms( 'layout_pack' );
$filter_pack = array();
$filter_pack[''] = esc_html_x( 'All Packs', 'Layout Packs', 'et_builder' );
if ( is_array( $layout_packs ) ) {
foreach ( $layout_packs as $pack ) {
$filter_pack[ $pack->slug ] = $pack->name;
}
}
$filter_layout_type = array(
'' => esc_html__( 'All Types', 'et_builder' ),
'module' => esc_html__( 'Modules', 'et_builder' ),
'row' => esc_html__( 'Rows', 'et_builder' ),
'section' => esc_html__( 'Sections', 'et_builder' ),
'layout' => esc_html__( 'Layouts', 'et_builder' ),
);
$filter_scope = array(
'' => esc_html__( 'All Scopes', 'et_builder' ),
'global' => esc_html__( 'Global', 'et_builder' ),
'not_global' => esc_html__( 'Not Global', 'et_builder' ),
);
?>
$label ) {
printf(
'%3$s ',
esc_attr( $value ),
selected( $value, $selected ),
esc_html( $label )
);
}
?>
$label ) {
printf(
'%3$s ',
esc_attr( $value ),
selected( $value, $selected ),
esc_html( $label )
);
}
?>
$label ) {
printf(
'%3$s ',
esc_attr( $value ),
selected( $value, $selected ),
esc_html( $label )
);
}
?>
$label ) {
printf(
'%3$s ',
esc_attr( $value ),
selected( $value, $selected ),
esc_html( $label )
);
}
?>
id ) {
// display wp error screen if library is disabled for current user.
if ( ! et_pb_is_allowed( 'divi_library' ) || ! et_pb_is_allowed( 'add_library' ) || ! et_pb_is_allowed( 'save_library' ) ) {
wp_die( esc_html__( "you don't have sufficient permissions to access this page", 'et_builder' ) );
}
add_action( 'all_admin_notices', 'et_pb_export_layouts_interface' );
}
}
endif;
add_action( 'load-edit.php', 'et_pb_load_export_section' );
if ( ! function_exists( 'et_pb_edit_library_categories' ) ) :
/**
* Enqueue script on Library Categories editing screen.
*/
function et_pb_edit_library_categories() {
$current_screen = get_current_screen();
if ( 'edit-layout_category' === $current_screen->id || 'edit-layout_pack' === $current_screen->id ) {
// display wp error screen if library is disabled for current user.
if ( ! et_pb_is_allowed( 'divi_library' ) || ! et_pb_is_allowed( 'add_library' ) || ! et_pb_is_allowed( 'save_library' ) ) {
wp_die( esc_html__( "you don't have sufficient permissions to access this page", 'et_builder' ) );
}
wp_enqueue_script( 'builder-library-category', ET_BUILDER_URI . '/scripts/library_category.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
}
}
endif;
add_action( 'load-edit-tags.php', 'et_pb_edit_library_categories' );
/**
* Check whether the library editor page should be displayed or not.
*/
function et_pb_check_library_permissions() {
$current_screen = get_current_screen();
if ( 'et_pb_layout' === $current_screen->id && ( ! et_pb_is_allowed( 'divi_library' ) || ! et_pb_is_allowed( 'save_library' ) ) ) {
// display wp error screen if library is disabled for current user.
wp_die( esc_html__( "you don't have sufficient permissions to access this page", 'et_builder' ) );
}
}
add_action( 'load-post.php', 'et_pb_check_library_permissions' );
if ( ! function_exists( 'exclude_premade_layouts_library' ) ) :
/**
* Exclude premade layouts from the list of all templates in the library.
*
* @param WP_Query $query Query.
*/
function exclude_premade_layouts_library( $query ) {
// phpcs:disable WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
global $pagenow;
$current_post_type = get_query_var( 'post_type' );
if ( is_admin() && 'edit.php' === $pagenow && $current_post_type && 'et_pb_layout' === $current_post_type ) {
$meta_query = array(
array(
'key' => '_et_pb_predefined_layout',
'value' => 'on',
'compare' => 'NOT EXISTS',
),
);
$used_built_for_post_types = ET_Builder_Library::built_for_post_types( 'all' );
if ( isset( $_GET['built_for'] ) && count( $used_built_for_post_types ) > 1 ) {
$built_for_post_type = sanitize_text_field( $_GET['built_for'] );
// get array of all standard post types if built_for is one of them.
$built_for_post_type_processed = in_array( $built_for_post_type, ET_Builder_Library::built_for_post_types(), true ) ? ET_Builder_Library::built_for_post_types() : $built_for_post_type;
if ( in_array( $built_for_post_type, $used_built_for_post_types, true ) ) {
$meta_query[] = array(
'key' => '_et_pb_built_for_post_type',
'value' => $built_for_post_type_processed,
'compare' => 'IN',
);
}
}
$query->set( 'meta_query', $meta_query );
//phpcs:enable
}
return $query;
}
endif;
add_action( 'pre_get_posts', 'exclude_premade_layouts_library' );
if ( ! function_exists( 'exclude_premade_layouts_library_count' ) ) :
/**
* Post count for "mine" in post table relies to fixed value set by WP_Posts_List_Table->user_posts_count
* Thus, exclude_premade_layouts_library() action doesn't automatically exclude premade layout and
* it has to be late filtered via this exclude_premade_layouts_library_count().
*
* @see WP_Posts_List_Table->user_posts_count to see how mine post value is retrieved.
*
* @param array $views All views in post list table.
* @return array
*/
function exclude_premade_layouts_library_count( $views ) {
if ( isset( $views['mine'] ) ) {
$current_user_id = get_current_user_id();
if ( isset( $_GET['author'] ) && ( $_GET['author'] === $current_user_id ) ) { // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
$class = 'current';
// Reuse current $wp_query global.
global $wp_query;
$mine_posts_count = $wp_query->found_posts;
} else {
$class = '';
// Use WP_Query instead of plain MySQL SELECT because the custom field filtering uses
// GROUP BY which needs FOUND_ROWS() and this has been automatically handled by WP_Query.
$query = new WP_Query(
array(
'post_type' => 'et_pb_layout',
'author' => $current_user_id,
'meta_query' => array(
'key' => '_et_pb_predefined_layout',
'value' => 'on',
'compare' => 'NOT EXISTS',
),
)
);
$mine_posts_count = $query->found_posts;
}
$url = add_query_arg(
array(
'post_type' => 'et_pb_layout',
'author' => $current_user_id,
),
'edit.php'
);
$views['mine'] = sprintf(
'%3$s (%4$s) ',
esc_url( $url ),
esc_attr( $class ),
esc_html__( 'Mine', 'et_builder' ),
esc_html( intval( $mine_posts_count ) )
);
}
return $views;
}
endif;
add_filter( 'views_edit-et_pb_layout', 'exclude_premade_layouts_library_count' );
if ( ! function_exists( 'et_pb_get_standard_post_types' ) ) :
/**
* Returns the standard '_et_pb_built_for_post_type' post types.
*
* @deprecated {@see ET_Builder_Post_Type_Layout::get_built_for_post_types()}
*
* @since 3.1 Deprecated.
* @since 1.8
*
* @return string[]
*/
function et_pb_get_standard_post_types() {
return ET_Builder_Library::built_for_post_types();
}
endif;
if ( ! function_exists( 'et_pb_get_used_built_for_post_types' ) ) :
/**
* Returns all current '_et_pb_built_for_post_type' post types.
*
* @deprecated {@see ET_Builder_Post_Type_Layout::get_built_for_post_types()}
*
* @since 3.1 Deprecated.
* @since 1.8
*
* @return string[]
*/
function et_pb_get_used_built_for_post_types() {
return ET_Builder_Library::built_for_post_types( 'all' );
}
endif;
if ( ! function_exists( 'et_pb_get_font_icon_symbols' ) ) :
/**
* Return fon icon symbols.
*/
function et_pb_get_font_icon_symbols() {
$symbols = array( '!', '"', '#', '$', '%', '&', ''', '(', ')', '*', '+', ',', '-', '.', '/', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', ':', ';', '<', '=', '>', '?', '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', '[', '\', ']', '^', '_', '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '{', '|', '}', '~', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '' );
$symbols = apply_filters( 'et_pb_font_icon_symbols', $symbols );
return $symbols;
}
endif;
if ( ! function_exists( 'et_pb_get_font_icon_list' ) ) :
/**
* Font icon list.
*/
function et_pb_get_font_icon_list() {
$output = is_customize_preview() ? et_pb_get_font_icon_list_items() : '<%= window.et_builder.font_icon_list_template() %>';
$output = sprintf( '
" . $output . '
';
return apply_filters( 'et_builder_include_categories_option_html', $output );
}
endif;
if ( ! function_exists( 'et_builder_include_categories_shop_option' ) ) :
/**
* Generate output string for `include_shop_categories` option used in backbone template.
*
* @param array $args arguments to get shop categories.
* @return string
*/
function et_builder_include_categories_shop_option( $args = array() ) {
if ( ! class_exists( 'WooCommerce' ) ) {
return '';
}
$output = "\t<% var et_pb_include_categories_shop_temp = typeof data !== 'undefined' && typeof data.et_pb_include_categories !== 'undefined' ? data.et_pb_include_categories.split( ',' ) : []; et_pb_include_categories_shop_temp = typeof data === 'undefined' && typeof et_pb_include_categories !== 'undefined' ? et_pb_include_categories.split( ',' ) : et_pb_include_categories_shop_temp; %>\n";
$product_categories = et_builder_get_shop_categories( $args );
$output .= '';
if ( is_array( $product_categories ) && ! empty( $product_categories ) ) {
foreach ( $product_categories as $category ) {
if ( is_object( $category ) && is_a( $category, 'WP_Term' ) ) {
$contains = sprintf(
'<%%= _.contains( et_pb_include_categories_shop_temp, "%1$s" ) ? checked="checked" : "" %%>',
esc_html( $category->term_id )
);
$output .= sprintf(
'%4$s %2$s ',
esc_attr( $category->term_id ),
esc_html( $category->name ),
$contains,
"\n\t\t\t\t\t"
);
}
}
}
$output .= '
';
return apply_filters( 'et_builder_include_categories_option_html', $output );
}
endif;
if ( ! function_exists( 'et_divi_get_projects' ) ) :
/**
* Return projects.
*
* @param array $args WP_Query arguments.
*/
function et_divi_get_projects( $args = array() ) {
$default_args = array(
'post_type' => 'project',
);
$args = wp_parse_args( $args, $default_args );
return new WP_Query( $args );
}
endif;
if ( ! function_exists( 'et_pb_extract_items' ) ) :
/**
* Return pricing table items html.
*
* @param string $content Content.
*/
function et_pb_extract_items( $content ) {
$output = '';
$first_character = '';
$lines = array_filter( explode( "\n", str_replace( array( '',
( $is_builder_used ? ' et_pb_builder_is_used' : '' ),
et_core_esc_previously( $buttons ),
( $is_builder_used ? ' class="et_pb_post_body_hidden"' : '' ),
( et_builder_bfb_enabled() ? ' style="opacity: 0;"' : '' )
);
} else {
printf(
'
',
( $is_builder_used ? ' class="et_pb_post_body_hidden"' : '' ),
( $is_builder_used ? ' et_pb_builder_is_used' : '' ),
( et_builder_bfb_enabled() ? ' style="opacity: 0;"' : '' )
);
}
if ( ! et_builder_bfb_enabled() ) {
$module_fields_dependencies = wp_json_encode( ET_Builder_Element::get_field_dependencies( $post->post_type ) );
echo et_core_esc_previously(
"
"
);
}
?>
ID ); ?>
ID ); ?>
ID ) ) {
return;
}
echo '
';
}
/**
* Setup Divi Builder in BFB.
*/
function et_pb_setup_main_editor() {
if ( ! et_core_is_gutenberg_enabled() ) {
add_action( 'edit_form_after_title', 'et_pb_before_main_editor' );
add_action( 'edit_form_after_editor', 'et_pb_after_main_editor' );
}
}
add_action( 'add_meta_boxes', 'et_pb_setup_main_editor', 11 );
/**
* Load scripts and styles in admin.
*
* @param string $hook The current admin page.
*/
function et_pb_admin_scripts_styles( $hook ) {
global $typenow, $pagenow;
// load css file for the Divi menu.
wp_enqueue_style( 'library-menu-styles', ET_BUILDER_URI . '/styles/library_menu.css', array(), ET_BUILDER_VERSION );
if ( 'widgets.php' === $hook ) {
wp_enqueue_script( 'et_pb_widgets_js', ET_BUILDER_URI . '/scripts/ext/widgets.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
$et_pb_options_admin = array(
'ajaxurl' => admin_url( 'admin-ajax.php' ),
'et_admin_load_nonce' => wp_create_nonce( 'et_admin_load_nonce' ),
'widget_info' => sprintf(
'
',
esc_html__( 'Here you can create new widget areas for use in the Sidebar module', 'et_builder' ),
esc_html__( 'Note: Naming your widget area "sidebar 1", "sidebar 2", "sidebar 3", "sidebar 4" or "sidebar 5" will cause conflicts with this theme', 'et_builder' ),
esc_html__( 'Widget Name', 'et_builder' ),
esc_html__( 'Create', 'et_builder' )
),
'delete_string' => esc_html__( 'Delete', 'et_builder' ),
);
wp_localize_script( 'et_pb_widgets_js', 'et_pb_options', apply_filters( 'et_pb_options_admin', $et_pb_options_admin ) );
wp_enqueue_style( 'et_pb_widgets_css', ET_BUILDER_URI . '/styles/widgets.css', array(), ET_BUILDER_VERSION );
return;
}
// Do not enqueue BB assets if GB is active on this page.
if ( et_core_is_gutenberg_enabled() ) {
return;
}
if ( ! in_array( $hook, array( 'post-new.php', 'post.php' ), true ) ) {
return;
}
/*
* Load the builder javascript and css files for custom post types
* custom post types can be added using et_builder_post_types filter
*/
$post_types = et_builder_get_builder_post_types();
$on_enabled_post_type = isset( $typenow ) && in_array( $typenow, $post_types, true );
$on_enabled_post = isset( $pagenow ) && 'post.php' === $pagenow && isset( $_GET['post'] ) && et_builder_enabled_for_post( intval( $_GET['post'] ) ); // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
if ( $on_enabled_post_type || $on_enabled_post ) {
wp_enqueue_style( 'et_bb_bfb_common', ET_BUILDER_URI . '/styles/bb_bfb_common.css', array(), ET_BUILDER_VERSION );
// Boot one builders assets or the other.
if ( et_builder_bfb_enabled() ) {
et_bfb_enqueue_scripts();
// do not load BFB if builder is disabled on page.
if ( ! et_pb_is_pagebuilder_used( get_the_ID() ) ) {
return;
}
// BFB loads builder modal outside the iframe using react portal. external scripts
// that is used on modal needs to be enqueued.
et_builder_enqueue_assets_main();
et_builder_enqueue_open_sans();
$secondary_css_bundles = glob( ET_BUILDER_DIR . 'frontend-builder/build/bundle.*.css' );
if ( $secondary_css_bundles ) {
$bundles = array( 'et-frontend-builder' );
foreach ( $secondary_css_bundles as $css_bundle ) {
$slug = basename( $css_bundle, '.css' );
$parts = explode( '.', $slug, -1 );
// Drop "bundle" from array.
array_shift( $parts );
$slug = implode( '-', $parts );
et_fb_enqueue_bundle( "et-fb-{$slug}", basename( $css_bundle ), $bundles, null );
$bundles[] = $slug;
}
}
// Hooks for theme/plugin specific styling which complements visual builder.
do_action( 'et_bfb_boot' );
} else {
et_pb_add_builder_page_js_css();
}
}
}
add_action( 'admin_enqueue_scripts', 'et_pb_admin_scripts_styles', 10, 1 );
/**
* Disable emoji detection script on edit page which has Backend Builder on it.
* WordPress automatically replaces emoji with plain image for backward compatibility
* on older browsers. This causes issue when emoji is used on header or other input
* text field because (when the modal is saved, shortcode is generated, and emoji
* is being replaced with plain image) it creates incorrect attribute markup
* such as `title="I
WP"` and causes
* the whole input text value to be disappeared
*
* @return void
*/
function et_pb_remove_emoji_detection_script() {
// phpcs:disable WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
global $pagenow;
$disable_emoji_detection = false;
// Disable emoji detection script on editing page which has Backend Builder
// global $post isn't available at admin_init, so retrieve $post data manually.
if ( 'post.php' === $pagenow && isset( $_GET['post'] ) ) {
$post_id = (int) $_GET['post'];
$post = get_post( $post_id );
if ( is_a( $post, 'WP_POST' ) && et_builder_enabled_for_post( $post->ID ) ) {
$disable_emoji_detection = true;
}
}
// Disable emoji detection script on post new page which has Backend Builder.
$has_post_type_query = isset( $_GET['post_type'] );
if ( 'post-new.php' === $pagenow && ( ! $has_post_type_query || ( $has_post_type_query && in_array( $_GET['post_type'], et_builder_get_builder_post_types(), true ) ) ) ) {
$disable_emoji_detection = true;
}
if ( $disable_emoji_detection ) {
remove_action( 'admin_print_scripts', 'print_emoji_detection_script' );
}
// phpcs:enable
}
add_action( 'admin_init', 'et_pb_remove_emoji_detection_script' );
/**
* Disable emoji detection script on visual builder
* WordPress automatically replaces emoji with plain image for backward compatibility
* on older browsers. This causes issue when emoji is used on header or other input
* text field because the staticize emoji creates HTML markup which appears to be
* invalid on input[type="text"] field such as `title="I
WP"` and causes the input text value to be escaped and
* disappeared
*
* @return void
*/
function et_fb_remove_emoji_detection_script() {
global $post;
// Disable emoji detection script on visual builder. React's auto escaping will
// remove all staticized emoji when being opened on modal's input field.
if ( isset( $post->ID ) && et_fb_is_enabled( $post->ID ) ) {
remove_action( 'wp_head', 'print_emoji_detection_script', 7 );
}
}
add_action( 'wp', 'et_fb_remove_emoji_detection_script' );
/**
* If the builder is used for the page, get rid of random p tags.
*
* @param string $content content.
*
* @return string|string[]|null
*/
function et_pb_fix_builder_shortcodes( $content ) {
if ( is_admin() ) {
// ET_Builder_Element is not loaded in the administration and some plugins call
// the_content there (e.g. WP File Manager).
return $content;
}
$is_theme_builder = ET_Builder_Element::is_theme_builder_layout();
$is_singular = is_singular() && 'on' === get_post_meta( get_the_ID(), '_et_pb_use_builder', true );
// if the builder is used for the page, get rid of random p tags.
if ( $is_theme_builder || $is_singular ) {
$content = et_pb_fix_shortcodes( $content );
}
return $content;
}
add_filter( 'the_content', 'et_pb_fix_builder_shortcodes' );
add_filter( 'et_builder_render_layout', 'et_pb_fix_builder_shortcodes' );
/**
* Prepare code module for wpautop.
*
* @param string $content content.
*
* @return string|string[]|null
*/
function et_pb_the_content_prep_code_module_for_wpautop( $content ) {
if ( 'on' === get_post_meta( get_the_ID(), '_et_pb_use_builder', true ) ) {
$content = et_pb_prep_code_module_for_wpautop( $content );
}
return $content;
}
add_filter( 'the_content', 'et_pb_the_content_prep_code_module_for_wpautop', 0 );
add_filter( 'et_builder_render_layout', 'et_pb_the_content_prep_code_module_for_wpautop', 0 );
if ( ! function_exists( 'et_pb_generate_new_layout_modal' ) ) {
/**
* Generate the html for "Add new template" Modal in Library.
*
* @return mixed|void
*/
function et_pb_generate_new_layout_modal() {
$template_type_option_output = '';
$layout_cat_option_output = '';
$layout_cats_list = '';
$layout_tags_list = '';
$template_type_options_list = '';
$new_layout_template_types = array(
'module' => esc_html__( 'Module', 'et_builder' ),
'fullwidth_module' => esc_html__( 'Fullwidth Module', 'et_builder' ),
'row' => esc_html__( 'Row', 'et_builder' ),
'section' => esc_html__( 'Section', 'et_builder' ),
'fullwidth_section' => esc_html__( 'Fullwidth Section', 'et_builder' ),
'specialty_section' => esc_html__( 'Specialty Section', 'et_builder' ),
'layout' => et_builder_i18n( 'Layout' ),
);
$template_type_options = apply_filters( 'et_pb_new_layout_template_types', $new_layout_template_types );
// construct output for the template type option.
if ( ! empty( $template_type_options ) ) {
foreach ( $template_type_options as $option_id => $option_name ) {
$template_type_options_list .= sprintf(
'
%2$s ',
esc_attr( $option_id ),
esc_html( $option_name )
);
}
$template_type_option_output = sprintf(
'
%1$s:
%2$s
',
esc_html__( 'Layout Type', 'et_builder' ),
$template_type_options_list
);
}
$template_global_option_output = apply_filters(
'et_pb_new_layout_global_option',
sprintf(
'
%1$s ',
esc_html__( 'Save as Global', 'et_builder' )
)
);
$layout_categories = apply_filters( 'et_pb_new_layout_cats_array', get_terms( 'layout_category', array( 'hide_empty' => false ) ) );
if ( is_array( $layout_categories ) && ! empty( $layout_categories ) ) {
foreach ( $layout_categories as $category ) {
$layout_cats_list .= sprintf(
'
%1$s ',
esc_html( $category->name ),
esc_attr( $category->term_id )
);
}
}
$layout_tags = apply_filters( 'et_pb_new_layout_tags_array', get_terms( 'layout_tag', array( 'hide_empty' => false ) ) );
if ( is_array( $layout_tags ) && ! empty( $layout_tags ) ) {
foreach ( $layout_tags as $tag ) {
$layout_tags_list .= sprintf(
'
%1$s ',
esc_html( $tag->name ),
esc_attr( $tag->term_id )
);
}
}
// Construct output for the layout Tag option.
$layout_cat_option_output = sprintf(
'
%1$s
%3$s
',
esc_html__( 'Add To Categories', 'et_builder' ),
esc_html__( 'Create new Category', 'et_builder' ),
$layout_cats_list
);
// Construct output for the layout Tag option.
$layout_tag_option_output = sprintf(
'
%1$s
%3$s
',
esc_html__( 'Add To Tags', 'et_builder' ),
esc_html__( 'Create new Tag', 'et_builder' ),
$layout_tags_list
);
$output = sprintf(
'
',
esc_html__( 'Add New Layout', 'et_builder' ),
esc_html__( 'Layout Name', 'et_builder' ),
$template_type_option_output,
$template_global_option_output,
$layout_cat_option_output, // #5
apply_filters( 'et_pb_new_layout_before_options', '' ),
apply_filters( 'et_pb_new_layout_after_options', '' ),
$layout_tag_option_output
);
return apply_filters( 'et_pb_new_layout_modal_output', $output );
}
}
if ( ! function_exists( 'et_pb_get_layout_type' ) ) :
/**
* Get layout type of given post ID.
*
* @param int $post_id post id.
*
* @return string|bool
*/
function et_pb_get_layout_type( $post_id ) {
// Get taxonomies.
$layout_type_data = wp_get_post_terms( $post_id, 'layout_type' );
if ( empty( $layout_type_data ) ) {
return false;
}
// Pluck name out of taxonomies.
$layout_type_array = wp_list_pluck( $layout_type_data, 'name' );
// Logically, a layout only have one layout type.
$layout_type = implode( '|', $layout_type_array );
return $layout_type;
}
endif;
if ( ! function_exists( 'et_pb_is_wp_old_version' ) ) :
/**
* Determine current wp version is less than 4.5.
*/
function et_pb_is_wp_old_version() {
global $wp_version;
$wp_major_version = substr( $wp_version, 0, 3 );
if ( version_compare( $wp_major_version, '4.5', '<' ) ) {
return true;
}
return false;
}
endif;
if ( ! function_exists( 'et_builder_theme_or_plugin_updated_cb' ) ) :
/**
* Delete cached definitions/helpers after theme or plugin update.
*/
function et_builder_theme_or_plugin_updated_cb() {
// Delete cached definitions / helpers.
et_fb_delete_builder_assets();
et_update_option( 'et_pb_clear_templates_cache', true );
}
add_action( 'after_switch_theme', 'et_builder_theme_or_plugin_updated_cb' );
add_action( 'activated_plugin', 'et_builder_theme_or_plugin_updated_cb', 10, 0 );
add_action( 'deactivated_plugin', 'et_builder_theme_or_plugin_updated_cb', 10, 0 );
add_action( 'upgrader_process_complete', 'et_builder_theme_or_plugin_updated_cb', 10, 0 );
add_action( 'et_support_center_toggle_safe_mode', 'et_builder_theme_or_plugin_updated_cb', 10, 0 );
endif;
/**
* Enqueue scripts that are required by BFB and Layout Block. These scripts are abstracted into
* separated file so Layout Block can enqueue the same sets of scripts without re-register and
* re-enqueue them
*
* @since 4.1.0
*/
function et_bfb_enqueue_scripts_dependencies() {
global $wp_version, $post;
$wp_major_version = substr( $wp_version, 0, 3 );
if ( et_pb_is_pagebuilder_used( get_the_ID() ) ) {
wp_enqueue_editor();
}
if ( version_compare( $wp_major_version, '4.5', '<' ) ) {
$jquery_ui = 'et_pb_admin_date_js';
wp_register_script( $jquery_ui, ET_BUILDER_URI . '/scripts/ext/jquery-ui-1.10.4.custom.min.js', array( 'jquery' ), ET_BUILDER_PRODUCT_VERSION, true );
} else {
$jquery_ui = 'jquery-ui-datepicker';
}
// Load timepicker script on admin page in case of BFB to make it work with modals loaded on WP admin DOM.
wp_enqueue_script( 'et_bfb_admin_date_addon_js', ET_BUILDER_URI . '/scripts/ext/jquery-ui-timepicker-addon.js', array( $jquery_ui ), ET_BUILDER_PRODUCT_VERSION, true );
// Load google maps script on admin page in case of BFB to make it work with modals loaded on WP admin DOM.
if ( et_pb_enqueue_google_maps_script() ) {
$bfb_google_maps_api_url_args = array(
'key' => et_pb_get_google_api_key(),
'callback' => 'initMap',
);
$bfb_google_maps_api_url = add_query_arg( $bfb_google_maps_api_url_args, is_ssl() ? 'https://maps.googleapis.com/maps/api/js' : 'http://maps.googleapis.com/maps/api/js' );
wp_enqueue_script( 'et_bfb_google_maps_api', esc_url( $bfb_google_maps_api_url ), array(), '3', true );
}
wp_enqueue_script( 'et_pb_media_library', ET_BUILDER_URI . '/scripts/ext/media-library.js', array( 'media-editor' ), ET_BUILDER_PRODUCT_VERSION, true );
if ( ! wp_script_is( 'wp-hooks', 'registered' ) ) {
// Use bundled wp-hooks script when WP < 5.0.
wp_enqueue_script( 'wp-hooks', ET_BUILDER_URI . '/frontend-builder/assets/backports/hooks.js', array(), ET_BUILDER_PRODUCT_VERSION, false );
}
}
if ( ! function_exists( 'et_bfb_enqueue_scripts' ) ) :
/**
* Register BFB scripts.
*/
function et_bfb_enqueue_scripts() {
global $post;
// Enqueue scripts required by BFB.
et_bfb_enqueue_scripts_dependencies();
wp_enqueue_script( 'et_bfb_admin_js', ET_BUILDER_URI . '/scripts/bfb_admin_script.js', array( 'jquery', 'et_pb_media_library' ), ET_BUILDER_PRODUCT_VERSION, true );
$bfb_options = array(
'ajaxurl' => admin_url( 'admin-ajax.php' ),
'et_enable_bfb_nonce' => wp_create_nonce( 'et_enable_bfb_nonce' ),
'default_initial_column_type' => apply_filters( 'et_builder_default_initial_column_type', '4_4' ),
'default_initial_text_module' => apply_filters( 'et_builder_default_initial_text_module', 'et_pb_text' ),
'skip_default_content_adding' => apply_filters( 'et_builder_skip_content_activation', false, $post ) ? 'skip' : '',
);
wp_localize_script( 'et_bfb_admin_js', 'et_bfb_options', apply_filters( 'et_bfb_options', $bfb_options ) );
// Add filter to register tinyMCE buttons that is missing from BFB.
add_filter( 'mce_external_plugins', 'et_bfb_filter_mce_plugin' );
}
endif;
/**
* BFB use built-in WordPress tinyMCE initialization while visual builder uses standalone tinyMCE
* initialization which leads to several buttons in VB not available in BFB. This function register
* them as plugins
*
* @since 4.0.9
*
* @param array $plugins tinyMCE plugin list.
*
* @return array
*/
function et_bfb_filter_mce_plugin( $plugins ) {
// NOTE: `ET_FB_ASSETS_URI` constant isn't available yet at this point, so use `ET_BUILDER_URI`.
$plugins['table'] = ET_BUILDER_URI . '/frontend-builder/assets/vendors/plugins/table/plugin.min.js';
return $plugins;
}
/**
* Tinymce to load in html mode for BB.
*
* @param array $settings Array of editor arguments.
* @param string $editor_id Unique editor identifier, e.g. 'content'. Accepts 'classic-block'
* when called from block editor's Classic block.
*
* @return mixed
*/
function et_pb_wp_editor_settings( $settings, $editor_id ) {
if ( 'content' === $editor_id ) {
$settings['default_editor'] = 'html';
}
return $settings;
}
if ( ! function_exists( 'et_pb_add_builder_page_js_css' ) ) :
/**
* Load builder js and css.
*/
function et_pb_add_builder_page_js_css() {
global $typenow, $post, $wp_version;
// Get WP major version.
$wp_major_version = substr( $wp_version, 0, 3 );
// Avoid serving any data from object cache.
if ( ! defined( 'DONOTCACHEPAGE' ) ) {
define( 'DONOTCACHEPAGE', true );
}
// fix tinymce to load in html mode for BB.
if ( et_pb_is_pagebuilder_used() ) {
add_filter( 'wp_editor_settings', 'et_pb_wp_editor_settings', 10, 2 );
}
// BEGIN Process shortcodes (for module settings migrations and Yoast SEO compatibility)
// Get list of shortcodes that causes issue if being triggered in admin.
$conflicting_shortcodes = et_pb_admin_excluded_shortcodes();
if ( ! empty( $conflicting_shortcodes ) ) {
foreach ( $conflicting_shortcodes as $shortcode ) {
remove_shortcode( $shortcode );
}
}
// save the original content of $post variable.
$post_original = $post;
// get the content for yoast.
$post_content_processed = do_shortcode( $post->post_content );
// set the $post to the original content to make sure it wasn't changed by do_shortcode().
$post = $post_original; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited -- were restoring it to what it was beforea few lines above.
// END Process shortcodes.
$is_global_template = '';
$post_id = '';
$post_type = $typenow;
$selective_sync_status = '';
$global_module_type = '';
$excluded_global_options = array();
$utils = ET_Core_Data_Utils::instance();
$updates_options = get_site_option( 'et_automatic_updates_options', array() );
$et_account = array(
'et_username' => $utils->array_get( $updates_options, 'username', '' ),
'et_api_key' => $utils->array_get( $updates_options, 'api_key', '' ),
'status' => get_site_option( 'et_account_status', 'not_active' ),
);
// we need some post data when editing saved templates.
if ( 'et_pb_layout' === $typenow ) {
$template_scope = wp_get_object_terms( get_the_ID(), 'scope' );
$template_type = wp_get_object_terms( get_the_ID(), 'layout_type' );
$is_global_template = ! empty( $template_scope[0] ) ? $template_scope[0]->slug : 'regular';
$global_module_type = ! empty( $template_type[0] ) ? $template_type[0]->slug : '';
$post_id = get_the_ID();
// Check whether it's a Global item's page and display wp error if Global items disabled for current user.
if ( ! et_pb_is_allowed( 'edit_global_library' ) && 'global' === $is_global_template ) {
wp_die( esc_html__( "you don't have sufficient permissions to access this page", 'et_builder' ) );
}
if ( 'global' === $is_global_template ) {
$excluded_global_options = get_post_meta( $post_id, '_et_pb_excluded_global_options' );
$selective_sync_status = empty( $excluded_global_options ) ? '' : 'updated';
}
$built_for_post_type = get_post_meta( get_the_ID(), '_et_pb_built_for_post_type', true );
$built_for_post_type = '' !== $built_for_post_type ? $built_for_post_type : 'page';
$post_type = apply_filters( 'et_pb_built_for_post_type', $built_for_post_type, get_the_ID() );
}
// we need this data to create the filter when adding saved modules.
$layout_categories = get_terms( 'layout_category' );
$layout_cat_data = array();
$layout_cat_data_json = '';
if ( is_array( $layout_categories ) && ! empty( $layout_categories ) ) {
foreach ( $layout_categories as $category ) {
$layout_cat_data[] = array(
'slug' => $category->slug,
'name' => $category->name,
);
}
}
if ( ! empty( $layout_cat_data ) ) {
$layout_cat_data_json = wp_json_encode( $layout_cat_data );
}
// Set fixed protocol for preview URL to prevent cross origin issue.
$preview_scheme = is_ssl() ? 'https' : 'http';
$preview_url = esc_url( home_url( '/' ) );
if ( 'https' === $preview_scheme && ! strpos( $preview_url, 'https://' ) ) {
$preview_url = str_replace( 'http://', 'https://', $preview_url );
}
// force update cache if et_pb_clear_templates_cache option is set to on.
$force_cache_value = et_get_option( 'et_pb_clear_templates_cache', '', '', true );
$force_cache_update = '' !== $force_cache_value ? $force_cache_value : ET_BUILDER_FORCE_CACHE_PURGE;
/**
* Whether or not the backend builder should clear its Backbone template cache.
*
* @param bool $force_cache_update
*/
$force_cache_update = apply_filters( 'et_pb_clear_template_cache', $force_cache_update );
// delete et_pb_clear_templates_cache option it's not needed anymore.
et_delete_option( 'et_pb_clear_templates_cache' );
wp_enqueue_script( 'jquery-ui-core' );
wp_enqueue_script( 'underscore' );
wp_enqueue_script( 'backbone' );
if ( et_pb_enqueue_google_maps_script() ) {
$google_maps_api_url_args = array(
'v' => 3,
'key' => et_pb_get_google_api_key(),
);
$google_maps_api_url = add_query_arg( $google_maps_api_url_args, is_ssl() ? 'https://maps.googleapis.com/maps/api/js' : 'http://maps.googleapis.com/maps/api/js' );
wp_enqueue_script( 'google-maps-api', esc_url_raw( $google_maps_api_url ), array(), '3', true );
}
wp_enqueue_script( 'wp-color-picker' );
wp_enqueue_style( 'wp-color-picker' );
if ( version_compare( $wp_major_version, '4.9', '>=' ) ) {
wp_enqueue_script( 'wp-color-picker-alpha', ET_BUILDER_URI . '/scripts/ext/wp-color-picker-alpha.min.js', array( 'jquery', 'wp-color-picker' ), ET_BUILDER_VERSION, true );
$color_picker_strings = array(
'legacy_pick' => esc_html__( 'Select', 'et_builder' ),
'legacy_current' => esc_html__( 'Current Color', 'et_builder' ),
);
wp_localize_script( 'wp-color-picker-alpha', 'et_pb_color_picker_strings', apply_filters( 'et_pb_color_picker_strings_builder', $color_picker_strings ) );
} else {
wp_enqueue_script( 'wp-color-picker-alpha', ET_BUILDER_URI . '/scripts/ext/wp-color-picker-alpha-48.min.js', array( 'jquery', 'wp-color-picker' ), ET_BUILDER_VERSION, true );
}
wp_register_script( 'chart', ET_BUILDER_URI . '/scripts/ext/chart.min.js', array(), ET_BUILDER_VERSION, true );
wp_register_script( 'jquery-tablesorter', ET_BUILDER_URI . '/scripts/ext/jquery.tablesorter.min.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
// load 1.10.4 versions of jQuery-ui scripts if WP version is less than 4.5, load 1.11.4 version otherwise.
if ( et_pb_is_wp_old_version() ) {
$jquery_ui = 'et_pb_admin_date_js';
wp_enqueue_script( $jquery_ui, ET_BUILDER_URI . '/scripts/ext/jquery-ui-1.10.4.custom.min.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
} else {
$jquery_ui = 'jquery-ui-datepicker';
}
wp_enqueue_script( 'et_pb_admin_date_addon_js', ET_BUILDER_URI . '/scripts/ext/jquery-ui-timepicker-addon.js', array( $jquery_ui ), ET_BUILDER_VERSION, true );
wp_enqueue_script( 'validation', ET_BUILDER_URI . '/scripts/ext/jquery.validate.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
wp_enqueue_script( 'minicolors', ET_BUILDER_URI . '/scripts/ext/jquery.minicolors.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
wp_enqueue_script( 'et_pb_cache_notice_js', ET_BUILDER_URI . '/scripts/cache_notice.js', array( 'jquery', 'et_pb_admin_js' ), ET_BUILDER_VERSION, true );
$pb_notice_options = array(
'product_version' => ET_BUILDER_PRODUCT_VERSION,
);
wp_localize_script( 'et_pb_cache_notice_js', 'et_pb_notice_options', apply_filters( 'et_pb_notice_options_builder', $pb_notice_options ) );
wp_enqueue_script( 'lz_string', ET_BUILDER_URI . '/scripts/ext/lz-string.min.js', array(), ET_BUILDER_VERSION, true );
// phpcs:disable WordPress.WP.EnqueuedResourceParameters -- The script version number are specified in the src. No need to set $ver explicitly.
wp_enqueue_script( 'es6-promise', '//cdn.jsdelivr.net/npm/es6-promise@4/dist/es6-promise.auto.min.js', array(), null, true );
wp_enqueue_script( 'postmate', '//cdn.jsdelivr.net/npm/postmate@1.1.9/build/postmate.min.js', array( 'es6-promise' ), null, true );
// phpcs:enable
wp_enqueue_script( 'et_pb_media_library', ET_BUILDER_URI . '/scripts/ext/media-library.js', array( 'media-editor' ), ET_BUILDER_PRODUCT_VERSION, true );
wp_enqueue_script( 'et_pb_admin_js', ET_BUILDER_URI . '/scripts/builder.js', array( 'jquery', 'jquery-ui-core', 'underscore', 'backbone', 'chart', 'jquery-tablesorter', 'et_pb_media_library', 'lz_string', 'es6-promise' ), ET_BUILDER_VERSION, true );
$saved_gutter_width = get_post_meta( get_the_ID(), '_et_pb_gutter_width', true );
$pb_options = array(
'debug' => defined( 'ET_DEBUG' ) && ET_DEBUG,
'wp_default_editor' => wp_default_editor(),
'et_account' => $et_account,
'ajaxurl' => admin_url( 'admin-ajax.php' ),
'home_url' => home_url(),
'cookie_path' => SITECOOKIEPATH,
'preview_url' => add_query_arg( 'et_pb_preview', 'true', $preview_url ),
'et_admin_load_nonce' => wp_create_nonce( 'et_admin_load_nonce' ),
'images_uri' => ET_BUILDER_URI . '/images',
'postId' => $post->ID,
'post_type' => $post_type,
'is_third_party_post_type' => et_builder_is_post_type_custom( $post_type ) ? 'yes' : 'no',
'et_builder_module_parent_shortcodes' => ET_Builder_Element::get_parent_slugs_regex( $post_type ),
'et_builder_module_child_shortcodes' => ET_Builder_Element::get_child_slugs_regex( $post_type ),
'et_builder_module_raw_content_shortcodes' => ET_Builder_Element::get_raw_content_slugs( $post_type ),
'et_builder_modules' => ET_Builder_Element::get_modules_js_array( $post_type ),
'et_builder_modules_count' => ET_Builder_Element::get_modules_count( $post_type ),
'et_builder_modules_with_children' => ET_Builder_Element::get_slugs_with_children( $post_type ),
'et_builder_modules_featured_image_background' => ET_Builder_Element::get_featured_image_background_modules( $post_type ),
'et_builder_templates_amount' => ET_BUILDER_AJAX_TEMPLATES_AMOUNT,
'et_builder_edit_global_library' => et_pb_is_allowed( 'edit_global_library' ),
'default_initial_column_type' => apply_filters( 'et_builder_default_initial_column_type', '4_4' ),
'default_initial_text_module' => apply_filters( 'et_builder_default_initial_text_module', 'et_pb_text' ),
'section_only_row_dragged_away' => esc_html__( 'The section should have at least one row.', 'et_builder' ),
'fullwidth_module_dragged_away' => esc_html__( 'Fullwidth module can\'t be used outside of the Fullwidth Section.', 'et_builder' ),
'stop_dropping_3_col_row' => esc_html__( "This number of columns can't be used on this row.", 'et_builder' ),
'preview_image' => esc_html__( 'Preview', 'et_builder' ),
'empty_admin_label' => esc_html__( 'Module', 'et_builder' ),
'video_module_image_error' => esc_html__( 'Still images cannot be generated from this video service and/or this video format', 'et_builder' ),
'geocode_error' => esc_html__( 'Geocode was not successful for the following reason', 'et_builder' ),
'geocode_error_2' => esc_html__( 'Geocoder failed due to', 'et_builder' ),
'no_results' => esc_html__( 'No results found', 'et_builder' ),
'all_tab_options_hidden' => esc_html__( 'No available options for this configuration.', 'et_builder' ),
'update_global_module' => esc_html__( 'You\'re about to update global module. This change will be applied to all pages where you use this module. Press OK if you want to update this module', 'et_builder' ),
'global_row_alert' => esc_html__( 'You cannot add global rows into global sections', 'et_builder' ),
'global_module_alert' => esc_html__( 'You cannot add global modules into global sections or rows', 'et_builder' ),
'all_cat_text' => esc_html__( 'All Categories', 'et_builder' ),
'font_name_error' => esc_html__( 'Name Cannot be Empty', 'et_builder' ),
'font_file_error' => esc_html__( 'Please Select Font File', 'et_builder' ),
'font_weight_error' => esc_html__( 'Please Select Font Weight', 'et_builder' ),
'is_global_template' => $is_global_template,
'selective_sync_status' => $selective_sync_status,
'global_module_type' => $global_module_type,
'excluded_global_options' => isset( $excluded_global_options[0] ) ? json_decode( $excluded_global_options[0] ) : array(),
'template_post_id' => $post_id,
'layout_categories' => $layout_cat_data_json,
'map_pin_address_error' => esc_html__( 'Map Pin Address cannot be empty', 'et_builder' ),
'map_pin_address_invalid' => esc_html__( 'Invalid Pin and address data. Please try again.', 'et_builder' ),
'locked_section_permission_alert' => esc_html__( 'You do not have permission to unlock this section.', 'et_builder' ),
'locked_row_permission_alert' => esc_html__( 'You do not have permission to unlock this row.', 'et_builder' ),
'locked_module_permission_alert' => esc_html__( 'You do not have permission to unlock this module.', 'et_builder' ),
'locked_item_permission_alert' => esc_html__( 'You do not have permission to perform this task.', 'et_builder' ),
'localstorage_unavailability_alert' => esc_html__( 'Unable to perform copy/paste process due to inavailability of localStorage feature in your browser. Please use latest modern browser (Chrome, Firefox, or Safari) to perform copy/paste process', 'et_builder' ),
'invalid_color' => esc_html__( 'Invalid Color', 'et_builder' ),
'et_pb_preview_nonce' => wp_create_nonce( 'et_pb_preview_nonce' ),
'is_divi_library' => 'et_pb_layout' === $typenow ? 1 : 0,
'layout_type' => 'et_pb_layout' === $typenow ? et_pb_get_layout_type( get_the_ID() ) : 0,
'is_plugin_used' => et_is_builder_plugin_active(),
'yoast_content' => et_is_yoast_seo_plugin_active() ? $post_content_processed : '',
'ab_db_status' => true === et_pb_db_status_up_to_date() ? 'exists' : 'not_exists',
'ab_testing_builder_nonce' => wp_create_nonce( 'ab_testing_builder_nonce' ),
'page_color_palette' => get_post_meta( get_the_ID(), '_et_pb_color_palette', true ),
'default_color_palette' => implode( '|', et_pb_get_default_color_palette() ),
'page_section_bg_color' => get_post_meta( get_the_ID(), '_et_pb_section_background_color', true ),
'page_gutter_width' => '' !== $saved_gutter_width ? $saved_gutter_width : et_get_option( 'gutter_width', '3' ),
'product_version' => ET_BUILDER_PRODUCT_VERSION,
'active_plugins' => et_builder_get_active_plugins(),
'force_cache_purge' => $force_cache_update ? 'true' : 'false',
'memory_limit_increased' => esc_html__( 'Your memory limit has been increased', 'et_builder' ),
'memory_limit_not_increased' => esc_html__( "Your memory limit can't be changed automatically", 'et_builder' ),
'google_api_key' => et_pb_get_google_api_key(),
'options_page_url' => et_pb_get_options_page_link(),
'et_pb_google_maps_script_notice' => et_pb_enqueue_google_maps_script(),
'select_text' => esc_html__( 'Select', 'et_builder' ),
'et_fb_autosave_nonce' => wp_create_nonce( 'et_fb_autosave_nonce' ),
'et_builder_email_fetch_lists_nonce' => wp_create_nonce( 'et_builder_email_fetch_lists_nonce' ),
'et_builder_email_add_account_nonce' => wp_create_nonce( 'et_builder_email_add_account_nonce' ),
'et_builder_email_remove_account_nonce' => wp_create_nonce( 'et_builder_email_remove_account_nonce' ),
'et_pb_module_settings_migrations' => ET_Builder_Module_Settings_Migration::$migrated,
'acceptable_css_string_values' => et_builder_get_acceptable_css_string_values( 'all' ),
'upload_font_nonce' => wp_create_nonce( 'et_fb_upload_font_nonce' ),
'user_fonts' => et_builder_get_custom_fonts(),
'google_fonts' => et_builder_get_google_fonts(),
'supported_font_weights' => et_builder_get_font_weight_list(),
'supported_font_formats' => et_pb_get_supported_font_formats(),
'all_svg_icons' => et_pb_get_svg_icons_list(),
'library_get_layouts_data_nonce' => wp_create_nonce( 'et_builder_library_get_layouts_data' ),
'library_get_layout_nonce' => wp_create_nonce( 'et_builder_library_get_layout' ),
'library_update_account_nonce' => wp_create_nonce( 'et_builder_library_update_account' ),
'library_custom_tabs' => ET_Builder_Library::builder_library_modal_custom_tabs( $post_type ),
);
$pb_options_builder = array_merge( $pb_options, et_pb_history_localization() );
wp_localize_script( 'et_pb_admin_js', 'et_pb_options', apply_filters( 'et_pb_options_builder', $pb_options_builder ) );
$ab_settings = et_builder_ab_labels();
$pb_ab_js_options = array(
'test_id' => $post->ID,
'has_report' => et_pb_ab_has_report( $post->ID ),
'has_permission' => et_pb_is_allowed( 'ab_testing' ),
'refresh_interval_duration' => et_pb_ab_get_refresh_interval_duration( $post->ID ),
'refresh_interval_durations' => et_pb_ab_refresh_interval_durations(),
'analysis_formula' => et_pb_ab_get_analysis_formulas(),
'have_conversions' => et_pb_ab_get_modules_have_conversions(),
'sales_title' => esc_html__( 'Sales', 'et_builder' ),
'force_cache_purge' => $force_cache_update,
'total_title' => esc_html__( 'Total', 'et_builder' ),
// Saved data.
'subjects_rank' => ( 'on' === get_post_meta( $post->ID, '_et_pb_use_builder', true ) ) ? et_pb_ab_get_saved_subjects_ranks( $post->ID ) : false,
// Rank color.
'subjects_rank_color' => et_pb_ab_get_subject_rank_colors(),
// Configuration.
'has_no_permission' => array(
'title' => esc_html__( 'Unauthorized Action', 'et_builder' ),
'desc' => esc_html__( 'You do not have permission to edit the module, row or section in this split test.', 'et_builder' ),
),
// AB Testing.
'select_ab_testing_subject' => $ab_settings['select_subject'],
'select_ab_testing_goal' => $ab_settings['select_goal'],
'configure_ab_testing_alternative' => $ab_settings['configure_alternative'],
'select_ab_testing_winner_first' => $ab_settings['select_winner_first'],
'select_ab_testing_subject_first' => $ab_settings['select_subject_first'],
'select_ab_testing_goal_first' => $ab_settings['select_goal_first'],
'cannot_select_subject_parent_as_goal' => $ab_settings['cannot_select_subject_parent_as_goal'],
'cannot_select_global_children_as_subject' => $ab_settings['cannot_select_global_children_as_subject'],
'cannot_select_global_children_as_goal' => $ab_settings['cannot_select_global_children_as_goal'],
// Save to Library.
'cannot_save_app_layout_has_ab_testing' => $ab_settings['cannot_save_app_layout_has_ab_testing'],
'cannot_save_section_layout_has_ab_testing' => $ab_settings['cannot_save_section_layout_has_ab_testing'],
'cannot_save_row_layout_has_ab_testing' => $ab_settings['cannot_save_row_layout_has_ab_testing'],
'cannot_save_row_inner_layout_has_ab_testing' => $ab_settings['cannot_save_row_inner_layout_has_ab_testing'],
'cannot_save_module_layout_has_ab_testing' => $ab_settings['cannot_save_module_layout_has_ab_testing'],
// Load / Clear Layout.
'cannot_load_layout_has_ab_testing' => $ab_settings['cannot_load_layout_has_ab_testing'],
'cannot_clear_layout_has_ab_testing' => $ab_settings['cannot_clear_layout_has_ab_testing'],
// Cannot Import / Export Layout (Portability).
'cannot_import_export_layout_has_ab_testing' => $ab_settings['cannot_import_export_layout_has_ab_testing'],
// Moving Goal / Subject.
'cannot_move_module_goal_out_from_subject' => $ab_settings['cannot_move_module_goal_out_from_subject'],
'cannot_move_row_goal_out_from_subject' => $ab_settings['cannot_move_row_goal_out_from_subject'],
'cannot_move_goal_into_subject' => $ab_settings['cannot_move_goal_into_subject'],
'cannot_move_subject_into_goal' => $ab_settings['cannot_move_subject_into_goal'],
// Cloning + Has Goal.
'cannot_clone_section_has_goal' => $ab_settings['cannot_clone_section_has_goal'],
'cannot_clone_row_has_goal' => $ab_settings['cannot_clone_row_has_goal'],
// Removing + Has Goal.
'cannot_remove_section_has_goal' => $ab_settings['cannot_remove_section_has_goal'],
'cannot_remove_row_has_goal' => $ab_settings['cannot_remove_row_has_goal'],
// Removing + Has Unremovable Subjects.
'cannot_remove_section_has_unremovable_subject' => $ab_settings['cannot_remove_section_has_unremovable_subject'],
'cannot_remove_row_has_unremovable_subject' => $ab_settings['cannot_remove_row_has_unremovable_subject'],
// View stats summary table heading.
'view_stats_thead_titles' => $ab_settings['view_stats_thead_titles'],
);
wp_localize_script( 'et_pb_admin_js', 'et_pb_ab_js_options', apply_filters( 'et_pb_ab_js_options', $pb_ab_js_options ) );
$pb_help_options = array(
'shortcuts' => et_builder_get_shortcuts( 'bb' ),
);
wp_localize_script( 'et_pb_admin_js', 'et_pb_help_options', apply_filters( 'et_pb_help_options', $pb_help_options ) );
et_core_load_main_fonts();
wp_enqueue_style( 'et_pb_admin_css', ET_BUILDER_URI . '/styles/style.css', array(), ET_BUILDER_VERSION );
wp_enqueue_style( 'et_pb_admin_date_css', ET_BUILDER_URI . '/styles/jquery-ui-1.12.1.custom.css', array(), ET_BUILDER_VERSION );
wp_add_inline_style( 'et_pb_admin_css', et_pb_ab_get_subject_rank_colors_style() );
ET_Cloud_App::load_js();
}
endif;
/**
* Set et-editor-available-post-* cookie
*/
function et_pb_set_editor_available_cookie() {
$post_id = isset( $_GET['post'] ) ? absint( $_GET['post'] ) : false; // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
$headers_sent = headers_sent();
if ( et_builder_should_load_framework() && is_admin() && ! $headers_sent && ! empty( $post_id ) ) {
setcookie( 'et-editor-available-post-' . $post_id . '-bb', 'bb', time() + ( MINUTE_IN_SECONDS * 30 ), SITECOOKIEPATH, false, is_ssl() );
}
}
add_action( 'admin_init', 'et_pb_set_editor_available_cookie' );
/**
* List of history meta.
*
* @return array History meta.
*/
function et_pb_history_localization() {
return array(
'verb' => array(
'did' => esc_html__( 'Did', 'et_builder' ),
'added' => esc_html__( 'Added', 'et_builder' ),
'edited' => esc_html__( 'Edited', 'et_builder' ),
'removed' => esc_html__( 'Removed', 'et_builder' ),
'moved' => esc_html__( 'Moved', 'et_builder' ),
'expanded' => esc_html__( 'Expanded', 'et_builder' ),
'collapsed' => esc_html__( 'Collapsed', 'et_builder' ),
'locked' => esc_html__( 'Locked', 'et_builder' ),
'unlocked' => esc_html__( 'Unlocked', 'et_builder' ),
'cloned' => esc_html__( 'Cloned', 'et_builder' ),
'cleared' => esc_html__( 'Cleared', 'et_builder' ),
'enabled' => esc_html__( 'Enabled', 'et_builder' ),
'disabled' => esc_html__( 'Disabled', 'et_builder' ),
'copied' => esc_html__( 'Copied', 'et_builder' ),
'reset' => esc_html__( 'Reset', 'et_builder' ),
'cut' => esc_html__( 'Cut', 'et_builder' ),
'pasted' => esc_html__( 'Pasted', 'et_builder' ),
'pasted_styles' => esc_html__( 'Pasted Styles', 'et_builder' ),
'renamed' => esc_html__( 'Renamed', 'et_builder' ),
'loaded' => esc_html__( 'Loaded', 'et_builder' ),
'turnon' => esc_html__( 'Turned On', 'et_builder' ),
'turnoff' => esc_html__( 'Turned Off', 'et_builder' ),
'globalon' => esc_html__( 'Made Global', 'et_builder' ),
'globaloff' => esc_html__( 'Disabled Global', 'et_builder' ),
'configured' => esc_html__( 'Configured', 'et_builder' ),
'find_replace' => esc_html__( 'Find & Replace', 'et_builder' ),
'extend_styles' => esc_html__( 'Extend Styles', 'et_builder' ),
'imported' => esc_html__( 'Imported From Layout', 'et_builder' ),
'presetCreated' => esc_html__( 'Preset Created For', 'et_builder' ),
'presetNameChanged' => esc_html__( 'Preset Name Changed For', 'et_builder' ),
'presetDeleted' => esc_html__( 'Preset Deleted For', 'et_builder' ),
'presetAssignedAsDefault' => esc_html__( 'Preset Assigned As Default For', 'et_builder' ),
),
'noun' => array(
'section' => esc_html__( 'Section', 'et_builder' ),
'saved_section' => esc_html__( 'Saved Section', 'et_builder' ),
'fullwidth_section' => esc_html__( 'Fullwidth Section', 'et_builder' ),
'specialty_section' => esc_html__( 'Specialty Section', 'et_builder' ),
'column' => esc_html__( 'Column', 'et_builder' ),
'row' => esc_html__( 'Row', 'et_builder' ),
'saved_row' => esc_html__( 'Saved Row', 'et_builder' ),
'module' => esc_html__( 'Module', 'et_builder' ),
'saved_module' => esc_html__( 'Saved Module', 'et_builder' ),
'page' => esc_html__( 'Page', 'et_builder' ),
'layout' => et_builder_i18n( 'Layout' ),
'abtesting' => esc_html__( 'Split Testing', 'et_builder' ),
'settings' => esc_html__( 'Settings', 'et_builder' ),
),
'addition' => array(
'phone' => esc_html__( 'on Phone', 'et_builder' ),
'tablet' => esc_html__( 'on Tablet', 'et_builder' ),
'desktop' => esc_html__( 'on Desktop', 'et_builder' ),
),
);
}
/**
* Page Settings Metabox code is included in builder.js which won't be loaded unless BB is.
* In such cases (eg BFB or GB are enabled) we provide the mbox js logic in a separate file.
*
* @return void
*/
function et_pb_metabox_scripts() {
// Only act if `builder.js` isn't enqueued.
if ( ! wp_script_is( 'et_pb_admin_js' ) ) {
global $typenow;
wp_enqueue_script( 'et_page_settings_metabox_js', ET_BUILDER_URI . '/scripts/page-settings-metabox.js', array( 'jquery' ), ET_BUILDER_PRODUCT_VERSION, true );
$pb_options = array(
'post_type' => $typenow,
'is_third_party_post_type' => et_builder_is_post_type_custom( $typenow ) ? 'yes' : 'no',
);
wp_localize_script( 'et_page_settings_metabox_js', 'et_pb_options', $pb_options );
}
}
/**
* Prevents the Builder mbox from being hidden.
*
* @param string[] $hidden all hidden metaboxes.
*
* @return mixed
*/
function et_pb_hidden_meta_boxes( $hidden ) {
$found = array_search( 'et_pb_layout', $hidden, true );
if ( false !== $found ) {
unset( $hidden[ $found ] );
}
return $hidden;
}
/**
* Add "The Divi Builder" BB metabox.
*
* @param string $post_type post type.
* @param WP_Post $post post object.
*/
function et_pb_add_custom_box( $post_type, $post ) {
add_action( 'admin_enqueue_scripts', 'et_pb_metabox_scripts', 99 );
// Do not add BB metabox if GB is active on this page.
if ( et_core_is_gutenberg_enabled() ) {
return;
}
// Do not add BB metabox if builder is not activate on this page.
if ( et_builder_bfb_enabled() && ! et_pb_is_pagebuilder_used( $post->ID ) ) {
return;
}
$post_types = et_builder_get_builder_post_types();
$add = in_array( $post_type, $post_types, true );
if ( ! $add && ! empty( $post ) && et_builder_enabled_for_post( $post->ID ) ) {
$add = true;
}
if ( $add ) {
add_meta_box( ET_BUILDER_LAYOUT_POST_TYPE, esc_html__( 'The Divi Builder', 'et_builder' ), 'et_pb_pagebuilder_meta_box', $post_type, 'normal', 'high' );
}
}
if ( ! function_exists( 'et_pb_get_the_author_posts_link' ) ) :
/**
* Return a post author link markup.
*/
function et_pb_get_the_author_posts_link() {
global $authordata, $post;
// Fallback for preview.
if ( empty( $authordata ) && isset( $post->post_author ) ) {
$authordata = get_userdata( $post->post_author ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited -- If $authordata is not set then set it.
}
// If $authordata is empty, don't continue.
if ( empty( $authordata ) ) {
return;
}
$link = sprintf(
'
%3$s ',
esc_url( get_author_posts_url( $authordata->ID, $authordata->user_nicename ) ),
// translators: post author name.
esc_attr( sprintf( __( 'Posts by %s', 'et_builder' ), get_the_author() ) ),
get_the_author()
);
return apply_filters( 'the_author_posts_link', $link );
}
endif;
if ( ! function_exists( 'et_pb_get_comments_popup_link' ) ) :
/**
* Return comments link.
*
* @param bool|string $zero text to display when 0 comments.
* @param bool|string $one text to display when 1 comment.
* @param bool|string $more text to display for more than 1 comments.
*/
function et_pb_get_comments_popup_link( $zero = false, $one = false, $more = false ) {
$id = get_the_ID();
$number = get_comments_number( $id );
if ( 0 === $number && ! comments_open() && ! pings_open() ) {
return;
}
if ( $number > 1 ) {
// translators: more comments text.
$output = str_replace( '%', number_format_i18n( $number ), ( false === $more ) ? __( '% Comments', 'et_builder' ) : $more );
} elseif ( 0 === $number ) {
$output = ( false === $zero ) ? __( 'No Comments', 'et_builder' ) : $zero;
} else { // must be one.
$output = ( false === $one ) ? __( '1 Comment', 'et_builder' ) : $one;
}
do_action( 'et_builder_before_comments_number' );
$link = '';
do_action( 'et_builder_after_comments_number' );
return $link;
}
endif;
if ( ! function_exists( 'et_pb_postinfo_meta' ) ) :
/**
* Return post meta.
*
* @param string[] $postinfo post info e.g date, author, categories.
* @param string $date_format date format.
* @param string $comment_zero text to display for 0 comments.
* @param string $comment_one text to display for 1 comments.
* @param string $comment_more text to display for more comments.
*/
function et_pb_postinfo_meta( $postinfo, $date_format, $comment_zero, $comment_one, $comment_more ) {
$postinfo_meta = array();
if ( in_array( 'author', $postinfo, true ) ) {
$postinfo_meta[] = ' ' . esc_html__( 'by', 'et_builder' ) . '
' . et_pb_get_the_author_posts_link() . ' ';
}
if ( in_array( 'date', $postinfo, true ) ) {
$postinfo_meta[] = '
' . esc_html( get_the_time( $date_format ) ) . ' ';
}
if ( in_array( 'categories', $postinfo, true ) ) {
$categories_list = get_the_category_list( ', ' );
// do not output anything if no categories retrieved.
if ( '' !== $categories_list ) {
$postinfo_meta[] = $categories_list;
}
}
if ( in_array( 'comments', $postinfo, true ) ) {
$postinfo_meta[] = et_pb_get_comments_popup_link( $comment_zero, $comment_one, $comment_more );
}
return implode( ' | ', array_filter( $postinfo_meta ) );
}
endif;
if ( ! function_exists( 'et_pb_fix_shortcodes' ) ) {
/**
* Fix shortcodes? @todo Add function doc.
*
* @param string $content post content.
* @param bool $is_raw_content whether content is row.
*
* @return string|string[]|null
*/
function et_pb_fix_shortcodes( $content, $is_raw_content = false ) {
// Turn back the "data-et-target-link" attribute as "target" attribte
// that has been made before saving the content in "et_fb_process_to_shortcode" function.
if ( false !== strpos( $content, 'data-et-target-link=' ) ) {
$content = str_replace( ' data-et-target-link=', ' target=', $content );
}
if ( $is_raw_content ) {
$content = et_builder_replace_code_content_entities( $content );
$content = ET_Builder_Element::convert_smart_quotes_and_amp( $content );
}
$slugs = ET_Builder_Element::get_module_slugs_by_post_type();
// The current patterns take care to replace only the shortcodes that extends `ET_Builder_Element` class
// In order to avoid cases like this: `[3:45]
`
// The pattern looks like this `(\[\/?(et_pb_section|et_pb_column|et_pb_row)[^\]]*\])`.
$shortcode_pattern = sprintf( '(\[\/?(%s)[^\]]*\])', implode( '|', $slugs ) );
$opening_pattern = '(
|
|\n)+';
$closing_pattern = '( |<\/p>|\n)+';
$space_pattern = '[\s*|\n]*';
// Replace `]
`, `]
` `]\n` with `]`
// Make sure to remove any closing `` tags or line breaks or new lines after shortcode tag.
$pattern_1 = sprintf( '/%1$s%2$s%3$s/', $shortcode_pattern, $space_pattern, $closing_pattern );
// Replace `
[`, ` [` `\n[` with `[`
// Make sure to remove any opening `
` tags or line breaks or new lines before shortcode tag.
$pattern_2 = sprintf( '/%1$s%2$s%3$s/', $opening_pattern, $space_pattern, $shortcode_pattern );
$content = preg_replace( $pattern_1, '$1', $content );
$content = preg_replace( $pattern_2, '$2', $content );
return $content;
}
}
if ( ! function_exists( 'et_pb_load_global_module' ) ) {
/**
* Return gloval module content.
*
* @param integer $global_id layout id.
* @param string $row_type row type.
* @param string $prev_bg Previous background color.
* @param string $next_bg next background color.
*
* @return string|string[]|null
*/
function et_pb_load_global_module( $global_id, $row_type = '', $prev_bg = '', $next_bg = '' ) {
$global_shortcode = '';
if ( '' !== $global_id ) {
$query = new WP_Query(
array(
'p' => (int) $global_id,
'post_type' => array(
ET_BUILDER_LAYOUT_POST_TYPE,
ET_THEME_BUILDER_HEADER_LAYOUT_POST_TYPE,
ET_THEME_BUILDER_BODY_LAYOUT_POST_TYPE,
ET_THEME_BUILDER_FOOTER_LAYOUT_POST_TYPE,
),
)
);
if ( ! empty( $query->post ) ) {
// Call the_post() to properly configure post data. Make sure to call the_post() and
// wp_reset_postdata() only if the posts result exist to avoid unexpected issues.
$query->the_post();
wp_reset_postdata();
$global_shortcode = $query->post->post_content;
if ( '' !== $row_type && 'et_pb_row_inner' === $row_type ) {
$global_shortcode = str_replace( 'et_pb_row', 'et_pb_row_inner', $global_shortcode );
$global_shortcode = str_replace( 'et_pb_column', 'et_pb_column_inner', $global_shortcode );
}
}
}
// Set provided prev_background_color.
if ( ! empty( $prev_bg ) ) {
$global_shortcode = preg_replace( '/prev_background_color="(.*?)"/', 'prev_background_color="' . $prev_bg . '"', $global_shortcode, 1 );
}
// Set provided next_background_color.
if ( ! empty( $next_bg ) ) {
$global_shortcode = preg_replace( '/next_background_color="(.*?)"/', 'next_background_color="' . $next_bg . '"', $global_shortcode, 1 );
}
return $global_shortcode;
}
}
if ( ! function_exists( 'et_pb_extract_shortcode_content' ) ) {
/**
* Return the shortcode content.
*
* @param string $content content.
* @param string $shortcode_name shortcode name.
*
* @return bool|false|string
*/
function et_pb_extract_shortcode_content( $content, $shortcode_name ) {
$start = strpos( $content, ']' ) + 1;
$end = strrpos( $content, '[/' . $shortcode_name );
if ( false !== $end ) {
$content = substr( $content, $start, $end - $start );
} else {
$content = (bool) false;
}
return $content;
}
}
if ( ! function_exists( 'et_pb_remove_shortcode_content' ) ) {
/**
* Remove the content part of the shortcode.
*
* @param string $content content.
* @param string $shortcode_name shortcode name.
*
* @return string|string[]
*/
function et_pb_remove_shortcode_content( $content, $shortcode_name ) {
$shortcode_content = et_pb_extract_shortcode_content( $content, $shortcode_name );
if ( $shortcode_content ) {
// Anchor to the ][ brackets around the content so content that appears in
// attributes does not get removed as well.
return str_replace( ']' . $shortcode_content . '[', '][', $content );
}
return $content;
}
}
if ( ! function_exists( 'et_pb_get_global_module_content' ) ) {
/**
* Return global module content.
*
* @param string $content content.
* @param string $shortcode_name shortcode slug.
* @param bool $for_inner_row whether we getting module content for inner row.
*
* @return bool|false|string|string[]|null
*/
function et_pb_get_global_module_content( $content, $shortcode_name, $for_inner_row = false ) {
/**
* Filter list of modules where we don't need to apply autop to the global module content.
*
* @param array Module slugs list.
*/
$custom_autop_ignored_modules = apply_filters( 'et_builder_global_modules_ignore_autop', array() );
$custom_autop_ignored_modules = is_array( $custom_autop_ignored_modules ) ? $custom_autop_ignored_modules : array();
$default_autop_ignored_modules = array_merge( array( 'et_pb_code', 'et_pb_fullwidth_code' ), $custom_autop_ignored_modules );
// Do not apply autop to code modules.
if ( in_array( $shortcode_name, $default_autop_ignored_modules, true ) ) {
return et_pb_extract_shortcode_content( $content, $shortcode_name );
}
$original_code_modules = array();
$shortcode_content = et_pb_extract_shortcode_content( $content, $shortcode_name );
// Getting content for Global row when it's turned to inner row in specialty section
// Need to make sure it wrapped in et_pb_column_inner, not et_pb_column.
if ( $for_inner_row && false === strpos( $shortcode_content, '[et_pb_column_inner' ) ) {
$shortcode_content = str_replace( 'et_pb_column', 'et_pb_column_inner', $shortcode_content );
}
// Get all the code and fullwidth code modules from content.
preg_match_all( '/(\[et_pb(_fullwidth_code|_code).+?\[\/et_pb(_fullwidth_code|_code)\])/s', $shortcode_content, $original_code_modules );
$global_content = et_pb_fix_shortcodes( wpautop( $shortcode_content ) );
// Replace content modified by wpautop for code and fullwidth code modules with original content.
if ( ! empty( $original_code_modules ) ) {
global $et_pb_global_code_replacements;
$et_pb_global_code_replacements = $original_code_modules[0];
$global_content = preg_replace_callback( '/(\[et_pb(_fullwidth_code|_code).+?\[\/et_pb(_fullwidth_code|_code)\])/s', 'et_builder_get_global_code_replacement', $global_content );
}
return $global_content;
}
}
if ( ! function_exists( 'et_builder_get_global_code_replacement' ) ) {
/**
* Retrieve the global code original instance to replace the modified in global code shortcode.
*
* @param array $matches found matches.
*
* @return mixed
*/
function et_builder_get_global_code_replacement( $matches ) {
global $et_pb_global_code_replacements;
return array_shift( $et_pb_global_code_replacements );
}
}
if ( ! function_exists( 'et_builder_activate_bfb_auto_draft' ) ) {
/**
* Force activate post_id which has auto-draft status
*/
function et_builder_activate_bfb_auto_draft() {
et_core_security_check( 'edit_posts', 'et_enable_bfb_nonce' );
$post_id = ! empty( $_POST['et_post_id'] ) ? absint( $_POST['et_post_id'] ) : 0;
if ( 0 === $post_id || ! current_user_can( 'edit_post', $post_id ) ) {
die();
}
// et_builder_activate_bfb_auto_draft() is executed when post title and content empty which means post_status is still lik. ely
// to be "auto-draft". "auto-draft" status returns 404 page; thus post status needs to be updated to "draft".
wp_update_post(
array(
'ID' => $post_id,
'post_status' => 'draft',
)
);
update_post_meta( $post_id, '_et_pb_use_builder', 'on' );
die();
}
}
add_action( 'wp_ajax_et_builder_activate_bfb_auto_draft', 'et_builder_activate_bfb_auto_draft' );
if ( ! function_exists( 'et_builder_ajax_toggle_bfb' ) ) {
/**
* Ajax Callback :: Switch To The New Divi Builder.
*/
function et_builder_ajax_toggle_bfb() {
et_core_security_check( 'manage_options', 'et_builder_toggle_bfb', 'nonce', '_GET' );
$enable = isset( $_GET['enable'] ) && '1' === $_GET['enable'];
$redirect = isset( $_GET['redirect'] ) ? esc_url_raw( $_GET['redirect'] ) : '';
if ( empty( $redirect ) && isset( $_SERVER['HTTP_REFERER'] ) ) {
$redirect = esc_url_raw( $_SERVER['HTTP_REFERER'] );
}
if ( empty( $redirect ) ) {
$redirect = esc_url_raw( admin_url( '/' ) );
}
et_builder_toggle_bfb( $enable );
set_transient( 'et_builder_show_bfb_welcome_modal', true, 0 );
wp_safe_redirect( $redirect );
exit;
}
}
add_action( 'wp_ajax_et_builder_toggle_bfb', 'et_builder_ajax_toggle_bfb' );
/**
* Return font weight select input element html.
*
* @return string
*/
function et_generate_font_weight_select_output() {
$all_weights = et_builder_get_font_weight_list();
$output = '';
foreach ( $all_weights as $number => $name ) {
$output .= sprintf(
' %2$s %3$s ',
esc_attr( $number ),
esc_html( $name ),
esc_html( $number )
);
}
return $output;
}
/**
* Return regular and specialty layouts.
*
* @return mixed|void
*/
function et_builder_get_columns() {
$columns = array(
'specialty' => array(
'1_2,1_2' => array(
'position' => '1,0',
'columns' => '3',
),
'1_2,1_2' => array(
'position' => '0,1',
'columns' => '3',
),
'1_4,3_4' => array(
'position' => '0,1',
'columns' => '3',
),
'3_4,1_4' => array(
'position' => '1,0',
'columns' => '3',
),
'1_4,1_4,1_2' => array(
'position' => '0,0,1',
'columns' => '3',
),
'1_2,1_4,1_4' => array(
'position' => '1,0,0',
'columns' => '3',
),
'1_4,1_2,1_4' => array(
'position' => '0,1,0',
'columns' => '3',
),
'1_3,2_3' => array(
'position' => '0,1',
'columns' => '4',
),
'2_3,1_3' => array(
'position' => '1,0',
'columns' => '4',
),
),
'regular' => array(
'4_4',
'1_2,1_2',
'1_3,1_3,1_3',
'1_4,1_4,1_4,1_4',
'1_5,1_5,1_5,1_5,1_5',
'1_6,1_6,1_6,1_6,1_6,1_6',
'2_5,3_5',
'3_5,2_5',
'1_3,2_3',
'2_3,1_3',
'1_4,3_4',
'3_4,1_4',
'1_4,1_2,1_4',
'1_5,3_5,1_5',
'1_4,1_4,1_2',
'1_2,1_4,1_4',
'1_5,1_5,3_5',
'3_5,1_5,1_5',
'1_6,1_6,1_6,1_2',
'1_2,1_6,1_6,1_6',
),
);
return apply_filters( 'et_builder_get_columns', $columns );
}
/**
* Return columns layout.
*
* @return mixed|void
*/
function et_builder_get_columns_layout() {
$layout_columns =
'<% if ( typeof et_pb_specialty !== \'undefined\' && et_pb_specialty === \'on\' ) { %>
<% } else if ( typeof view !== \'undefined\' && typeof view.model.attributes.specialty_columns !== \'undefined\' ) { %>
<% if ( view.model.attributes.layout === "2_3" ) { %>
<% } else { %>
<% } %>
<% } else { %>
<%
}
%>';
return apply_filters( 'et_builder_layout_columns', $layout_columns );
}
/**
* Display meta box in admin screen.
*/
function et_pb_pagebuilder_meta_box() {
global $typenow, $post;
do_action( 'et_pb_before_page_builder' );
if ( et_builder_bfb_enabled() ) {
$new_page_url = false;
$is_new_page = false;
$edit_page_id = get_the_ID();
$no_rtl_class = is_rtl() && 'on' === et_get_option( 'divi_disable_translations', 'off' ) ? 'et-fb-no-rtl' : '';
// Polylang creates copy of page and BFB should be loaded on page which is not saved yet and cannot be loaded on FE
// Therefore load the homepage and replace the content for BFB to make it load with content from other post.
if ( 'add' === get_current_screen()->action || (int) get_option( 'page_for_posts' ) === $edit_page_id ) {
$new_page_url = get_home_url();
$is_new_page = true;
}
$bfb_url = et_core_intentionally_unescaped( et_fb_get_bfb_url( $new_page_url, $is_new_page, $edit_page_id ), 'fixed_string' );
// 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.
$bfb_url = set_url_scheme( $bfb_url, is_ssl() ? 'https' : 'http' );
// phpcs:disable WordPress.Security.EscapeOutput -- XSS safe.
echo "
";
// phpcs:enable
return;
}
$new_builder_url_args = array(
'action' => 'et_builder_toggle_bfb',
'enable' => '1',
'nonce' => wp_create_nonce( 'et_builder_toggle_bfb' ),
);
$new_builder_url = add_query_arg( $new_builder_url_args, admin_url( 'admin-ajax.php' ) );
// Disable BFB notification on Extra category builder. BFB support for Extra category builder will be added post inital launch
// This option available for admins only.
if ( apply_filters( 'et_pb_display_bfb_notification_under_bb', true ) && current_user_can( 'manage_options' ) && et_pb_is_allowed( 'use_visual_builder' ) && et_pb_is_allowed( 'divi_builder_control' ) ) {
echo '
';
}
echo '
';
echo '
';
$content_editor_settings = array(
'media_buttons' => true,
'tinymce' => array(
'wp_autoresize_on' => true,
),
);
wp_editor( '', 'et_pb_content', $content_editor_settings );
echo '
';
echo '
';
$description_editor_settings = array(
'media_buttons' => true,
'tinymce' => array(
'wp_autoresize_on' => true,
),
);
wp_editor( '', 'et_pb_description', $description_editor_settings );
echo '
';
echo '';
echo '
';
printf(
'
',
esc_attr( $typenow ),
! et_pb_is_allowed( 'move_module' ) ? ' et-pb-disable-sort' : ''
);
$rename_module_menu = et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) ) ? sprintf(
'<%% if ( this.hasOption( "rename" ) ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Rename', 'et_builder' )
) : '';
$copy_module_menu = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( this.hasOption( "copy" ) ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Copy', 'et_builder' )
) : '';
$paste_after_menu = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( this.hasOption( "paste-after" ) ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Paste After', 'et_builder' )
) : '';
$paste_menu_item = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( this.hasOption( "paste-column" ) ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Paste', 'et_builder' )
) : '';
$paste_app_menu_item = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( this.hasOption( "paste-app" ) ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Paste', 'et_builder' )
) : '';
$save_to_lib_menu = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'save_library' ) ? sprintf(
'<%% if ( this.hasOption( "save-to-library") ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Save to Library', 'et_builder' )
) : '';
$lock_unlock_menu = et_pb_is_allowed( 'lock_module' ) ? sprintf(
'<%% if ( this.hasOption( "lock" ) ) { %%>
%1$s %2$s
<%% } %%>',
esc_html__( 'Unlock', 'et_builder' ),
esc_html__( 'Lock', 'et_builder' )
) : '';
$enable_disable_menu = et_pb_is_allowed( 'disable_module' ) ? sprintf(
'<%% if ( this.hasOption( "disable" ) ) { %%>
%1$s %2$s
<%% } %%>',
esc_html__( 'Enable', 'et_builder' ),
esc_html__( 'Disable', 'et_builder' )
) : '';
// Hide AB Testing menu if current post is Divi Library.
$is_divi_library = 'et_pb_layout' === $post->post_type;
$start_ab_testing_menu = et_pb_is_allowed( 'ab_testing' ) && ! $is_divi_library ? sprintf(
'<%% if ( this.hasOption( "start-ab-testing") ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Split Test', 'et_builder' )
) : '';
$end_ab_testing_menu = et_pb_is_allowed( 'ab_testing' ) && ! $is_divi_library ? sprintf(
'<%% if ( this.hasOption( "end-ab-testing") ) { %%>
%1$s
<%% } %%>',
esc_html__( 'End Split Test', 'et_builder' )
) : '';
$disable_global_menu = et_pb_is_allowed( 'edit_module' ) && et_pb_is_allowed( 'edit_global_library' ) ? sprintf(
'<%% if ( this.hasOption( "disable-global") ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Disable Global', 'et_builder' )
) : '';
// Right click options Template.
printf(
'',
et_core_esc_previously( $rename_module_menu ),
et_core_esc_previously( $enable_disable_menu ),
et_core_esc_previously( $lock_unlock_menu ),
et_builder_i18n( 'Expand' ),
esc_html__( 'Collapse', 'et_builder' ), // #5
et_core_esc_previously( $copy_module_menu ),
et_core_esc_previously( $paste_after_menu ),
et_core_esc_previously( $save_to_lib_menu ),
esc_html__( 'Undo', 'et_builder' ),
esc_html__( 'Redo', 'et_builder' ), // #10
et_core_esc_previously( $paste_menu_item ),
et_core_esc_previously( $paste_app_menu_item ),
et_core_esc_previously( et_pb_allowed_modules_list() ),
esc_html__( 'Preview', 'et_builder' ),
et_core_esc_previously( $start_ab_testing_menu ), // #15
et_core_esc_previously( $end_ab_testing_menu ),
et_core_esc_previously( $disable_global_menu )
);
// "Rename Module Admin Label" Modal Window Template
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_attr__( 'Save', 'et_builder' )
);
// "Rename Module Admin Label" Modal Content Template
printf(
'',
esc_html__( 'Rename', 'et_builder' ),
esc_html__( 'Enter a new name for this module', 'et_builder' )
);
// Builder's Main Buttons.
$save_to_lib_button = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'save_library' ) ? sprintf(
'
%2$s
',
esc_attr__( 'Save to Library', 'et_builder' ),
esc_html__( 'Save to Library', 'et_builder' )
) : '';
$load_from_lib_button = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'load_layout' ) && et_pb_is_allowed( 'add_library' ) && et_pb_is_allowed( 'add_module' ) ? sprintf(
'
%2$s
',
esc_attr__( 'Load From Library', 'et_builder' ),
esc_html__( 'Load Layout', 'et_builder' )
) : '';
$clear_layout_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'
%2$s
',
esc_attr__( 'Clear Layout', 'et_builder' ),
esc_html__( 'Clear Layout', 'et_builder' )
) : '';
// Builder's History Buttons.
$history_button = sprintf(
'
%2$s
',
esc_attr__( 'See History', 'et_builder' ),
esc_html__( 'See History', 'et_builder' )
);
$redo_button = sprintf(
'
%2$s
',
esc_attr__( 'Redo', 'et_builder' ),
esc_html__( 'Redo', 'et_builder' )
);
$undo_button = sprintf(
'
%2$s
',
esc_attr__( 'Undo', 'et_builder' ),
esc_html__( 'Undo', 'et_builder' )
);
// App View Stats Button.
$view_ab_stats_button = sprintf(
'
%2$s
',
esc_attr__( 'View Stats', 'et_builder' ),
esc_html__( 'View Stats', 'et_builder' )
);
// App Settings Button.
$settings_button = sprintf(
'
%2$s
',
esc_attr__( 'Settings', 'et_builder' ),
esc_html__( 'Settings', 'et_builder' )
);
// App Template.
printf(
'',
et_core_esc_previously( $save_to_lib_button ),
et_core_esc_previously( $load_from_lib_button ),
et_core_esc_previously( $clear_layout_button ),
et_core_esc_previously( $history_button ),
et_core_esc_previously( $redo_button ),
et_core_esc_previously( $undo_button ),
et_core_esc_previously( $view_ab_stats_button ),
et_core_esc_previously( $settings_button )
);
// App Settings Buttons Template.
$builder_button_ab_testing_conditional = '( typeof et_pb_ab_goal === "undefined" || et_pb_ab_goal === "off" || typeof et_pb_ab_subject !== "undefined" )';
$is_ab_active = isset( $post->ID ) && 'on' === get_post_meta( $post->ID, '_et_pb_use_ab_testing', true );
$view_stats_active_class = $is_ab_active ? 'active' : '';
$view_stats_button = et_pb_is_allowed( 'ab_testing' ) ? sprintf(
'
%2$s
',
esc_attr( $view_stats_active_class ),
esc_attr__( 'View Split Testing Stats', 'et_builder' ),
esc_url( ET_BUILDER_URI )
) : '';
$portability_class = 'et-pb-app-portability-button';
if ( $is_ab_active ) {
$portability_class .= ' et-core-disabled';
}
$page_settings_button = et_pb_is_allowed( 'page_options' ) ? sprintf(
'
%2$s
',
esc_attr__( 'Settings', 'et_builder' ),
esc_html__( 'Settings', 'et_builder' ),
esc_url( ET_BUILDER_URI )
) : '';
printf(
'',
et_core_esc_previously( $page_settings_button ),
et_core_esc_previously( et_builder_portability_link( 'et_builder', array( 'class' => $portability_class ) ) ),
et_core_esc_previously( $view_stats_button )
);
// do not display settings on global sections if not allowed for current user.
$global_settings_logic = ! et_pb_is_allowed( 'edit_global_library' ) ? ' && typeof et_pb_global_module === "undefined"' : '';
$section_settings_button = sprintf(
'<%% if ( ( typeof et_pb_template_type === \'undefined\' || \'section\' === et_pb_template_type || \'\' === et_pb_template_type )%3$s ) { %%>
%2$s
<%% } %%>',
esc_attr__( 'Settings', 'et_builder' ),
esc_html__( 'Settings', 'et_builder' ),
et_core_esc_previously( $global_settings_logic )
);
$section_clone_button = sprintf(
'%3$s
%2$s
%4$s',
esc_attr__( 'Clone Section', 'et_builder' ),
esc_html__( 'Clone Section', 'et_builder' ),
'<% if ( ' . et_core_esc_previously( $builder_button_ab_testing_conditional ) . ' ) { %>',
'<% } %>'
);
$section_remove_button = sprintf(
'%3$s
%2$s
%4$s',
esc_attr__( 'Delete Section', 'et_builder' ),
esc_html__( 'Delete Section', 'et_builder' ),
'<% if ( ' . et_core_esc_previously( $builder_button_ab_testing_conditional ) . ' ) { %>',
'<% } %>'
);
$section_unlock_button = sprintf(
'
%2$s ',
esc_attr__( 'Unlock Section', 'et_builder' ),
esc_html__( 'Unlock Section', 'et_builder' )
);
// Section Template.
$settings_controls = sprintf(
'
%1$s
<%% if ( typeof et_pb_template_type === \'undefined\' || ( \'section\' !== et_pb_template_type && \'row\' !== et_pb_template_type && \'module\' !== et_pb_template_type ) ) { %%>
%2$s
%3$s
<%% } %%>
%5$s
%6$s
',
et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) ) ? et_core_esc_previously( $section_settings_button ) : '',
et_pb_is_allowed( 'add_module' ) ? et_core_esc_previously( $section_clone_button ) : '',
et_pb_is_allowed( 'add_module' ) ? et_core_esc_previously( $section_remove_button ) : '',
esc_attr__( 'Expand Section', 'et_builder' ),
esc_html__( 'Expand Section', 'et_builder' ),
et_pb_is_allowed( 'lock_module' ) ? et_core_esc_previously( $section_unlock_button ) : ''
);
$settings_controls = apply_filters( 'et_builder_section_settings_controls', $settings_controls );
$add_from_lib_section = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'add_library' ) ? sprintf(
'
%1$s ',
esc_html__( 'Add From Library', 'et_builder' )
) : '';
$add_standard_section_button = sprintf(
'
%1$s ',
esc_html__( 'Standard Section', 'et_builder' )
);
$add_standard_section_button = apply_filters( 'et_builder_add_main_section_button', $add_standard_section_button );
$add_fullwidth_section_button = sprintf(
'
%1$s ',
esc_html__( 'Fullwidth Section', 'et_builder' )
);
$add_fullwidth_section_button = apply_filters( 'et_builder_add_fullwidth_section_button', $add_fullwidth_section_button );
$add_specialty_section_button = sprintf(
'
%1$s ',
esc_html__( 'Specialty Section', 'et_builder' )
);
$add_specialty_section_button = apply_filters( 'et_builder_add_specialty_section_button', $add_specialty_section_button );
$settings_add_controls = sprintf(
'<%% if ( typeof et_pb_template_type === \'undefined\' || ( \'section\' !== et_pb_template_type && \'row\' !== et_pb_template_type && \'module\' !== et_pb_template_type ) ) { %%>
%1$s
%2$s
%3$s
%4$s
<%% } %%>',
et_core_esc_previously( $add_standard_section_button ),
et_core_esc_previously( $add_fullwidth_section_button ),
et_core_esc_previously( $add_specialty_section_button ),
et_core_esc_previously( $add_from_lib_section )
);
$settings_add_controls = et_pb_is_allowed( 'add_module' ) ? apply_filters( 'et_builder_section_add_controls', $settings_add_controls ) : '';
$insert_first_row_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'
%1$s
',
esc_html__( 'Insert Row(s)', 'et_builder' )
) : '';
$disable_sort_logic = ! et_pb_is_allowed( 'move_module' ) ? ' et-pb-disable-sort' : '';
$disable_global_sort_logic = ! et_pb_is_allowed( 'edit_global_library' )
? '<%= typeof et_pb_global_module !== \'undefined\' ? \' et-pb-disable-sort\' : \'\' %>'
: '';
printf(
'',
et_core_esc_previously( $settings_controls ),
et_core_esc_previously( $settings_add_controls ),
et_core_intentionally_unescaped( $disable_sort_logic, 'fixed_string' ),
et_core_intentionally_unescaped( $disable_global_sort_logic, 'fixed_string' ),
et_core_esc_previously( $insert_first_row_button )
);
$row_settings_button = sprintf(
'<%% if ( ( typeof et_pb_template_type === \'undefined\' || et_pb_template_type !== \'module\' )%3$s ) { %%>
%2$s
<%% } %%>',
esc_attr__( 'Settings', 'et_builder' ),
esc_html__( 'Settings', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? ' && ( typeof et_pb_global_module === "undefined" || "" === et_pb_global_module ) && ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent )' : '' // do not display settings button on global rows if not allowed for current user.
);
$row_clone_button = sprintf(
'%3$s
%2$s
%4$s',
esc_attr__( 'Clone Row', 'et_builder' ),
esc_html__( 'Clone Row', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? '<% if ( ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent ) && ' . $builder_button_ab_testing_conditional . ' ) { %>' : '<% if ( ' . $builder_button_ab_testing_conditional . ' ) { %>', // do not display clone button on rows within global sections if not allowed for current user.
'<% } %>'
);
$row_remove_button = sprintf(
'%3$s
%2$s
%4$s',
esc_attr__( 'Delete Row', 'et_builder' ),
esc_html__( 'Delete Row', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? '<% if ( ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent ) && ' . $builder_button_ab_testing_conditional . ') { %>' : '<% if ( ' . $builder_button_ab_testing_conditional . ' ) { %>', // do not display clone button on rows within global sections if not allowed for current user.
'<% } %>'
);
$row_change_structure_button = sprintf(
'%3$s
%2$s
%4$s',
esc_attr__( 'Change Structure', 'et_builder' ),
esc_html__( 'Change Structure', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? '<% if ( ( typeof et_pb_global_module === "undefined" || "" === et_pb_global_module ) && ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent ) ) { %>' : '', // do not display change structure button on global rows if not allowed for current user.
! et_pb_is_allowed( 'edit_global_library' ) ? '<% } %>' : ''
);
$row_unlock_button = sprintf(
'
%2$s ',
esc_attr__( 'Unlock Row', 'et_builder' ),
esc_html__( 'Unlock Row', 'et_builder' )
);
// Row Template.
$settings = sprintf(
'
%1$s
<%% if ( typeof et_pb_template_type === \'undefined\' || \'section\' === et_pb_template_type ) { %%>
%2$s
<%% }
if ( typeof et_pb_template_type === \'undefined\' || et_pb_template_type !== \'module\' ) { %%>
%4$s
<%% }
if ( typeof et_pb_template_type === \'undefined\' || \'section\' === et_pb_template_type ) { %%>
%3$s
<%% } %%>
%6$s
%7$s
',
et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) ) ? $row_settings_button : '',
et_pb_is_allowed( 'add_module' ) ? $row_clone_button : '',
et_pb_is_allowed( 'add_module' ) ? $row_remove_button : '',
et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) ) ? $row_change_structure_button : '',
esc_attr__( 'Expand Row', 'et_builder' ),
esc_html__( 'Expand Row', 'et_builder' ),
et_pb_is_allowed( 'lock_module' ) ? $row_unlock_button : ''
);
$settings = apply_filters( 'et_builder_row_settings_controls', $settings );
$row_class = sprintf(
'class="et-pb-row-content et-pb-data-cid%1$s%2$s <%%= typeof et_pb_template_type !== \'undefined\' && \'module\' === et_pb_template_type ? \' et_pb_hide_insert\' : \'\' %%>"',
! et_pb_is_allowed( 'move_module' ) ? ' et-pb-disable-sort' : '',
! et_pb_is_allowed( 'edit_global_library' )
? sprintf( '<%%= typeof et_pb_global_parent !== \'undefined\' || typeof et_pb_global_module !== \'undefined\' ? \' et-pb-disable-sort\' : \'\' %%>' )
: ''
);
$data_skip = 'data-skip="<%= typeof( et_pb_skip_module ) === \'undefined\' ? \'false\' : \'true\' %>"';
$add_row_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( ( typeof et_pb_template_type === \'undefined\' || \'section\' === et_pb_template_type )%2$s ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Add Row', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? ' && typeof et_pb_global_parent === "undefined"' : '' // do not display add row buton on global sections if not allowed for current user.
) : '';
$insert_column_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'
%1$s
',
esc_html__( 'Insert Column(s)', 'et_builder' )
) : '';
printf(
'',
et_core_esc_previously( $settings ),
et_core_intentionally_unescaped( $row_class, 'fixed_string' ),
et_core_intentionally_unescaped( $data_skip, 'fixed_string' ),
et_core_esc_previously( $insert_column_button ),
et_core_esc_previously( $add_row_button )
);
// Module Block Template.
$clone_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( ( typeof et_pb_template_type === \'undefined\' || et_pb_template_type !== \'module\' )%3$s && _.contains(%4$s, module_type) && ' . $builder_button_ab_testing_conditional . ' ) { %%>
%2$s
<%% } %%>',
esc_attr__( 'Clone Module', 'et_builder' ),
esc_html__( 'Clone Module', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? ' && ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent )' : '',
et_pb_allowed_modules_list()
) : '';
$remove_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'<%% if ( ( typeof et_pb_template_type === \'undefined\' || et_pb_template_type !== \'module\' )%3$s && (_.contains(%4$s, module_type) || "removed" === component_status) && ' . $builder_button_ab_testing_conditional . ' ) { %%>
%2$s
<%% } %%>',
esc_attr__( 'Remove Module', 'et_builder' ),
esc_html__( 'Remove Module', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? ' && ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent )' : '',
et_pb_allowed_modules_list()
) : '';
$unlock_button = et_pb_is_allowed( 'lock_module' ) ? sprintf(
'<%% if ( typeof et_pb_template_type === \'undefined\' || et_pb_template_type !== \'module\' ) { %%>
%2$s
<%% } %%>',
esc_html__( 'Unlock Module', 'et_builder' ),
esc_attr__( 'Unlock Module', 'et_builder' )
) : '';
$settings_button = et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) ) ? sprintf(
'<%% if (%3$s _.contains( %4$s, module_type ) ) { %%>
%2$s
<%% } %%>',
esc_attr__( 'Module Settings', 'et_builder' ),
esc_html__( 'Module Settings', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? ' ( typeof et_pb_global_parent === "undefined" || "" === et_pb_global_parent ) && ( typeof et_pb_global_module === "undefined" || "" === et_pb_global_module ) &&' : '',
et_pb_allowed_modules_list()
) : '';
printf(
'',
et_core_esc_previously( $settings_button ),
et_core_esc_previously( $clone_button ),
et_core_esc_previously( $remove_button ),
et_core_esc_previously( $unlock_button )
);
// Modal Template.
$can_edit_or_has_modal_view_tab = et_pb_is_allowed( 'edit_module' ) && ( et_pb_is_allowed( 'general_settings' ) || et_pb_is_allowed( 'advanced_settings' ) || et_pb_is_allowed( 'custom_css_settings' ) );
$save_exit_button = $can_edit_or_has_modal_view_tab ? sprintf(
'
%1$s
',
esc_html__( 'Save & Exit', 'et_builder' )
) : '';
$save_template_button = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'save_library' ) ? sprintf(
'<%% if ( typeof et_pb_template_type === \'undefined\' || \'\' === et_pb_template_type ) { %%>
%1$s
<%% } %%>',
esc_html__( 'Save & Add To Library', 'et_builder' )
) : '';
$preview_template_button = sprintf(
'
%1$s
',
esc_html__( 'Preview', 'et_builder' )
);
$single_button_class = ! et_pb_is_allowed( 'divi_library' ) || ! et_pb_is_allowed( 'save_library' ) ? ' et_pb_single_button' : '';
$no_editing_class = $can_edit_or_has_modal_view_tab ? '' : ' et_pb_no_editing';
printf(
'',
et_builder_i18n( 'Cancel' ),
et_core_esc_previously( $save_template_button ),
et_core_esc_previously( $save_exit_button ),
et_core_intentionally_unescaped( $single_button_class, 'fixed_string' ),
et_core_esc_previously( $preview_template_button ),
et_core_intentionally_unescaped( $no_editing_class, 'fixed_string' )
);
// Column Settings Template.
$columns_number =
'<% if ( view.model.attributes.specialty_columns === 3 ) { %>
3
<% } else { %>
2
<% } %>';
$data_specialty_columns = sprintf(
'<%% if ( typeof view !== \'undefined\' && typeof view.model.attributes.specialty_columns !== \'undefined\' ) { %%>
data-specialty_columns="%1$s"
<%% } %%>',
$columns_number
);
$saved_row_tab = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'add_library' ) ? sprintf(
'
%1$s
',
esc_html__( 'Add From Library', 'et_builder' )
) : '';
$saved_row_container = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'add_library' )
? '<% if ( ( typeof change_structure === \'undefined\' || \'true\' !== change_structure ) && ( typeof et_pb_specialty === \'undefined\' || et_pb_specialty !== \'on\' ) ) { %>
<% } %>'
: '';
printf(
'',
esc_html__( 'Insert Columns', 'et_builder' ),
et_core_intentionally_unescaped( $data_specialty_columns, 'fixed_string' ),
esc_html__( 'New Row', 'et_builder' ),
et_core_esc_previously( $saved_row_tab ),
et_core_intentionally_unescaped( et_builder_get_columns_layout(), 'fixed_string' ),
et_core_intentionally_unescaped( $saved_row_container, 'fixed_string' )
);
// "Add Module" Template
$fullwidth_class =
'<% if ( typeof module.fullwidth_only !== \'undefined\' && module.fullwidth_only === \'on\' ) { %> et_pb_fullwidth_only_module<% } %>';
$saved_modules_tab = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'add_library' ) ? sprintf(
'
%1$s
',
esc_html__( 'Add From Library', 'et_builder' )
) : '';
$saved_modules_container = et_pb_is_allowed( 'divi_library' ) && et_pb_is_allowed( 'add_library' )
? '
'
: '';
printf(
'',
esc_html__( 'Insert Module', 'et_builder' ),
esc_html__( 'New Module', 'et_builder' ),
et_core_esc_previously( $saved_modules_tab ),
et_core_intentionally_unescaped( $fullwidth_class, 'fixed_string' ),
et_core_intentionally_unescaped( $saved_modules_container, 'fixed_string' ),
et_core_esc_previously( et_pb_allowed_modules_list() )
);
// Load Layout Template.
printf(
'',
esc_html__( 'Load Layout', 'et_builder' ),
esc_html__( 'Premade Layouts', 'et_builder' ),
esc_html__( 'Your Saved Layouts', 'et_builder' )
);
// Library Account Status Error.
$root_directory = defined( 'ET_BUILDER_PLUGIN_ACTIVE' ) ? ET_BUILDER_PLUGIN_DIR : get_template_directory();
$library_i18n = require $root_directory . '/cloud/i18n/library.php';
printf(
'
',
et_core_esc_previously( $library_i18n['Uh Oh!'] ),
et_core_esc_previously( $library_i18n['$expiredAccount'] ),
et_core_esc_previously( $library_i18n['Authentication Required'] ),
et_core_esc_previously( $library_i18n['$noAccount'] ),
et_core_esc_previously( $library_i18n['Username'] ),
et_core_esc_previously( $library_i18n['$usernameHelp'] ),
et_core_esc_previously( $library_i18n['API Key'] ),
et_core_esc_previously( $library_i18n['$apiKeyHelp'] ),
et_core_esc_previously( $library_i18n['Submit'] )
);
// Library Back Button.
echo '
';
$insert_module_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'%2$s
%1$s
%3$s',
esc_html__( 'Insert Module(s)', 'et_builder' ),
! et_pb_is_allowed( 'edit_global_library' ) ? '<% if ( typeof et_pb_global_parent === "undefined" ) { %>' : '',
! et_pb_is_allowed( 'edit_global_library' ) ? '<% } %>' : ''
) : '';
// Column Template.
printf(
'',
et_core_esc_previously( $insert_module_button )
);
// Insert Row(s).
$insert_row_button = et_pb_is_allowed( 'add_module' ) ? sprintf(
'
%1$s
',
esc_html__( 'Insert Row(s)', 'et_builder' )
) : '';
// Insert Row Template.
printf(
'',
et_core_esc_previously( $insert_row_button )
);
// Advanced Settings Buttons Module.
printf(
'',
esc_html__( 'Delete', 'et_builder' ),
esc_html__( 'Settings', 'et_builder' ),
esc_html__( 'Clone Module', 'et_builder' )
);
// Advanced Settings Modal Buttons Template.
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_html__( 'Save', 'et_builder' )
);
// "Deactivate Builder" Modal Message Template
printf(
'',
esc_html__( 'Disable Builder', 'et_builder' ),
esc_html__( 'All content created in the Divi Builder will be lost. Previous content will be restored.', 'et_builder' ),
esc_html__( 'Do you wish to proceed?', 'et_builder' )
);
// "Clear Layout" Modal Window Template
printf(
'',
esc_html__( 'Clear Layout', 'et_builder' ),
esc_html__( 'All of your current page content will be lost.', 'et_builder' ),
esc_html__( 'Do you wish to proceed?', 'et_builder' )
);
// "Reset Advanced Settings" Modal Template
printf(
'',
esc_html__( 'All advanced module settings in will be lost.', 'et_builder' ),
esc_html__( 'Do you wish to proceed?', 'et_builder' )
);
// "Save Layout" Modal Window Template
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_html__( 'Save', 'et_builder' )
);
// "Save Layout" Modal Content Template
printf(
'',
esc_html__( 'Save To Library', 'et_builder' ),
esc_html__( 'Save your current page to the Divi Library for later use.', 'et_builder' ),
esc_html__( 'Layout Name', 'et_builder' )
);
// "Delete Font" Modal Text
printf(
'',
esc_html__( 'Delete Font', 'et_builder' ),
sprintf( '%1$s %2$s?', esc_html__( 'Are you sure want to delete', 'et_builder' ), '<%= font_name %>' )
);
// "Upload Font" Modal Template
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_html__( 'Upload', 'et_builder' )
);
// "Upload Font" Modal Text
printf(
'',
esc_html__( 'Upload Font', 'et_builder' ),
esc_html__( 'Font Name', 'et_builder' ),
esc_html__( 'Drag Files Here', 'et_builder' ),
esc_html__( 'Choose Font Files', 'et_builder' ),
esc_html__( 'Supported Font Weights', 'et_builder' ),
esc_html__( 'All', 'et_builder' ),
esc_html__( 'Supported File Formats', 'et_builder' ),
esc_html__( 'Selected Font Files', 'et_builder' ),
esc_html__( 'Choose the font weights supported by your font. Select "All" if you don\'t know this information or if your font includes all weights.', 'et_builder' ),
et_core_esc_previously( et_generate_font_weight_select_output() )
);
// "Save Template" Modal Window Layout
printf(
'',
esc_attr__( 'Save And Add To Library', 'et_builder' )
);
// "Save Template" Content Layout
$layout_categories = get_terms( 'layout_category', array( 'hide_empty' => false ) );
$categories_output = sprintf(
'
';
}
$categories_output .= sprintf(
'
',
esc_html__( 'Create New Category', 'et_builder' )
);
$general_checkbox = sprintf(
'
%1$s
',
esc_html__( 'Include General settings', 'et_builder' )
);
$advanced_checkbox = sprintf(
'
%1$s
',
esc_html__( 'Include Advanced Design settings', 'et_builder' )
);
$css_checkbox = sprintf(
'
%1$s
',
esc_html__( 'Include Custom CSS', 'et_builder' )
);
printf(
'',
esc_html__( 'Here you can save the current item and add it to your Divi Library for later use as well.', 'et_builder' ),
esc_html__( 'Template Name', 'et_builder' ),
esc_html__( 'Save as Global:', 'et_builder' ),
esc_html__( 'Make this a global item', 'et_builder' ),
et_core_esc_previously( $categories_output )
);
// Prompt Modal Window Template.
printf(
'',
et_builder_i18n( 'No' ),
et_builder_i18n( 'Yes' )
);
// "Open Settings" Modal Window Template
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_html__( 'Save', 'et_builder' )
);
$utils = ET_Core_Data_Utils::instance();
$fields = array();
// Filter out fields not supposed to show in BB.
foreach ( ET_Builder_Settings::get_fields() as $key => $field ) {
if ( true === $utils->array_get( $field, 'show_in_bb', true ) ) {
$fields[ $key ] = $field;
}
}
// "Open Settings" Modal Content Template
printf(
'',
esc_html__( 'Divi Builder Settings', 'et_builder' ),
et_core_esc_previously( et_pb_get_builder_settings_fields( $fields ) )
);
// AB Testing.
$ab_testing = et_builder_ab_labels();
// "Turn off AB Testing" Modal Window Template
printf(
'',
et_builder_i18n( 'Cancel' ),
et_builder_i18n( 'Yes' )
);
// "Turn off AB Testing" Modal Content Template
printf(
'',
esc_html__( 'End Split Test?', 'et_builder' ),
esc_html__( 'Upon ending your split test, you will be asked to select which subject variation you would like to keep. Remaining subjects will be removed.', 'et_builder' ),
esc_html__( 'Note: this process cannot be undone.', 'et_builder' )
);
// AB Testing Alert :: Modal Window Template.
printf(
'',
esc_html__( 'Ok', 'et_builder' )
);
// AB Testing Alert :: Modal Content Template.
printf(
'',
esc_html__( 'An Error Occurred', 'et_builder' ),
esc_html__( 'For some reason, you cannot perform this task.', 'et_builder' )
);
// AB Testing Alert Yes/No :: Modal Window Template.
printf(
'',
et_builder_i18n( 'Cancel' ),
esc_html__( 'Proceed', 'et_builder' )
);
// AB Testing Alert Yes/No :: Modal Content Template.
printf(
'',
esc_html__( 'An Error Occurred', 'et_builder' ),
esc_html__( 'For some reason, you cannot perform this task.', 'et_builder' )
);
/**
* Split Testing :: Set global item winner status
*/
printf(
'',
et_core_esc_previously( $ab_testing['set_global_winner_status']['cancel'] ),
et_core_esc_previously( $ab_testing['set_global_winner_status']['proceed'] )
);
// AB Testing :: Set global item winner status template.
printf(
'',
et_core_esc_previously( $ab_testing['set_global_winner_status']['title'] ),
et_core_esc_previously( $ab_testing['set_global_winner_status']['desc'] ),
et_core_esc_previously( $ab_testing['set_global_winner_status']['option_1'] ),
et_core_esc_previously( $ab_testing['set_global_winner_status']['option_2'] )
);
/**
* AB Testing :: View Stats Template
*/
printf(
'',
et_builder_i18n( 'Cancel' )
);
$view_stats_tabs = '';
foreach ( et_pb_ab_get_analysis_types() as $analysis ) {
$view_stats_tabs .= sprintf(
'
',
esc_attr( $analysis ),
esc_html__( 'Last 24 Hours', 'et_builder' ),
esc_html__( 'Last 7 Days', 'et_builder' ),
esc_html__( 'Last Month', 'et_builder' ),
esc_html__( 'All Time', 'et_builder' ),
esc_html__( 'Summary & Data', 'et_builder' ),
esc_url( ET_BUILDER_URI ),
esc_html__( 'Statistics are still being collected for this time frame', 'et_builder' ),
esc_html__( 'Stats will be displayed upon sufficient data collection', 'et_builder' )
);
}
// AB Testing :: View Stats content.
printf(
'',
esc_html__( 'Split Testing Statistics', 'et_builder' ),
esc_html__( 'Clicks', 'et_builder' ),
esc_html__( 'Reads', 'et_builder' ),
esc_html__( 'Bounces', 'et_builder' ),
esc_html__( 'Goal Engagement', 'et_builder' ), // 5
esc_html__( 'Conversions', 'et_builder' ),
et_core_esc_previously( $view_stats_tabs ),
esc_url( ET_BUILDER_URI ),
esc_html__( 'Statistics are being collected', 'et_builder' ),
esc_html__( 'Stats will be displayed upon sufficient data collection', 'et_builder' ), // 10
esc_attr__( 'Refresh Stats', 'et_builder' ),
esc_html__( 'Refresh Stats', 'et_builder' ),
esc_html__( 'Shortcode Conversions', 'et_builder' ),
esc_attr__( 'End Split Test & Pick Winner', 'et_builder' )
);
// "Add Specialty Section" Button Template
printf(
'',
esc_html__( 'Add Specialty Section', 'et_builder' )
);
// Saved Entry Template.
echo '';
// Font Family Select Template.
$font_marker = et_pb_is_allowed( 'custom_fonts_management' ) ? '
' : '';
$upload_button = et_pb_is_allowed( 'custom_fonts_management' ) ? sprintf( '
', esc_html__( 'Upload', 'et_builder' ) ) : '';
printf(
'',
esc_html__( 'Uploaded', 'et_builder' ),
et_core_intentionally_unescaped( $font_marker, 'fixed_string' ),
et_core_esc_previously( $upload_button ),
et_core_esc_previously( et_builder_get_google_font_items() )
);
// Font Icons Template.
printf(
'',
et_core_esc_previously( et_pb_get_font_icon_list_items() )
);
// Histories Visualizer Item Template.
printf(
''
);
// Font Down Icons Template.
printf(
'',
et_core_esc_previously( et_pb_get_font_down_icon_list_items() )
);
printf(
'',
esc_html__( 'Mobile', 'et_builder' ),
et_builder_i18n( 'Tablet' ),
et_builder_i18n( 'Desktop' )
);
printf(
''
);
printf(
'',
et_builder_i18n( 'Desktop' ),
et_builder_i18n( 'Tablet' ),
esc_html__( 'Smartphone', 'et_builder' )
);
printf(
'',
et_builder_i18n( 'Default' ),
esc_html__( 'Hover', 'et_builder' )
);
printf(
'',
et_builder_i18n( 'Desktop' ),
esc_html__( 'Hover', 'et_builder' ),
et_builder_i18n( 'Tablet' ),
esc_html__( 'Smartphone', 'et_builder' )
);
printf(
''
);
printf(
''
);
printf(
''
);
print(
''
);
print(
''
);
print(
''
);
print(
''
);
printf(
''
);
printf(
'',
esc_html__( 'Font Weight', 'et_builder' )
);
printf(
'',
esc_html__( 'Font Style', 'et_builder' )
);
printf(
'',
esc_html__( 'Line Color', 'et_builder' ),
esc_attr__( 'Hex Value', 'et_builder' ),
esc_attr__( 'Line Style', 'et_builder' ),
esc_attr__( 'Choose Custom Color', 'et_builder' ),
esc_attr__( 'Underline', 'et_builder' ),
esc_attr__( 'Strikethrough', 'et_builder' ),
esc_attr__( 'Color', 'et_builder' ),
esc_attr__( 'Style', 'et_builder' )
);
printf(
''
);
printf(
''
);
printf(
'',
et_core_esc_previously( et_builder_get_failure_notification_modal() )
);
printf(
'',
et_core_esc_previously( et_builder_get_cache_notification_modal() )
);
printf(
'',
et_core_esc_previously( et_builder_page_creation_modal() )
);
// Help Template.
printf(
'',
esc_html__( 'Divi Builder Helper', 'et_builder' ),
esc_html__( 'Shortcuts', 'et_builder' )
);
do_action( 'et_pb_after_page_builder' );
}
/**
* Returns builder settings markup
*
* @param array $options builder settings' configuration.
* @return string builder settings' markup
*/
function et_pb_get_builder_settings_fields( $options ) {
$outputs = '';
$defaults = et_pb_get_builder_settings_configuration_default();
foreach ( $options as $option ) {
$option = wp_parse_args( $option, $defaults );
$type = $option['type'];
$field_list_class = $type;
$affecting = ! empty( $option['affects'] ) ? implode( '|', $option['affects'] ) : '';
if ( $option['depends_show_if'] ) {
$field_list_class .= ' et-pb-display-conditionally';
}
if ( isset( $option['class'] ) ) {
$field_list_class .= ' ' . $option['class'];
}
$outputs .= sprintf(
'
',
esc_attr( $field_list_class ),
esc_attr( $option['id'] ),
esc_attr( $type ),
esc_attr( $option['autoload'] ),
esc_attr( $affecting ),
esc_attr( $option['depends_show_if'] )
);
switch ( $option['type'] ) {
case 'yes_no_button':
$outputs .= sprintf(
'
%2$s
',
esc_attr( $option['id'] ),
esc_html( $option['label'] ),
isset( $option['values'] ) ? esc_html( $option['values']['yes'] ) : et_builder_i18n( 'Yes' ),
isset( $option['values'] ) ? esc_html( $option['values']['no'] ) : et_builder_i18n( 'No' ),
et_builder_i18n( 'Off' ),
et_builder_i18n( 'On' )
);
break;
case 'codemirror':
case 'textarea':
$outputs .= sprintf(
'
%2$s
',
esc_attr( $option['id'] ),
esc_html( $option['label'] ),
isset( $option['readonly'] ) && 'readonly' === $option['readonly'] ? ' readonly' : ''
);
break;
case 'colorpalette':
$outputs .= sprintf( '
%1$s ', esc_html( $option['label'] ) );
$outputs .= '
';
for ( $colorpalette_index = 1; $colorpalette_index < 9; $colorpalette_index++ ) {
$outputs .= sprintf( ' ', esc_attr( $colorpalette_index ) );
}
$outputs .= '
';
for ( $colorpicker_index = 1; $colorpicker_index < 9; $colorpicker_index++ ) {
$outputs .= sprintf(
'
',
esc_attr( $option['id'] ),
esc_attr( $colorpicker_index )
);
}
$outputs .= '
';
break;
case 'color-alpha':
$outputs .= sprintf(
'
%2$s
',
esc_attr( $option['id'] ),
esc_html( $option['label'] ),
esc_attr( $option['default'] )
);
break;
case 'range':
$outputs .= sprintf(
'
%2$s
',
esc_attr( $option['id'] ),
esc_html( $option['label'] ),
esc_attr( $option['range_settings']['step'] ),
esc_attr( $option['range_settings']['min'] ),
esc_attr( $option['range_settings']['max'] )
);
break;
case 'select':
$options = '';
foreach ( $option['options'] as $value => $text ) {
$options .= sprintf(
'
%2$s ',
esc_attr( $value ),
esc_html( $text )
);
}
$outputs .= sprintf(
'
%2$s
%3$s
',
esc_attr( $option['id'] ),
esc_html( $option['label'] ),
et_core_esc_previously( $options )
);
break;
}
$outputs .= sprintf( '
', esc_attr( $option['type'] ) );
}
return $outputs;
}
/**
* Prints hidden inputs for passing settings data to database
*
* @param integer $post_id post id.
*
* @return void|bool
*/
function et_pb_builder_settings_hidden_inputs( $post_id ) {
if ( ! class_exists( 'ET_Builder_Settings' ) ) {
return false;
}
$settings = ET_Builder_Settings::get_fields();
$defaults = et_pb_get_builder_settings_configuration_default();
if ( empty( $settings ) ) {
return;
}
if ( empty( $settings ) ) {
return;
}
foreach ( $settings as $setting ) {
$setting = wp_parse_args( $setting, $defaults );
if ( ! $setting['autoload'] ) {
continue;
}
$id = '_' . $setting['id'];
$meta_key = isset( $setting['meta_key'] ) ? $setting['meta_key'] : $id;
$value = get_post_meta( $post_id, $meta_key, true );
if ( ( ! $value || '' === $value ) && $setting['default'] ) {
$value = $setting['default'];
}
printf(
'
',
esc_attr( $id ),
esc_attr( $value )
);
}
}
/**
* Prints hidden inputs for passing global modules data to database.
*
* @param integer $post_id post id.
*
* @return void
*/
function et_pb_builder_global_library_inputs( $post_id ) {
global $typenow;
if ( 'et_pb_layout' !== $typenow ) {
return;
}
$template_scope = wp_get_object_terms( get_the_ID(), 'scope' );
$template_type = wp_get_object_terms( get_the_ID(), 'layout_type' );
$is_global_template = ! empty( $template_scope[0] ) ? $template_scope[0]->slug : 'regular';
$template_type_slug = ! empty( $template_type[0] ) ? $template_type[0]->slug : '';
if ( 'global' !== $is_global_template || 'module' !== $template_type_slug ) {
return;
}
$excluded_global_options = get_post_meta( $post_id, '_et_pb_excluded_global_options' );
printf(
'
',
isset( $excluded_global_options[0] ) ? esc_attr( $excluded_global_options[0] ) : wp_json_encode( array() )
);
}
/**
* Returns array of default builder settings configuration item
*
* @return array
*/
function et_pb_get_builder_settings_configuration_default() {
return array(
'id' => '',
'type' => '',
'label' => '',
'min' => '',
'max' => '',
'step' => '',
'autoload' => true,
'default' => false,
'affects' => array(),
'depends_show_if' => false,
);
}
/**
* Update a builder setting.
*
* @param array $settings The new option value.
* @param string $post_id The post id or 'global' for global settings.
*/
function et_builder_update_settings( $settings, $post_id = 'global' ) {
// Allow the use of uppercase in $is_bb variable as BB is common abbreviation.
et_core_nonce_verified_previously();
$is_global = 'global' === $post_id;
$is_bb = null === $settings;
$settings = $is_bb ? $_POST : $settings;
$fields = $is_global ? ET_Builder_Settings::get_fields( 'builder' ) : ET_Builder_Settings::get_fields();
$utils = ET_Core_Data_Utils::instance();
$update = array();
foreach ( (array) $settings as $setting_key => $setting_value ) {
$raw_setting_value = $setting_value;
$setting_key = $is_bb ? substr( $setting_key, 1 ) : $setting_key;
// Verify setting key.
if ( ! isset( $fields[ $setting_key ] ) || ! isset( $fields[ $setting_key ]['type'] ) ) {
continue;
}
// Auto-formatting subjects' value format.
if ( 'et_pb_ab_subjects' === $setting_key && is_array( $setting_value ) ) {
$setting_value = implode( ',', $setting_value );
}
// TODO Possibly move sanitization.php to builder dir
// Sanitize value.
switch ( $fields[ $setting_key ]['type'] ) {
case 'colorpalette':
$palette_colors = explode( '|', $setting_value );
$setting_value = implode( '|', array_map( 'et_sanitize_alpha_color', $palette_colors ) );
break;
case 'range':
// Avoid setting absolute value for range if option is z_index.
if ( 'et_pb_page_z_index' === $setting_key ) {
break;
}
$setting_value = absint( $setting_value );
$range_min = isset( $fields[ $setting_key ]['range_settings'] ) && isset( $fields[ $setting_key ]['range_settings']['min'] ) ?
absint( $fields[ $setting_key ]['range_settings']['min'] ) : -1;
$range_max = isset( $fields[ $setting_key ]['range_settings'] ) && isset( $fields[ $setting_key ]['range_settings']['max'] ) ?
absint( $fields[ $setting_key ]['range_settings']['max'] ) : -1;
if ( $setting_value < $range_min || $range_max < $setting_value ) {
continue 2;
}
break;
case 'color-alpha':
$setting_value = et_sanitize_alpha_color( $setting_value );
break;
case 'codemirror':
case 'textarea':
// Allow HTML content on Excerpt field.
if ( 'et_pb_post_settings_excerpt' === $setting_key ) {
$setting_value = wp_kses_post( $setting_value );
} else {
$setting_value = sanitize_textarea_field( $setting_value );
}
break;
case 'categories':
$setting_value = array_map( 'intval', explode( ',', $setting_value ) );
break;
default:
$setting_value = sanitize_text_field( $setting_value );
break;
}
// check whether or not the defined value === default value.
$is_default = isset( $fields[ $setting_key ]['default'] ) && $setting_value === $fields[ $setting_key ]['default'];
// Auto-formatting AB Testing status' meta key.
if ( 'et_pb_enable_ab_testing' === $setting_key ) {
$setting_key = 'et_pb_use_ab_testing';
}
/**
* Fires before updating a builder setting in the database.
*
* @param string $setting_key The option name/id.
* @param string $setting_value The new option value.
* @param string|int $post_id The post id or 'global' for global settings.
*/
do_action( 'et_builder_settings_update_option', $setting_key, $setting_value, $post_id );
// If `post_field` is defined, we need to update the post.
$post_field = $utils->array_get( $fields, "{$setting_key}.post_field", false );
if ( false !== $post_field ) {
// Only allowed in VB.
if ( ! ( $is_global || $is_bb ) ) {
// Save the post field so we can do a single update.
// Use the raw value and rely on wp_update_post to sanitize it in order to allow certain HTML tags.
$update[ $post_field ] = $raw_setting_value;
}
continue;
}
// If `taxonomy_name` is defined, we need to update the post terms.
$taxonomy_name = $utils->array_get( $fields, "{$setting_key}.taxonomy_name", false );
if ( false !== $taxonomy_name ) {
// Only allowed in VB.
if ( ! ( $is_global || $is_bb ) ) {
$post_type = $utils->array_get( $fields, "{$setting_key}.post_type", false );
if ( get_post_type( $post_id ) === $post_type ) {
// Only update if the post type matches.
wp_set_object_terms( $post_id, $setting_value, $taxonomy_name );
}
}
continue;
}
// Save the setting in a post meta.
$meta_key = $utils->array_get( $fields, $setting_key . '.meta_key', false ) ? $fields[ $setting_key ]['meta_key'] : "_{$setting_key}";
$save_post = $utils->array_get( $fields, $setting_key . '.save_post', true );
if ( $is_bb && false === $save_post ) {
// This meta key must be ignored during classic-editor / BB save action or it will
// overwrite values in the WP edit page.
continue;
}
// remove if value is default.
if ( $is_default ) {
$is_global ? et_delete_option( $setting_key ) : delete_post_meta( $post_id, $meta_key );
} else {
// Update.
$is_global ? et_update_option( $setting_key, $setting_value ) : update_post_meta( $post_id, $meta_key, $setting_value );
}
// Removing autosave.
delete_post_meta( $post_id, "{$meta_key}_draft" );
}
// Removing builder settings autosave.
$current_user_id = get_current_user_id();
delete_post_meta( $post_id, "_et_builder_settings_autosave_{$current_user_id}" );
if ( count( $update ) > 0 ) {
// This MUST NOT be executed while saving data in the BB or it will generate
// an update loop that will end the universe as we know it.
if ( ! ( $is_bb || wp_is_post_revision( $post_id ) ) ) {
$update['ID'] = $post_id;
wp_update_post( $update );
}
}
}
/**
* Returns array of default color pallete.
*
* @param integer $post_id post id.
*
* @return array default color palette
*/
function et_pb_get_default_color_palette( $post_id = 0 ) {
$default_palette = array(
'#000000',
'#FFFFFF',
'#E02B20',
'#E09900',
'#EDF000',
'#7CDA24',
'#0C71C3',
'#8300E9',
);
$saved_global_palette = et_get_option( 'divi_color_palette', false );
$palette = $saved_global_palette && '' !== str_replace( '|', '', $saved_global_palette ) ? explode( '|', $saved_global_palette ) : $default_palette;
return apply_filters( 'et_pb_get_default_color_palette', $palette, $post_id );
}
/**
* Modify builder editor's TinyMCE configuration
*
* @param array $mce_init An array with TinyMCE config.
* @param string $editor_id Unique editor identifier.
*
* @return array
*/
function et_pb_content_mce_config( $mce_init, $editor_id ) {
if ( 'et_pb_content' === $editor_id && isset( $mce_init['toolbar1'] ) ) {
// Get toolbar as array.
$toolbar1 = explode( ',', $mce_init['toolbar1'] );
// Look for read more (wp_more)'s array' key.
$wp_more_key = array_search( 'wp_more', $toolbar1, true );
if ( $wp_more_key ) {
unset( $toolbar1[ $wp_more_key ] );
}
// Update toolbar1 configuration.
$mce_init['toolbar1'] = implode( ',', $toolbar1 );
}
return $mce_init;
}
add_filter( 'tiny_mce_before_init', 'et_pb_content_mce_config', 10, 2 );
/**
* Get post format with filterable output
*
* @todo once WordPress provide filter for get_post_format() output, this function can be retired
* @see get_post_format()
*
* @return mixed string|bool string of post format or false for default
*/
function et_pb_post_format() {
return apply_filters( 'et_pb_post_format', get_post_format(), get_the_ID() );
}
/**
* Return post format into false when using pagebuilder.
*
* @param string $post_format post format.
* @param integer $post_id post id.
*
* @return mixed string|bool string of post format or false for default
*/
function et_pb_post_format_in_pagebuilder( $post_format, $post_id ) {
if ( et_pb_is_pagebuilder_used( $post_id ) ) {
return false;
}
return $post_format;
}
add_filter( 'et_pb_post_format', 'et_pb_post_format_in_pagebuilder', 10, 2 );
if ( ! function_exists( 'et_get_first_audio_block' ) ) :
/**
* Return the first audio block from the post content.
*/
function et_get_first_audio_block() {
$content = get_the_content();
// It is assumed that audio module figures will not contain other figures.
preg_match( '/
]*?class=([\'"])[^\'"]*?wp-block-audio[^\'"]*?\1[^>]*?>.*?<\/figure>/', $content, $matches );
if ( empty( $matches ) ) {
return '';
}
return $matches[0];
}
endif;
if ( ! function_exists( 'et_pb_get_audio_player' ) ) :
/**
* Return audio player.
*/
function et_pb_get_audio_player() {
global $_et_pbgap_audio_to_remove;
$shortcode_audio = '';
$regex = get_shortcode_regex( array( 'audio' ) );
preg_match_all( "/{$regex}/s", get_the_content(), $matches );
foreach ( $matches[2] as $key => $shortcode_match ) {
// Remove audio shortcode if its contains first attached audio file URL
// first attached audio file is automatically appended on post's format content.
if ( 'audio' === $shortcode_match ) {
$_et_pbgap_audio_to_remove = $matches[0][0];
$shortcode_audio = do_shortcode( $_et_pbgap_audio_to_remove );
break;
}
}
if ( '' === $shortcode_audio ) {
$_et_pbgap_audio_to_remove = et_get_first_audio_block();
$shortcode_audio = $_et_pbgap_audio_to_remove;
}
if ( '' === $shortcode_audio ) {
return false;
}
$output = sprintf(
'
%1$s
',
$shortcode_audio
);
add_filter( 'the_content', 'et_delete_post_audio' );
return $output;
}
endif;
if ( ! function_exists( 'et_divi_post_format_content' ) ) :
/**
* Displays post audio, quote and link post formats content
*/
function et_divi_post_format_content() {
$post_format = et_pb_post_format();
$text_color_class = et_divi_get_post_text_color();
$inline_style = et_divi_get_post_bg_inline_style();
global $post;
if ( post_password_required( $post ) ) {
return;
}
switch ( $post_format ) {
case 'audio':
printf(
'',
esc_html( get_the_title() ),
et_core_intentionally_unescaped( et_pb_get_audio_player(), 'html' ),
esc_url( get_permalink() ),
esc_attr( $text_color_class ),
et_core_esc_previously( $inline_style )
);
break;
case 'quote':
printf(
'',
et_core_intentionally_unescaped( et_get_blockquote_in_content(), 'html' ),
esc_url( get_permalink() ),
esc_html__( 'Read more', 'et_builder' ),
esc_attr( $text_color_class ),
et_core_esc_previously( $inline_style )
);
break;
case 'link':
printf(
'',
esc_html( get_the_title() ),
esc_url( get_permalink() ),
esc_url( et_get_link_url() ),
esc_html( et_get_link_url() ),
esc_attr( $text_color_class ),
et_core_esc_previously( $inline_style )
);
break;
}
}
endif;
if ( ! function_exists( 'et_get_blockquote_in_content' ) ) :
/**
* Extract and return the first blockquote from content.
*/
function et_get_blockquote_in_content() {
global $more;
$more_default = $more;
$more = 1; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited -- Disable `$more` flag to get the blockquote that may exist after tag.
remove_filter( 'the_content', 'et_remove_blockquote_from_content' );
$content = apply_filters( 'the_content', get_the_content() );
add_filter( 'the_content', 'et_remove_blockquote_from_content' );
$more = $more_default; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited -- Restore default `$more` flag.
if ( preg_match( '//is', $content, $matches ) ) {
return $matches[0];
} else {
return false;
}
}
endif;
if ( ! function_exists( 'et_get_link_url' ) ) :
/**
* Return link from the post content.
*/
function et_get_link_url() {
$link_url = get_post_meta( get_the_ID(), '_format_link_url', true );
if ( '' !== $link_url ) {
return $link_url;
}
$content = get_the_content();
$has_url = get_url_in_content( $content );
return ( $has_url ) ? $has_url : apply_filters( 'the_permalink', get_permalink() );
}
endif;
if ( ! function_exists( 'et_get_first_video' ) ) :
/**
* Fix the issue with thumbnail video player, not working, when video url is added to content without shortcode
*/
function et_get_first_video() {
$first_url = '';
$first_video = '';
$video_width = (int) apply_filters( 'et_blog_video_width', 1080 );
$video_height = (int) apply_filters( 'et_blog_video_height', 630 );
$i = 0;
$content = get_the_content();
preg_match_all( '|^\s*https?://[^\s"]+\s*$|im', $content, $urls );
foreach ( $urls[0] as $url ) {
$i++;
if ( 1 === $i ) {
$first_url = trim( $url );
}
$oembed = wp_oembed_get( esc_url( $url ) );
if ( ! $oembed ) {
continue;
}
$first_video = $oembed;
$first_video = preg_replace( '//', ' ', $first_video );
// If the url comes from a GB embed block.
if ( preg_match( '|wp-block-embed.+?' . preg_quote( $url, null ) . '|s', $content ) ) {
// We need to remove some useless markup later.
add_filter( 'the_content', 'et_delete_post_video' );
}
break;
}
if ( '' === $first_video ) {
// Gutenberg compatibility.
if ( ! has_shortcode( $content, 'video' ) && empty( $first_url ) ) {
preg_match( '//', $content, $gb_video );
$first_url = isset( $gb_video[1] ) ? $gb_video[1] : false;
}
if ( ! has_shortcode( $content, 'video' ) && ! empty( $first_url ) ) {
$video_shortcode = sprintf( '[video src="%1$s" /]', esc_attr( $first_url ) );
if ( ! empty( $gb_video ) ) {
$content = str_replace( $gb_video[0], $video_shortcode, $content );
} else {
$content = str_replace( $first_url, $video_shortcode, $content );
}
}
if ( has_shortcode( $content, 'video' ) ) {
$regex = get_shortcode_regex();
preg_match( "/{$regex}/s", $content, $match );
$first_video = preg_replace( '/width="[0-9]*"/', "width=\"{$video_width}\"", $match[0] );
$first_video = preg_replace( '/height="[0-9]*"/', "height=\"{$video_height}\"", $first_video );
add_filter( 'the_content', 'et_delete_post_video' );
$first_video = do_shortcode( et_pb_fix_shortcodes( $first_video ) );
}
}
return ( '' !== $first_video ) ? $first_video : false;
}
endif;
if ( ! function_exists( 'et_delete_post_video' ) ) :
/**
* Removes the first video shortcode from content on single pages since it is displayed
* at the top of the page. This will also remove the video shortcode url from archive pages content
*
* @param string $content post content.
*/
function et_delete_post_video( $content ) {
if ( has_post_format( 'video' ) ) :
if ( has_shortcode( $content, 'video' ) ) {
$regex = get_shortcode_regex();
preg_match_all( "/{$regex}/s", $content, $matches );
// $matches[2] holds an array of shortcodes names in the post
foreach ( $matches[2] as $key => $shortcode_match ) {
if ( 'video' === $shortcode_match ) {
$content = str_replace( $matches[0][ $key ], '', $content );
if ( is_single() && is_main_query() ) {
break;
}
}
}
} else {
// Gutenberg compatibility.
preg_match( '//', $content, $gb_video );
if ( ! empty( $gb_video[0] ) ) {
$content = str_replace( $gb_video[0], '', $content );
} else {
// Remove GB embed caption for the first video.
$content = preg_replace( '| $post ) {
if ( isset( $post['excerpt'] ) ) {
// Remove ET shortcodes from JetPack excerpt.
$results[ $key ]['excerpt'] = preg_replace( '#\[et_pb(.*)\]#', '', $post['excerpt'] );
}
}
return $results;
}
add_filter( 'jetpack_relatedposts_returned_results', 'et_jetpack_post_excerpt' );
/**
* Adds a Divi gallery type when the Jetpack plugin is enabled.
*
* @param array $types gallery types.
*/
function et_jetpack_gallery_type( $types ) {
$types['divi'] = 'Divi';
return $types;
}
add_filter( 'jetpack_gallery_types', 'et_jetpack_gallery_type' );
if ( ! function_exists( 'et_get_gallery_attachments' ) ) :
/**
* Fetch the gallery attachments
*
* @param array $attr gallery shortcode attributes.
*/
function et_get_gallery_attachments( $attr ) {
// We're trusting author input, so let's at least make sure it looks like a valid orderby statement.
if ( isset( $attr['orderby'] ) ) {
$attr['orderby'] = sanitize_sql_orderby( $attr['orderby'] );
if ( ! $attr['orderby'] ) {
unset( $attr['orderby'] );
}
}
$html5 = current_theme_supports( 'html5', 'gallery' );
$pairs = array(
'order' => 'ASC',
'orderby' => 'menu_order ID',
'id' => get_the_ID() ? get_the_ID() : 0,
'itemtag' => $html5 ? 'figure' : 'dl',
'icontag' => $html5 ? 'div' : 'dt',
'captiontag' => $html5 ? 'figcaption' : 'dd',
'columns' => 3,
'size' => 'thumbnail',
'include' => '',
'exclude' => '',
'link' => '',
);
$atts = shortcode_atts( $pairs, $attr, 'gallery' );
$id = intval( $atts['id'] );
if ( 'RAND' === $atts['order'] ) {
$atts['orderby'] = 'none';
}
if ( ! empty( $atts['include'] ) ) {
$_attachments = get_posts(
array(
'include' => $atts['include'],
'post_status' => 'inherit',
'post_type' => 'attachment',
'post_mime_type' => 'image',
'order' => $atts['order'],
'orderby' => $atts['orderby'],
)
);
$attachments = array();
foreach ( $_attachments as $key => $val ) {
$attachments[ $val->ID ] = $_attachments[ $key ];
}
} elseif ( ! empty( $atts['exclude'] ) ) {
$attachments = get_children(
array(
'post_parent' => $id,
'exclude' => $atts['exclude'],
'post_status' => 'inherit',
'post_type' => 'attachment',
'post_mime_type' => 'image',
'order' => $atts['order'],
'orderby' => $atts['orderby'],
)
);
} else {
$attachments = get_children(
array(
'post_parent' => $id,
'post_status' => 'inherit',
'post_type' => 'attachment',
'post_mime_type' => 'image',
'order' => $atts['order'],
'orderby' => $atts['orderby'],
)
);
}
return $attachments;
}
endif;
/**
* Generate the HTML for custom gallery layouts.
*
* @param string $val output.
* @param string $attr gallery settings.
*
* @return string
*/
function et_gallery_layout( $val, $attr ) {
// check to see if the gallery output is already rewritten.
if ( ! empty( $val ) ) {
return $val;
}
// Do not filter gallery items in plugin to not break custom styles which may be applied by 3rd party theme.
if ( et_is_builder_plugin_active() ) {
return $val;
}
if ( ! apply_filters( 'et_gallery_layout_enable', false ) ) {
return $val;
}
$output = '';
if ( ! is_singular() && ! et_pb_is_pagebuilder_used( get_the_ID() ) && ! is_et_pb_preview() ) {
$attachments = et_get_gallery_attachments( $attr );
$gallery_output = '';
foreach ( $attachments as $attachment ) {
$attachment_image = wp_get_attachment_url( $attachment->ID, 'et-pb-post-main-image-fullwidth' );
$gallery_output .= sprintf(
'
',
esc_attr( $attachment_image )
);
}
$output = sprintf(
'',
$gallery_output
);
} else {
if ( ! isset( $attr['type'] ) || ! in_array( $attr['type'], array( 'rectangular', 'square', 'circle', 'rectangle' ), true ) ) {
$attachments = et_get_gallery_attachments( $attr );
$gallery_output = '';
foreach ( $attachments as $attachment ) {
$gallery_output .= sprintf(
'
%2$s
%4$s
',
esc_url( wp_get_attachment_url( $attachment->ID, 'full' ) ),
wp_get_attachment_image( $attachment->ID, 'et-pb-portfolio-image' ),
esc_attr( $attachment->post_title ),
! empty( $attachment->post_excerpt )
? sprintf( '%1$s
', esc_html( $attachment->post_excerpt ) )
: ''
);
}
$output = sprintf(
'',
$gallery_output
);
}
}
return $output;
}
add_filter( 'post_gallery', 'et_gallery_layout', 1000, 2 );
if ( ! function_exists( 'et_pb_gallery_images' ) ) :
/**
* Display image gallery.
*
* @param string $force_gallery_layout Optional. Slider gallery layout.
*/
function et_pb_gallery_images( $force_gallery_layout = '' ) {
if ( 'slider' === $force_gallery_layout ) {
// Get the post content.
$post = get_post( get_the_ID() );
if ( ! $post ) {
return '';
}
// We want to include GB galleries in results so need to convert them to shortcodes
// because `get_post_gallery` won't find them otherwise.
$post->post_content = apply_filters( 'et_gb_galleries_to_shortcodes', $post->post_content );
$attachments = get_post_gallery( $post, false );
$gallery_output = '';
$output = '';
$images_array = ! empty( $attachments['ids'] ) ? explode( ',', $attachments['ids'] ) : array();
if ( empty( $images_array ) ) {
return $output;
}
foreach ( $images_array as $attachment ) {
$image_src = wp_get_attachment_url( $attachment, 'et-pb-post-main-image-fullwidth' );
$gallery_output .= sprintf(
'
',
esc_url( $image_src )
);
}
printf(
'',
et_core_esc_previously( $gallery_output )
);
} else {
add_filter( 'et_gallery_layout_enable', 'et_gallery_layout_turn_on' );
printf(
do_shortcode( '%1$s' ),
et_core_intentionally_unescaped( get_post_gallery(), 'html' )
);
remove_filter( 'et_gallery_layout_enable', 'et_gallery_layout_turn_on' );
}
}
endif;
/**
* Used to always use divi gallery on et_pb_gallery_images
*/
function et_gallery_layout_turn_on() {
return true;
}
/**
* Remove Elegant Builder plugin filter, that activates visual mode on each page load in WP-Admin
*/
function et_pb_remove_lb_plugin_force_editor_mode() {
remove_filter( 'wp_default_editor', 'et_force_tmce_editor' );
}
add_action( 'admin_init', 'et_pb_remove_lb_plugin_force_editor_mode' );
/**
*
* Generates array of all Role options
*/
function et_pb_all_role_options() {
// get all the modules and build array of capabilities for them.
$all_modules_array = ET_Builder_Element::get_modules_array();
$custom_user_tabs = ET_Builder_Element::get_tabs();
$options_categories = ET_Builder_Element::get_options_categories();
$module_capabilies = array();
$tabs_array = array(
'general_settings' => array(
'name' => esc_html__( 'Content Settings', 'et_builder' ),
),
'advanced_settings' => array(
'name' => esc_html__( 'Design Settings', 'et_builder' ),
),
'custom_css_settings' => array(
'name' => esc_html__( 'Advanced Settings', 'et_builder' ),
),
);
// add all custom user tabs into list.
if ( ! empty( $custom_user_tabs ) ) {
foreach ( $custom_user_tabs as $module => $tabs_data ) {
if ( ! empty( $tabs_data ) ) {
foreach ( $tabs_data as $tab_slug => $tab_data ) {
$tabs_array[ $tab_slug ] = array(
'name' => $tab_data['name'],
);
}
}
}
}
foreach ( $all_modules_array as $module => $module_details ) {
if ( ! in_array( $module_details['label'], array( 'et_pb_section', 'et_pb_row', 'et_pb_row_inner', 'et_pb_column' ), true ) ) {
$module_capabilies[ $module_details['label'] ] = array(
'name' => sanitize_text_field( $module_details['title'] ),
);
}
}
// we need to display some options only when theme activated.
$theme_only_options = ! et_is_builder_plugin_active()
? array(
'theme_customizer' => array(
'name' => esc_html__( 'Theme Customizer', 'et_builder' ),
'applicability' => array( 'administrator' ),
),
'page_options' => array(
'name' => esc_html__( 'Page Options', 'et_builder' ),
),
)
: array();
$all_role_options = array(
'general_capabilities' => array(
'section_title' => '',
'options' => array(
'theme_options' => array(
'name' => et_is_builder_plugin_active() ? esc_html__( 'Plugin Options', 'et_builder' ) : esc_html__( 'Theme Options', 'et_builder' ),
'applicability' => array( 'administrator' ),
),
'divi_library' => array(
'name' => esc_html__( 'Divi Library', 'et_builder' ),
),
'theme_builder' => array(
'name' => esc_html__( 'Theme Builder', 'et_builder' ),
'applicability' => array( 'administrator', 'editor' ),
),
'divi_ai' => array(
'name' => esc_html__( 'Divi AI', 'et_builder' ),
'applicability' => array( 'administrator', 'editor' ),
),
'ab_testing' => array(
'name' => esc_html__( 'Split Testing', 'et_builder' ),
),
),
),
'builder_capabilities' => array(
'section_title' => esc_html__( 'Builder Interface', 'et_builder' ),
'options' => array(
'add_module' => array(
'name' => esc_html__( 'Add/Delete Item', 'et_builder' ),
),
'edit_module' => array(
'name' => esc_html__( 'Edit Item', 'et_builder' ),
),
'move_module' => array(
'name' => esc_html__( 'Move Item', 'et_builder' ),
),
'disable_module' => array(
'name' => esc_html__( 'Disable Item', 'et_builder' ),
),
'lock_module' => array(
'name' => esc_html__( 'Lock Item', 'et_builder' ),
),
'divi_builder_control' => array(
'name' => esc_html__( 'Toggle Divi Builder', 'et_builder' ),
),
'load_layout' => array(
'name' => esc_html__( 'Load Layout', 'et_builder' ),
),
'use_visual_builder' => array(
'name' => esc_html__( 'Use Visual Builder', 'et_builder' ),
),
'custom_fonts_management' => array(
'name' => esc_html__( 'Upload/Remove Fonts', 'et_builder' ),
),
'read_dynamic_content_custom_fields' => array(
'name' => esc_html__( 'Dynamic Content Custom Fields', 'et_builder' ),
),
),
),
'library_capabilities' => array(
'section_title' => esc_html__( 'Library Settings', 'et_builder' ),
'options' => array(
'save_library' => array(
'name' => esc_html__( 'Save To Library', 'et_builder' ),
),
'add_library' => array(
'name' => esc_html__( 'Add From Library', 'et_builder' ),
),
'edit_global_library' => array(
'name' => esc_html__( 'Edit Global Items', 'et_builder' ),
),
),
),
'module_tabs' => array(
'section_title' => esc_html__( 'Settings Tabs', 'et_builder' ),
'options' => $tabs_array,
),
'general_module_capabilities' => array(
'section_title' => esc_html__( 'Settings Types', 'et_builder' ),
'options' => $options_categories,
),
'module_capabilies' => array(
'section_title' => esc_html__( 'Module Use', 'et_builder' ),
'options' => $module_capabilies,
),
);
$all_role_options = apply_filters( 'add_et_builder_role_options', $all_role_options );
$all_role_options['general_capabilities']['options'] = array_merge( $all_role_options['general_capabilities']['options'], $theme_only_options );
// Set portability capabilities.
$registered_portabilities = et_core_cache_get_group( 'et_core_portability' );
if ( ! empty( $registered_portabilities ) ) {
$all_role_options['general_capabilities']['options']['portability'] = array(
'name' => esc_html__( 'Portability', 'et_builder' ),
);
$all_role_options['portability'] = array(
'section_title' => esc_html__( 'Portability', 'et_builder' ),
'options' => array(),
);
// Dynamically create an option foreach portability.
foreach ( $registered_portabilities as $portability_context => $portability_instance ) {
$all_role_options['portability']['options'][ "{$portability_context}_portability" ] = array(
'name' => esc_html( $portability_instance->name ),
);
}
}
return $all_role_options;
}
/**
*
* Prints the admin page for Role Editor
*/
function et_pb_display_role_editor() {
$all_role_options = et_pb_all_role_options();
$option_tabs = '';
$menu_tabs = '';
$builder_roles_array = et_pb_get_all_roles_list();
foreach ( $builder_roles_array as $role => $role_title ) {
$option_tabs .= et_pb_generate_roles_tab( $all_role_options, $role );
$menu_tabs .= sprintf(
'
%2$s
',
esc_attr( $role_title ),
esc_html( $role_title ),
esc_attr( $role ),
'administrator' === $role ? ' et_pb_roles_active_menu' : ''
);
}
printf(
'',
et_core_esc_previously( $menu_tabs ),
esc_html__( 'Divi Role Editor', 'et_builder' ),
esc_html__( 'Save Divi Roles', 'et_builder' ),
et_core_esc_previously( et_builder_portability_link( 'et_pb_roles', array( 'class' => 'et-pb-layout-buttons et-pb-portability-button' ) ) ),
et_core_esc_previously( $option_tabs )
);
}
/**
* Generates the options tab for specified role.
*
* @param array $all_role_options array of all Role options.
* @param string $role role.
*
* @return string
*/
function et_pb_generate_roles_tab( $all_role_options, $role ) {
$form_sections = '';
// generate all sections of the form for current role.
if ( ! empty( $all_role_options ) ) {
foreach ( $all_role_options as $capability_id => $capability_options ) {
$form_sections .= sprintf(
'',
! empty( $capability_options['section_title'] )
? sprintf( '%1$s ', esc_html( $capability_options['section_title'] ) )
: '',
et_pb_generate_capabilities_output( $capability_options['options'], $role )
);
}
}
$output = sprintf(
'',
esc_html__( 'Using the Divi Role Editor, you can limit the types of actions that can be taken by WordPress users of different roles. This is a great way to limit the functionality available to your customers or guest authors to ensure that they only have the necessary options available to them.', 'et_builder' ),
esc_attr( $role ),
'administrator' === $role ? ' active-container' : '',
$form_sections // #4
);
return $output;
}
/**
* Generates the enable/disable buttons list based on provided capabilities array and role.
*
* @param array $cap_array capabilities.
* @param string $role user role.
*
* @return string
*/
function et_pb_generate_capabilities_output( $cap_array, $role ) {
$output = '';
if ( ! empty( $cap_array ) ) {
foreach ( $cap_array as $capability => $capability_details ) {
if ( empty( $capability_details['applicability'] ) || ( ! empty( $capability_details['applicability'] ) && in_array( $role, $capability_details['applicability'], true ) ) ) {
$output .= sprintf(
'',
esc_html__( 'Enabled', 'et_builder' ),
esc_html__( 'Disabled', 'et_builder' ),
esc_attr( $capability ),
esc_html( $capability_details['name'] ),
selected( true, et_pb_is_allowed( $capability, $role ), false ),
selected( false, et_pb_is_allowed( $capability, $role ), false )
);
}
}
}
return $output;
}
/**
* Loads scripts and styles for Role Editor Admin page
*
* @param string $hook hook name.
*/
function et_pb_load_roles_admin( $hook ) {
// load scripts only on role editor page.
if ( apply_filters( 'et_pb_load_roles_admin_hook', 'divi_page_et_divi_role_editor' ) !== $hook ) {
return;
}
et_core_load_main_fonts();
wp_enqueue_style( 'builder-roles-editor-styles', ET_BUILDER_URI . '/styles/roles_style.css', array( 'et-core-admin' ), ET_BUILDER_VERSION );
wp_enqueue_script( 'builder-roles-editor-scripts', ET_BUILDER_URI . '/scripts/roles_admin.js', array( 'jquery' ), ET_BUILDER_VERSION, true );
$pb_roles_options = array(
'ajaxurl' => admin_url( 'admin-ajax.php' ),
'et_roles_nonce' => wp_create_nonce( 'et_roles_nonce' ),
'modal_title' => esc_html__( 'Reset Roles', 'et_builder' ),
'modal_message' => esc_html__( 'All of your current role settings will be set to defaults. Do you wish to proceed?', 'et_builder' ),
'modal_yes' => et_builder_i18n( 'Yes' ),
'modal_no' => et_builder_i18n( 'No' ),
);
wp_localize_script( 'builder-roles-editor-scripts', 'et_pb_roles_options', $pb_roles_options );
}
add_action( 'admin_enqueue_scripts', 'et_pb_load_roles_admin' );
/**
* Generates the array of allowed modules in jQuery Array format.
*
* @param string $role the user role.
*
* @return string
*/
function et_pb_allowed_modules_list( $role = '' ) {
global $typenow;
// always return empty array if user doesn't have the edit_posts capability.
if ( ! current_user_can( 'edit_posts' ) ) {
return '[]';
}
$saved_capabilities = et_pb_get_role_settings();
$role = '' === $role ? et_pb_get_current_user_role() : $role;
$all_modules_array = ET_Builder_Element::get_modules_array( $typenow );
$saved_modules_capabilities = isset( $saved_capabilities[ $role ] ) ? $saved_capabilities[ $role ] : array();
$alowed_modules = '[';
foreach ( $all_modules_array as $module => $module_details ) {
if ( ! in_array( $module_details['label'], array( 'et_pb_section', 'et_pb_row', 'et_pb_row_inner', 'et_pb_column' ), true ) ) {
// Add module into the list if it's not saved or if it's saved not with "off" state.
if ( ! isset( $saved_modules_capabilities[ $module_details['label'] ] ) || ( isset( $saved_modules_capabilities[ $module_details['label'] ] ) && 'off' !== $saved_modules_capabilities[ $module_details['label'] ] ) ) {
$alowed_modules .= "'" . esc_attr( $module_details['label'] ) . "',";
}
}
}
$alowed_modules .= ']';
return $alowed_modules;
}
if ( ! function_exists( 'et_divi_get_post_text_color' ) ) {
/**
* Return text color css class.
*
* @return mixed|string
*/
function et_divi_get_post_text_color() {
$text_color_class = '';
$post_format = et_pb_post_format();
if ( in_array( $post_format, array( 'audio', 'link', 'quote' ), true ) ) {
$text_color = get_post_meta( get_the_ID(), '_et_post_bg_layout', true );
$text_color_class = $text_color ? $text_color : 'light';
$text_color_class = ' et_pb_text_color_' . $text_color_class;
}
return $text_color_class;
}
}
if ( ! function_exists( 'et_divi_get_post_bg_inline_style' ) ) {
/**
* Return css style attribute that ho;d background color inline style.
*
* @return string
*/
function et_divi_get_post_bg_inline_style() {
$inline_style = '';
$post_id = get_the_ID();
$post_use_bg_color = get_post_meta( $post_id, '_et_post_use_bg_color', true ) ? true : false;
$bg_color = get_post_meta( $post_id, '_et_post_bg_color', true );
$post_bg_color = $bg_color && '' !== $bg_color ? $bg_color : '#ffffff';
if ( $post_use_bg_color ) {
$inline_style = sprintf( ' style="background-color: %1$s;"', esc_html( $post_bg_color ) );
}
return $inline_style;
}
}
/**
* Remove the blockquote from post content.
*
* @param string $content post content.
*
* @return string|string[]|null
*/
function et_remove_blockquote_from_content( $content ) {
if ( 'quote' !== et_pb_post_format() ) {
return $content;
}
if ( et_theme_builder_overrides_layout( ET_THEME_BUILDER_BODY_LAYOUT_POST_TYPE ) ) {
// Do not remove quotes when TB has taken over.
return $content;
}
$content = preg_replace( '//is', '', $content, 1 );
return $content;
}
add_filter( 'the_content', 'et_remove_blockquote_from_content' );
/**
* Register rewrite rule and tag for preview page
*
* @return void
*/
function et_pb_register_preview_endpoint() {
add_rewrite_tag( '%et_pb_preview%', 'true' );
}
add_action( 'init', 'et_pb_register_preview_endpoint', 11 );
/**
* Flush rewrite rules to fix the issue "preg_match" issue with 2.5
*
* @return void
*/
function et_pb_maybe_flush_rewrite_rules() {
et_builder_maybe_flush_rewrite_rules( '2_5_flush_rewrite_rules' );
}
add_action( 'init', 'et_pb_maybe_flush_rewrite_rules', 9 );
/**
* Register template for preview page.
*
* @param string $template The path of the template to include.
*
* @return string path to template file
*/
function et_pb_register_preview_page( $template ) {
global $wp_query;
if ( 'true' === $wp_query->get( 'et_pb_preview' ) && isset( $_GET['et_pb_preview_nonce'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
show_admin_bar( false );
return ET_BUILDER_DIR . 'template-preview.php';
}
return $template;
}
add_action( 'template_include', 'et_pb_register_preview_page' );
/**
* Disable all the dynamic assets for preview page so all the styles can be rendered correctly.
*
* @return void
*/
function et_pb_preview_page_disable_dynamic_assets() {
global $wp_query;
if ( 'true' === $wp_query->get( 'et_pb_preview' ) && isset( $_GET['et_pb_preview_nonce'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
// Instruct Shortcode Manager to register/load all modules/shortcodes.
add_filter( 'et_builder_should_load_all_module_data', '__return_true' );
// Disable Feature: Dynamic Assets.
add_filter( 'et_disable_js_on_demand', '__return_true' );
add_filter( 'et_use_dynamic_css', '__return_false' );
add_filter( 'et_should_generate_dynamic_assets', '__return_false' );
// Disable Feature: Critical CSS.
add_filter( 'et_builder_critical_css_enabled', '__return_false' );
// Disable Cache in Feature Manager.
add_filter( 'et_builder_post_feature_cache_enabled', '__return_false' );
}
}
add_action( 'wp', 'et_pb_preview_page_disable_dynamic_assets' );
if ( ! function_exists( 'et_builder_replace_code_content_entities' ) ) :
/**
* The do_shortcode() replaces square brackers with html entities,
* convert them back to make sure js code works ok
*
* @param string $content post content.
*/
function et_builder_replace_code_content_entities( $content ) {
$content = str_replace( '[', '[', $content );
$content = str_replace( ']', ']', $content );
$content = str_replace( '×', 'x', $content );
return $content;
}
endif;
if ( ! function_exists( 'et_builder_convert_line_breaks' ) ) :
/**
* We use placeholders to preserve the line-breaks, convert them back to "\n".
*
* @param string $content content.
* @param string $line_breaks_format line break format e.g \n or .
*/
function et_builder_convert_line_breaks( $content, $line_breaks_format = "\n" ) {
// before we swap out the placeholders,
// remove all the tags and \n that wpautop added!
$content = preg_replace( '/\n/smi', '', $content );
$content = preg_replace( '/
/smi', '', $content );
$content = preg_replace( '/<\/p>/smi', '', $content );
$content = str_replace( array( '', '', '||et_pb_line_break_holder||' ), $line_breaks_format, $content );
$content = str_replace( '', ' ', $content );
// convert the ', '
', $content );
return $content;
}
endif;
/**
* Adjust the number of all layouts displayed on library page to exclude predefined layouts.
*
* @param object $counts An object containing the current post_type's post
* counts by status.
*
* @return mixed
*/
function et_pb_fix_count_library_items( $counts ) {
// do nothing if get_current_screen function doesn't exists at this point to avoid php errors in some plugins.
if ( ! function_exists( 'get_current_screen' ) ) {
return $counts;
}
$current_screen = get_current_screen();
if ( isset( $current_screen->id ) && 'edit-et_pb_layout' === $current_screen->id && isset( $counts->publish ) ) {
// perform query to get all the not predefined layouts.
$query = new WP_Query(
array(
'meta_query' => array(
array(
'key' => '_et_pb_predefined_layout',
'value' => 'on',
'compare' => 'NOT EXISTS',
),
),
'post_type' => ET_BUILDER_LAYOUT_POST_TYPE,
'posts_per_page' => '-1',
)
);
// set the $counts->publish = amount of non predefined layouts.
$counts->publish = isset( $query->post_count ) ? (int) $query->post_count : 0;
}
return $counts;
}
add_filter( 'wp_count_posts', 'et_pb_fix_count_library_items' );
/**
* Mobile settings tabs: Desktop, Table, Smartphone.
*
* @return string
*/
function et_pb_generate_mobile_settings_tabs() {
$mobile_settings_tabs = '<%= window.et_builder.mobile_tabs_output() %>';
return $mobile_settings_tabs;
}
/**
* Generates the css code for responsive options.
*
* Uses array of values for each device as input parameter and css_selector with property to
* apply the css
*
* @deprecated See ET_Builder_Module_Helper_ResponsiveOptions::instance()->generate_responsive_css().
*
* @since 3.23 Deprecated.
*
* @param array $values_array All device values.
* @param mixed $css_selector CSS selector.
* @param string $css_property CSS property.
* @param string $function_name Module slug.
* @param string $additional_css Additional CSS.
*/
function et_pb_generate_responsive_css( $values_array, $css_selector, $css_property, $function_name, $additional_css = '' ) {
if ( ! empty( $values_array ) ) {
foreach ( $values_array as $device => $current_value ) {
if ( '' === $current_value ) {
continue;
}
$declaration = '';
// value can be provided as a string or array in following format - array( 'property_1' => 'value_1', 'property_2' => 'property_2', ... , 'property_n' => 'value_n' ).
if ( is_array( $current_value ) && ! empty( $current_value ) ) {
foreach ( $current_value as $this_property => $this_value ) {
if ( '' === $this_value ) {
continue;
}
$declaration .= sprintf(
'%1$s: %2$s%3$s',
$this_property,
esc_html( et_builder_process_range_value( $this_value, $this_property ) ),
'' !== $additional_css ? $additional_css : ';'
);
}
} else {
$declaration = sprintf(
'%1$s: %2$s%3$s',
$css_property,
esc_html( et_builder_process_range_value( $current_value, $css_property ) ),
'' !== $additional_css ? $additional_css : ';'
);
}
if ( '' === $declaration ) {
continue;
}
$style = array(
'selector' => $css_selector,
'declaration' => $declaration,
);
if ( 'desktop_only' === $device ) {
$style['media_query'] = ET_Builder_Element::get_media_query( 'min_width_981' );
} elseif ( 'desktop' !== $device ) {
$current_media_query = 'tablet' === $device ? 'max_width_980' : 'max_width_767';
$style['media_query'] = ET_Builder_Element::get_media_query( $current_media_query );
}
ET_Builder_Element::set_style( $function_name, $style );
}
}
}
/**
* Search module: search the posts.
*
* @param mixed $query search WP_Query object.
*/
function et_pb_custom_search( $query = false ) {
if ( is_admin() || ! is_a( $query, 'WP_Query' ) || ! $query->is_search ) {
return;
}
$utils = ET_Core_Data_Utils::instance();
// phpcs:disable WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
if ( isset( $_GET['et_pb_searchform_submit'] ) ) {
$post_types = array();
if ( ! isset( $_GET['et_pb_include_posts'] ) && ! isset( $_GET['et_pb_include_pages'] ) ) {
$post_types = array( 'post' );
}
if ( isset( $_GET['et_pb_include_pages'] ) ) {
$post_types = array( 'page' );
}
if ( isset( $_GET['et_pb_include_posts'] ) ) {
$post_types[] = 'post';
}
// $postTypes is allowlisted values only
$query->set( 'post_type', $post_types );
if ( ! empty( $_GET['et_pb_search_cat'] ) ) {
// phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- Search categories sanitization has been done below on each cat.
$categories_array = explode( ',', $_GET['et_pb_search_cat'] );
$categories_array = $utils->sanitize_text_fields( $categories_array );
$query->set( 'category__not_in', $categories_array );
}
if ( isset( $_GET['et-posts-count'] ) ) {
$query->set( 'posts_per_page', (int) $_GET['et-posts-count'] );
}
}
// phpcs:enable
}
add_action( 'pre_get_posts', 'et_pb_custom_search' );
if ( ! function_exists( 'et_custom_comments_display' ) ) :
/**
* Custom callback function to control the look of the comment.
*
* @param WP_Comment $comment comment object.
* @param WP_Comment $args args.
* @param WP_Comment $depth comment depth.
*/
function et_custom_comments_display( $comment, $args, $depth ) {
// phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited -- Set the current comment.
$GLOBALS['comment'] = $comment;
$default_avatar = get_option( 'avatar_default' ) ? get_option( 'avatar_default' ) : 'mystery';
?>
id="li-comment-">
comment_count ) ? $post->comment_count : $count;
}
return $count;
}
add_filter( 'get_comments_number', 'et_pb_preview_comment_count', 10, 2 );
/**
* List of shortcodes that triggers error if being used in admin
*
* @return array shortcode tag
*/
function et_pb_admin_excluded_shortcodes() {
$shortcodes = array();
// Triggers issue if Sensei and YOAST SEO are activated.
if ( et_is_yoast_seo_plugin_active() && function_exists( 'Sensei' ) ) {
$shortcodes[] = 'usercourses';
}
// WPL real estate prints unwanted on-page JS that caused an issue on BB.
if ( class_exists( 'wpl_extensions' ) ) {
$shortcodes[] = 'WPL';
}
// [submit_job_form] shortcode prints wp_editor this creating problems post edit page render
if ( et_is_wp_job_manager_plugin_active() ) {
$shortcodes[] = 'submit_job_form';
}
// [shop_messages] shortcode causes a fatal error when rendered too soon
if ( et_is_woocommerce_plugin_active() ) {
$shortcodes[] = 'shop_messages';
}
return apply_filters( 'et_pb_admin_excluded_shortcodes', $shortcodes );
}
/**
* Get GMT offset string that can be used for parsing date into correct timestamp
*
* @return string
*/
function et_pb_get_gmt_offset_string() {
$gmt_offset = get_option( 'gmt_offset' );
$gmt_divider = '-' === substr( $gmt_offset, 0, 1 ) ? '-' : '+';
$gmt_offset_hour = str_pad( abs( intval( $gmt_offset ) ), 2, '0', STR_PAD_LEFT );
$gmt_offset_minute = str_pad( ( ( abs( $gmt_offset ) * 100 ) % 100 ) * ( 60 / 100 ), 2, '0', STR_PAD_LEFT );
$gmt_offset_string = "GMT{$gmt_divider}{$gmt_offset_hour}{$gmt_offset_minute}";
return $gmt_offset_string;
}
/**
* Get post's category label and permalink to be used on frontend
*
* @param int $post_id post id.
* @param WP_Term[] $default term objects.
*
* @return array categories
*/
function et_pb_get_post_categories( $post_id, $default = array() ) {
$categories = get_the_category( $post_id );
$post_categories = array();
if ( $default && ! $categories ) {
$categories = array_values( $default );
foreach ( array_keys( $categories ) as $key ) {
_make_cat_compat( $categories[ $key ] );
}
}
// Filter out any falsy values that may appear due from $default.
$categories = array_filter( $categories );
if ( ! empty( $categories ) ) {
foreach ( $categories as $category ) {
$post_categories[ $category->cat_ID ] = array(
'id' => $category->cat_ID,
'label' => $category->cat_name,
'permalink' => get_category_link( $category->cat_ID ),
);
}
}
return $post_categories;
}
/**
* Generates "Use Visual Builder" button url
*
* @return string
*/
function et_fb_get_page_url() {
global $wp;
$post_id = get_the_ID();
$is_divi_library = 'et_pb_layout' === get_post_type( $post_id );
if ( $is_divi_library ) {
return get_edit_post_link( $post_id );
} elseif ( is_singular() ) {
return get_permalink( $post_id );
}
return add_query_arg( $wp->query_string, '', home_url( $wp->request ) );
}
/**
* Add "Use Visual Builder" link to WP-Admin bar
*
* @return void
*/
function et_fb_add_admin_bar_link() {
$is_not_builder_enabled_single = ! is_singular() || ! et_builder_fb_enabled_for_post( get_the_ID() );
$is_not_in_wc_shop = ! et_builder_used_in_wc_shop();
$not_allowed_fb_access = ! et_pb_is_allowed( 'use_visual_builder' );
$app_preferences = et_fb_app_preferences_settings();
$default_visual_theme_builder = et_()->array_get( $app_preferences, 'enable_visual_theme_builder.default' );
$is_visual_theme_builder_enabled = et_get_option( 'et_fb_pref_builder_enable_visual_theme_builder', $default_visual_theme_builder );
$is_not_theme_builder_used = ! et_fb_is_theme_builder_used_on_page() || ! et_pb_is_allowed( 'theme_builder' ) || ! filter_var( $is_visual_theme_builder_enabled, FILTER_VALIDATE_BOOLEAN );
$is_enabled_for_post_type = ! is_singular() || et_builder_enabled_for_post_type( get_post_type( get_the_ID() ) );
// Return if builder is not allowed for the user.
if ( $not_allowed_fb_access ) {
return;
}
// Return if builder is not enabled for the post type.
if ( ! $is_enabled_for_post_type && $is_not_builder_enabled_single ) {
return;
}
// Return for non-singular pages if they are not WC shop page and Theme Builder is not used.
if ( $is_not_builder_enabled_single && $is_not_in_wc_shop && $is_not_theme_builder_used ) {
return;
}
global $wp_admin_bar, $wp_the_query;
$post_id = get_the_ID();
// WooCommerce Shop Page replaces main query, thus it has to be normalized.
if ( et_builder_used_in_wc_shop() && method_exists( $wp_the_query, 'get_queried_object' ) && isset( $wp_the_query->get_queried_object()->ID ) ) {
$post_id = $wp_the_query->get_queried_object()->ID;
}
$page_url = et_fb_get_page_url();
// Don't add the link, if Frontend Builder has been loaded already.
if ( et_fb_is_enabled() || et_fb_is_enabled_on_any_template() ) {
$wp_admin_bar->add_menu(
array(
'id' => 'et-disable-visual-builder',
'title' => esc_html__( 'Exit Visual Builder', 'et_builder' ),
'href' => esc_url( $page_url ),
)
);
return;
}
$current_object = $wp_the_query->get_queried_object();
$user_is_allowed_to_edit = isset( $current_object->ID ) ? current_user_can( 'edit_post', $current_object->ID ) : ( et_fb_is_theme_builder_used_on_page() && et_pb_is_allowed( 'theme_builder' ) );
if ( is_admin() || ! $user_is_allowed_to_edit || ! et_pb_is_allowed( 'divi_builder_control' ) ) {
return;
}
$use_visual_builder_url = et_pb_is_pagebuilder_used( $post_id ) || ( et_fb_is_theme_builder_used_on_page() && ! is_singular() ) ?
et_fb_get_builder_url( $page_url ) :
add_query_arg(
array(
'et_fb_activation_nonce' => wp_create_nonce( 'et_fb_activation_nonce_' . $post_id ),
),
$page_url
);
$wp_admin_bar->add_menu(
array(
'id' => 'et-use-visual-builder',
'title' => esc_html__( 'Enable Visual Builder', 'et_builder' ),
'href' => esc_url( $use_visual_builder_url ),
)
);
}
add_action( 'admin_bar_menu', 'et_fb_add_admin_bar_link', 999 );
/**
* Retrieve and process saved Layouts.
* It different than the function which retrieves saved Sections, Rows and Modules from library because layouts require different processing
*/
function et_fb_get_saved_layouts() {
if ( ! isset( $_POST['et_fb_retrieve_library_modules_nonce'] ) || ! wp_verify_nonce( $_POST['et_fb_retrieve_library_modules_nonce'], 'et_fb_retrieve_library_modules_nonce' ) ) { // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- The nonce value is used only for comparision in the `wp_verify_nonce`.
die( -1 );
}
if ( ! current_user_can( 'edit_posts' ) ) {
die( -1 );
}
// Reduce number of results per page if we're hosted on wpengine to avoid 500 error due to memory allocation.
// This is caused by one of their custom mu-plugins doing additional stuff but we have no control over there.
$page_size = function_exists( 'is_wpe' ) || function_exists( 'is_wpe_snapshot' ) ? 25 : 50;
$post_type = ! empty( $_POST['et_post_type'] ) ? sanitize_text_field( $_POST['et_post_type'] ) : 'post';
$layouts_type = ! empty( $_POST['et_load_layouts_type'] ) ? sanitize_text_field( $_POST['et_load_layouts_type'] ) : 'all';
$start_from = ! empty( $_POST['et_templates_start_page'] ) ? sanitize_text_field( $_POST['et_templates_start_page'] ) : 0;
$post_type = apply_filters( 'et_pb_show_all_layouts_built_for_post_type', $post_type, $layouts_type );
$all_layouts_data = et_pb_retrieve_templates( 'layout', '', 'false', '0', $post_type, $layouts_type, array( $start_from, $page_size ) );
$all_layouts_data_processed = $all_layouts_data;
$next_page = 'none';
if ( 0 !== $start_from && empty( $all_layouts_data ) ) {
$all_layouts_data_processed = array();
} else {
if ( empty( $all_layouts_data ) ) {
$all_layouts_data_processed = array( 'error' => esc_html__( 'You have not saved any items to your Divi Library yet. Once an item has been saved to your library, it will appear here for easy use.', 'et_builder' ) );
} else {
foreach ( $all_layouts_data as $index => $data ) {
$all_layouts_data_processed[ $index ]['shortcode'] = et_fb_process_shortcode( $data['shortcode'] );
}
$next_page = $start_from + $page_size;
}
}
$json_templates = wp_json_encode(
array(
'templates_data' => $all_layouts_data_processed,
'next_page' => $next_page,
)
);
die( et_core_intentionally_unescaped( $json_templates, 'html' ) );
}
add_action( 'wp_ajax_et_fb_get_saved_layouts', 'et_fb_get_saved_layouts' );
/**
* Ajax Callback: Process imported content.
*/
function et_fb_process_imported_content() {
if ( ! isset( $_POST['et_fb_process_imported_data_nonce'] ) || ! wp_verify_nonce( $_POST['et_fb_process_imported_data_nonce'], 'et_fb_process_imported_data_nonce' ) ) { // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- The nonce value is used only for comparision in the `wp_verify_nonce`.
die( -1 );
}
if ( ! current_user_can( 'edit_posts' ) ) {
die( -1 );
}
// phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- Value in `$_POST['et_raw_shortcode']` is processed by `et_fb_process_shortcode` and being returned in ajax response.
$processed_shortcode = isset( $_POST['et_raw_shortcode'] ) ? et_fb_process_shortcode( stripslashes( $_POST['et_raw_shortcode'] ) ) : '';
die( wp_json_encode( $processed_shortcode ) );
}
add_action( 'wp_ajax_et_fb_process_imported_content', 'et_fb_process_imported_content' );
/**
* Builder initial content.
*
* @param string $content post content.
* @param integer $post_id post id.
*
* @return string
*/
function et_fb_maybe_get_bfb_initial_content( $content, $post_id ) {
$from_post = isset( $_GET['from_post'] ) ? sanitize_text_field( $_GET['from_post'] ) : ''; // phpcs:ignore WordPress.Security.NonceVerification -- This function does not change any state, and is therefore not susceptible to CSRF.
if ( ! empty( $from_post ) && 'empty' !== $from_post ) {
$copy_content_from = get_post( $from_post );
$existing_content = $copy_content_from->post_content;
if ( '' !== $existing_content && has_shortcode( $existing_content, 'et_pb_section' ) ) {
return $existing_content;
}
}
// process the content only for BFB.
if ( ! et_builder_bfb_enabled() ) {
return $content;
}
// If content already has a section, it means builder is active and activation has to be
// skipped to avoid nested and unwanted builder structure.
if ( has_shortcode( $content, 'et_pb_section' ) ) {
return $content;
}
// Save old content.
$saved_old_content = get_post_meta( $post_id, '_et_pb_old_content', true );
$save_old_content = false;
$post = get_post( $post_id );
if ( '' !== $content ) {
$save_old_content = update_post_meta( $post_id, '_et_pb_old_content', $content );
}
/**
* Filters the flag that sets default Content during Builder activation.
*
* @since 3.29
*
* @used-by et_builder_wc_init()
*/
if ( apply_filters( 'et_builder_skip_content_activation', false, $post ) ) {
return $content;
}
if ( true !== $save_old_content && $saved_old_content !== $content && '' !== $content ) {
return $content;
}
$text_module = '' !== $content ? '[et_pb_text admin_label="Text"]' . $content . '[/et_pb_text]' : '';
// Re-format content.
$updated_content =
'[et_pb_section admin_label="section"]
[et_pb_row admin_label="row"]
[et_pb_column type="4_4"]' . $text_module . '[/et_pb_column]
[/et_pb_row]
[/et_pb_section]';
return $updated_content;
}
/**
* Called via async AJAX call after the builder rendered. It will regenerate both helper/definitions files.
* If their content changed, the builder will trigger a page reload to use the updated cached files.
*/
function et_fb_update_builder_assets() {
if ( ! isset( $_POST['et_fb_helper_nonce'] ) || ! wp_verify_nonce( $_POST['et_fb_helper_nonce'], 'et_fb_update_helper_assets_nonce' ) ) { // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- The nonce value is used only for comparision in the `wp_verify_nonce`.
die( -1 );
}
$post_id = ! empty( $_POST['et_post_id'] ) ? sanitize_text_field( $_POST['et_post_id'] ) : '';
if ( ! current_user_can( 'edit_post', $post_id ) ) {
die( -1 );
}
// Set current post as global $post.
$post = get_post( $post_id ); // phpcs:ignore WordPress.Variables.GlobalVariables.OverrideProhibited
$post_type = ! empty( $_POST['et_post_type'] ) ? sanitize_text_field( $_POST['et_post_type'] ) : 'post';
// Update helpers cached js file.
$helpers = et_fb_get_dynamic_asset( 'helpers', $post_type, true );
// Update definitions cached js file.
$definitions = et_fb_get_dynamic_asset( 'definitions', $post_type, true );
// When either definitions or helpers needs an update, also clear modules cache.
if ( $definitions['updated'] || $helpers['updated'] ) {
$modules_cache = ET_Builder_Element::get_cache_filename( $post_type );
if ( file_exists( $modules_cache ) ) {
@unlink( $modules_cache ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged -- `unlink` may fail with the permissions denied error.
}
}
die(
wp_json_encode(
array(
'helpers' => $helpers,
'definitions' => $definitions,
)
)
);
}
add_action( 'wp_ajax_et_fb_update_builder_assets', 'et_fb_update_builder_assets' );
/**
* Returns builder definitions.
*
* @param string $post_type post type.
*
* @return array
*/
function et_fb_get_builder_definitions( $post_type ) {
// force render builder data when retrieving builder definition to ensure definitions retrieved via ajax call
// equal to definitions retrieved on wp_footer when no dynamic asset cache found.
add_filter( 'et_builder_module_force_render', '__return_true' );
$fields_data = array();
$fields_data['custom_css'] = ET_Builder_Element::get_custom_css_fields( $post_type );
$fields_data['advanced_fields'] = ET_Builder_Element::get_advanced_fields( $post_type );
$fields_data['general_fields'] = ET_Builder_Element::get_general_fields( $post_type );
$fields_data['childModuleTitles'] = ET_Builder_Element::get_child_module_titles( $post_type );
$fields_data['optionsToggles'] = ET_Builder_Element::get_toggles( $post_type );
$fields_data['customTabs'] = ET_Builder_Element::get_tabs( $post_type );
$fields_data['customTabsFields'] = ET_Builder_Element::get_settings_modal_tabs_fields( $post_type );
$fields_data['customLayoutsTabs'] = ET_Builder_Library::builder_library_modal_custom_tabs( $post_type );
$fields_data['moduleItemsConfig'] = ET_Builder_Element::get_module_items_configs( $post_type );
$fields_data['moduleTransitions'] = ET_Builder_Element::get_modules_transitions( $post_type );
$fields_data['contact_form_input_defaults'] = et_fb_process_shortcode(
sprintf(
'[et_pb_contact_field field_title="%1$s" field_type="input" field_id="Name" required_mark="on" fullwidth_field="off" /][et_pb_contact_field field_title="%2$s" field_type="email" field_id="Email" required_mark="on" fullwidth_field="off" /][et_pb_contact_field field_title="%3$s" field_type="text" field_id="Message" required_mark="on" fullwidth_field="on" /]',
esc_attr__( 'Name', 'et_builder' ),
esc_attr__( 'Email Address', 'et_builder' ),
esc_attr__( 'Message', 'et_builder' )
)
);
// Remove duplicates from field definitions.
$map = array();
$unique_fields = array();
$unique_count = 0;
foreach ( array( 'custom_css', 'general_fields', 'advanced_fields' ) as $source ) {
$definitions = &$fields_data[ $source ];
$module_names = array_keys( $definitions );
foreach ( $module_names as $module_name ) {
$module = &$definitions[ $module_name ];
$setting_names = array_keys( $module );
foreach ( $setting_names as $setting_name ) {
$setting = &$module[ $setting_name ];
if ( 'advanced_defaults' === $setting_name ) {
// advanced_defaults are just duplicated data, we can rebuilt them later.
$setting = false;
continue;
}
$key = wp_json_encode( $setting );
if ( ! isset( $map[ $key ] ) ) {
// Found a duplicate here.
$unique_fields[] = $setting;
$map[ $key ] = $unique_count++;
}
$setting = $map[ $key ];
}
}
}
// Remove force builder data render.
remove_filter( 'et_builder_module_force_render', '__return_true' );
// No longer needed.
unset( $map );
// Include the unique fields in the AJAX payload.
$fields_data['unique_fields'] = $unique_fields;
return $fields_data;
}
/**
* Returns builder shortcode object.
*
* @param string $post_type the post type.
* @param integer $post_id the post id.
* @param string $layout_type layout type.
*
* @return array
*/
function et_fb_get_builder_shortcode_object( $post_type, $post_id, $layout_type ) {
// We need to store the current post when this function is executed in a wp-admin page
// to prevent post based modules included in the shortcode from altering the loop.
global $post;
$backup = $post;
$fields_data = array();
add_filter( 'et_builder_module_force_render', '__return_true' );
$post_data = get_post( $post_id );
$post_data_post_modified = is_object( $post_data ) ? gmdate( 'U', strtotime( $post_data->post_modified ) ) : 0;
$post_content = is_object( $post_data ) ? $post_data->post_content : '';
$is_theme_builder = et_builder_tb_enabled();
$is_backend_builder = et_builder_bfb_enabled();
$theme_builder_layouts = et_theme_builder_get_template_layouts();
// Unset main template from Theme Builder layouts to avoid PHP Notices.
if ( isset( $theme_builder_layouts['et_template'] ) ) {
unset( $theme_builder_layouts['et_template'] );
}
// Get the content for all theme builder posts.
foreach ( $theme_builder_layouts as $key => $theme_builder_layout ) {
if ( 0 !== $theme_builder_layout['id'] && $theme_builder_layout['enabled'] && $theme_builder_layout['override'] ) {
$post_data = get_post( $theme_builder_layout['id'] );
$theme_builder_layouts[ $key ]['post_content'] = $post_data->post_content;
}
}
// if autosave exists here, return it with the real content, autosave.js and getServerSavedPostData() will look for it.
$current_user_id = get_current_user_id();
// Store one autosave per author. If there is already an autosave, overwrite it.
$autosave = wp_get_post_autosave( $post_id, $current_user_id );
if ( ! empty( $autosave ) ) {
$autosave_post_modified = gmdate( 'U', strtotime( $autosave->post_modified ) );
if ( $autosave_post_modified > $post_data_post_modified ) {
$fields_data['autosave_shortcode_object'] = et_fb_process_shortcode( $autosave->post_content );
$fields_data['has_newer_autosave'] = true;
} else {
$fields_data['has_newer_autosave'] = false;
}
// Delete the autosave, becuase we will present the option to use the autosave to the user, and they will use it or not
// we need to delete the db copy now.
wp_delete_post_revision( $autosave->ID );
}
switch ( $layout_type ) {
case 'module':
$use_fullwidth_section = false !== strpos( $post_content, '[et_pb_fullwidth_' ) ? true : false;
// Remove module placeholders.
$post_content = false !== strpos( $post_content, 'et_pb_fullwidth_module_placeholder' ) || false !== strpos( $post_content, 'et_pb_module_placeholder' ) ? '' : $post_content;
if ( ! $use_fullwidth_section ) {
$post_content = sprintf( '[et_pb_row][et_pb_column type="4_4"]%1$s[/et_pb_column][/et_pb_row]', $post_content );
}
$post_content = sprintf(
'[et_pb_section%2$s]%1$s[/et_pb_section]',
$post_content,
$use_fullwidth_section ? ' fullwidth="on"' : ''
);
break;
case 'row':
$post_content = '[et_pb_section]' . $post_content . '[/et_pb_section]';
break;
}
$post_content = et_fb_maybe_get_bfb_initial_content( $post_content, $post_id );
/**
* Filters the raw post content when the Builder is loaded.
*
* @since 3.29
*
* @param string $post_content The raw/unprocessed post content.
* @param int $post_id Post ID.
*/
$post_content = apply_filters( 'et_fb_load_raw_post_content', $post_content, $post_id );
$fields_data['shortcode_object'] = array();
// Process main post content fields data.
$post_fields_data = et_fb_process_shortcode( $post_content );
$fields_data['shortcode_object'] = array();
// In Visual Builder get All Theme Builder Areas.
if ( et_pb_is_allowed( 'theme_builder' ) && ! $is_theme_builder && ! $is_backend_builder && ! empty( $theme_builder_layouts ) ) {
// Process Theme Builder Header area fields data.
if ( isset( $theme_builder_layouts['et_header_layout']['post_content'] ) ) {
$theme_builder_header = apply_filters( 'et_fb_load_raw_post_content', $theme_builder_layouts['et_header_layout']['post_content'], $theme_builder_layouts['et_header_layout']['id'] );
$tb_header_fields_data = et_fb_process_shortcode( $theme_builder_header, '', '', '', '', 'et_header_layout' );
$processed_fields_data[] = $tb_header_fields_data;
}
// Process Theme Builder Body area.
if ( isset( $theme_builder_layouts['et_body_layout']['post_content'] ) ) {
$theme_builder_body = apply_filters( 'et_fb_load_raw_post_content', $theme_builder_layouts['et_body_layout']['post_content'], $theme_builder_layouts['et_body_layout']['id'] );
$tb_body_fields_data = et_fb_process_shortcode( $theme_builder_body, '', '', '', '', 'et_body_layout' );
$theme_builder_post_content_selector = et_fb_generate_post_content_module_selector( $tb_body_fields_data, 'theme_builder_content' );
if ( $theme_builder_post_content_selector && ! is_home() && ! is_archive() && ! is_404() ) {
// If Theme Builder Body contains Post Content Module, replace it with real post content.
$post_content_fields_data = et_fb_process_shortcode( $post_content, '', '', '', '', 'post_content' );
$processed_fields_data[] = et_fb_generate_tb_body_area_with_post_content( $tb_body_fields_data, $theme_builder_post_content_selector, $post_content_fields_data );
} else {
// if not, just add Theme Builder Body area content.
$processed_fields_data[] = $tb_body_fields_data;
// Add Post content too, so it can be loaded when post content module is added.
if ( ! is_home() && ! is_archive() && ! is_404() ) {
$initial_post_content_fields_data = et_fb_process_shortcode( $post_content, '', '', '', '', 'initial_post_content' );
$processed_fields_data[] = $initial_post_content_fields_data;
}
}
} elseif ( ! is_home() && et_pb_is_pagebuilder_used( $post_id ) ) {
// If there is no Theme Builder Body, load post content.
$processed_fields_data[] = $post_fields_data;
}
// Process Theme Builder Header area fields data.
if ( isset( $theme_builder_layouts['et_footer_layout']['post_content'] ) ) {
$theme_builder_footer = apply_filters( 'et_fb_load_raw_post_content', $theme_builder_layouts['et_footer_layout']['post_content'], $theme_builder_layouts['et_footer_layout']['id'] );
$tb_footer_fields_data = et_fb_process_shortcode( $theme_builder_footer, '', '', '', '', 'et_footer_layout' );
$processed_fields_data[] = $tb_footer_fields_data;
}
// Build the shortcode_object from Theme Builder Areas.
foreach ( $processed_fields_data as $processed_field_data ) {
if ( is_array( $processed_field_data ) ) {
$fields_data['shortcode_object'] = array_merge( $fields_data['shortcode_object'], $processed_field_data );
}
}
} else {
// In Theme Builder and Backend Builder show only main post.
$fields_data['shortcode_object'] = $post_fields_data;
}
remove_filter( 'et_builder_module_force_render', '__return_true' );
// Restore post.
$post = $backup; // phpcs:ignore WordPress.WP.GlobalVariablesOverride -- This is legit way of setting global $post.
setup_postdata( $post );
return $fields_data;
}
/**
* This function searches the multidimensional array of post content fields for post content module
* and generates array of keys that are used to select post content module.
*
* @param array $array Array of post fields data.
* @param string $element_type Type of an element that is analyzes at the moment.
*
* @return array
*/
function et_fb_generate_post_content_module_selector( array $array, $element_type ) {
global $current_section;
global $current_row;
global $current_row_inner;
global $current_column;
global $current_column_inner;
global $current_module;
global $post_content_module_selector;
if ( 'theme_builder_content' === $element_type ) {
// Loop through Theme Builder Area sections.
foreach ( $array as $key => $value ) {
if ( isset( $array[ $key ]['content'] ) && is_array( $array[ $key ]['content'] ) ) {
$current_section = $key;
et_fb_generate_post_content_module_selector( $array[ $key ]['content'], $array[ $key ]['type'] );
}
}
} elseif ( 'et_pb_section' === $element_type ) {
// Loop through rows.
foreach ( $array as $key => $value ) {
if ( 'et_pb_fullwidth_post_content' === $array[ $key ]['type'] ) {
et_fb_generate_post_content_module_selector( $array, 'et_pb_column' );
} elseif ( isset( $array[ $key ]['content'] ) && is_array( $array[ $key ]['content'] ) ) {
$current_row = $key;
et_fb_generate_post_content_module_selector( $array[ $key ]['content'], $array[ $key ]['type'] );
}
}
} elseif ( 'et_pb_row' === $element_type ) {
// Loop through columns.
foreach ( $array as $key => $value ) {
if ( isset( $array[ $key ]['content'] ) && is_array( $array[ $key ]['content'] ) ) {
$current_column = $key;
et_fb_generate_post_content_module_selector( $array[ $key ]['content'], $array[ $key ]['type'] );
}
}
} elseif ( 'et_pb_row_inner' === $element_type ) {
// Loop through columns.
foreach ( $array as $key => $value ) {
if ( isset( $array[ $key ]['content'] ) && is_array( $array[ $key ]['content'] ) ) {
$current_column_inner = $key;
et_fb_generate_post_content_module_selector( $array[ $key ]['content'], $array[ $key ]['type'] );
}
}
} elseif ( 'et_pb_column' === $element_type || 'et_pb_column_inner' === $element_type ) {
// Loop through modules.
foreach ( $array as $key => $value ) {
if ( 'et_pb_row_inner' === $array[ $key ]['type'] ) {
foreach ( $array as $key => $value ) {
if ( isset( $array[ $key ]['content'] ) && is_array( $array[ $key ]['content'] ) ) {
$current_row_inner = $key;
et_fb_generate_post_content_module_selector( $array[ $key ]['content'], $array[ $key ]['type'] );
}
}
} elseif ( 'et_pb_post_content' === $array[ $key ]['type'] ) {
// If Post Content Module is Found build the selector from current Section, Row, and Column.
$current_module = $key;
$is_column_inner = 'et_pb_column_inner' === $array[ $key ]['parent_slug'];
$post_content_module_selector = $is_column_inner ? array(
'section' => $current_section,
'row_inner' => $current_row_inner,
'column' => $current_row,
'column_inner' => $current_column_inner,
'module' => $current_module,
) : array(
'section' => $current_section,
'row' => $current_row,
'column' => $current_column,
'module' => $current_module,
);
} elseif ( 'et_pb_fullwidth_post_content' === $array[ $key ]['type'] ) {
// If Post Content Module is FullWidth create selector with section and module id.
$current_module = $key;
$post_content_module_selector = array(
'section' => $current_section,
'module' => $current_module,
);
}
}
}
return $post_content_module_selector;
}
/**
* This function is used to generate the Theme Builder Body area that has Post
* Content module inside. It replaces Post Content module with sections of a
* post that is being currently edited.
*
* @param array $theme_builder_body_fields Theme Builder Body area fields data.
* @param array $selector Array of keys for selecting Post Content Module.
* @param array $post_content_fields Post Content fields data that should
* replace the Post Content Module.
*
* @return array
*/
function et_fb_generate_tb_body_area_with_post_content( $theme_builder_body_fields, $selector, $post_content_fields ) {
if ( ! isset( $selector['row'] ) && ! isset( $selector['column'] ) ) {
$original_post_content_module = $theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['module'] ];
$theme_builder_body_fields[ $selector['section'] ]['attrs']['post_content_module_attrs'] = $original_post_content_module['attrs'];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['module'] ]['content'] = $post_content_fields;
} elseif ( null === $selector['column'] ) {
$original_post_content_module = $theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['content'][ $selector['module'] ];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['attrs']['post_content_module_attrs'] = $original_post_content_module['attrs'];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['content'][ $selector['module'] ]['content'] = $post_content_fields;
} elseif ( isset( $selector['row_inner'] ) ) {
$original_post_content_module = $theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['column'] ]['content'][ $selector['row_inner'] ]['content'][ $selector['column_inner'] ]['content'][ $selector['module'] ];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['column'] ]['content'][ $selector['row_inner'] ]['content'][ $selector['column_inner'] ]['attrs']['post_content_module_attrs'] = $original_post_content_module['attrs'];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['column'] ]['content'][ $selector['row_inner'] ]['content'][ $selector['column_inner'] ]['content'][ $selector['module'] ]['content'] = $post_content_fields;
} else {
$original_post_content_module = $theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['content'][ $selector['column'] ]['content'][ $selector['module'] ];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['content'][ $selector['column'] ]['attrs']['post_content_module_attrs'] = $original_post_content_module['attrs'];
$theme_builder_body_fields[ $selector['section'] ]['content'][ $selector['row'] ]['content'][ $selector['column'] ]['content'][ $selector['module'] ]['content'] = $post_content_fields;
}
return $theme_builder_body_fields;
}
/**
* Ajax Callback: Retrieve builder data on frontend app load.
*/
function et_fb_retrieve_builder_data() {
if ( ! isset( $_POST['et_fb_helper_nonce'] ) || ! wp_verify_nonce( $_POST['et_fb_helper_nonce'], 'et_fb_load_helper_assets_nonce' ) ) { // phpcs:ignore ET.Sniffs.ValidatedSanitizedInput -- The nonce value is used only for comparision in the `wp_verify_nonce`.
die( -1 );
}
$post_id = ! empty( $_POST['et_post_id'] ) ? sanitize_text_field( $_POST['et_post_id'] ) : '';
if ( ! current_user_can( 'edit_posts' ) || ! current_user_can( 'edit_post', $post_id ) ) {
die( -1 );
}
$post_type = ! empty( $_POST['et_post_type'] ) ? sanitize_text_field( $_POST['et_post_type'] ) : 'post';
$layout_type = ! empty( $_POST['et_layout_type'] ) ? sanitize_text_field( $_POST['et_layout_type'] ) : '';
$fields_data = array_merge(
et_fb_get_builder_definitions( $post_type ),
et_fb_get_builder_shortcode_object( $post_type, $post_id, $layout_type )
);
// Enable zlib compression.
et_builder_enable_zlib_compression();
die( wp_json_encode( $fields_data ) );
}
add_action( 'wp_ajax_et_fb_retrieve_builder_data', 'et_fb_retrieve_builder_data' );
/**
* Replaces site_url in a json string with its protocol-less version.
*
* @param string $json The json string that contain site url.
*
* @return string
*/
function et_fb_remove_site_url_protocol( $json ) {
$no_proto = str_replace( '/', '\/', preg_replace( '#^\w+:#', '', get_site_url() ) );
$from = array(
"https:$no_proto" => $no_proto,
"http:$no_proto" => $no_proto,
);
return strtr( $json, $from );
}
/**
* Used to update the content of the cached definitions js file.
*
* @param string $content content? @todo Add param description.
* @param string $post_type Post type? @todo Add param description.
*
* @return string
*/
function et_fb_get_asset_definitions( $content, $post_type ) {
$definitions = et_fb_get_builder_definitions( $post_type );
return sprintf(
'window.ETBuilderBackend=jQuery.extend(true,%s,window.ETBuilderBackend)',
et_fb_remove_site_url_protocol( wp_json_encode( $definitions, ET_BUILDER_JSON_ENCODE_OPTIONS ) )
);
}
add_filter( 'et_fb_get_asset_definitions', 'et_fb_get_asset_definitions', 10, 2 );
/**
* Return Divi options setting page link.
*
* @return mixed|string|void
*/
function et_pb_get_options_page_link() {
// Builder plugin has different path to options page.
if ( et_is_builder_plugin_active() ) {
return admin_url( 'admin.php?page=et_divi_options#tab_et_dashboard_tab_content_api_main' );
}
return apply_filters( 'et_pb_theme_options_link', admin_url( 'admin.php?page=et_divi_options' ) );
}
/**
* Localization: Product tour text.
*
* @param integer $post_id The post id to determine the Save/Publish button text from post status.
*
* @return array
*/
function et_fb_get_product_tour_text( $post_id ) {
$post_status = get_post_status( $post_id );
$product_tour_text = array(
'start' => array(
'title' => esc_html__( 'Welcome To The Divi Builder', 'et_builder' ),
'description' => sprintf(
// translators: %10$s: Tour video overlay, %1$s: "Section" - label, %2$s: Add icon markup, %3$s: "Row" - label, %4$s: Add icon markup, %5$s: "Modules" - label, %6$s: Add icon markup, %7$s: Settings gear icon markup, %9$s: Documentation link markup.
__( '%10$sBuilding beautiful pages is a breeze using the Visual Builder. To get started, add a new %1$s to your page by pressing the %2$s button. Next, add a %3$s of columns inside your section by pressing the %4$s button. Finally, start adding some content %5$s inside your columns by pressing the %6$s button. You can customize the design and content of any element on the page by pressing the %7$s button. If you ever need help, visit our %9$s page for a full list of tutorials.', 'et_builder' ),
sprintf( '%1$s ', esc_html__( 'Section' ) ),
' ',
sprintf( '%1$s ', esc_html__( 'Row' ) ),
' ',
sprintf( '%1$s ', esc_html__( 'Modules' ) ),
' ',
' ',
'? ',
sprintf( '%1$s ', esc_html__( 'Documentation' ) ),
sprintf(
'',
esc_url( ET_BUILDER_URI . '/frontend-builder/assets/img/product-tour-intro.jpg' )
)
),
'endButtonText' => esc_html__( 'Start Building', 'et_builder' ),
'skipButtonText' => esc_html__( 'Take the Tour', 'et_builder' ),
),
'loadLayout' => array(
'title' => esc_html__( 'Load A New Layout', 'et_builder' ),
'description' => esc_html__( 'Loading pre-made layouts is a great way to jump-start your new page. The Divi Builder comes with dozens of layouts to choose from, and you can find lots of great free layouts online too. You can save your favorite layouts to the Divi Library and load them on new pages or share them with the community. Click the highlighted button to open the layouts menu and select a pre-made layout.', 'et_builder' ),
),
'selectLayoutPack' => array(
'title' => esc_html__( 'Choose A Layout Pack', 'et_builder' ),
'description' => esc_html__( 'Here you can see a list of pre-made layout packs that ship with the Divi Builder. You can also access layouts that you have saved to your Divi Library. Choose the “Divi Builder Demo” layout pack to see the layouts it includes.', 'et_builder' ),
),
'loadLayoutItem' => array(
'title' => esc_html__( 'Choose A Layout To Start With', 'et_builder' ),
'description' => esc_html__( 'Now you can see more details about the layout pack as well as a list of the layouts it includes. Click “Use Layout” to apply the layout to your page.', 'et_builder' ),
),
'addSection' => array(
'title' => esc_html__( 'Add A New Section', 'et_builder' ),
'description' => sprintf(
// translators: %1$s: "Sections" - label, %2$s: "Rows" - label.
__( 'Now that your pre-made layout has been loaded, we can start adding new content to the page. The Divi Builder organizes content using %1$s, %2$s and Modules. Sections are the largest organizational element. Click the highlighted button to add a new section to the page.', 'et_builder' ),
sprintf( '%1$s ', esc_html__( 'Sections' ) ),
sprintf( '%1$s ', esc_html__( 'Rows' ) )
),
),
'selectSectionType' => array(
'title' => esc_html__( 'Choose A Section Type', 'et_builder' ),
'description' => sprintf(
// translators: %1$s: "Regular" - label text, %2$s: "Specialty" - label tex, %3$s: "Fullwidth" - label tex.
__( 'The Divi Builder has three different section types. %1$s sections conform to the standard width of your page layout. %2$s Sections can be used to create advanced sidebar layouts. %3$s sections extend the full width of your page and can be used with fullwidth modules. Click the “Regular” section button to add a new section to your page.', 'et_builder' ),
sprintf( '%1$s ', esc_html__( 'Regular' ) ),
sprintf( '%1$s ', esc_html__( 'Specialty' ) ),
sprintf( '%1$s ', esc_html__( 'Fullwidth' ) )
),
),
'selectRow' => array(
'title' => esc_html__( 'Add A New Row Of Columns', 'et_builder' ),
'description' => sprintf(
// translators: %1$s: "Rows" - label.
__( 'Every section contains one or more %1$s of columns. You can choose between various column layouts for each row you add to your page. Click the highlighted three-column layout to add a new row to your section.', 'et_builder' ),
sprintf( '%1$s ', esc_html__( 'Rows' ) )
),
),
'selectModule' => array(
'title' => esc_html__( 'Add A Module To The Column', 'et_builder' ),
'description' => esc_html__( 'Within each column you can add one or more Modules. A module is basic content element. The Divi Builder comes with over 40 different content elements to choose from, such as Images, Videos, Text, and Buttons. Click the highlighted Blurb button to add a new Blurb module to the first column in your row.', 'et_builder' ),
),
'configureModule' => array(
'title' => esc_html__( 'Adjust Your Module Settings', 'et_builder' ),
'description' => esc_html__( 'Each Module comes with various settings. These settings are separated into three tabs: Content, Design and Advanced. Inside the content tab you can modify the module content elements, such as text and images. If you need more control over the appearance of your module, head over to the Design tab. For more advanced modifications, such as custom CSS and HTML attributes, explore the Advanced tab. Try adjusting the Title of your blurb by clicking into the highlighted field.', 'et_builder' ),
),
'saveModule' => array(
'title' => esc_html__( 'Accept Or Discard Your Changes', 'et_builder' ),
'description' => esc_html__( 'Whenever you make changes in the Divi Builder, these changes can be Undone, Redone, Discarded or Accepted. Now that you have adjusted your module’s title, you can click the red discard button to cancel these changes, or your can click the green button to accept them.', 'et_builder' ),
),
'duplicateModule' => array(
'title' => esc_html__( 'Hover To Access Action Buttons', 'et_builder' ),
'description' => esc_html__( 'Whenever you hover over a Section, Row or Module in the Divi Builder, action buttons will appear. These buttons can be used to move, modify, duplicate or delete your content. Click the highlighted “duplicate” icon to duplicate the blurb module that you just added to the page.', 'et_builder' ),
),
'moveModule' => array(
'title' => __( 'Drag & Drop Content', 'et_builder' ),
'description' => esc_html__( 'Every item on the page can be dragged and dropped to new locations. Using your mouse, click the highlighted move icon and hold down the mouse button. While holding down the mouse button, move your cursor over to the empty column and then release your mouse button to drop the module into the new column.', 'et_builder' ),
),
'rightClickCopy' => array(
'title' => esc_html__( 'Access Right Click Options', 'et_builder' ),
'description' => esc_html__( 'In addition to hover actions, additional options can be accessed by Right Clicking or Cmd + Clicking on any module, row or section. Using the right click menu shown, click the highlighted “Copy Module” button to copy the blurb module that you just moved.', 'et_builder' ),
),
'rightClickPaste' => array(
'title' => esc_html__( 'Paste Your Copied Module', 'et_builder' ),
'description' => esc_html__( 'Now that you have copied a module using the Right Click menu, you can Right Click in a new location to paste that module. Using the right click options shown, click the “Paste Module” button to paste the module you just copied into the empty column.', 'et_builder' ),
),
'rowOptions' => array(
'title' => esc_html__( 'Access Your Row Options', 'et_builder' ),
'description' => esc_html__( 'Every Row and Section has its own set of options that can be used to adjust the item’s appearance. You can adjust its width, padding, background and more. To access a row’s settings, hover over the row and click the highlighted options button.', 'et_builder' ),
),
'editRow' => array(
'title' => esc_html__( 'Adjust Your Row Setting', 'et_builder' ),
'description' => esc_html__( 'Just like Modules, Rows come with a lot of settings that are separated into the Content, Design and Advanced tabs. Click the highlighted button to add a new background color to your row.', 'et_builder' ),
),
'saveRow' => array(
'title' => esc_html__( 'Accept Your Changes', 'et_builder' ),
'description' => esc_html__( 'Click the highlighted green check mark button to accept your changes. ', 'et_builder' ),
),
'pageSettings' => array(
'title' => esc_html__( 'Open Your Page Settings', 'et_builder' ),
'description' => esc_html__( 'While using the Divi Builder, you can access your page settings by toggling the page settings bar at the bottom of your screen. Click the highlighted button to reveal your page settings.', 'et_builder' ),
),
'tabletPreview' => array(
'title' => esc_html__( 'Preview Your Page On Mobile', 'et_builder' ),
'description' => esc_html__( 'While editing your page, it’s easy to see what your design will look like on mobile devices. You can also make adjustments to your module, row and section settings for each mobile breakpoint. Click the highlighted “Tablet” icon to enter Tablet preview mode. ', 'et_builder' ),
),
'desktopPreview' => array(
'title' => esc_html__( 'Switch Back To Desktop Mode', 'et_builder' ),
'description' => esc_html__( 'You can switch back and forth between each preview mode freely while editing your page. Now that we have previewed our page on Tablet, let’s switch back to Desktop preview mode by clicking the highlighted button.', 'et_builder' ),
),
'openHistory' => array(
'title' => esc_html__( 'Access Your Editing History', 'et_builder' ),
'description' => esc_html__( 'Every change you make while editing your page is saved in your editing history. You can navigate backwards and forwards through time to any point during your current editing session, as well as undo and redo recent changes. Click the highlighted History button to access your editing history. ', 'et_builder' ),
),
'editHistory' => array(
'title' => esc_html__( 'Undo, Redo And Restore', 'et_builder' ),
'description' => esc_html__( 'Here you can undo, redo or restore a saved history state. If you change your mind about recent changes, simply click back in time and start building again. You can also undo and redo recent changes. Click the undo and redo buttons and then accept your changes by clicking the green check mark.', 'et_builder' ),
),
'savePage' => array(
'title' => esc_html__( 'Save Your Page', 'et_builder' ),
'description' => sprintf(
// translators: %1$s: "Save" or "Publish" - label.
esc_html__( 'When you are all done, you can save your changes by clicking the %1$s button inside of your page settings bar. You can also press Ctrl + S at any time to save your changes. Click the highlighted Save button to save your changes. Don’t worry, the page you were working on before starting this tour will not be lost!', 'et_builder' ),
in_array( $post_status, array( 'private', 'publish' ), true ) ? esc_html__( 'Save', 'et_builder' ) : esc_html__( 'Publish', 'et_builder' )
),
),
'finish' => array(
'title' => esc_html__( 'You’re Ready To Go!', 'et_builder' ),
'description' => sprintf(
// translators: %10$s: Tour video overlay, %1$s: "Section" - label, %2$s: Add icon markup, %3$s: "Row" - label, %4$s: Add icon markup, %5$s: "Modules" - label, %6$s: Add icon markup, %7$s: Settings gear icon markup, %9$s: Documentation link markup.
__( '%10$sBuilding beautiful pages is a breeze using the Visual Builder. To get started, add a new %1$s to your page by pressing the %2$s button. Next, add a %3$s of columns inside your section by pressing the %4$s button. Finally, start adding some content %5$s inside your columns by pressing the %6$s button. You can customize the design and content of any element on the page by pressing the %7$s button. If you ever need help, visit our %9$s page for a full list of tutorials.', 'et_builder' ),
sprintf( '%1$s ', esc_html__( 'Section' ) ),
' ',
sprintf( '%1$s ', esc_html__( 'Row' ) ),
' ',
sprintf( '%1$s ', esc_html__( 'Modules' ) ),
' ',
' ',
'? ',
sprintf( '%1$s ', esc_html__( 'Documentation' ) ),
sprintf(
'',
esc_url( ET_BUILDER_URI . '/frontend-builder/assets/img/product-tour-intro.jpg' )
)
),
'endButtonText' => esc_html__( 'Start Building', 'et_builder' ),
),
'endButtonTextDefault' => esc_html__( 'End the Tour', 'et_builder' ),
'skipButtonTextDefault' => esc_html__( 'Skip This Step', 'et_builder' ),
);
return $product_tour_text;
}
/**
* Process builder shortcode into object.
*
* The standard do_shortcode filter should be removed, and
* this function hooked instead.
*
* This function is very similar to `do_shortcode`,
* with the main differences being:
* - Its main design is to allow recursive array to be built out of wp shortcode
* - Allows shortcode callback to return an array rather than a string
* - It tracks the inner `index` / `_i` of each child shortcode to the passed content, which is used in the address creation as well
* - It uses and passes `$address` & `$parent_address`, which are used by FB app.
*
* @param string $content post content.
* @param string $parent_address parent shortcode address.
* @param string $global_parent ?? @todo Add param doc.
* @param string $global_parent_type ?? @todo Add param doc.
* @param string $parent_type ?? @todo Add param doc.
*
* @return mixed
*/
function et_fb_process_shortcode( $content, $parent_address = '', $global_parent = '', $global_parent_type = '', $parent_type = '', $theme_builder_area = '' ) {
global $shortcode_tags, $fb_processing_counter;
if ( false === strpos( $content, '[' ) ) {
return $content;
}
// Count started processes.
$fb_processing_counter = isset( $fb_processing_counter ) ? $fb_processing_counter + 1 : 1;
// Find all registered tag names in $content.
preg_match_all( '@\[([^<>&/\[\]\x00-\x20=]++)@', $content, $matches );
// Only need unique tag names.
$unique_matches = array_unique( $matches[1] );
$tagnames = array_intersect( array_keys( $shortcode_tags ), $unique_matches );
$pattern = get_shortcode_regex( $unique_matches );
$content = preg_match_all( "/$pattern/", $content, $matches, PREG_SET_ORDER );
$_matches = array();
$_index = 0;
foreach ( $matches as $match ) {
$tag = $match[2];
// reset global parent data to calculate it correctly for next modules.
if ( $global_parent_type === $tag && '' !== $global_parent ) {
$global_parent = '';
$global_parent_type = '';
}
$attr = shortcode_parse_atts( $match[3] );
if ( ! is_array( $attr ) ) {
$attr = array();
}
$index = $_index++;
$address = isset( $parent_address ) && '' !== $parent_address ? (string) $parent_address . '.' . (string) $index : (string) $index;
// set global parent and global parent tag if current module is global and can be a parent.
$possible_global_parents = array( 'et_pb_section', 'et_pb_row', 'et_pb_row_inner' );
if ( '' === $global_parent && in_array( $tag, $possible_global_parents, true ) ) {
$global_parent = isset( $attr['global_module'] ) ? $attr['global_module'] : '';
$global_parent_type = $tag;
}
// As responsive content attributes value might be has been encoded before saving to database,
// so we need to decode it before passing back to builder.
if ( $attr ) {
$decoded_content_fields = array(
'content__hover',
'content_tablet',
'content_phone',
'raw_content__hover',
'raw_content_tablet',
'raw_content_phone',
);
foreach ( $decoded_content_fields as $decoded_content_field ) {
if ( array_key_exists( $decoded_content_field, $attr ) ) {
$attr[ $decoded_content_field ] = str_replace( array( '%22', '%92', '%91', '%93' ), array( '"', '\\', '[', ']' ), $attr[ $decoded_content_field ] );
}
}
}
$attr['_i'] = $index;
$attr['_address'] = $address;
// Builder shortcode which exist on page but not registered in WP i.e. 3rd party shortcode when 3rd party module disabled
// Add dummy object to render it in Divi Builder.
if ( ! in_array( $tag, $tagnames, true ) ) {
$_matches[] = array(
'_i' => $index,
'_order' => $index,
'address' => $address,
'vb_support' => 'off',
'component_path' => 'et-fb-removed-component',
'type' => $tag,
'attrs' => $attr,
);
continue;
}
// Flag that the shortcode object is being built.
$GLOBALS['et_fb_processing_shortcode_object'] = true;
if ( isset( $match[5] ) ) {
// phpcs:ignore Generic.PHP.ForbiddenFunctions.Found -- `call_user_func` calls the registered shortcode callback.
$output = call_user_func( $shortcode_tags[ $tag ], $attr, $match[5], $tag, $parent_address, $global_parent, $global_parent_type, $parent_type, $theme_builder_area );
} else {
// self-closing tag.
// phpcs:ignore Generic.PHP.ForbiddenFunctions.Found -- `call_user_func` calls the registered shortcode callback.
$output = call_user_func( $shortcode_tags[ $tag ], $attr, null, $tag );
}
$_matches[] = et_fb_add_additional_attrs( $attr, $output );
}
// Count finished processes.
$fb_processing_counter = $fb_processing_counter - 1; // phpcs:ignore Squiz.Operators.IncrementDecrementUsage -- This is more readable.
// Make sure ALL the processes finished to avoid wrong disabling of `et_fb_processing_shortcode_object` when several concurrent instances of `et_fb_process_shortcode` running.
if ( 0 === $fb_processing_counter ) {
// Turn off the flag since the shortcode object is done being built.
et_fb_reset_shortcode_object_processing();
}
return $_matches;
}
/**
* Allowlist any additional attributes.
*
* @param array $processed_attrs Shortcode's processed attributes.
* @param array $output Shortcode output.
*
* @return mixed
*/
function et_fb_add_additional_attrs( $processed_attrs, $output ) {
if ( empty( $output['attrs'] ) ) {
return $output;
}
// A list of all the attributes that are already returned after the shortcode is processed.
$safe_attrs = array_keys( $output['attrs'] );
$allowed_attrs = array();
foreach ( $processed_attrs as $attr => $value ) {
if ( ! preg_match( '~_hover(_enabled)?$~', $attr ) ) {
continue;
}
// if color value includes `gcid-`, check for associated Global Color value.
if ( empty( $value ) || false === strpos( $value, 'gcid-' ) ) {
continue;
}
$global_color_info = et_builder_get_all_global_colors();
// If there are no matching Global Colors, return null.
if ( ! is_array( $global_color_info ) ) {
continue;
}
foreach ( $global_color_info as $gcid => $details ) {
if ( false !== strpos( $value, $gcid ) ) {
// Match substring (needed for attrs like gradient stops).
$value = str_replace( $gcid, $details['color'], $value );
}
}
// Finally, escape the output.
if ( ! empty( $global_color_info['color'] ) ) {
$value = esc_attr( $value );
}
$allowed_attrs[ $attr ] = $value;
}
// Extra conversion for the case with the `font_icon__hover` option.
if ( ! empty( $allowed_attrs['font_icon__hover'] ) && et_pb_maybe_old_divi_font_icon( $allowed_attrs['font_icon__hover'] ) ) {
$allowed_attrs['font_icon__hover'] = et_pb_build_extended_font_icon_value( $allowed_attrs['font_icon__hover'], null, null, true );
}
if ( $allowed_attrs ) {
$output['attrs'] = array_merge( $output['attrs'], $allowed_attrs );
}
return $output;
}
/**
* Parse builder shortcode into an array.
*
* @param string $content Builder built post content.
*
* @return array Array representation of the builder shortcode.
*/
function et_pb_parse_shortcode_to_array( $content ) {
global $shortcode_tags;
if ( false === strpos( $content, '[' ) ) {
return $content;
}
// Find all registered tag names in $content.
preg_match_all( '@\[([^<>&/\[\]\x00-\x20=]++)@', $content, $matches );
$tagnames = array_intersect( array_keys( $shortcode_tags ), $matches[1] );
$pattern = get_shortcode_regex( $matches[1] );
$content = preg_match_all( "/$pattern/", $content, $matches, PREG_SET_ORDER );
$shortcode_data = array();
foreach ( $matches as $match ) {
$tag = $match[2];
$attr = shortcode_parse_atts( $match[3] );
if ( ! is_array( $attr ) ) {
$attr = array();
}
$_shortcode_data = array(
'type' => $tag,
'attrs' => $attr,
);
if ( ! empty( $match[5] ) ) {
$_shortcode_data['content'] = et_pb_parse_shortcode_to_array( $match[5] );
}
$shortcode_data[] = $_shortcode_data;
}
return $shortcode_data;
}
/**
* Parse post content that includes builder shortcode to an array,
* then run it back through "some sanity check" and "some sanitization"
* and then form into a shortcode again.
*
* @see et_fb_process_to_shortcode() for exact sanitizations performed.
*
* @param string $content Builder built post content.
* @param bool $force_valid_builder_slugs Whether to force the shortcode to allow valid builder shortcode slugs only.
*
* @return string Sanitized builder built post content.
*/
function et_pb_sanitize_shortcode( $content, $force_valid_builder_slugs = false ) {
global $shortcode_tags;
if ( false === strpos( $content, '[' ) ) {
return $content;
}
$content_array = et_pb_parse_shortcode_to_array( $content );
$options = array(
'force_valid_slugs' => $force_valid_builder_slugs,
);
$new_content = et_fb_process_to_shortcode( $content_array, $options );
return $new_content;
}
/**
* Use shortcode tag which renders the content to correctly display its properties.
*
* @param string $tag shortcode tag.
*
* @return string
*/
function et_fb_prepare_tag( $tag ) {
// List of aliases.
$aliases = apply_filters(
'et_fb_prepare_tag_aliases',
array(
'et_pb_accordion_item' => 'et_pb_toggle',
)
);
return isset( $aliases[ $tag ] ) ? $aliases[ $tag ] : $tag;
}
if ( ! function_exists( 'et_strip_shortcodes' ) ) :
/**
* Strip builder shortcodes only, leaving default WordPress shortcodes intact.
*
* @param string $content the content.
* @param string $truncate_post_based_shortcodes_only Optional. Whether trunct only post based shortcodes.
*/
function et_strip_shortcodes( $content, $truncate_post_based_shortcodes_only = false ) {
global $shortcode_tags;
$content = trim( $content );
$strip_content_shortcodes = array(
'et_pb_code',
'et_pb_fullwidth_code',
'et_pb_social_media_follow_network',
);
// list of post-based shortcodes.
if ( $truncate_post_based_shortcodes_only ) {
$strip_content_shortcodes = array(
'et_pb_post_slider',
'et_pb_fullwidth_post_slider',
'et_pb_blog',
'et_pb_comments',
);
}
foreach ( $strip_content_shortcodes as $shortcode_name ) {
$regex = sprintf(
'(\[%1$s[^\]]*\][^\[]*\[\/%1$s\]|\[%1$s[^\]]*\])',
esc_html( $shortcode_name )
);
$content = preg_replace( $regex, '', $content );
}
// do not proceed if we need to truncate post-based shortcodes only.
if ( $truncate_post_based_shortcodes_only ) {
return $content;
}
$shortcode_tag_names = array();
foreach ( $shortcode_tags as $shortcode_tag_name => $shortcode_tag_cb ) {
if ( 0 !== strpos( $shortcode_tag_name, 'et_pb_' ) ) {
continue;
}
$shortcode_tag_names[] = $shortcode_tag_name;
}
$et_shortcodes = implode( '|', $shortcode_tag_names );
$regex_opening_shortcodes = sprintf( '(\[(%1$s)[^\]]+\])', esc_html( $et_shortcodes ) );
$regex_closing_shortcodes = sprintf( '(\[\/(%1$s)\])', esc_html( $et_shortcodes ) );
$content = preg_replace( $regex_opening_shortcodes, '', $content );
$content = preg_replace( $regex_closing_shortcodes, '', $content );
return $content;
}
endif;
/**
* Reset shortcode object processing.
*/
function et_fb_reset_shortcode_object_processing() {
$GLOBALS['et_fb_processing_shortcode_object'] = false;
}
add_action( 'et_fb_enqueue_assets', 'et_fb_backend_helpers' );
if ( ! function_exists( 'et_builder_maybe_flush_rewrite_rules' ) ) :
/**
* Flush rewrite rules if theme option saved value and passed $value are not same.
*
* @param string $setting_name The theme option.
* @param string $value The value to be compared.
*/
function et_builder_maybe_flush_rewrite_rules( $setting_name, $value = 'done' ) {
$string_value = (string) $value;
$saved_value = et_get_option( $setting_name );
if ( $saved_value && $saved_value === $string_value ) {
return;
}
flush_rewrite_rules();
et_update_option( $setting_name, $string_value );
}
endif;
/**
* Flush rewrite rules to fix the issue Layouts, not being visible on front-end and visual builder,
* if pretty permalinks were enabled
*
* @return void
*/
function et_pb_maybe_flush_rewrite_rules_library() {
// Run flush rewrite only when et_pb_layout post type registered.
if ( post_type_exists( 'et_pb_layout' ) ) {
et_builder_maybe_flush_rewrite_rules( 'et_flush_rewrite_rules_library', ET_BUILDER_PRODUCT_VERSION );
}
}
add_action( 'init', 'et_pb_maybe_flush_rewrite_rules_library', 9 );
/**
* Remove et_builder_maybe_flush_rewrite_rules flag if flush_rewrite_rules() is called while
* `et_pb_layout` post type hasn't been registered
*
* @since 3.19.18
*
* @param string|array $old_value old option value.
* @param string|array $value new option value.
* @param string $option option name.
*/
function et_pb_maybe_remove_flush_rewrite_rules_library_flag( $old_value, $value, $option ) {
// rewrite rules for CPT that are rebuilt by flush_rewrite_rules() are based on
// get_post_types( array( '_builtin' => false ) ) value; Hence if flush_rewrite_rules() is
// executed while `et_pb_layout` CPT hasn't been registered (usually by third party plugin)
// et_pb_maybe_flush_rewrite_rules_library() flag has to be removed to trigger flush_rewrite_rules()
// via et_pb_maybe_flush_rewrite_rules_library() which contains `et_pb_layout` rewrite rules
// because et_pb_maybe_flush_rewrite_rules_library() checks for `et_pb_layout` first.
if ( '' === $value && ! post_type_exists( 'et_pb_layout' ) ) {
et_update_option( 'et_flush_rewrite_rules_library', '' );
}
}
add_action( 'update_option_rewrite_rules', 'et_pb_maybe_remove_flush_rewrite_rules_library_flag', 10, 3 );
if ( ! function_exists( 'et_builder_get_shortcuts' ) ) :
/**
* Get list of shortcut available on BB and FB
*
* @param string $on (fb|bb) shortcut mode.
* @return array shortcut list
*/
function et_builder_get_shortcuts( $on = 'fb' ) {
$shortcuts = array(
'page' => array(
'page_title' => array(
'title' => esc_html__( 'Page Shortcuts', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'undo' => array(
'kbd' => array( 'super', 'z' ),
'desc' => esc_html__( 'Undo', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'redo' => array(
'kbd' => array( 'super', 'y' ),
'desc' => esc_html__( 'Redo', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'save' => array(
'kbd' => array( 'super', 's' ),
'desc' => esc_html__( 'Save Page', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'save_as_draft' => array(
'kbd' => array( 'super', 'shift', 's' ),
'desc' => esc_html__( 'Save Page As Draft', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'exit' => array(
'kbd' => array( 'super', 'e' ),
'desc' => esc_html__( 'Exit Visual Builder', 'et_builder' ),
'on' => array(
'fb',
),
),
'exit_to_backend_builder' => array(
'kbd' => array( 'super', 'shift', 'e' ),
'desc' => esc_html__( 'Exit To Backend Builder', 'et_builder' ),
'on' => array(
'fb',
),
),
'toggle_settings_bar' => array(
'kbd' => array( 't' ),
'desc' => esc_html__( 'Toggle Settings Bar', 'et_builder' ),
'on' => array(
'fb',
),
),
'open_page_settings' => array(
'kbd' => array( 'o' ),
'desc' => esc_html__( 'Open Page Settings', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'open_history' => array(
'kbd' => array( 'h' ),
'desc' => esc_html__( 'Open History Window', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'open_portability' => array(
'kbd' => array( 'p' ),
'desc' => esc_html__( 'Open Portability Window', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'zoom_in' => array(
'kbd' => array( 'super', '+' ),
'desc' => esc_html__( 'Responsive Zoom In', 'et_builder' ),
'on' => array(
'fb',
),
),
'zoom_out' => array(
'kbd' => array( 'super', '-' ),
'desc' => esc_html__( 'Responsive Zoom Out', 'et_builder' ),
'on' => array(
'fb',
),
),
'wireframe' => array(
'kbd' => array( 'shift', 'w' ),
'desc' => esc_html__( 'Wireframe Mode', 'et_builder' ),
'on' => array(
'fb',
),
),
'click_mode' => array(
'kbd' => array( 'super', 'shift', 'c' ),
'desc' => esc_html__( 'Click Mode', 'et_builder' ),
'on' => array(
'fb',
),
),
'grid_mode' => array(
'kbd' => array( 'super', 'shift', 'g' ),
'desc' => esc_html__( 'Grid Mode', 'et_builder' ),
'on' => array(
'fb',
),
),
'hover_mode' => array(
'kbd' => array( 'super', 'shift', 'h' ),
'desc' => esc_html__( 'Hover Mode', 'et_builder' ),
'on' => array(
'fb',
),
),
'help' => array(
'kbd' => array( '?' ),
'desc' => esc_html__( 'List All Shortcuts', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
),
'inline' => array(
'inline_title' => array(
'title' => esc_html__( 'Inline Editor Shortcuts', 'et_builder' ),
'on' => array(
'fb',
),
),
'escape' => array(
'kbd' => array( 'esc' ),
'desc' => esc_html__( 'Exit Inline Editor', 'et_builder' ),
'on' => array(
'fb',
),
),
),
'module' => array(
'module_title' => array(
'title' => esc_html__( 'Module Shortcuts', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'module_copy' => array(
'kbd' => array( 'super', 'c' ),
'desc' => esc_html__( 'Copy Module', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'module_cut' => array(
'kbd' => array( 'super', 'x' ),
'desc' => esc_html__( 'Cut Module', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'module_paste' => array(
'kbd' => array( 'super', 'v' ),
'desc' => esc_html__( 'Paste Module', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'module_copy_styles' => array(
'kbd' => array( 'super', 'alt', 'c' ),
'desc' => esc_html__( 'Copy Module Styles', 'et_builder' ),
'on' => array(
'fb',
),
),
'module_paste_styles' => array(
'kbd' => array( 'super', 'alt', 'v' ),
'desc' => esc_html__( 'Paste Module Styles', 'et_builder' ),
'on' => array(
'fb',
),
),
'module_reset_styles' => array(
'kbd' => array( 'super', 'alt', 'r' ),
'desc' => esc_html__( 'Reset Module Styles', 'et_builder' ),
'on' => array(
'fb',
),
),
'module_lock' => array(
'kbd' => array( 'super', 'shift', 'l' ),
'desc' => esc_html__( 'Lock Module', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'module_disable' => array(
'kbd' => array( 'super', 'shift', 'd' ),
'desc' => esc_html__( 'Disable Module', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'drag_auto_copy' => array(
'kbd' => array( 'alt', 'module move' ),
'desc' => esc_html__( 'Move and copy module into dropped location', 'et_builder' ),
'on' => array(
'fb',
),
),
'column_change_structure' => array(
'kbd' => array( 'c', array( '1', '2', '3', '4', '5', '...' ) ),
'desc' => esc_html__( 'Change Column Structure', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'row_make_fullwidth' => array(
'kbd' => array( 'r', 'f' ),
'desc' => esc_html__( 'Make Row Fullwidth', 'et_builder' ),
'on' => array(
'fb',
),
),
'row_edit_gutter' => array(
'kbd' => array( 'g', array( '1', '2', '3', '4' ) ),
'desc' => esc_html__( 'Change Gutter Width', 'et_builder' ),
'on' => array(
'fb',
),
),
'add_new_row' => array(
'kbd' => array( 'r', array( '1', '2', '3', '4', '5', '...' ) ),
'desc' => esc_html__( 'Add New Row', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'add_new_section' => array(
'kbd' => array( 's', array( '1', '2', '3' ) ),
'desc' => esc_html__( 'Add New Section', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'resize_padding_auto_opposite' => array(
'kbd' => array( 'shift', 'Drag Padding' ),
'desc' => esc_html__( 'Restrict padding to 10px increments', 'et_builder' ),
'on' => array(
'fb',
),
),
'resize_padding_limited' => array(
'kbd' => array( 'alt', 'Drag Padding' ),
'desc' => esc_html__( 'Padding limited to opposing value', 'et_builder' ),
'on' => array(
'fb',
),
),
'resize_padding_10' => array(
'kbd' => array( 'shift', 'alt', 'Drag Padding' ),
'desc' => esc_html__( 'Mirror padding on both sides', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_row' => array(
'kbd' => array( 'r', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Row Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_row' => array(
'kbd' => array( 'r', 'alt', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Row Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_section' => array(
'kbd' => array( 's', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Section Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_section' => array(
'kbd' => array( 's', 'alt', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Section Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_module' => array(
'kbd' => array( 'm', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Module Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_module' => array(
'kbd' => array( 'm', 'alt', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Module Padding', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_row_10' => array(
'kbd' => array( 'r', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Row Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_row_10' => array(
'kbd' => array( 'r', 'alt', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Row Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_section_10' => array(
'kbd' => array( 's', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Section Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_section_10' => array(
'kbd' => array( 's', 'alt', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Section Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
'increase_padding_module_10' => array(
'kbd' => array( 'm', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Increase Module Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
'decrease_padding_module_10' => array(
'kbd' => array( 'm', 'alt', 'shift', array( 'left', 'right', 'up', 'down' ) ),
'desc' => esc_html__( 'Decrease Module Padding By 10px', 'et_builder' ),
'on' => array(
'fb',
),
),
),
'modal' => array(
'modal_title' => array(
'title' => esc_html__( 'Modal Shortcuts', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'escape' => array(
'kbd' => array( 'esc' ),
'desc' => esc_html__( 'Close Modal', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'save_changes' => array(
'kbd' => array( 'enter' ),
'desc' => esc_html__( 'Save Changes', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'undo' => array(
'kbd' => array( 'super', 'z' ),
'desc' => esc_html__( 'Undo', 'et_builder' ),
'on' => array(
'fb',
),
),
'redo' => array(
'kbd' => array( 'super', 'shift', 'z' ),
'desc' => esc_html__( 'Redo', 'et_builder' ),
'on' => array(
'fb',
),
),
'switch_tabs' => array(
'kbd' => array( 'shift', 'tab' ),
'desc' => esc_html__( 'Switch Tabs', 'et_builder' ),
'on' => array(
'fb',
'bb',
),
),
'toggle_expand' => array(
'kbd' => array( 'super', 'enter' ),
'desc' => esc_html__( 'Expand Modal Fullscreen', 'et_builder' ),
'on' => array(
'fb',
),
),
'toggle_snap' => array(
'kbd' => array( 'super', array( 'left', 'right' ) ),
'desc' => esc_html__( 'Snap Modal Left / Right', 'et_builder' ),
'on' => array(
'fb',
),
),
'quick_actions' => array(
'kbd' => array( 'shift', 'space' ),
'desc' => esc_html__( 'Quick Actions', 'et_builder' ),
'on' => array(
'fb',
),
),
'layers_view' => array(
'kbd' => array( 'super', 'l' ),
'desc' => esc_html__( 'Layers View', 'et_builder' ),
'on' => array(
'fb',
),
),
),
);
$shortcuts = apply_filters( 'et_builder_get_shortcuts', $shortcuts );
// Filter shortcuts.
$filtered_shortcuts = array();
foreach ( $shortcuts as $group_key => $group ) {
foreach ( $group as $shortcut_key => $shortcut ) {
if ( in_array( $on, $shortcut['on'], true ) ) {
$filtered_shortcuts[ $group_key ][ $shortcut_key ] = $shortcut;
}
}
}
return $filtered_shortcuts;
}
endif;
if ( ! function_exists( 'et_pb_get_responsive_status' ) ) :
/**
* Parsed *_last_edited value and determine wheter the passed string means it has responsive value or not
* *_last_edited holds two values (responsive status and last opened tabs) in the following format: status|last_opened_tab
*
* @param string $last_edited last_edited data.
* @return bool
*/
function et_pb_get_responsive_status( $last_edited ) {
$parsed_last_edited = is_string( $last_edited ) ? explode( '|', $last_edited ) : array( 'off', 'desktop' );
return isset( $parsed_last_edited[0] ) ? 'on' === $parsed_last_edited[0] : false;
}
endif;
if ( ! function_exists( 'et_pb_get_value_unit' ) ) :
/**
* Get unit of given value
*
* @param string $value string with unit.
* @param string $default_unit default unit.
*
* @return string unit name
*/
function et_pb_get_value_unit( $value, $default_unit = 'px' ) {
$value = isset( $value ) ? $value : '';
$valid_one_char_units = array( '%', 'x' );
$valid_two_chars_units = array( 'em', 'px', 'cm', 'mm', 'in', 'pt', 'pc', 'ex', 'vh', 'vw', 'ms' );
$valid_three_chars_units = array( 'deg', 'rem' );
$important = '!important';
$important_length = strlen( $important );
$value_length = strlen( $value );
if ( '' === $value || is_numeric( $value ) ) {
return $default_unit;
}
if ( substr( $value, ( 0 - $important_length ), $important_length ) === $important ) {
$value_length = $value_length - $important_length;
$value = trim( substr( $value, 0, $value_length ) );
}
if ( in_array( substr( $value, -3, 3 ), $valid_three_chars_units, true ) ) {
return substr( $value, -3, 3 );
}
if ( in_array( substr( $value, -2, 2 ), $valid_two_chars_units, true ) ) {
return substr( $value, -2, 2 );
}
if ( in_array( substr( $value, -1, 1 ), $valid_one_char_units, true ) ) {
return substr( $value, -1, 1 );
}
return $default_unit;
}
endif;
if ( ! function_exists( 'et_sanitize_input_unit' ) ) :
/**
* Sanitized value and its unit
*
* @param mixed $value Input value.
* @param string $auto_important Whether add !important specificity. Default false.
* @param string|bool $default_unit The default unit.
*
* @return string sanitized input and its unit
*/
function et_sanitize_input_unit( $value = '', $auto_important = false, $default_unit = false ) {
$value = (string) $value;
$valid_one_char_units = array( '%', 'x' );
$valid_two_chars_units = array( 'em', 'px', 'cm', 'mm', 'in', 'pt', 'pc', 'ex', 'vh', 'vw', 'ms' );
$valid_three_chars_units = array( 'deg', 'rem' );
$important = '!important';
$important_length = strlen( $important );
$has_important = false;
$value_length = strlen( $value );
$unit_value;
// Check for important.
if ( substr( $value, ( 0 - $important_length ), $important_length ) === $important ) {
$has_important = true;
$value_length = $value_length - $important_length;
$value = trim( substr( $value, 0, $value_length ) );
}
if ( in_array( substr( $value, -3, 3 ), $valid_three_chars_units, true ) ) {
$unit_value = floatval( $value ) . substr( $value, -3, 3 );
// Re-add !important tag.
if ( $has_important && ! $auto_important ) {
$unit_value = $unit_value . ' ' . $important;
}
return $unit_value;
}
if ( in_array( substr( $value, -2, 2 ), $valid_two_chars_units, true ) ) {
$unit_value = floatval( $value ) . substr( $value, -2, 2 );
// Re-add !important tag.
if ( $has_important && ! $auto_important ) {
$unit_value = $unit_value . ' ' . $important;
}
return $unit_value;
}
if ( in_array( substr( $value, -1, 1 ), $valid_one_char_units, true ) ) {
$unit_value = floatval( $value ) . substr( $value, -1, 1 );
// Re-add !important tag.
if ( $has_important && ! $auto_important ) {
$unit_value = $unit_value . ' ' . $important;
}
return $unit_value;
}
$result = floatval( $value );
if ( 'no_default_unit' === $default_unit ) {
return $result;
}
if ( $default_unit ) {
return $result . $default_unit;
}
if ( ! $default_unit ) {
$result .= 'px';
}
// Return and automatically append px (default value).
return $result;
}
endif;
if ( ! function_exists( 'et_builder_get_taxonomies' ) ) :
/**
* Get taxonomies for modules
*
* @param array $args Optional. Uses 'use_terms' argument to retrieve the terms in a given taxonomy.
* Uses 'term_name' to specify Taxonomy to retrieve terms.
*
* @return array Array of WP taxonomies splitted into the taxonomy types
*/
function et_builder_get_shop_categories( $args = array() ) {
$defaults = apply_filters(
'et_builder_include_categories_shop_defaults',
array(
'use_terms' => true,
'term_name' => 'product_cat',
)
);
$term_args = apply_filters( 'et_builder_include_categories_shop_args', array( 'hide_empty' => false ) );
$args = wp_parse_args( $args, $defaults );
$product_categories = $args['use_terms'] ? get_terms( $args['term_name'], $term_args ) : get_categories( apply_filters( 'et_builder_get_categories_shop_args', 'hide_empty=0' ) );
return $product_categories;
}
endif;
if ( ! function_exists( 'et_pb_get_spacing' ) ) :
/**
* Return spacing value.
*
* @param string $spacing spacing string.
* @param string $corner spacing corner.
* @param string $default default value.
*/
function et_pb_get_spacing( $spacing, $corner, $default = '0px' ) {
$corners = array( 'top', 'right', 'bottom', 'left' );
$corner_index = array_search( $corner, $corners, true );
$spacing_array = explode( '|', $spacing );
return isset( $spacing_array[ $corner_index ] ) && '' !== $spacing_array[ $corner_index ] ? $spacing_array[ $corner_index ] : $default;
}
endif;
if ( ! function_exists( 'et_fb_enqueue_bundle' ) ) :
/**
* Enqueue a bundle
*
* @param string $id Name of the stylesheet.
* @param string $resource Resource file name.
* @param array $deps Resource dependencies.
* @param mixed $ver Resource version number.
*/
function et_fb_enqueue_bundle( $id, $resource, $deps, $ver = false ) {
$debug = defined( 'ET_DEBUG' ) && ET_DEBUG;
$ver = false === $ver ? ET_BUILDER_VERSION : $ver;
$build = 'frontend-builder/build';
$bundle = sprintf( '%s/%s/%s', ET_BUILDER_URI, $build, $resource );
$type = pathinfo( $resource, PATHINFO_EXTENSION );
switch ( $type ) {
case 'css':
if ( file_exists( sprintf( '%s%s/%s', ET_BUILDER_DIR, $build, $resource ) ) || ! $debug ) {
wp_enqueue_style( $id, $bundle, $deps, $ver );
} elseif ( $debug ) {
// Style is already embedded in the bundle but we still need to enqueue its deps.
foreach ( $deps as $dep ) {
wp_enqueue_style( $dep );
}
}
break;
case 'js':
if ( file_exists( sprintf( '%s%s/%s', ET_BUILDER_DIR, $build, $resource ) ) || ! $debug ) {
// If the file exists on disk, enqueue it.
wp_enqueue_script( $id, $bundle, $deps, $ver, true );
} else {
// Otherwise load `hot` from webpack-dev-server.
$site_url = wp_parse_url( get_site_url() );
$hot_bundle_url = "{$site_url['scheme']}://{$site_url['host']}:31495/$resource";
wp_enqueue_script( $id, $hot_bundle_url, $deps, $ver, true );
}
wp_add_inline_script( $id, 'window.et_gb = (window.top && window.top.Cypress && window.parent === window.top && window) || (window.top && window.top.Cypress && window.parent !== window.top && window.parent) || window.top || window;', 'before' );
break;
}
}
endif;
if ( ! function_exists( 'et_builder_get_active_plugins' ) ) :
/**
* Get list of all active plugins (single, network active, and mu)
*
* @return array active plugins
*/
function et_builder_get_active_plugins() {
$active_plugins = get_option( 'active_plugins' );
// Returned format must be array.
if ( ! is_array( $active_plugins ) ) {
$active_plugins = array();
}
// Get mu-plugins (must-use)
// mu-plugins data is returned in array( "plugin/name.php" => array( 'data' => 'value' ) ) format.
$mu_plugins = get_mu_plugins();
if ( is_array( $mu_plugins ) ) {
$active_plugins = array_merge( $active_plugins, array_keys( $mu_plugins ) );
}
// Get network active plugins
// Network active plugin data is returned in array( "plugin/name.php" => active_timestamp_int format.
if ( is_multisite() ) {
$network_active_plugins = get_site_option( 'active_sitewide_plugins' );
if ( is_array( $network_active_plugins ) ) {
$active_plugins = array_merge( $active_plugins, array_keys( $network_active_plugins ) );
}
}
return apply_filters( 'et_builder_get_active_plugins', $active_plugins );
}
endif;
if ( ! function_exists( 'et_has_hover_enabled' ) ) :
/**
* Determine whether hover is enable or not.
*
* @param array $props element's props.
*/
function et_has_hover_enabled( $props ) {
$et_has_hover_enabled = false;
$prop_names = array_keys( $props );
$suffix = et_pb_hover_options()->get_enabled_suffix();
foreach ( $prop_names as $prop_name ) {
if ( preg_match( "~{$suffix}$~", $prop_name ) && 'on' === $props[ $prop_name ] ) {
$et_has_hover_enabled = true;
break;
}
}
return $et_has_hover_enabled;
}
endif;
if ( ! function_exists( 'et_builder_is_hover_enabled' ) ) :
/**
* Check if the setting has enabled hover options
*
* @param string $setting setting name.
* @param array $props element's props.
*/
function et_builder_is_hover_enabled( $setting, $props ) {
return et_pb_hover_options()->is_enabled( $setting, $props );
}
endif;
if ( ! function_exists( 'et_builder_add_prefix' ) ) {
/**
* Prefixes a string key with a prefix string using the provided delimiter
* In case the prefix is empty, original key is returned
*
* @param string $prefix prefix string.
* @param string $key string key.
* @param string $delimiter delimiter.
*
* @return string
*/
function et_builder_add_prefix( $prefix, $key, $delimiter = '_' ) {
return '' === $prefix ? $key : $prefix . $delimiter . $key;
}
}
if ( ! function_exists( 'et_builder_has_value' ) ) {
/**
* Check if value is not an empty value
* Empty values are considered:
* - null
* - ''
* - false
*
* @param string $value value to check.
*
* @return bool
*/
function et_builder_has_value( $value ) {
return null !== $value && '' !== $value && false !== $value;
}
}
if ( ! function_exists( 'et_builder_get_or' ) ) {
/**
* Returns the value in case it is not empty
* Otherwise, return the default value
*
* @param string $value the builder value.
* @param string $default default value.
*
* @return string
*/
function et_builder_get_or( $value, $default = '' ) {
return et_builder_has_value( $value ) ? $value : $default;
}
}
if ( ! function_exists( 'et_builder_module_prop' ) ) {
/**
* Returns props value by provided key, if the value is empty, returns the default value
*
* @param string $prop provided key.
* @param array $props all props.
* @param mixed $default default value.
*
* @return mixed|null
*/
function et_builder_module_prop( $prop, $props, $default ) {
return et_builder_get_or( et_()->array_get( $props, $prop ), $default );
}
}
if ( ! function_exists( 'et_pb_get_column_svg' ) ) {
/**
* Returns svg which represents the requried columns type
*
* @param string $type Column layout type.
*
* @return string svg code.
*/
function et_pb_get_column_svg( $type ) {
$svg = '';
switch ( $type ) {
case '4_4':
$svg = ' ';
break;
case '1_2,1_2':
$svg = '
';
break;
case '1_3,1_3,1_3':
$svg = '
';
break;
case '1_4,1_4,1_4,1_4':
$svg = '
';
break;
case '1_5,1_5,1_5,1_5,1_5':
$svg = '
';
break;
case '1_6,1_6,1_6,1_6,1_6,1_6':
$svg = '
';
break;
case '2_5,3_5':
$svg = '
';
break;
case '3_5,2_5':
$svg = '
';
break;
case '1_3,2_3':
$svg = '
';
break;
case '2_3,1_3':
$svg = '
';
break;
case '1_4,3_4':
$svg = '
';
break;
case '3_4,1_4':
$svg = '
';
break;
case '1_4,1_2,1_4':
$svg = '
';
break;
case '1_5,3_5,1_5':
$svg = '
';
break;
case '1_4,1_4,1_2':
$svg = '
';
break;
case '1_2,1_4,1_4':
$svg = '
';
break;
case '1_5,1_5,3_5':
$svg = '
';
break;
case '3_5,1_5,1_5':
$svg = '
';
break;
case '1_6,1_6,1_6,1_2':
$svg = '
';
break;
case '1_2,1_6,1_6,1_6':
$svg = '
';
break;
}
return $svg;
}
}
/**
* Get image metadata responsive sizes
*
* @since 3.27.3
*
* @param string $image_src The 'src' of the image.
* @param array $image_meta The image meta data as returned by 'wp_get_attachment_metadata()'.
* @param array $size Array of width and height values in pixels (in that order).
*
* @return array|bool
*/
function et_builder_responsive_image_metadata( $image_src, $image_meta = null, $size = null ) {
$cache = ET_Core_Cache_File::get( 'image_responsive_metadata' );
// Normalize image URL.
$normalized_url = et_attachment_normalize_url( $image_src );
if ( isset( $cache[ $normalized_url ] ) ) {
if ( et_core_is_uploads_dir_url( $normalized_url ) ) {
return $cache[ $normalized_url ];
}
unset( $cache[ $normalized_url ] );
ET_Core_Cache_File::set( 'image_responsive_metadata', $cache );
}
$responsive_sizes = array();
$image_id = is_numeric( $image_src ) ? intval( $image_src ) : et_get_attachment_id_by_url( $image_src );
if ( ! $image_id ) {
return array();
}
if ( is_null( $image_meta ) ) {
$image_meta = wp_get_attachment_metadata( $image_id );
}
if ( ! $image_meta || empty( $image_meta['sizes'] ) ) {
return array();
}
if ( is_null( $size ) ) {
$size = et_get_attachment_size_by_url( $image_src );
}
if ( 'full' === $size && isset( $image_meta['width'] ) && isset( $image_meta['height'] ) ) {
$size = array(
absint( $image_meta['width'] ),
absint( $image_meta['height'] ),
);
} elseif ( is_string( $size ) && ! empty( $image_meta['sizes'][ $size ] ) ) {
$size = array(
absint( $image_meta['sizes'][ $size ]['width'] ),
absint( $image_meta['sizes'][ $size ]['height'] ),
);
}
if ( ! $size || ! is_array( $size ) ) {
return array();
}
foreach ( $image_meta['sizes'] as $size_key => $size_data ) {
if ( strpos( $size_key, 'et-pb-image--responsive--' ) !== 0 ) {
continue;
}
if ( is_array( $size ) && $size[0] < $size_data['width'] ) {
$responsive_sizes[ $size_data['width'] ] = false;
} else {
$responsive_sizes[ $size_data['width'] ] = $size_data;
}
}
if ( $responsive_sizes ) {
ksort( $responsive_sizes );
// Cache the responsive sizes data.
if ( et_core_is_uploads_dir_url( $normalized_url ) ) {
$cache[ $normalized_url ] = $responsive_sizes;
ET_Core_Cache_File::set( 'image_responsive_metadata', $cache );
}
}
return $responsive_sizes;
}
if ( ! function_exists( 'et_filter_wp_calculate_image_srcset' ) ) :
/**
* Filters an image's 'srcset' sources.
*
* @since 3.27
*
* @param array $sources {
* One or more arrays of source data to include in the 'srcset'.
*
* @type array $width {
* @type string $url The URL of an image source.
* @type string $descriptor The descriptor type used in the image candidate string,
* either 'w' or 'x'.
* @type int $value The source width if paired with a 'w' descriptor, or a
* pixel density value if paired with an 'x' descriptor.
* }
* }
* @param array $size_array Array of width and height values in pixels (in that order).
* @param string $image_src The 'src' of the image.
* @param array $image_meta The image meta data as returned by 'wp_get_attachment_metadata()'.
*
* @return array
*/
function et_filter_wp_calculate_image_srcset( $sources, $size_array, $image_src, $image_meta ) {
// Do not filter when in wp-admin area.
if ( is_admin() ) {
return $sources;
}
$responsive_sources = array();
if ( ! et_is_responsive_images_enabled() ) {
return $responsive_sources;
}
if ( is_string( $size_array ) ) {
$size_array = et_get_attachment_size_by_url( $image_src );
}
if ( is_string( $size_array ) && $image_meta ) {
if ( 'full' === $size_array ) {
$size_array = array(
absint( $image_meta['width'] ),
absint( $image_meta['height'] ),
);
} elseif ( ! empty( $image_meta['sizes'][ $size_array ] ) ) {
$size_array = array(
absint( $image_meta['sizes'][ $size_array ]['width'] ),
absint( $image_meta['sizes'][ $size_array ]['height'] ),
);
}
}
if ( ! is_array( $size_array ) ) {
return $responsive_sources;
}
$responsive_metadata = et_builder_responsive_image_metadata( $image_src, $image_meta, $size_array );
if ( $responsive_metadata ) {
foreach ( $responsive_metadata as $max_width => $size_data ) {
if ( ! $size_data ) {
continue;
}
// In some SVG images, the value of `$max_width` is 0, in those cases we can set `$max_width` from `$size_array`.
if ( ! $max_width ) {
$max_width = $size_array[0];
}
$responsive_sources[ $max_width ] = array(
'url' => str_replace( basename( $image_src ), $size_data['file'], $image_src ),
'descriptor' => 'w',
'value' => $max_width,
);
}
if ( $responsive_sources && $size_array[0] > $max_width ) {
$responsive_sources[ $size_array[0] ] = array(
'url' => $image_src,
'descriptor' => 'w',
'value' => $size_array[0],
);
}
if ( $responsive_sources ) {
krsort( $responsive_sources );
}
} else {
$responsive_sources = $sources;
}
return $responsive_sources;
}
endif;
add_filter( 'wp_calculate_image_srcset', 'et_filter_wp_calculate_image_srcset', 10, 4 );
if ( ! function_exists( 'et_filter_wp_calculate_image_sizes' ) ) :
/**
* Filters the output of 'wp_calculate_image_sizes()'.
*
* @since 3.27.3
*
* @param string $sizes A source size value for use in a 'sizes' attribute.
* @param array|string $size Requested size. Image size or array of width and height values
* in pixels (in that order).
* @param string|null $image_src The URL to the image file or null.
* @param array|null $image_meta The image meta data as returned by wp_get_attachment_metadata() or null.
*
* @return string|bool A valid source size value for use in a 'sizes' attribute or false.
*/
function et_filter_wp_calculate_image_sizes( $sizes, $size, $image_src, $image_meta ) {
// Do not filter when in wp-admin area.
if ( is_admin() ) {
return $sizes;
}
$responsive_sizes = '';
if ( ! et_is_responsive_images_enabled() ) {
return $responsive_sizes;
}
if ( is_string( $size ) ) {
$size = et_get_attachment_size_by_url( $image_src );
}
if ( is_string( $size ) && $image_meta ) {
if ( 'full' === $size ) {
$size = array(
absint( $image_meta['width'] ),
absint( $image_meta['height'] ),
);
} elseif ( ! empty( $image_meta['sizes'][ $size ] ) ) {
$size = array(
absint( $image_meta['sizes'][ $size ]['width'] ),
absint( $image_meta['sizes'][ $size ]['height'] ),
);
}
}
if ( ! is_array( $size ) ) {
return $responsive_sizes;
}
$responsive_metadata = et_builder_responsive_image_metadata( $image_src, $image_meta, $size );
if ( $responsive_metadata ) {
$max_width = 0;
$prev_width = 0;
$sizes_temp = array();
foreach ( $responsive_metadata as $max_width => $size_data ) {
if ( ! $size_data ) {
continue;
}
if ( $prev_width ) {
$sizes_temp[ $max_width ] = sprintf( '(min-width: %2$dpx) and (max-width: %1$dpx) %1$dpx', $max_width, ( $prev_width + 1 ) );
} else {
$sizes_temp[ $max_width ] = sprintf( '(min-width: %2$dpx) and (max-width: %1$dpx) %1$dpx', $max_width, $prev_width );
}
$prev_width = $max_width;
}
if ( $sizes_temp && $size[0] > $prev_width ) {
$sizes_temp[ $size[0] ] = sprintf( '(min-width: %2$dpx) %1$dpx', $size[0], ( $prev_width + 1 ) );
}
if ( $sizes_temp ) {
$sizes_temp[] = '100vw';
}
$responsive_sizes = implode( ', ', $sizes_temp );
} else {
$responsive_sizes = $sizes;
}
return $responsive_sizes;
}
endif;
add_filter( 'wp_calculate_image_sizes', 'et_filter_wp_calculate_image_sizes', 10, 4 );
/**
* Register and localize assets early enough to avoid conflicts
* with third party plugins that use the same assets.
*
* @since 4.0.9
*/
function et_builder_register_assets() {
global $wp_version;
$root = ET_BUILDER_URI;
$wp_major_version = substr( $wp_version, 0, 3 );
// phpcs:disable WordPress.WP.EnqueuedResourceParameters -- These scripts are inside WordPress core. In order to always load latest script, version number is set false.
wp_register_script( 'iris', admin_url( 'js/iris.min.js' ), array( 'jquery-ui-draggable', 'jquery-ui-slider', 'jquery-touch-punch' ), false, 1 );
wp_register_script( 'wp-color-picker', admin_url( 'js/color-picker.min.js' ), array( 'iris' ), false, 1 );
// phpcs:enable
$wp_color_picker_l10n = array(
'clear' => esc_html__( 'Clear', 'et_builder' ),
'defaultString' => et_builder_i18n( 'Default' ),
'pick' => esc_html__( 'Select Color', 'et_builder' ),
);
if ( version_compare( $wp_major_version, '4.9', '>=' ) ) {
wp_register_script( 'wp-color-picker-alpha', "{$root}/scripts/ext/wp-color-picker-alpha.min.js", array( 'jquery', 'wp-color-picker' ), ET_BUILDER_VERSION, true );
} else {
wp_register_script( 'wp-color-picker-alpha', "{$root}/scripts/ext/wp-color-picker-alpha-48.min.js", array( 'jquery', 'wp-color-picker' ), ET_BUILDER_VERSION, true );
$wp_color_picker_l10n['current'] = esc_html__( 'Current Color', 'et_builder' );
}
wp_localize_script( 'wp-color-picker', 'wpColorPickerL10n', $wp_color_picker_l10n );
}
add_action( 'init', 'et_builder_register_assets', 11 );
if ( ! function_exists( 'et_set_parallax_bg_wrap_border_radius' ) ) :
/**
* Set border radius to parallax background wrapper.
*
* @since 4.4.8
*
* @param array $props Module settings.
* @param string $module Module slug.
* @param string $order_class Module main css element.
*
* @return void
*/
function et_set_parallax_bg_wrap_border_radius( $props, $module, $order_class ) {
$border_radius_values = et_pb_responsive_options()->get_property_values( $props, 'border_radii' );
$border_radius_hover_enabled = et_builder_module_prop( 'border_radii__hover_enabled', $props, '' );
$border_radius_hover_values = et_builder_module_prop( 'border_radii__hover', $props, '' );
foreach ( et_pb_responsive_options()->get_modes() as $device ) {
if ( 'on||||' === $border_radius_values[ $device ] ) {
$border_radius_values[ $device ] = '';
continue;
}
$border_radius_values[ $device ] = et_format_parallax_bg_wrap_radius_values( $border_radius_values[ $device ] );
}
et_pb_responsive_options()->generate_responsive_css(
$border_radius_values,
$order_class . ' .et_parallax_bg_wrap',
'border-radius',
$module,
'',
'border'
);
if ( 'on|hover' === $border_radius_hover_enabled ) {
$radius_hover_values = et_format_parallax_bg_wrap_radius_values( $border_radius_hover_values );
} else {
$radius_hover_values = $border_radius_values['desktop'];
}
if ( $radius_hover_values ) {
$el_style = array(
'selector' => $order_class . ':hover .et_parallax_bg_wrap',
'declaration' => esc_html(
sprintf(
'border-radius: %1$s;',
$radius_hover_values
)
),
);
ET_Builder_Element::set_style( $module, $el_style );
}
}
endif;
if ( ! function_exists( 'et_format_parallax_bg_wrap_radius_values' ) ) :
/**
* Get formatted border radius of parallax background wrapper
*
* @since 4.4.8
*
* @param string $border_radius_values border radius values.
*
* @return string
*/
function et_format_parallax_bg_wrap_radius_values( $border_radius_values ) {
$radius_values = array();
$radius_array = explode( '|', $border_radius_values );
$radius_count = count( $radius_array );
for ( $i = 1; $i < $radius_count; $i++ ) {
$radius_values[] = $radius_array[ $i ] ? $radius_array[ $i ] : 0;
}
return trim( implode( ' ', $radius_values ) );
}
endif;
if ( ! function_exists( 'et_builder_generate_css' ) ) {
/**
* Generate CSS.
*
* @param array $args Styles arg.
*
* @return string|void
*/
function et_builder_generate_css( $args ) {
$defaults = array(
'prefix' => '',
'suffix' => '',
);
$args = wp_parse_args( $args, $defaults );
/*
* Bail early if we have no $selector elements or properties and $value.
*/
if ( ! $args['value'] || ! $args['selector'] ) {
return;
}
return sprintf( '%s { %s: %s; }', $args['selector'], $args['style'], $args['prefix'] . $args['value'] . $args['suffix'] );
}
}
if ( ! function_exists( 'et_builder_generate_css_style' ) ) {
/**
* Generate CSS property.
*
* @param array $args Styles arg.
*
* @return string|void
*/
function et_builder_generate_css_style( $args ) {
$defaults = array(
'prefix' => '',
'suffix' => '',
);
$args = wp_parse_args( $args, $defaults );
/*
* Bail early if we have no style and $value.
*/
if ( ! $args['value'] || ! $args['style'] ) {
return;
}
return sprintf( '%s: %s;', $args['style'], $args['prefix'] . $args['value'] . $args['suffix'] );
}
}
if ( ! function_exists( 'et_builder_default_colors_ajax_update_handler' ) ) :
/**
* Default colors AJAX update handler.
*
* @since 4.9.0
*/
function et_builder_default_colors_ajax_update_handler() {
// Get nonce from $_POST.
$nonce = isset( $_POST['et_builder_default_colors_nonce'] ) ? sanitize_text_field( $_POST['et_builder_default_colors_nonce'] ) : '';
if ( empty( $nonce ) || ! wp_verify_nonce( $nonce, 'et_builder_default_colors_update' ) ) {
wp_send_json_error();
}
if ( ! current_user_can( 'edit_posts' ) ) {
wp_send_json_error();
}
// Get default_colors from $_POST.
$default_colors = isset( $_POST['default_colors'] ) ? sanitize_text_field( wp_unslash( $_POST['default_colors'] ) ) : '';
et_update_option( 'divi_color_palette', str_replace( ',', '|', $default_colors ) );
wp_send_json_success();
}
endif;
add_action( 'wp_ajax_et_builder_default_colors_update', 'et_builder_default_colors_ajax_update_handler' );
if ( ! function_exists( 'et_builder_global_colors_ajax_save_handler' ) ) :
/**
* Global colors AJAX save handler.
*
* @since 4.9.0
*/
function et_builder_global_colors_ajax_save_handler() {
// Get nonce from $_POST.
$nonce = isset( $_POST['et_builder_global_colors_save_nonce'] ) ? sanitize_text_field( $_POST['et_builder_global_colors_save_nonce'] ) : '';
if ( empty( $nonce ) || ! wp_verify_nonce( $nonce, 'et_builder_global_colors_save' ) ) {
wp_send_json_error();
}
if ( ! current_user_can( 'edit_posts' ) ) {
wp_send_json_error();
}
// Get colors from $_POST.
$post_colors = filter_input( INPUT_POST, 'global_colors', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY );
$global_colors = array();
if ( is_array( $post_colors ) ) {
foreach ( $post_colors as $data_id => $data ) {
// Drop bad data.
if ( 'undefined' === $data_id || empty( $data ) ) {
continue;
}
// Sanitize data_id (e.g: gcid-3330f0vf7 ).
$global_id = sanitize_text_field( $data_id );
foreach ( $data as $type => $value ) {
// Sanitize both type (e.g: color, active) and value (color value, yes/no).
$global_colors[ $global_id ][ sanitize_text_field( $type ) ] = sanitize_text_field( $value );
}
}
}
if ( empty( $global_colors ) ) {
wp_send_json_error();
}
// Global Color data has been sanitized above.
et_update_option( 'et_global_colors', $global_colors );
ET_Core_PageResource::remove_static_resources( 'all', 'all' );
wp_send_json_success();
}
endif;
add_action( 'wp_ajax_et_builder_global_colors_save', 'et_builder_global_colors_ajax_save_handler' );
/**
* Get all global colors.
*
* @since 4.9.0
*
* @return array
*/
function et_builder_get_all_global_colors() {
return et_get_option( 'et_global_colors' );
}
if ( ! function_exists( 'et_builder_global_colors_ajax_get_handler' ) ) :
/**
* Global colors AJAX get handler.
*
* @since 4.19.2
*/
function et_builder_global_colors_ajax_get_handler() {
// Get nonce from $_GET.
et_core_security_check( 'edit_posts', 'et_builder_global_colors_get', 'et_builder_global_colors_get_nonce', '_GET' );
wp_send_json_success( [ 'global_colors' => et_builder_get_all_global_colors() ] );
}
endif;
add_action( 'wp_ajax_et_builder_global_colors_get', 'et_builder_global_colors_ajax_get_handler' );
/**
* Get a global color info by id.
*
* @since 4.9.0
*
* @param string $color_id Id of global color.
*
* @return array
*/
function et_builder_get_global_color_info( $color_id ) {
$colors = et_builder_get_all_global_colors();
if ( empty( $colors ) || ! array_key_exists( $color_id, $colors ) ) {
return null;
}
// if replaced value exists, return color info with that replaced id.
if ( isset( $colors[ $color_id ]['replaced_with'] ) ) {
$replaced_id = $colors[ $color_id ]['replaced_with'];
return $colors[ $replaced_id ];
}
return $colors[ $color_id ];
}
/**
* Check if given value is a Global Color Id.
*
* @since 4.21.1
*
* @param string $attr_value Color value.
* @return bool
*/
function et_builder_is_global_color( $attr_value ) {
return 0 === strpos( $attr_value, 'gcid-' );
}
/**
* Get Global Color by Color Id.
*
* @since 4.21.1
*
* @param string $color_id Color ID.
* @return string
*/
function et_builder_get_global_color( $color_id ) {
$color_info = et_builder_get_global_color_info( $color_id );
return isset( $color_info['color'] ) ? $color_info['color'] : $color_id;
}
/**
* Checks if overflow CSS property should be set or not.
*
* @since 4.17.4
*
* @param bool|string $overflow_enabled If overflow is enabled (true) or disabled (false) or -x or -y.
* @param string $function_name Module slug.
* @param ET_Builder_Element $module Module object.
*
* @return bool|string
*/
function et_process_border_radii_options_overflow( $overflow_enabled, $function_name, $module ) {
if ( in_array( $function_name, [ 'et_pb_section', 'et_pb_row' ], true ) &&
ET_Builder_Element::module_contains( $module->_original_content, [ 'et_pb_menu', 'et_pb_fullwidth_menu' ] ) ) {
$overflow_enabled = false;
}
return $overflow_enabled;
}
add_filter( 'et_builder_process_advanced_borders_options_radii_overflow_enabled', 'et_process_border_radii_options_overflow', 10, 3 );
/**
* Adds `fitvidsignore` class to vimeo videos parent tags which have unusual aspect ratios.
*
* WordPress adds extra `div` tag as a parent on vimeo videos with unusual aspect ratios so
* videos would have proper aspect ratio responsively. That causes issues with jQuery `fitvids()`.
* Ref: https://github.com/elegantthemes/Divi/issues/16116
*
* @since 4.17.5
*/
add_filter(
'oembed_dataparse',
function( $html, $data ) {
if ( ! class_exists( 'DOMDocument' ) ) {
return $html;
}
if ( 'Vimeo' !== $data->provider_name ) {
return $html;
}
$doc = new DOMDocument();
$doc_load_html_state = $doc->loadHTML( $html, LIBXML_HTML_NOIMPLIED | LIBXML_HTML_NODEFDTD );
if ( false === $doc_load_html_state ) {
return $html;
}
$extra_div_nodes = $doc->getElementsByTagName( 'div' );
if ( 0 === $extra_div_nodes->length ) {
return $html;
}
$extra_div_node = $extra_div_nodes[0];
$extra_div_node->setAttribute( 'class', 'fitvidsignore' );
$output = $doc->saveHTML();
$output = false === $output ? $html : $output;
return $output;
},
10,
2
);
if ( ! function_exists( 'et_pb_get_youtube_url_regex' ) ) :
/**
* Regex to match a YouTube URL from any known/common YouTube URL format.
*
* Expected YouTube URL Formats.
* - https://www.youtube.com/watch?v=XXXX.
* - https://www.youtube.com/embed/XXXX.
* - https://youtu.be/XXXX.
*
* To check regex, see: https://regex101.com/r/4FbeMZ/1.
*
* @since 4.18.1
*
* @return string YouTube video URL regex.
*/
function et_pb_get_youtube_url_regex() {
return '/^(?:https?:\/\/)?(?:m\.|www\.)?(?:youtu\.be\/|youtube\.com\/(?:embed\/|v\/|watch\?v=|watch\?.+&v=))((\w|-){11})(?:\S+)?$/i';
}
endif;
if ( ! function_exists( 'et_pb_normalize_youtube_url' ) ) :
/**
* Normalize a YouTube URL from any known/common YouTube URL format.
*
* Convert YouTube URL into normalized form: https://www.youtube.com/watch?v=XXXX.
* For https://www.youtube.com/watch?v=XXXX to check regex is https://regex101.com/r/B2qLJy/1.
* For https://www.youtube.com/embed/XXXX to check regex is https://regex101.com/r/oZ3iNP/1.
* For https://youtu.be/XXXX to check regex is https://regex101.com/r/5nqmhF/1.
*
* @param string $url youtube video url.
*
* @since 4.18.1
*
* @return string Normalized YouTube URL.
*/
function et_pb_normalize_youtube_url( $url ) {
preg_match( et_pb_get_youtube_url_regex(), esc_url( $url ), $youtube_embed_video );
return 'https://www.youtube.com/watch?v=' . $youtube_embed_video[1];
}
endif;
if ( ! function_exists( 'et_pb_validate_youtube_url' ) ) :
/**
* Validate a YouTube URL from any known/common YouTube URL format.
*
* For https://www.youtube.com/watch?v=XXXX to check regex is https://regex101.com/r/B2qLJy/1.
* For https://www.youtube.com/embed/XXXX to check regex is https://regex101.com/r/oZ3iNP/1.
* For https://youtu.be/XXXX to check regex is https://regex101.com/r/5nqmhF/1.
*
* @param string $url youtube video url.
*
* @since 4.18.1
*
* @return bool Whether provided URL is a valid YouTube URL or not.
*/
function et_pb_validate_youtube_url( $url ) {
preg_match( et_pb_get_youtube_url_regex(), $url, $youtube_embed_video );
return is_array( $youtube_embed_video ) && ! empty( $youtube_embed_video );
}
endif;