true,
'wpautop' => true,
'textarea_rows' => 16,
) ) );
return ob_get_clean();
}
/**
* Set the default text editor to tinymce when the builder is active.
*
* @since 1.0
* @param string $type The current default editor type.
* @return string
*/
static public function default_editor( $type ) {
return FLBuilderModel::is_builder_active() ? 'tinymce' : $type;
}
/**
* Add custom CSS for the builder to the text editor.
*
* @since 1.0
* @param string $mce_css
* @return string
*/
static public function add_editor_css( $mce_css ) {
if ( FLBuilderModel::is_builder_active() ) {
if ( ! empty( $mce_css ) ) {
$mce_css .= ',';
}
$mce_css .= FL_BUILDER_URL . 'css/editor.css';
}
return $mce_css;
}
/**
* Filter text editor buttons for the first row
*
* @since 1.0
* @param array $buttons The current buttons array.
* @return array
*/
static public function editor_buttons( $buttons ) {
if ( FLBuilderModel::is_builder_active() ) {
if ( ( $key = array_search( 'wp_more', $buttons ) ) !== false ) { // @codingStandardsIgnoreLine
unset( $buttons[ $key ] );
}
}
return $buttons;
}
/**
* Add additional buttons to the text editor.
*
* @since 1.0
* @param array $buttons The current buttons array.
* @return array
*/
static public function editor_buttons_2( $buttons ) {
global $wp_version;
if ( FLBuilderModel::is_builder_active() ) {
array_shift( $buttons );
array_unshift( $buttons, 'fontsizeselect' );
if ( version_compare( $wp_version, '4.6.9', '<=' ) ) {
array_unshift( $buttons, 'formatselect' );
}
if ( ( $key = array_search( 'wp_help', $buttons ) ) !== false ) { // @codingStandardsIgnoreLine
unset( $buttons[ $key ] );
}
}
return $buttons;
}
/**
* Custom font size options for the editor font size select.
*
* @since 1.6.3
* @param array $init The TinyMCE init array.
* @return array
*/
static public function editor_font_sizes( $init ) {
if ( FLBuilderModel::is_builder_active() ) {
$init['fontsize_formats'] = implode( ' ', array(
'10px',
'12px',
'14px',
'16px',
'18px',
'20px',
'22px',
'24px',
'26px',
'28px',
'30px',
'32px',
'34px',
'36px',
'38px',
'40px',
'42px',
'44px',
'46px',
'48px',
));
}
return $init;
}
/**
* Only allows certain text editor plugins to avoid conflicts
* with third party plugins.
*
* @since 1.0
* @param array $plugins The current editor plugins.
* @return array
*/
static public function editor_external_plugins( $plugins ) {
if ( FLBuilderModel::is_builder_active() ) {
$allowed = array(
'anchor',
'code',
'insertdatetime',
'nonbreaking',
'print',
'searchreplace',
'table',
'visualblocks',
'visualchars',
'emoticons',
'advlist',
'wptadv',
);
foreach ( $plugins as $key => $val ) {
if ( ! in_array( $key, $allowed ) ) {
unset( $plugins[ $key ] );
}
}
}
return $plugins;
}
/**
* Register the styles and scripts for builder layouts.
*
* @since 1.7.4
* @return void
*/
static public function register_layout_styles_scripts() {
$ver = FL_BUILDER_VERSION;
$css_url = plugins_url( '/css/', FL_BUILDER_FILE );
$js_url = plugins_url( '/js/', FL_BUILDER_FILE );
$min = ( self::is_debug() ) ? '' : '.min';
// Register additional CSS
wp_register_style( 'fl-slideshow', $css_url . 'fl-slideshow.css', array( 'yui3' ), $ver );
wp_register_style( 'jquery-bxslider', $css_url . 'jquery.bxslider.css', array(), $ver );
wp_register_style( 'jquery-magnificpopup', $css_url . 'jquery.magnificpopup.css', array(), $ver );
wp_register_style( 'yui3', $css_url . 'yui3.css', array(), $ver );
// Register icon CDN CSS
wp_register_style( 'font-awesome', self::$fa4_url, array(), $ver );
wp_register_style( 'font-awesome-5', self::get_fa5_url(), array(), $ver );
wp_register_style( 'foundation-icons', 'https://cdnjs.cloudflare.com/ajax/libs/foundicons/3.0.0/foundation-icons.css', array(), $ver );
// Register additional JS
wp_register_script( 'fl-slideshow', $js_url . 'fl-slideshow' . $min . '.js', array( 'yui3' ), $ver, true );
wp_register_script( 'fl-gallery-grid', $js_url . 'fl-gallery-grid.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-bxslider', $js_url . 'jquery.bxslider.js', array( 'jquery-easing', 'jquery-fitvids' ), $ver, true );
wp_register_script( 'jquery-easing', $js_url . 'jquery.easing.min.js', array( 'jquery' ), '1.4', true );
wp_register_script( 'jquery-fitvids', $js_url . 'jquery.fitvids.min.js', array( 'jquery' ), '1.2', true );
wp_register_script( 'jquery-imagesloaded', $js_url . 'jquery.imagesloaded.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-infinitescroll', $js_url . 'jquery.infinitescroll.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-magnificpopup', $js_url . 'jquery.magnificpopup.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-mosaicflow', $js_url . 'jquery.mosaicflow.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-waypoints', $js_url . 'jquery.waypoints.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'jquery-wookmark', $js_url . 'jquery.wookmark.min.js', array( 'jquery' ), $ver, true );
wp_register_script( 'yui3', $js_url . 'yui3.min.js', array(), $ver, true );
wp_register_script( 'youtube-player', 'https://www.youtube.com/iframe_api', array(), $ver, true );
wp_register_script( 'vimeo-player', 'https://player.vimeo.com/api/player.js', array(), $ver, true );
}
/**
* Enqueue the styles and scripts for all builder layouts
* in the main WordPress query.
*
* @since 1.7.4
* @return void
*/
static public function enqueue_all_layouts_styles_scripts() {
global $wp_query;
global $post;
$original_post = $post;
$is_archive = is_archive() || is_home() || is_search();
// Enqueue assets for posts in the main query.
if ( ! $is_archive && isset( $wp_query->posts ) ) {
foreach ( $wp_query->posts as $post ) {
self::enqueue_layout_styles_scripts();
}
}
// Enqueue assets for posts via the fl_builder_global_posts filter.
$post_ids = FLBuilderModel::get_global_posts();
if ( count( $post_ids ) > 0 ) {
$posts = get_posts(array(
'post__in' => $post_ids,
'post_type' => get_post_types(),
'posts_per_page' => -1,
));
foreach ( $posts as $post ) {
self::enqueue_layout_styles_scripts();
}
}
// Reset the global post variable.
$post = $original_post;
}
/**
* Enqueue the styles and scripts for a single layout.
*
* @since 1.0
* @param bool $rerender Whether to rerender the CSS and JS.
* @return void
*/
static public function enqueue_layout_styles_scripts( $rerender = false ) {
if ( FLBuilderModel::is_builder_enabled() ) {
$nodes = FLBuilderModel::get_categorized_nodes();
// Enqueue required row CSS and JS
foreach ( $nodes['rows'] as $row ) {
if ( 'slideshow' == $row->settings->bg_type ) {
wp_enqueue_script( 'yui3' );
wp_enqueue_script( 'fl-slideshow' );
wp_enqueue_script( 'jquery-imagesloaded' );
wp_enqueue_style( 'fl-slideshow' );
} elseif ( 'video' == $row->settings->bg_type ) {
wp_enqueue_script( 'jquery-imagesloaded' );
if ( 'video_service' == $row->settings->bg_video_source ) {
$video_data = FLBuilderUtils::get_video_data( $row->settings->bg_video_service_url );
if ( 'youtube' == $video_data['type'] ) {
wp_enqueue_script( 'youtube-player' );
} elseif ( 'vimeo' == $video_data['type'] ) {
wp_enqueue_script( 'vimeo-player' );
}
}
}
}
// Enqueue required module CSS and JS
foreach ( $nodes['modules'] as $module ) {
$module->enqueue_icon_styles();
$module->enqueue_font_styles();
$module->enqueue_scripts();
foreach ( $module->css as $handle => $props ) {
wp_enqueue_style( $handle, $props[0], $props[1], $props[2], $props[3] );
}
foreach ( $module->js as $handle => $props ) {
wp_enqueue_script( $handle, $props[0], $props[1], $props[2], $props[3] );
}
if ( ! empty( $module->settings->animation ) ) {
wp_enqueue_script( 'jquery-waypoints' );
}
}
// Enqueue Google Fonts
FLBuilderFonts::enqueue_google_fonts();
// Enqueue layout CSS
self::enqueue_layout_cached_asset( 'css', $rerender );
// Enqueue layout JS
self::enqueue_layout_cached_asset( 'js', $rerender );
}
}
/**
* Enqueue the styles and scripts for a single layout
* using the provided post ID.
*
* @since 1.10
* @param int $post_id
* @return void
*/
static public function enqueue_layout_styles_scripts_by_id( $post_id ) {
FLBuilderModel::set_post_id( $post_id );
FLBuilder::enqueue_layout_styles_scripts();
FLBuilderModel::reset_post_id();
}
/**
* Enqueues the cached CSS or JS asset for a layout.
*
* @since 1.8.2
* @access private
* @param string $type The type of asset. Either CSS or JS.
* @param bool $rerender Whether to rerender the CSS or JS.
* @return string
*/
static private function enqueue_layout_cached_asset( $type = 'css', $rerender = false ) {
$post_id = FLBuilderModel::get_post_id();
$asset_info = FLBuilderModel::get_asset_info();
$asset_ver = FLBuilderModel::get_asset_version();
$active = FLBuilderModel::is_builder_active();
$preview = FLBuilderModel::is_builder_draft_preview();
$handle = 'fl-builder-layout-' . $post_id;
/**
* Use this filter to add dependencies to the dependency array when the main builder layout CSS file is enqueued using wp_enqueue_style.
* @see fl_builder_layout_style_dependencies
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$css_deps = apply_filters( 'fl_builder_layout_style_dependencies', array() );
$css_media = apply_filters( 'fl_builder_layout_style_media', 'all' );
// Enqueue with the global code included?
if ( in_array( 'global-' . $type, self::$enqueued_global_assets ) ) {
$path = $asset_info[ $type . '_partial' ];
$url = $asset_info[ $type . '_partial_url' ];
$global = false;
} else {
$path = $asset_info[ $type ];
$url = $asset_info[ $type . '_url' ];
$global = true;
self::$enqueued_global_assets[] = 'global-' . $type;
}
// Render the asset inline instead of enqueuing the file?
if ( ! $active && apply_filters( 'fl_builder_render_assets_inline', false ) ) {
// Bail if we've already rendered this.
if ( in_array( $path, self::$rendered_assets ) ) {
return;
}
// Enqueue inline.
if ( 'css' === $type ) {
wp_register_style( $handle, false, $css_deps, $asset_ver, $css_media );
wp_enqueue_style( $handle );
wp_add_inline_style( $handle, self::render_css( $global, false ) );
} else {
self::$inline_js .= self::render_js( $global, false );
if ( ! has_action( 'wp_footer', __CLASS__ . '::render_inline_js' ) ) {
add_action( 'wp_footer', __CLASS__ . '::render_inline_js', PHP_INT_MAX );
}
}
} else {
// Render if the file doesn't exist.
if ( ! in_array( $path, self::$rendered_assets ) && ( ! fl_builder_filesystem()->file_exists( $path ) || $rerender || $preview || self::is_debug() ) ) {
call_user_func_array( array( 'FLBuilder', 'render_' . $type ), array( $global ) );
self::$rendered_assets[] = $path;
}
// Don't enqueue if we don't have a file after trying to render.
if ( ! fl_builder_filesystem()->file_exists( $path ) || 0 === fl_builder_filesystem()->filesize( $path ) ) {
return;
}
// Enqueue.
if ( 'css' == $type ) {
wp_enqueue_style( $handle, $url, $css_deps, $asset_ver, $css_media );
} elseif ( 'js' == $type ) {
wp_enqueue_script( $handle, $url, array( 'jquery' ), $asset_ver, true );
}
}
}
/**
*
*
* @since 2.1
* @return void
*/
static public function render_inline_js() {
echo '';
}
/**
* Clears the enqueued global assets cache to ensure new asset
* renders include global node assets.
*
* @since 1.10.2
* @return void
*/
static public function clear_enqueued_global_assets() {
self::$enqueued_global_assets = array();
}
/**
* Register and enqueue the styles and scripts for the builder UI.
*
* @since 1.7.4
* @return void
*/
static public function enqueue_ui_styles_scripts() {
if ( FLBuilderModel::is_builder_active() ) {
global $wp_the_query;
// Remove wp admin bar top margin
remove_action( 'wp_head', '_admin_bar_bump_cb' );
$ver = FL_BUILDER_VERSION;
$css_url = plugins_url( '/css/', FL_BUILDER_FILE );
$js_url = plugins_url( '/js/', FL_BUILDER_FILE );
/* Frontend builder styles */
wp_enqueue_style( 'dashicons' );
/**
* FA4 css and FA5 css do not mix well and actually break some of the icvons in the selector.
*/
if ( in_array( 'font-awesome', FLBuilderModel::get_enabled_icons() ) ) {
wp_enqueue_style( 'font-awesome' );
}
wp_enqueue_style( 'font-awesome-5' );
wp_enqueue_style( 'foundation-icons' );
wp_enqueue_style( 'jquery-nanoscroller', $css_url . 'jquery.nanoscroller.css', array(), $ver );
wp_enqueue_style( 'jquery-autosuggest', $css_url . 'jquery.autoSuggest.min.css', array(), $ver );
wp_enqueue_style( 'jquery-tiptip', $css_url . 'jquery.tiptip.css', array(), $ver );
wp_enqueue_style( 'bootstrap-tour', $css_url . 'bootstrap-tour-standalone.min.css', array(), $ver );
// Enqueue individual builder styles if WP_DEBUG is on.
if ( self::is_debug() ) {
wp_enqueue_style( 'fl-color-picker', $css_url . 'fl-color-picker.css', array(), $ver );
wp_enqueue_style( 'fl-lightbox', $css_url . 'fl-lightbox.css', array(), $ver );
wp_enqueue_style( 'fl-icon-selector', $css_url . 'fl-icon-selector.css', array(), $ver );
wp_enqueue_style( 'fl-builder', $css_url . 'fl-builder.css', array(), $ver );
// skins need to come after default ui styles
wp_enqueue_style( 'fl-builder-ui-skin-dark', $css_url . 'fl-builder-ui-skin-dark.css', array(), $ver );
wp_enqueue_style( 'fl-builder-bundle', $css_url . 'build/builder.bundle.css', array(), $ver );
} else {
wp_enqueue_style( 'fl-builder-min', $css_url . 'fl-builder.min.css', array(), $ver );
wp_enqueue_style( 'fl-builder-bundle', $css_url . 'build/builder.bundle.min.css', array(), $ver );
}
/* Custom Icons */
FLBuilderIcons::enqueue_all_custom_icons_styles();
/* RTL Support */
if ( is_rtl() ) {
wp_enqueue_style( 'fl-builder-rtl', $css_url . 'fl-builder-rtl.css', array(), $ver );
}
/* We have a custom version of sortable that fixes a bug. */
wp_deregister_script( 'jquery-ui-sortable' );
/* Frontend builder scripts */
wp_enqueue_media();
wp_enqueue_script( 'heartbeat' );
wp_enqueue_script( 'wpdialogs' );
wp_enqueue_script( 'wpdialogs-popup' );
wp_enqueue_script( 'wplink' );
wp_enqueue_script( 'editor' );
wp_enqueue_script( 'quicktags' );
wp_enqueue_script( 'json2' );
wp_enqueue_script( 'jquery-ui-droppable' );
wp_enqueue_script( 'jquery-ui-draggable' );
wp_enqueue_script( 'jquery-ui-slider' );
wp_enqueue_script( 'jquery-ui-widget' );
wp_enqueue_script( 'jquery-ui-position' );
do_action( 'fl_before_sortable_enqueue' );
wp_enqueue_script( 'jquery-ui-sortable', $js_url . 'jquery.ui.sortable.js', array( 'jquery-ui-core', 'jquery-ui-widget', 'jquery-ui-mouse' ), $ver );
wp_enqueue_script( 'jquery-nanoscroller', $js_url . 'jquery.nanoscroller.min.js', array(), $ver );
wp_enqueue_script( 'jquery-autosuggest', $js_url . 'jquery.autoSuggest.min.js', array(), $ver );
wp_enqueue_script( 'jquery-tiptip', $js_url . 'jquery.tiptip.min.js', array(), $ver );
wp_enqueue_script( 'jquery-showhideevents', $js_url . 'jquery.showhideevents.js', array(), $ver );
wp_enqueue_script( 'jquery-simulate', $js_url . 'jquery.simulate.js', array(), $ver );
wp_enqueue_script( 'jquery-validate', $js_url . 'jquery.validate.min.js', array(), $ver );
wp_enqueue_script( 'bootstrap-tour', $js_url . 'bootstrap-tour-standalone.min.js', array(), $ver );
wp_enqueue_script( 'ace', $js_url . 'ace/ace.js', array(), $ver );
wp_enqueue_script( 'ace-language-tools', $js_url . 'ace/ext-language_tools.js', array(), $ver );
wp_enqueue_script( 'mousetrap', $js_url . 'mousetrap-custom.js', array(), $ver );
// Enqueue individual builder scripts if WP_DEBUG is on.
if ( self::is_debug() ) {
wp_enqueue_script( 'fl-color-picker', $js_url . 'fl-color-picker.js', array(), $ver );
wp_enqueue_script( 'fl-lightbox', $js_url . 'fl-lightbox.js', array(), $ver );
wp_enqueue_script( 'fl-icon-selector', $js_url . 'fl-icon-selector.js', array(), $ver );
wp_enqueue_script( 'fl-stylesheet', $js_url . 'fl-stylesheet.js', array(), $ver );
wp_enqueue_script( 'fl-builder', $js_url . 'fl-builder.js', array(), $ver );
wp_enqueue_script( 'fl-builder-ajax-layout', $js_url . 'fl-builder-ajax-layout.js', array(), $ver );
wp_enqueue_script( 'fl-builder-preview', $js_url . 'fl-builder-preview.js', array(), $ver );
wp_enqueue_script( 'fl-builder-simulate-media-query', $js_url . 'fl-builder-simulate-media-query.js', array(), $ver );
wp_enqueue_script( 'fl-builder-responsive-editing', $js_url . 'fl-builder-responsive-editing.js', array(), $ver );
wp_enqueue_script( 'fl-builder-responsive-preview', $js_url . 'fl-builder-responsive-preview.js', array(), $ver );
wp_enqueue_script( 'fl-builder-services', $js_url . 'fl-builder-services.js', array(), $ver );
wp_enqueue_script( 'fl-builder-tour', $js_url . 'fl-builder-tour.js', array(), $ver );
wp_enqueue_script( 'fl-builder-ui', $js_url . 'fl-builder-ui.js', array( 'fl-builder', 'mousetrap' ), $ver );
wp_enqueue_script( 'fl-builder-ui-main-menu', $js_url . 'fl-builder-ui-main-menu.js', array( 'fl-builder-ui' ), $ver );
wp_enqueue_script( 'fl-builder-ui-panel-content', $js_url . 'fl-builder-ui-panel-content-library.js', array( 'fl-builder-ui' ), $ver );
wp_enqueue_script( 'fl-builder-ui-settings-forms', $js_url . 'fl-builder-ui-settings-forms.js', array(), $ver );
wp_enqueue_script( 'fl-builder-ui-pinned', $js_url . 'fl-builder-ui-pinned.js', array(), $ver );
wp_enqueue_script( 'fl-builder-revisions', $js_url . 'fl-builder-revisions.js', array(), $ver );
wp_enqueue_script( 'fl-builder-search', $js_url . 'fl-builder-search.js', array( 'jquery' ), $ver );
wp_enqueue_script( 'fl-builder-save-manager', $js_url . 'fl-builder-save-manager.js', array( 'jquery' ), $ver );
wp_enqueue_script( 'fl-builder-bundle', $js_url . 'build/builder.bundle.js', array(), $ver, true );
} else {
wp_enqueue_script( 'fl-builder-min', $js_url . 'fl-builder.min.js', array( 'jquery', 'mousetrap' ), $ver );
wp_enqueue_script( 'fl-builder-bundle', $js_url . 'build/builder.bundle.min.js', array(), $ver, true );
}
/* Additional module styles and scripts */
foreach ( FLBuilderModel::$modules as $module ) {
$module->enqueue_scripts();
foreach ( $module->css as $handle => $props ) {
wp_enqueue_style( $handle, $props[0], $props[1], $props[2], $props[3] );
}
foreach ( $module->js as $handle => $props ) {
wp_enqueue_script( $handle, $props[0], $props[1], $props[2], $props[3] );
}
}
}
wp_add_inline_style( 'admin-bar', '#wp-admin-bar-fl-builder-frontend-edit-link .ab-icon:before { content: "\f116" !important; top: 2px; margin-right: 3px; }' );
}
/**
* Include a jQuery fallback script when the builder is
* enabled for a page.
*
* @since 1.0
* @return void
*/
static public function include_jquery() {
if ( FLBuilderModel::is_builder_enabled() ) {
include FL_BUILDER_DIR . 'includes/jquery.php';
}
}
/**
* Adds builder classes to the body class.
*
* @since 1.0
* @param array $classes An array of existing classes.
* @return array
*/
static public function body_class( $classes ) {
$do_render = apply_filters( 'fl_builder_do_render_content', true, FLBuilderModel::get_post_id() );
$simple_ui = ! FLBuilderUserAccess::current_user_can( 'unrestricted_editing' );
$template_type = FLBuilderModel::get_user_template_type();
if ( $do_render && FLBuilderModel::is_builder_enabled() && ! is_archive() ) {
$classes[] = 'fl-builder';
}
if ( FLBuilderModel::is_builder_active() ) {
$classes[] = 'fl-builder-edit';
// Lite version
if ( true === FL_BUILDER_LITE ) {
$classes[] = 'fl-builder-lite';
}
// Simple UI
if ( $simple_ui ) {
$classes[] = 'fl-builder-simple';
}
// Simple pinned UI
if ( $simple_ui || 'module' === $template_type ) {
$classes[] = 'fl-builder-simple-pinned';
}
// Skin
$user_settings = FLBuilderUserSettings::get();
$classes[] = 'fl-builder-ui-skin--' . $user_settings['skin'];
// Draft changes
if ( FLBuilderModel::layout_has_drafted_changes() ) {
$classes[] = 'fl-builder--layout-has-drafted-changes';
}
// RTL
if ( is_rtl() ) {
$classes[] = 'fl-builder-direction-rtl';
} else {
$classes[] = 'fl-builder-direction-ltr';
}
// Has notifications
$has_new_notifications = FLBuilderNotifications::get_notifications();
if ( ! $has_new_notifications['read'] ) {
$classes[] = 'fl-builder-has-new-notifications';
}
}
return $classes;
}
/**
* Adds the page builder button to the WordPress admin bar.
*
* @since 1.0
* @param object $wp_admin_bar An instance of the WordPress admin bar.
* @return void
*/
static public function admin_bar_menu( $wp_admin_bar ) {
global $wp_the_query;
if ( FLBuilderModel::is_post_editable() && is_object( $wp_the_query->post ) ) {
$enabled = get_post_meta( $wp_the_query->post->ID, '_fl_builder_enabled', true );
$dot = ' •';
$wp_admin_bar->add_node( array(
'id' => 'fl-builder-frontend-edit-link',
'title' => '' . FLBuilderModel::get_branding() . $dot,
'href' => FLBuilderModel::get_edit_url( $wp_the_query->post->ID ),
));
}
}
static public function locate_template_file( $template_base, $slug ) {
$specific_template = $template_base . '-' . $slug . '.php';
$general_template = $template_base . '.php';
$default_dir = trailingslashit( FL_BUILDER_DIR ) . 'includes/';
// Try to find the specific template, then repeat the same process for general.
$locate_template_order = apply_filters( 'fl_builder_locate_template_order', array(
trailingslashit( self::$template_dir ) . $specific_template,
trailingslashit( self::$template_dir ) . $general_template,
), self::$template_dir, $template_base, $slug );
$template_path = locate_template( $locate_template_order );
if ( ! $template_path ) {
if ( file_exists( $default_dir . $specific_template ) ) {
$template_path = $default_dir . $specific_template;
} elseif ( file_exists( $default_dir . $general_template ) ) {
$template_path = $default_dir . $general_template;
}
}
return apply_filters( 'fl_builder_template_path', $template_path, $template_base, $slug );
}
/**
* Initializes the builder interface.
*
* @since 1.0
* @since 1.8 Method name changed from init to init_ui.
* @return void
*/
static public function init_ui() {
// Enable editing if the builder is active.
if ( FLBuilderModel::is_builder_active() && ! FLBuilderAJAX::doing_ajax() ) {
// Tell W3TC not to minify while the builder is active.
define( 'DONOTMINIFY', true );
// Tell Autoptimize not to minify while the builder is active.
add_filter( 'autoptimize_filter_noptimize', '__return_true' );
// Remove 3rd party editor buttons.
remove_all_actions( 'media_buttons', 999999 );
remove_all_actions( 'media_buttons_context', 999999 );
// Increase available memory.
if ( function_exists( 'wp_raise_memory_limit' ) ) {
wp_raise_memory_limit( 'bb-plugin' );
}
// Get the post.
require_once ABSPATH . 'wp-admin/includes/post.php';
$post_id = FLBuilderModel::get_post_id();
// Check to see if the post is locked.
if ( wp_check_post_lock( $post_id ) !== false ) {
header( 'Location: ' . admin_url( '/post.php?post=' . $post_id . '&action=edit' ) );
} else {
FLBuilderModel::enable_editing();
}
}
}
/**
* Renders the markup for the builder interface.
*
* @since 1.0
* @return void
*/
static public function render_ui() {
global $wp_the_query;
if ( FLBuilderModel::is_builder_active() ) {
$post_id = is_object( $wp_the_query->post ) ? $wp_the_query->post->ID : null;
$unrestricted = FLBuilderUserAccess::current_user_can( 'unrestricted_editing' );
$simple_ui = ! $unrestricted;
$global_settings = FLBuilderModel::get_global_settings();
include FL_BUILDER_DIR . 'includes/ui-extras.php';
include FL_BUILDER_DIR . 'includes/ui-js-templates.php';
include FL_BUILDER_DIR . 'includes/ui-js-config.php';
}
}
/**
* Get data structure for main builder menu.
*
* @since 2.0
* @return array
*/
static function get_main_menu_data() {
global $post;
$views = array();
$is_lite = true === FL_BUILDER_LITE; // @codingStandardsIgnoreLine
$is_user_template = FLBuilderModel::is_post_user_template();
$enabled_templates = FLBuilderModel::get_enabled_templates();
$is_simple_ui = ! FLBuilderUserAccess::current_user_can( 'unrestricted_editing' );
$key_shortcuts = self::get_keyboard_shortcuts();
$help = FLBuilderModel::get_help_button_settings();
$default_view = array(
'name' => __( 'Unnamed Menu', 'fl-builder' ),
'isShowing' => false,
'isRootView' => false,
'items' => array(),
);
// Tools
$tools_view = array(
'name' => __( 'Tools', 'fl-builder' ),
'isShowing' => true,
'isRootView' => true,
'items' => array(),
);
if ( ! $is_lite && ! $is_user_template && ( 'enabled' == $enabled_templates || 'user' == $enabled_templates ) ) {
$tools_view['items'][10] = array(
'label' => __( 'Save Template', 'fl-builder' ),
'type' => 'event',
'eventName' => 'saveTemplate',
'accessory' => $key_shortcuts['saveTemplate']['keyLabel'],
);
}
$tools_view['items'][20] = array(
'label' => __( 'Duplicate Layout', 'fl-builder' ),
'type' => 'event',
'eventName' => 'duplicateLayout',
);
$tools_view['items'][30] = array(
'label' => __( 'Preview Layout', 'fl-builder' ),
'type' => 'event',
'eventName' => 'previewLayout',
'accessory' => $key_shortcuts['previewLayout']['keyLabel'],
);
$tools_view['items'][40] = array(
'type' => 'separator',
);
$tools_view['items'][50] = array(
'label' => __( 'Layout CSS & Javascript', 'fl-builder' ),
'type' => 'event',
'eventName' => 'showLayoutSettings',
'accessory' => $key_shortcuts['showLayoutSettings']['keyLabel'],
);
$tools_view['items'][60] = array(
'label' => __( 'Global Settings', 'fl-builder' ),
'type' => 'event',
'eventName' => 'showGlobalSettings',
'accessory' => $key_shortcuts['showGlobalSettings']['keyLabel'],
);
$tools_view['items'][70] = array(
'type' => 'separator',
);
$tools_view['items'][80] = array(
'label' => __( 'Change UI Brightness', 'fl-builder' ),
'type' => 'event',
'eventName' => 'toggleUISkin',
'accessory' => $key_shortcuts['toggleUISkin']['keyLabel'],
);
$tools_view['items'][100] = array(
'label' => __( 'WordPress Admin', 'fl-builder' ),
'type' => 'view',
'view' => 'admin',
);
if ( $help['enabled'] && ! $is_simple_ui ) {
$tools_view['items'][110] = array(
'label' => __( 'Help', 'fl-builder' ),
'type' => 'view',
'view' => 'help',
);
}
$tools_view['items'][120] = array(
'label' => __( 'Keyboard Shortcuts', 'fl-builder' ),
'type' => 'event',
'eventName' => 'showKeyboardShortcuts',
);
$views['main'] = wp_parse_args( $tools_view, $default_view );
// Admin
$admin_view = array(
'name' => __( 'WordPress Admin', 'fl-builder' ),
'items' => array(),
);
// Edit current post/page/cpt
if ( is_single( $post->ID ) || is_page( $post->ID ) ) {
$edit_label = get_post_type_object( $post->post_type )->labels->edit_item;
$admin_view['items'][10] = array(
'label' => $edit_label,
'type' => 'link',
'url' => get_edit_post_link( $post->ID ),
);
}
$admin_view['items'][15] = array(
'type' => 'separator',
);
// Dashboard
$admin_view['items'][17] = array(
'label' => _x( 'Dashboard', 'label for the WordPress Dashboard link', 'fl-builder' ),
'type' => 'link',
'url' => admin_url( 'index.php' ),
);
$templates_enabled = FLBuilderUserAccess::current_user_can( 'builder_admin' );
if ( $templates_enabled ) {
$admin_view['items'][20] = array(
'label' => __( 'Manage Templates', 'fl-builder' ),
'type' => 'link',
'url' => admin_url( 'edit.php?post_type=fl-builder-template' ),
);
}
if ( current_user_can( 'customize' ) ) {
$post_url = get_permalink( $post->ID );
if ( $post_url ) {
$url = admin_url( 'customize.php?url=' . $post_url );
} else {
$url = admin_url( 'customize.php' );
}
$admin_view['items'][30] = array(
'label' => __( 'Customize Theme', 'fl-builder' ),
'type' => 'link',
'url' => $url,
);
}
$views['admin'] = wp_parse_args( $admin_view, $default_view );
// Help
if ( $help['enabled'] && ! $is_simple_ui ) {
$help_view = array(
'name' => __( 'Help', 'fl-builder' ),
'items' => array(),
);
if ( $help['video'] && isset( $help['video_embed'] ) ) {
// Disable Auto Play
$help['video_embed'] = str_replace( 'autoplay=1', 'autoplay=0', $help['video_embed'] );
// Remove Height from iframe
$help['video_embed'] = str_replace( 'height="315"', 'height="173"', $help['video_embed'] );
$help_view['items'][10] = array(
'type' => 'video',
'embed' => $help['video_embed'],
);
}
if ( $help['tour'] ) {
$help_view['items'][20] = array(
'label' => __( 'Take A Tour', 'fl-builder' ),
'type' => 'event',
'eventName' => 'beginTour',
);
}
if ( $help['knowledge_base'] && isset( $help['knowledge_base_url'] ) ) {
$help_view['items'][30] = array(
'label' => __( 'View Knowledge Base', 'fl-builder' ),
'type' => 'link',
'url' => $help['knowledge_base_url'],
);
}
if ( $help['forums'] && isset( $help['forums_url'] ) ) {
$help_view['items'][40] = array(
'label' => __( 'Contact Support', 'fl-builder' ),
'type' => 'link',
'url' => $help['forums_url'],
);
}
$views['help'] = wp_parse_args( $help_view, $default_view );
}
return apply_filters( 'fl_builder_main_menu', $views );
}
/**
* Get array of registered keyboard shortcuts. The key corresponds to
* an event to be triggered by FLbuilder.triggerHook()
*
* @since 2.0
* @return array
*/
static function get_keyboard_shortcuts() {
$default_action = array(
'label' => _x( 'Untitled Shortcut', 'A keyboard shortcut with no label given', 'fl-builder' ),
'keyCode' => '',
'keyLabel' => '',
'isGlobal' => false,
'enabled' => true,
);
$data = array(
'showModules' => array(
'label' => _x( 'Open Modules Tab', 'Keyboard action to show modules tab', 'fl-builder' ),
'keyCode' => 'j',
),
'showRows' => array(
'label' => _x( 'Open Rows Tab', 'Keyboard action to show rows tab', 'fl-builder' ),
'keyCode' => 'k',
),
'showTemplates' => array(
'label' => _x( 'Open Templates Tab', 'Keyboard action to show templates tab', 'fl-builder' ),
'keyCode' => 'l',
),
'showSaved' => array(
'label' => _x( 'Open Saved Tab', 'Keyboard action to show saved tab', 'fl-builder' ),
'keyCode' => ';',
'enabled' => true !== FL_BUILDER_LITE,
),
'saveTemplate' => array(
'label' => _x( 'Save New Template', 'Keyboard action to open save template form', 'fl-builder' ),
'keyCode' => 'mod+j',
'enabled' => true !== FL_BUILDER_LITE,
),
'previewLayout' => array(
'label' => _x( 'Toggle Preview Mode', 'Keyboard action to toggle preview mode', 'fl-builder' ),
'keyCode' => 'p',
),
'showGlobalSettings' => array(
'label' => _x( 'Open Global Settings', 'Keyboard action to open the global settings panel', 'fl-builder' ),
'keyCode' => 'mod+u',
),
'showLayoutSettings' => array(
'label' => _x( 'Open Layout Settings', 'Keyboard action to open the layout settings panel', 'fl-builder' ),
'keyCode' => 'mod+y',
),
'toggleUISkin' => array(
'label' => _x( 'Change UI Brightness', 'Keyboard action to switch between light and dark UI brightness', 'fl-builder' ),
'keyCode' => 'o',
),
'showSearch' => array(
'label' => _x( 'Display Module Search', 'Keyboard action to open the module search panel', 'fl-builder' ),
'keyCode' => 'mod+i',
'enabled' => true !== FL_BUILDER_LITE,
),
'showSavedMessage' => array(
'label' => _x( 'Save Layout', 'Keyboard action to save changes', 'fl-builder' ),
'keyCode' => 'mod+s',
'isGlobal' => true,
),
'publishAndRemain' => array(
'label' => _x( 'Publish changes without leaving builder', 'Keyboard action to publish any pending changes', 'fl-builder' ),
'keyCode' => 'mod+p',
'isGlobal' => true,
),
'cancelTask' => array(
'label' => _x( 'Dismiss Active Panel', 'Keyboard action to dismiss the current task or panel', 'fl-builder' ),
'keyCode' => 'esc',
'isGlobal' => true,
),
);
$data = apply_filters( 'fl_builder_keyboard_shortcuts', $data );
foreach ( $data as $hook => $args ) {
// Check for old (alpha) format and normalize
if ( is_string( $args ) ) {
$args = array(
'label' => ucwords( preg_replace( '/([^A-Z])([A-Z])/', '$1 $2', $hook ) ),
'keyCode' => $args,
);
}
$args = wp_parse_args( $args, $default_action );
// Unset this shortcut if it's not enabled.
if ( ! $args['enabled'] ) {
unset( $data[ $hook ] );
continue;
}
// Map 'mod' to mac or pc equivalent
$code = $args['keyCode'];
$code = str_replace( '+', '', $code );
if ( false !== strpos( $code, 'mod' ) ) {
$is_mac = strpos( $_SERVER['HTTP_USER_AGENT'], 'Macintosh' ) ? true : false;
if ( $is_mac ) {
$code = str_replace( 'mod', 'command', $code );
} else {
$code = str_replace( 'mod', 'Ctrl+', $code );
}
}
// Replace 'command'
$code = str_replace( 'command', '⌘', $code );
// Replace 'shift'
$code = str_replace( 'shift', '⇧', $code );
// Replace 'delete'
$code = str_replace( 'delete', '⌫', $code );
// Replace 'left' arrow
$code = str_replace( 'left', '←', $code );
// Replace 'right' arrow
$code = str_replace( 'right', '→', $code );
$args['keyLabel'] = $code;
$data[ $hook ] = $args;
}
return $data;
}
/**
* Renders the markup for the title in the builder's bar.
*
* @since 1.6.3
* @return void
*/
static public function render_ui_bar_title() {
global $post;
$simple_ui = ! FLBuilderUserAccess::current_user_can( 'unrestricted_editing' );
$title = apply_filters( 'fl_builder_ui_bar_title', get_the_title( $post->ID ) );
$icon_url = FLBuilderModel::get_branding_icon();
$wrapper_classes = array( 'fl-builder-bar-title' );
if ( '' == $icon_url ) {
$wrapper_classes[] = 'fl-builder-bar-title-no-icon';
}
$edited_object_label = get_post_type_object( $post->post_type )->labels->singular_name;
$pretitle = sprintf( _x( 'Currently Editing %s', 'Currently editing message', 'fl-builder' ), $edited_object_label );
$pretitle = apply_filters( 'fl_builder_ui_bar_pretitle', $pretitle );
// Render the bar title.
include FL_BUILDER_DIR . 'includes/ui-bar-title-area.php';
}
/**
* Renders the markup for the buttons in the builder's bar.
*
* @since 1.6.3
* @return void
*/
static public function render_ui_bar_buttons() {
$help_button = FLBuilderModel::get_help_button_settings();
$simple_ui = ! FLBuilderUserAccess::current_user_can( 'unrestricted_editing' );
$should_display_search = ! FLBuilderModel::is_post_user_template( 'module' ) && ! $simple_ui;
$add_btn_svg = '';
$notifications = FLBuilderNotifications::get_notifications();
$buttons = apply_filters( 'fl_builder_ui_bar_buttons', array(
'upgrade' => array(
'label' => __( 'Upgrade Today', 'fl-builder' ) . ' ',
'show' => true === FL_BUILDER_LITE,
),
'buy' => array(
'label' => __( 'Buy Now', 'fl-builder' ) . ' ',
'show' => stristr( home_url(), 'demo.wpbeaverbuilder.com' ),
),
'done' => array(
'label' => __( 'Done', 'fl-builder' ),
),
'content-panel' => array(
'label' => $add_btn_svg,
'show' => ! $simple_ui,
),
'add-content' => array( // Only added here for backwards compat.
'label' => $add_btn_svg,
'show' => ! $simple_ui,
),
) );
echo '
';
$i = 0;
foreach ( $buttons as $slug => $button ) {
if ( 'add-content' == $slug ) {
continue; // The old add content button is no longer supported.
}
if ( isset( $button['show'] ) && ! $button['show'] ) {
continue;
}
echo '';
$i++;
}
echo '';
if ( ! $simple_ui && ! FLBuilderModel::is_white_labeled() && $notifications['data'] && '{}' !== $notifications['data'] && ! apply_filters( 'fl_disable_notifications', false ) ) {
echo '';
echo '';
}
echo '
';
}
/**
* Renders layouts using a new instance of WP_Query with the provided
* args and enqueues the necessary styles and scripts. We set the global
* $wp_query variable so the builder thinks we are in the loop when content
* is rendered without having to call query_posts.
*
* @link https://codex.wordpress.org/Class_Reference/WP_Query See for a complete list of args.
*
* @since 1.7
* @param array|string $args An array or string of args to be passed to a new instance of WP_Query.
* @param int $site_id The ID of a site on a network to pull the query from.
* @return void
*/
static public function render_query( $args, $site_id = null ) {
global $post;
$switched = false;
// Pull from a site on the network?
if ( $site_id && is_multisite() ) {
switch_to_blog( $site_id );
$switched = true;
}
// Get the query.
$query = new WP_Query( $args );
// Loop through the posts.
foreach ( $query->posts as $query_post ) {
// Make sure this isn't the same post as the original post to prevent infinite loops.
if ( is_object( $post ) && $post->ID === $query_post->ID && ! $switched ) {
continue;
}
if ( FLBuilderModel::is_builder_enabled( $query_post->ID ) ) {
// Enqueue styles and scripts for this post.
self::enqueue_layout_styles_scripts_by_id( $query_post->ID );
// Print the styles if we are outside of the head tag.
if ( ! doing_action( 'wp_enqueue_scripts' ) ) {
wp_print_styles();
}
// Render the builder content.
FLBuilder::render_content_by_id( $query_post->ID );
} else {
// Render the WP editor content if the builder isn't enabled.
echo apply_filters( 'the_content', $query_post->post_content );
}
}
// Reset the site data?
if ( $site_id && is_multisite() ) {
restore_current_blog();
}
}
/**
* Renders the layout for a post with the given post ID.
* This is useful for rendering builder content outside
* of the loop.
*
* @since 1.10
* @param int $post_id The ID of the post with the layout to render.
* @param string $tag The HTML tag for the content wrapper.
* @param array $attrs An array of key/value attribute data for the content wrapper.
* @return void
*/
static public function render_content_by_id( $post_id, $tag = 'div', $attrs = array() ) {
// Force the builder to use this post ID.
FLBuilderModel::set_post_id( $post_id );
// Build the attributes string.
$attr_string = '';
foreach ( $attrs as $attr_key => $attr_value ) {
$attr_string .= ' ' . $attr_key . '="' . $attr_value . '"';
}
// Prevent the builder's render_content filter from running.
add_filter( 'fl_builder_do_render_content', '__return_false' );
// Fire the render content start action.
do_action( 'fl_builder_render_content_start' );
// Render the content.
ob_start();
do_action( 'fl_builder_before_render_content' );
echo '<' . $tag . ' class="' . self::render_content_classes() . '" data-post-id="' . $post_id . '"' . $attr_string . '>';
self::render_nodes();
echo '' . $tag . '>';
do_action( 'fl_builder_after_render_content' );
$content = ob_get_clean();
// Allow the builder's render_content filter to run again.
remove_filter( 'fl_builder_do_render_content', '__return_false' );
// Process shortcodes.
if ( apply_filters( 'fl_builder_render_shortcodes', true ) ) {
global $wp_embed;
$content = apply_filters( 'fl_builder_before_render_shortcodes', $content );
$pattern = get_shortcode_regex();
$content = preg_replace_callback( "/$pattern/s", 'FLBuilder::double_escape_shortcodes', $content );
$content = $wp_embed->run_shortcode( $content );
$content = do_shortcode( $content );
}
// Add srcset attrs to images with the class wp-image-.
if ( function_exists( 'wp_make_content_images_responsive' ) ) {
$content = wp_make_content_images_responsive( $content );
}
// Fire the render content complete action.
do_action( 'fl_builder_render_content_complete' );
// Stop forcing the builder to use this post ID.
FLBuilderModel::reset_post_id();
echo $content;
}
/**
* Renders the content for a builder layout while in the loop.
* This method should only be called by the_content filter as
* defined in this class. To output builder content, use
* the_content function while in a WordPress loop or use
* the FLBuilder::render_content_by_id method.
*
* @since 1.0
* @param string $content The existing content.
* @return string
*/
static public function render_content( $content ) {
$post_id = FLBuilderModel::get_post_id( true );
$enabled = FLBuilderModel::is_builder_enabled( $post_id );
$rendering = $post_id === self::$post_rendering;
$do_render = apply_filters( 'fl_builder_do_render_content', true, $post_id );
$in_loop = in_the_loop();
$is_global = in_array( $post_id, FLBuilderModel::get_global_posts() );
if ( $enabled && ! $rendering && $do_render && ( $in_loop || $is_global ) ) {
// Set the post rendering ID.
self::$post_rendering = $post_id;
// Try to enqueue here in case it didn't happen in the head for this layout.
self::enqueue_layout_styles_scripts();
// Render the content.
ob_start();
self::render_content_by_id( $post_id );
$content = ob_get_clean();
// Clear the post rendering ID.
self::$post_rendering = null;
}
return $content;
}
/**
* Escaped shortcodes need to be double escaped or they will
* be parsed by WP's shortcodes filter.
*
* @since 1.6.4.1
* @param array $matches The existing content.
* @return string
*/
static public function double_escape_shortcodes( $matches ) {
if ( '[' == $matches[1] && ']' == $matches[6] ) {
return '[' . $matches[0] . ']';
}
return $matches[0];
}
/**
* Renders the CSS classes for the main content div tag.
*
* @since 1.6.4
* @return string
*/
static public function render_content_classes() {
global $wp_the_query;
$post_id = FLBuilderModel::get_post_id();
// Build the content class.
$classes = 'fl-builder-content fl-builder-content-' . $post_id;
// Add the primary content class.
if ( isset( $wp_the_query->post ) && $wp_the_query->post->ID == $post_id ) {
$classes .= ' fl-builder-content-primary';
}
// Add browser specific classes.
if ( isset( $_SERVER['HTTP_USER_AGENT'] ) ) {
if ( stristr( $_SERVER['HTTP_USER_AGENT'], 'Trident/7.0' ) && stristr( $_SERVER['HTTP_USER_AGENT'], 'rv:11.0' ) ) {
$classes .= ' fl-builder-ie-11';
}
}
return apply_filters( 'fl_builder_content_classes', $classes );
}
/**
* Renders the markup for all nodes in a layout.
*
* @since 1.6.3
* @return void
*/
static public function render_nodes() {
do_action( 'fl_builder_before_render_nodes' );
if ( apply_filters( 'fl_builder_render_nodes', true ) ) {
self::render_rows();
}
do_action( 'fl_builder_after_render_nodes' );
}
/**
* Renders the markup for a node's attributes.
*
* @since 1.8
* @param array $attrs
* @return void
*/
static public function render_node_attributes( $attrs ) {
foreach ( $attrs as $attr_key => $attr_value ) {
if ( empty( $attr_value ) ) {
continue;
} elseif ( is_string( $attr_value ) ) {
echo ' ' . $attr_key . '="' . $attr_value . '"';
} elseif ( is_array( $attr_value ) && ! empty( $attr_value ) ) {
echo ' ' . $attr_key . '="';
for ( $i = 0; $i < count( $attr_value ); $i++ ) {
echo $attr_value[ $i ];
if ( $i < count( $attr_value ) - 1 ) {
echo ' ';
}
}
echo '"';
}
}
}
/**
* Renders the stripped down content for a layout
* that is saved to the WordPress editor.
*
* @since 1.0
* @param string $content The existing content.
* @return string
*/
static public function render_editor_content() {
$rows = FLBuilderModel::get_nodes( 'row' );
ob_start();
// Render the modules.
foreach ( $rows as $row ) {
$groups = FLBuilderModel::get_nodes( 'column-group', $row );
foreach ( $groups as $group ) {
$cols = FLBuilderModel::get_nodes( 'column', $group );
foreach ( $cols as $col ) {
$col_children = FLBuilderModel::get_nodes( null, $col );
foreach ( $col_children as $col_child ) {
if ( 'module' == $col_child->type ) {
$module = FLBuilderModel::get_module( $col_child );
if ( $module && $module->editor_export ) {
// Don't crop photos to ensure media library photos are rendered.
if ( 'photo' == $module->settings->type ) {
$module->settings->crop = false;
}
FLBuilder::render_module_html( $module->settings->type, $module->settings, $module );
}
} elseif ( 'column-group' == $col_child->type ) {
$group_cols = FLBuilderModel::get_nodes( 'column', $col_child );
foreach ( $group_cols as $group_col ) {
$modules = FLBuilderModel::get_modules( $group_col );
foreach ( $modules as $module ) {
if ( $module->editor_export ) {
// Don't crop photos to ensure media library photos are rendered.
if ( 'photo' == $module->settings->type ) {
$module->settings->crop = false;
}
FLBuilder::render_module_html( $module->settings->type, $module->settings, $module );
}
}
}
}
}
}
}
}
// Get the content.
$content = ob_get_clean();
// Remove unnecessary tags and attributes.
$content = preg_replace( '/<\/?div[^>]*\>/i', '', $content );
$content = preg_replace( '/<\/?span[^>]*\>/i', '', $content );
$content = preg_replace( '##is', '', $content );
$content = preg_replace( '/<\/?noscript[^>]*\>/i', '', $content );
$content = preg_replace( '##is', '', $content );
$content = preg_replace( '/]*><\\/i[^>]*>/', '', $content );
$content = preg_replace( '/ class=".*?"/', '', $content );
$content = preg_replace( '/ style=".*?"/', '', $content );
// Remove empty lines.
$content = preg_replace( '/(^[\r\n]*|[\r\n]+)[\s\t]*[\r\n]+/', "\n", $content );
return apply_filters( 'fl_builder_editor_content', $content );
}
/**
* Renders a settings via PHP. This method is only around for
* backwards compatibility with third party settings forms that are
* still being rendered via AJAX. Going forward, all settings forms
* should be rendered on the frontend using FLBuilderSettingsForms.render.
*
* @since 1.0
* @param array $form The form data.
* @param object $settings The settings data.
* @return array
*/
static public function render_settings( $form = array(), $settings ) {
return FLBuilderUISettingsForms::render_settings( $form, $settings );
}
/**
* Renders a settings form via PHP. This method is only around for
* backwards compatibility with third party settings forms that are
* still being rendered via AJAX. Going forward, all settings forms
* should be rendered on the frontend using FLBuilderSettingsForms.render.
*
* @since 1.0
* @param string $type The type of form to render.
* @param object $settings The settings data.
* @return array
*/
static public function render_settings_form( $type = null, $settings = null ) {
return FLBuilderUISettingsForms::render_settings_form( $type, $settings );
}
/**
* Renders a settings field via PHP. This method is only around for
* backwards compatibility with third party settings forms that are
* still being rendered via AJAX. Going forward, all settings forms
* should be rendered on the frontend using FLBuilderSettingsForms.render.
*
* @since 1.0
* @param string $name The field name.
* @param array $field An array of setup data for the field.
* @param object $settings Form settings data object.
* @return void
*/
static public function render_settings_field( $name, $field, $settings = null ) {
return FLBuilderUISettingsForms::render_settings_field( $name, $field, $settings );
}
/**
* Renders the markup for the icon selector.
*
* @since 1.0
* @return array
*/
static public function render_icon_selector() {
return FLBuilderUISettingsForms::render_icon_selector();
}
/**
* Renders the markup for all of the rows in a layout.
*
* @since 1.0
* @return void
*/
static public function render_rows() {
$rows = FLBuilderModel::get_nodes( 'row' );
do_action( 'fl_builder_before_render_rows', $rows );
foreach ( $rows as $row ) {
self::render_row( $row );
}
do_action( 'fl_builder_after_render_rows', $rows );
}
/**
* Renders the markup for a single row.
*
* @since 1.0
* @param object $row The row to render.
* @return void
*/
static public function render_row( $row ) {
global $wp_the_query;
$groups = FLBuilderModel::get_nodes( 'column-group', $row );
$post_id = FLBuilderModel::get_post_id();
$active = FLBuilderModel::is_builder_active() && $post_id == $wp_the_query->post->ID;
$visible = FLBuilderModel::is_node_visible( $row );
if ( $active || $visible ) {
do_action( 'fl_builder_before_render_row', $row, $groups );
$template_file = self::locate_template_file(
apply_filters( 'fl_builder_row_template_base', 'row', $row ),
apply_filters( 'fl_builder_row_template_slug', '', $row )
);
if ( $template_file ) {
include $template_file;
}
do_action( 'fl_builder_after_render_row', $row, $groups );
}
}
/**
* Renders the HTML attributes for a single row.
*
* @since 1.0
* @param object $row A row node object.
* @return void
*/
static public function render_row_attributes( $row ) {
/**
* Use this filter to work with the custom class a user adds to a row under Row Settings > Advanced > Class.
* @see fl_builder_row_custom_class
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$custom_class = apply_filters( 'fl_builder_row_custom_class', $row->settings->class, $row );
$overlay_bgs = array( 'photo', 'parallax', 'slideshow', 'video' );
$active = FLBuilderModel::is_builder_active();
$visible = FLBuilderModel::is_node_visible( $row );
$has_rules = FLBuilderModel::node_has_visibility_rules( $row );
$attrs = array(
'id' => $row->settings->id,
'class' => array(
'fl-row',
'fl-row-' . $row->settings->width . '-width',
'fl-row-bg-' . $row->settings->bg_type,
'fl-node-' . $row->node,
),
'data-node' => $row->node,
);
// Classes
if ( ! empty( $row->settings->full_height ) && 'full' == $row->settings->full_height ) {
$attrs['class'][] = 'fl-row-full-height';
if ( isset( $row->settings->content_alignment ) ) {
$attrs['class'][] = 'fl-row-align-' . $row->settings->content_alignment;
}
}
if ( in_array( $row->settings->bg_type, $overlay_bgs ) && ! empty( $row->settings->bg_overlay_color ) ) {
$attrs['class'][] = 'fl-row-bg-overlay';
}
if ( ! empty( $row->settings->responsive_display ) ) {
$attrs['class'][] = 'fl-visible-' . $row->settings->responsive_display;
}
if ( ! empty( $custom_class ) ) {
$attrs['class'][] = trim( esc_attr( $custom_class ) );
}
if ( $active && ! $visible ) {
$attrs['class'][] = 'fl-node-hidden';
}
if ( $active && $has_rules ) {
$attrs['class'][] = 'fl-node-has-rules';
}
// Data
if ( 'parallax' == $row->settings->bg_type && ! empty( $row->settings->bg_parallax_image_src ) ) {
$attrs['data-parallax-speed'] = $row->settings->bg_parallax_speed;
$attrs['data-parallax-image'] = $row->settings->bg_parallax_image_src;
}
self::render_node_attributes( apply_filters( 'fl_builder_row_attributes', $attrs, $row ) );
}
/**
* Renders the markup for a row's background.
*
* @since 1.0
* @param object $row A row node object.
* @return void
*/
static public function render_row_bg( $row ) {
do_action( 'fl_builder_before_render_row_bg', $row );
if ( 'video' == $row->settings->bg_type ) {
$vid_data = FLBuilderModel::get_row_bg_data( $row );
if ( $vid_data || in_array( $row->settings->bg_video_source, array( 'video_url', 'video_service' ) ) ) {
$template_file = self::locate_template_file(
apply_filters( 'fl_builder_row_video_bg_template_base', 'row-video', $row ),
apply_filters( 'fl_builder_row_video_bg_template_slug', '', $row )
);
if ( $template_file ) {
include $template_file;
}
}
} elseif ( 'slideshow' == $row->settings->bg_type ) {
echo '';
}
do_action( 'fl_builder_after_render_row_bg', $row );
}
/**
* Renders the HTML class for a row's content wrapper.
*
* @since 1.0
* @param object $row A row node object.
* @return void
*/
static public function render_row_content_class( $row ) {
echo 'fl-row-content';
echo ' fl-row-' . $row->settings->content_width . '-width';
echo ' fl-node-content';
}
/**
* Renders the markup for a column group.
*
* @since 1.0
* @param object $group A column group node object.
* @return void
*/
static public function render_column_group( $group ) {
$cols = FLBuilderModel::get_nodes( 'column', $group );
do_action( 'fl_builder_before_render_column_group', $group, $cols );
$template_file = self::locate_template_file(
apply_filters( 'fl_builder_column_group_template_base', 'column-group', $group ),
apply_filters( 'fl_builder_column_group_template_slug', '', $group )
);
if ( $template_file ) {
include $template_file;
}
do_action( 'fl_builder_after_render_column_group', $group, $cols );
}
/**
* Renders the attrs for a column group.
*
* @since 1.0
* @param object $group
* @return void
*/
static public function render_column_group_attributes( $group ) {
$cols = FLBuilderModel::get_nodes( 'column', $group );
$parent = FLBuilderModel::get_node_parent( $group );
$attrs = array(
'class' => array(
'fl-col-group',
'fl-node-' . $group->node,
),
'data-node' => $group->node,
);
if ( 'column' == $parent->type ) {
$attrs['class'][] = 'fl-col-group-nested';
}
foreach ( $cols as $col ) {
if ( isset( $col->settings->equal_height ) && 'yes' == $col->settings->equal_height ) {
if ( ! in_array( 'fl-col-group-equal-height', $attrs['class'] ) ) {
$attrs['class'][] = 'fl-col-group-equal-height';
}
if ( isset( $col->settings->content_alignment ) ) {
if ( ! in_array( 'fl-col-group-align-' . $col->settings->content_alignment, $attrs['class'] ) ) {
$attrs['class'][] = 'fl-col-group-align-' . $col->settings->content_alignment;
}
}
}
if ( isset( $col->settings->responsive_size ) && 'custom' == $col->settings->responsive_size ) {
if ( ! in_array( 'fl-col-group-custom-width', $attrs['class'] ) ) {
$attrs['class'][] = 'fl-col-group-custom-width';
}
}
if ( isset( $col->settings->responsive_order ) && 'reversed' == $col->settings->responsive_order ) {
if ( ! in_array( 'fl-col-group-responsive-reversed', $attrs['class'] ) ) {
$attrs['class'][] = 'fl-col-group-responsive-reversed';
}
}
}
self::render_node_attributes( apply_filters( 'fl_builder_column_group_attributes', $attrs, $group ) );
}
/**
* Renders the markup for a single column.
*
* @since 1.7
* @param string|object $col_id A column ID or object.
* @return void
*/
static public function render_column( $col_id = null ) {
global $wp_the_query;
$col = is_object( $col_id ) ? $col_id : FLBuilderModel::get_node( $col_id );
$post_id = FLBuilderModel::get_post_id();
$active = FLBuilderModel::is_builder_active() && $post_id == $wp_the_query->post->ID;
$visible = FLBuilderModel::is_node_visible( $col );
if ( $active || $visible ) {
include FL_BUILDER_DIR . 'includes/column.php';
}
}
/**
* Renders the HTML attributes for a single column.
*
* @since 1.0
* @param object $col A column node object.
* @return void
*/
static public function render_column_attributes( $col ) {
/**
* Use this filter to work with the custom class a user adds to a column under Column Settings > Advanced > Class.
* @see fl_builder_column_custom_class
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$custom_class = apply_filters( 'fl_builder_column_custom_class', $col->settings->class, $col );
$overlay_bgs = array( 'photo' );
$nested = FLBuilderModel::get_nodes( 'column-group', $col );
$active = FLBuilderModel::is_builder_active();
$visible = FLBuilderModel::is_node_visible( $col );
$has_rules = FLBuilderModel::node_has_visibility_rules( $col );
$attrs = array(
'id' => $col->settings->id,
'class' => array(
'fl-col',
'fl-node-' . $col->node,
),
'data-node' => $col->node,
'style' => array(),
);
// Classes
if ( $col->settings->size <= 50 ) {
$attrs['class'][] = 'fl-col-small';
}
if ( count( $nested ) > 0 ) {
$attrs['class'][] = 'fl-col-has-cols';
}
if ( in_array( $col->settings->bg_type, $overlay_bgs ) && ! empty( $col->settings->bg_overlay_color ) ) {
$attrs['class'][] = 'fl-col-bg-overlay';
}
if ( ! empty( $col->settings->responsive_display ) ) {
$attrs['class'][] = 'fl-visible-' . $col->settings->responsive_display;
}
if ( ! empty( $custom_class ) ) {
$attrs['class'][] = trim( esc_attr( $custom_class ) );
}
if ( $active && ! $visible ) {
$attrs['class'][] = 'fl-node-hidden';
}
if ( $active && $has_rules ) {
$attrs['class'][] = 'fl-node-has-rules';
}
// Style
if ( $active ) {
$attrs['style'][] = 'width: ' . $col->settings->size . '%;';
}
// Render the attrs
self::render_node_attributes( apply_filters( 'fl_builder_column_attributes', $attrs, $col ) );
}
/**
* Renders the markup for all modules in a column.
*
* @since 1.0
* @param string|object $col_id A column ID or object.
* @return void
*/
static public function render_modules( $col_id = null ) {
$nodes = FLBuilderModel::get_nodes( null, $col_id );
do_action( 'fl_builder_before_render_modules', $nodes, $col_id );
foreach ( $nodes as $node ) {
if ( 'module' == $node->type && FLBuilderModel::is_module_registered( $node->settings->type ) ) {
self::render_module( $node );
} elseif ( 'column-group' == $node->type ) {
self::render_column_group( $node );
}
}
do_action( 'fl_builder_after_render_modules', $nodes, $col_id );
}
/**
* Renders the markup for a single module.
*
* @since 1.7
* @param string|object $module_id A module ID or object.
* @return void
*/
static public function render_module( $module_id = null ) {
global $wp_the_query;
$module = FLBuilderModel::get_module( $module_id );
$settings = $module->settings;
$id = $module->node;
$post_id = FLBuilderModel::get_post_id();
$active = FLBuilderModel::is_builder_active() && $post_id == $wp_the_query->post->ID;
$visible = FLBuilderModel::is_node_visible( $module );
if ( $active || $visible ) {
do_action( 'fl_builder_before_render_module', $module );
$template_file = self::locate_template_file(
apply_filters( 'fl_builder_module_template_base', 'module', $module ),
apply_filters( 'fl_builder_module_template_slug', '', $module )
);
if ( $template_file ) {
include $template_file;
}
do_action( 'fl_builder_after_render_module', $module );
}
}
/**
* Renders the markup for a single module. This can be used to render
* the markup of a module within another module by passing the type
* and settings params and leaving the module param null.
*
* @since 1.0
* @param string $type The type of module.
* @param object $settings A module settings object.
* @param object $module Optional. An existing module object to use.
* @return void
*/
static public function render_module_html( $type, $settings, $module = null ) {
// Settings
$defaults = FLBuilderModel::get_module_defaults( $type );
$settings = (object) array_merge( (array) $defaults, (array) $settings );
// Module
$class = get_class( FLBuilderModel::$modules[ $type ] );
$module = new $class();
$module->settings = $settings;
// Shorthand reference to the module's id.
$id = $module->node;
do_action( 'fl_builder_render_module_html_before', $type, $settings, $module );
ob_start();
include apply_filters( 'fl_builder_render_module_html', $module->dir . 'includes/frontend.php', $type, $settings, $module );
$content = ob_get_clean();
echo apply_filters( 'fl_builder_render_module_html_content', $content, $type, $settings, $module );
do_action( 'fl_builder_render_module_html_after', $type, $settings, $module );
}
/**
* Renders the HTML attributes for a single module.
*
* @since 1.0
* @param object $module A module node object.
* @return void
*/
static public function render_module_attributes( $module ) {
/**
* Use this filter to work with the custom class a user adds to a module in the Class field on the Advanced tab.
* @see fl_builder_module_custom_class
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$custom_class = apply_filters( 'fl_builder_module_custom_class', $module->settings->class, $module );
$active = FLBuilderModel::is_builder_active();
$visible = FLBuilderModel::is_node_visible( $module );
$has_rules = FLBuilderModel::node_has_visibility_rules( $module );
$attrs = array(
'id' => esc_attr( $module->settings->id ),
'class' => array(
'fl-module',
'fl-module-' . $module->settings->type,
'fl-node-' . $module->node,
),
'data-node' => $module->node,
);
// Classes
if ( ! empty( $module->settings->responsive_display ) ) {
$attrs['class'][] = 'fl-visible-' . $module->settings->responsive_display;
}
if ( ! empty( $module->settings->animation ) && is_string( $module->settings->animation ) ) {
$attrs['class'][] = 'fl-animation fl-' . $module->settings->animation;
$attrs['data-animation-delay'][] = $module->settings->animation_delay;
}
if ( ! empty( $custom_class ) ) {
$attrs['class'][] = trim( esc_attr( $custom_class ) );
}
if ( $active && ! $visible ) {
$attrs['class'][] = 'fl-node-hidden';
}
if ( $active && $has_rules ) {
$attrs['class'][] = 'fl-node-has-rules';
}
// Data
if ( $active ) {
$attrs['data-parent'] = $module->parent;
$attrs['data-type'] = $module->settings->type;
$attrs['data-name'] = $module->name;
}
// Render the attrs
self::render_node_attributes( apply_filters( 'fl_builder_module_attributes', $attrs, $module ) );
}
/**
* Renders the CSS for a single module.
*
* NOTE: This is not used to render CSS for modules in the FLBuilder::render_css
* method. Instead it is used to render CSS for one module inside of another.
* For example, you can use this along with FLBuilder::render_module_html to
* render a button module inside of a callout module. If you need to filter the
* CSS for the layout, consider using the fl_builder_render_css filter instead.
*
* @since 1.0
* @param string $type The type of module.
* @param object $id A module node ID.
* @param object $settings A module settings object.
* @return void
*/
static public function render_module_css( $type, $id, $settings ) {
// Settings
$global_settings = FLBuilderModel::get_global_settings();
$defaults = FLBuilderModel::get_module_defaults( $type );
$settings = (object) array_merge( (array) $defaults, (array) $settings );
$settings = apply_filters( 'fl_builder_render_module_css_settings', $settings, $id, $type );
// Module
$class = get_class( FLBuilderModel::$modules[ $type ] );
$module = new $class();
$module->settings = $settings;
// CSS
ob_start();
include $module->dir . 'includes/frontend.css.php';
$css = ob_get_clean();
echo apply_filters( 'fl_builder_render_module_css', $css, $module, $id );
}
/**
* Renders the CSS and JS assets.
*
* @since 1.7
* @return void
*/
static public function render_assets() {
self::render_css();
self::render_js();
}
/**
* Renders custom CSS in a style tag so it can be edited
* using the builder interface.
*
* @since 1.7
* @return void
*/
static public function render_custom_css_for_editing() {
if ( ! FLBuilderModel::is_builder_active() ) {
return;
}
$global_settings = FLBuilderModel::get_global_settings();
$layout_settings = FLBuilderModel::get_layout_settings();
echo '';
echo '';
}
/**
* Renders and caches the CSS for a builder layout.
*
* @since 1.0
* @param bool $include_global
* @param bool $save
* @return string
*/
static public function render_css( $include_global = true, $save = true ) {
// Get info on the new file.
$nodes = FLBuilderModel::get_categorized_nodes();
$node_status = FLBuilderModel::get_node_status();
$global_settings = FLBuilderModel::get_global_settings();
$asset_info = FLBuilderModel::get_asset_info();
$post_id = FLBuilderModel::get_post_id();
$post = get_post( $post_id );
$css = '';
$path = $include_global ? $asset_info['css'] : $asset_info['css_partial'];
// Render the global css.
if ( $include_global ) {
$css .= self::render_global_css();
}
// Loop through rows
foreach ( $nodes['rows'] as $row ) {
// Instance row css
ob_start();
include FL_BUILDER_DIR . 'includes/row-css.php';
$css .= ob_get_clean();
// Instance row margins
$css .= self::render_row_margins( $row );
// Instance row padding
$css .= self::render_row_padding( $row );
// Instance row border
$css .= self::render_row_border( $row );
}
// Loop through the columns.
foreach ( $nodes['columns'] as $col ) {
// Instance column css
ob_start();
include FL_BUILDER_DIR . 'includes/column-css.php';
$css .= ob_get_clean();
// Instance column margins
$css .= self::render_column_margins( $col );
// Instance column padding
$css .= self::render_column_padding( $col );
// Instance column border
$css .= self::render_column_border( $col );
// Get the modules in this column.
$modules = FLBuilderModel::get_modules( $col );
}
// Loop through the modules.
foreach ( $nodes['modules'] as $module ) {
// Global module css
$file = $module->dir . 'css/frontend.css';
$file_responsive = $module->dir . 'css/frontend.responsive.css';
// Only include global module css that hasn't been included yet.
if ( ! in_array( $module->settings->type . '-module-css', self::$enqueued_global_assets ) ) {
// Add to the compiled array so we don't include it again.
self::$enqueued_global_assets[] = $module->settings->type . '-module-css';
// Get the standard module css.
if ( fl_builder_filesystem()->file_exists( $file ) ) {
$css .= fl_builder_filesystem()->file_get_contents( $file );
}
// Get the responsive module css.
if ( $global_settings->responsive_enabled && fl_builder_filesystem()->file_exists( $file_responsive ) ) {
$css .= '@media (max-width: ' . $global_settings->responsive_breakpoint . 'px) { ';
$css .= fl_builder_filesystem()->file_get_contents( $file_responsive );
$css .= ' }';
}
}
// Instance module css
$file = $module->dir . 'includes/frontend.css.php';
$settings = $module->settings;
$id = $module->node;
if ( fl_builder_filesystem()->file_exists( $file ) ) {
ob_start();
include $file;
$css .= ob_get_clean();
}
// Instance module margins
$css .= self::render_module_margins( $module );
if ( ! isset( $global_settings->auto_spacing ) || $global_settings->auto_spacing ) {
$css .= self::render_responsive_module_margins( $module );
}
}
// Custom Global CSS (included here for proper specificity)
if ( 'published' == $node_status && $include_global ) {
$css .= $global_settings->css;
}
// Custom Global Nodes CSS
$css .= self::render_global_nodes_custom_code( 'css' );
// Custom Layout CSS
if ( 'published' == $node_status ) {
$css .= FLBuilderModel::get_layout_settings()->css;
}
/**
* Use this filter to modify the CSS that is compiled and cached for each builder layout.
* @see fl_builder_render_css
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$css = apply_filters( 'fl_builder_render_css', $css, $nodes, $global_settings, $include_global );
// Minify the CSS.
if ( ! self::is_debug() ) {
$css = preg_replace( '!/\*[^*]*\*+([^/][^*]*\*+)*/!', '', $css );
$css = str_replace( array( "\r\n", "\r", "\n", "\t", ' ', ' ', ' ' ), '', $css );
}
// Save the CSS.
if ( $save ) {
fl_builder_filesystem()->file_put_contents( $path, $css );
}
do_action( 'fl_builder_after_render_css' );
return $css;
}
/**
* Renders the CSS used for all builder layouts.
*
* @since 1.8.2
* @return string
*/
static public function render_global_css() {
// Get info on the new file.
$global_settings = FLBuilderModel::get_global_settings();
// Core layout css
$css = fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout.css' );
// Core button defaults
if ( ! defined( 'FL_THEME_VERSION' ) ) {
$css .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout-button-defaults.css' );
}
// Core layout RTL css
if ( is_rtl() ) {
$css .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout-rtl.css' );
}
// Global node css
foreach ( array(
array( 'row_margins', '.fl-row-content-wrap { margin: ' ),
array( 'row_padding', '.fl-row-content-wrap { padding: ' ),
array( 'row_width', '.fl-row-fixed-width { max-width: ' ),
array( 'module_margins', '.fl-module-content { margin: ' ),
) as $data ) {
if ( '' !== $global_settings->{ $data[0] } ) {
$value = preg_replace( self::regex( 'css_unit' ), '', strtolower( $global_settings->{ $data[0] } ) );
$css .= $data[1] . esc_attr( $value );
$css .= ( is_numeric( $value ) ) ? ( 'px; }' ) : ( '; }' );
}
}
// Responsive layout css
if ( $global_settings->responsive_enabled ) {
// Medium devices
$css .= '@media (max-width: ' . $global_settings->medium_breakpoint . 'px) { ';
// Core medium layout css
$css .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout-medium.css' );
// Global node medium css
foreach ( array(
array( 'row_margins_medium', '.fl-row[data-node] > .fl-row-content-wrap { margin: ' ),
array( 'row_padding_medium', '.fl-row[data-node] > .fl-row-content-wrap { padding: ' ),
array( 'module_margins_medium', '.fl-module[data-node] > .fl-module-content { margin: ' ),
) as $data ) {
if ( '' !== $global_settings->{ $data[0] } ) {
$value = preg_replace( self::regex( 'css_unit' ), '', strtolower( $global_settings->{ $data[0] } ) );
$css .= $data[1] . esc_attr( $value );
$css .= ( is_numeric( $value ) ) ? ( 'px; }' ) : ( '; }' );
}
}
$css .= ' }';
// Responsive devices
$css .= '@media (max-width: ' . $global_settings->responsive_breakpoint . 'px) { ';
// Core responsive layout css
$css .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout-responsive.css' );
// Auto spacing
if ( ! isset( $global_settings->auto_spacing ) || $global_settings->auto_spacing ) {
$css .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . '/css/fl-builder-layout-auto-spacing.css' );
}
// Global node responsive css
foreach ( array(
array( 'row_margins_responsive', '.fl-row[data-node] > .fl-row-content-wrap { margin: ' ),
array( 'row_padding_responsive', '.fl-row[data-node] > .fl-row-content-wrap { padding: ' ),
array( 'module_margins_responsive', '.fl-module[data-node] > .fl-module-content { margin: ' ),
) as $data ) {
if ( '' !== $global_settings->{ $data[0] } ) {
$value = preg_replace( self::regex( 'css_unit' ), '', strtolower( $global_settings->{ $data[0] } ) );
$css .= $data[1] . esc_attr( $value );
$css .= ( is_numeric( $value ) ) ? ( 'px; }' ) : ( '; }' );
}
}
$css .= ' }';
}
// Default page heading
if ( FLBuilderModel::is_builder_enabled() ) {
if ( ! $global_settings->show_default_heading && ! empty( $global_settings->default_heading_selector ) ) {
$heading_selector = esc_attr( $global_settings->default_heading_selector );
// If the value starts with `body` or `.fl-builder` selector, we use custom selectors
if ( 0 === strpos( $heading_selector, 'body' ) || 0 === strpos( $heading_selector, '.fl-builder' ) ) {
$css .= $heading_selector;
} else {
$css .= '.page ' . $heading_selector . ', .single-fl-builder-template ' . $heading_selector;
}
$css .= ' { display:none; }';
}
}
return $css;
}
/**
* Forcing HTTPS in URLs when `FLBuilderModel::is_ssl()` returns TRUE
*
* @since 1.7.6
* @param string $content A string where the URLs will be modified.
* @return string String with SSL ready URLs.
*/
static public function rewrite_css_cache_urls( $content ) {
if ( FLBuilderModel::is_ssl() ) {
$content = str_ireplace( 'http:', 'https:', $content );
}
return $content;
}
/**
* Regular expressions.
*
* @since 1.9
* @param string $scope What regular expression to return?
* @return string Regular expression.
*/
static public function regex( $scope ) {
$regex = array(
'css_unit' => '/[^a-z0-9%.\-]/',
);
return ( isset( $regex[ $scope ] ) ) ? $regex[ $scope ] : null;
}
/**
* Renders the CSS spacing and border properties for a node.
*
* @param object $node A generic node object.
* @param string $prop_type One of [ 'padding', 'margin', 'border' ].
* @param string $selector_prefix Optional CSS selector prefix for better overrides.
* @return string A CSS string.
*/
static public function render_node_spacing( $node = null, $prop_type = '', $selector_prefix = '' ) {
// Exit early if incorrect parameters
if ( ! is_object( $node ) || empty( $prop_type ) ) {
return;
}
$prop_type = strtolower( $prop_type );
// Ensure type is valid
if ( ! in_array( $prop_type, array( 'margin', 'padding', 'border' ), true ) ) {
return;
}
$global_settings = FLBuilderModel::get_global_settings();
$settings = $node->settings;
$css = '';
$selector_prefix .= ' .fl-node-' . $node->node;
// Determine selector suffix to apply spacing to
switch ( $node->type ) {
case 'row':
$selector_suffix = ' > .fl-row-content-wrap';
break;
case 'column':
$selector_suffix = ' > .fl-col-content';
break;
case 'module':
$selector_suffix = ' > .fl-module-content';
break;
}
// Create rules for each breakpoint
foreach ( array( 'default', 'medium', 'responsive' ) as $breakpoint ) {
$breakpoint_css = '';
$setting_suffix = ( 'default' !== $breakpoint ) ? '_' . $breakpoint : '';
// Iterate over each direction
foreach ( array( 'top', 'right', 'bottom', 'left' ) as $dir ) {
$setting = $prop_type . '_' . $dir . $setting_suffix;
if ( ! isset( $settings->{ $setting } ) ) {
continue;
}
$prop = $prop_type . '-' . $dir;
$value = preg_replace( self::regex( 'css_unit' ), '', strtolower( $settings->{ $setting } ) );
if ( 'border' === $prop_type ) {
if ( empty( $settings->border_type ) ) {
continue;
} else {
$prop .= '-width';
}
}
if ( '' !== $value ) {
$breakpoint_css .= "\t";
$breakpoint_css .= $prop . ':' . esc_attr( $value );
$breakpoint_css .= ( is_numeric( trim( $value ) ) ) ? ( 'px;' ) : ( ';' );
$breakpoint_css .= "\r\n";
}
}
if ( ! empty( $breakpoint_css ) ) {
// Build the selector
if ( 'default' !== $breakpoint ) {
$selector = $selector_prefix . '.fl-' . str_replace( 'column', 'col', $node->type ) . $selector_suffix;
} else {
$selector = $selector_prefix . $selector_suffix;
}
// Wrap css in selector
$breakpoint_css = $selector . ' {' . "\r\n" . $breakpoint_css . '}' . "\r\n";
// Wrap css in media query
if ( 'default' !== $breakpoint ) {
$breakpoint_css = '@media ( max-width: ' . $global_settings->{ $breakpoint . '_breakpoint' } . 'px ) {' . "\r\n" . $breakpoint_css . '}' . "\r\n";
}
$css .= $breakpoint_css;
}
}
return $css;
}
/**
* Renders the CSS margins for a row.
*
* @since 1.0
* @param object $row A row node object.
* @return string The row CSS margins string.
*/
static public function render_row_margins( $row ) {
return self::render_node_spacing( $row, 'margin' );
}
/**
* Renders the CSS padding for a row.
*
* @since 1.0
* @param object $row A row node object.
* @return string The row CSS padding string.
*/
static public function render_row_padding( $row ) {
return self::render_node_spacing( $row, 'padding' );
}
/**
* Renders the CSS border widths for a row.
*
* @since 1.9
* @param object $row A row node object.
* @return string The row CSS border-width string.
*/
static public function render_row_border( $row ) {
return self::render_node_spacing( $row, 'border' );
}
/**
* Renders the CSS margins for a column.
*
* @since 1.0
* @param object $col A column node object.
* @return string The column CSS margins string.
*/
static public function render_column_margins( $col ) {
return self::render_node_spacing( $col, 'margin' );
}
/**
* Renders the CSS padding for a column.
*
* @since 1.0
* @param object $col A column node object.
* @return string The column CSS padding string.
*/
static public function render_column_padding( $col ) {
return self::render_node_spacing( $col, 'padding' );
}
/**
* Renders the CSS border widths for a column.
*
* @since 1.9
* @param object $col A column node object.
* @return string The column CSS border-width string.
*/
static public function render_column_border( $col ) {
return self::render_node_spacing( $col, 'border', '.fl-builder-content' );
}
/**
* Renders the CSS margins for a module.
*
* @since 1.0
* @param object $module A module node object.
* @return string The module CSS margins string.
*/
static public function render_module_margins( $module ) {
return self::render_node_spacing( $module, 'margin' );
}
/**
* Renders the (auto) responsive CSS margins for a module.
*
* @since 1.0
* @param object $module A module node object.
* @return string The module CSS margins string.
*/
static public function render_responsive_module_margins( $module ) {
$global_settings = FLBuilderModel::get_global_settings();
$settings = $module->settings;
$margins = '';
$css = '';
// Bail early if we have global responsive margins.
if ( '' != $global_settings->module_margins_responsive ) {
return $css;
}
// Get the global default margin value to use.
if ( '' != $global_settings->module_margins_medium ) {
$default = trim( $global_settings->module_margins_medium );
} else {
$default = trim( $global_settings->module_margins );
}
// Set the responsive margin CSS if necessary.
foreach ( array( 'top', 'bottom', 'left', 'right' ) as $dimension ) {
$responsive = 'margin_' . $dimension . '_responsive';
$medium = 'margin_' . $dimension . '_responsive';
$desktop = 'margin_' . $dimension;
if ( '' == $settings->$responsive ) {
$value = '' == $settings->$medium ? $settings->$desktop : $settings->$medium;
if ( '' != $value && ( $value > $default || $value < 0 ) ) {
$margins .= 'margin-' . $dimension . ':' . esc_attr( $default ) . 'px;';
}
}
}
// Set the media query if we have margins.
if ( '' !== $margins ) {
$css .= '@media (max-width: ' . esc_attr( $global_settings->responsive_breakpoint ) . 'px) { ';
$css .= '.fl-node-' . $module->node . ' > .fl-module-content { ' . $margins . ' }';
$css .= ' }';
}
return $css;
}
/**
* Renders and caches the JavaScript for a builder layout.
*
* @since 1.0
* @param bool $include_global
* @param bool $save
* @return string
*/
static public function render_js( $include_global = true, $save = true ) {
// Get info on the new file.
$nodes = FLBuilderModel::get_categorized_nodes();
$global_settings = FLBuilderModel::get_global_settings();
$layout_settings = FLBuilderModel::get_layout_settings();
$rows = FLBuilderModel::get_nodes( 'row' );
$asset_info = FLBuilderModel::get_asset_info();
$js = '';
$path = $include_global ? $asset_info['js'] : $asset_info['js_partial'];
// Render the global js.
if ( $include_global && ! isset( $_GET['safemode'] ) ) {
$js .= self::render_global_js();
}
// Loop through the rows.
foreach ( $nodes['rows'] as $row ) {
$js .= self::render_row_js( $row );
}
// Loop through the modules.
foreach ( $nodes['modules'] as $module ) {
$js .= self::render_module_js( $module );
}
// Add the layout settings JS.
if ( ! isset( $_GET['safemode'] ) ) {
$js .= self::render_global_nodes_custom_code( 'js' );
$js .= ( is_array( $layout_settings->js ) || is_object( $layout_settings->js ) ) ? json_encode( $layout_settings->js ) : $layout_settings->js;
}
// Call the FLBuilder._renderLayoutComplete method if we're currently editing.
if ( stristr( $asset_info['js'], '-draft.js' ) || stristr( $asset_info['js'], '-preview.js' ) ) {
$js .= "; if(typeof FLBuilder !== 'undefined' && typeof FLBuilder._renderLayoutComplete !== 'undefined') FLBuilder._renderLayoutComplete();";
}
// Include FLJSMin
if ( ! class_exists( 'FLJSMin' ) ) {
include FL_BUILDER_DIR . 'classes/class-fl-jsmin.php';
}
/**
* Use this filter to modify the JavaScript that is compiled and cached for each builder layout.
* @see fl_builder_render_js
* @link https://kb.wpbeaverbuilder.com/article/117-plugin-filter-reference
*/
$js = apply_filters( 'fl_builder_render_js', $js, $nodes, $global_settings, $include_global );
// Only proceed if we have JS.
if ( ! empty( $js ) ) {
// Minify the JS.
if ( ! self::is_debug() ) {
try {
$min = FLJSMin::minify( $js );
} catch ( Exception $e ) {}
if ( isset( $min ) ) {
$js = $min;
}
}
// Save the JS.
if ( $save ) {
fl_builder_filesystem()->file_put_contents( $path, $js );
}
do_action( 'fl_builder_after_render_js' );
}
return $js;
}
/**
* Renders the JS used for all builder layouts.
*
* @since 1.8.2
* @return string
*/
static public function render_global_js() {
$global_settings = FLBuilderModel::get_global_settings();
$js = '';
// Add the path legacy vars (FLBuilderLayoutConfig.paths should be used instead).
$js .= "var wpAjaxUrl = '" . admin_url( 'admin-ajax.php' ) . "';";
$js .= "var flBuilderUrl = '" . FL_BUILDER_URL . "';";
// Layout config object.
ob_start();
include FL_BUILDER_DIR . 'includes/layout-js-config.php';
$js .= ob_get_clean();
// Core layout JS.
$js .= fl_builder_filesystem()->file_get_contents( FL_BUILDER_DIR . 'js/fl-builder-layout.js' );
// Add the global settings JS.
$js .= $global_settings->js;
return $js;
}
/**
* Renders the JavaScript for a single row.
*
* @since 1.7
* @param string|object $row_id A row ID or object.
* @return string
*/
static public function render_row_js( $row_id ) {
$row = is_object( $row_id ) ? $row_id : FLBuilderModel::get_node( $row_id );
$settings = $row->settings;
$id = $row->node;
ob_start();
include FL_BUILDER_DIR . 'includes/row-js.php';
return ob_get_clean();
}
/**
* Renders the JavaScript for all modules in a single row.
*
* @since 1.7
* @param string|object $row_id A row ID or object.
* @return string
*/
static public function render_row_modules_js( $row_id ) {
$row = is_object( $row_id ) ? $row_id : FLBuilderModel::get_node( $row_id );
$nodes = FLBuilderModel::get_categorized_nodes();
$template_post_id = FLBuilderModel::is_node_global( $row );
$js = '';
// Render the JS.
foreach ( $nodes['groups'] as $group ) {
if ( $row->node == $group->parent || ( $template_post_id && $row->template_node_id == $group->parent ) ) {
foreach ( $nodes['columns'] as $column ) {
if ( $group->node == $column->parent ) {
foreach ( $nodes['modules'] as $module ) {
if ( $column->node == $module->parent ) {
$js .= self::render_module_js( $module );
}
}
}
}
}
}
// Return the JS.
return $js;
}
/**
* Renders the JavaScript for all modules in a single column.
*
* @since 1.7
* @param string|object $col_id A column ID or object.
* @return string
*/
static public function render_column_modules_js( $col_id ) {
$col = is_object( $col_id ) ? $col_id : FLBuilderModel::get_node( $col_id );
$nodes = FLBuilderModel::get_categorized_nodes();
$js = '';
// Render the JS.
foreach ( $nodes['modules'] as $module ) {
if ( $col->node == $module->parent ) {
$js .= self::render_module_js( $module );
}
}
// Return the JS.
return $js;
}
/**
* Renders the JavaScript for a single module.
*
* @since 1.7
* @param string|object $module_id A module ID or object.
* @return string
*/
static public function render_module_js( $module_id ) {
$module = is_object( $module_id ) ? $module_id : FLBuilderModel::get_module( $module_id );
$global_settings = FLBuilderModel::get_global_settings();
$js = '';
// Global module JS
$file = $module->dir . 'js/frontend.js';
if ( fl_builder_filesystem()->file_exists( $file ) && ! in_array( $module->settings->type . '-module-js', self::$enqueued_global_assets ) ) {
$js .= "\n" . fl_builder_filesystem()->file_get_contents( $file );
self::$enqueued_global_assets[] = $module->settings->type . '-module-js';
}
// Instance module JS
$file = $module->dir . 'includes/frontend.js.php';
$settings = $module->settings;
$id = $module->node;
if ( fl_builder_filesystem()->file_exists( $file ) ) {
ob_start();
include $file;
$js .= ob_get_clean();
}
// Return the JS.
return $js;
}
/**
* Renders the custom CSS or JS for all global nodes in a layout.
*
* @since 1.7
*/
static public function render_global_nodes_custom_code( $type = 'css' ) {
$code = '';
$rendered = array();
if ( ! FLBuilderModel::is_post_node_template() ) {
$nodes = FLBuilderModel::get_layout_data();
$node_status = FLBuilderModel::get_node_status();
foreach ( $nodes as $node_id => $node ) {
$template_post_id = FLBuilderModel::is_node_global( $node );
if ( $template_post_id && ! in_array( $template_post_id, $rendered ) ) {
$rendered[] = $template_post_id;
$code .= FLBuilderModel::get_layout_settings( $node_status, $template_post_id )->{ $type };
}
}
}
return $code;
}
/**
* Check if publish should require page to refresh.
*
* @since 2.0
* @return void
*/
static public function should_refresh_on_publish() {
$refresh = ! is_admin_bar_showing();
return apply_filters( 'fl_builder_should_refresh_on_publish', $refresh );
}
/**
* Custom logging function that handles objects and arrays.
*
* @since 1.0
* @return void
*/
static public function log() {
foreach ( func_get_args() as $arg ) {
ob_start();
print_r( $arg );
error_log( ob_get_clean() );
}
}
/**
* Filter WP uploads and check filetype is valid for photo and video modules.
* @since 1.10.8
*/
static public function wp_handle_upload_prefilter_filter( $file ) {
$type = isset( $_POST['fl_upload_type'] ) ? $_POST['fl_upload_type'] : false;
$ext = pathinfo( $file['name'], PATHINFO_EXTENSION );
$regex = array(
'photo' => '#(jpe?g|png|gif|bmp|tiff?)#i',
'video' => '#(mp4|m4v|webm)#i',
);
if ( ! $type ) {
return $file;
}
$regex = apply_filters( 'fl_module_upload_regex', $regex, $type, $ext, $file );
if ( ! preg_match( $regex[ $type ], $ext ) ) {
$file['error'] = sprintf( __( 'The uploaded file is not a valid %s extension.', 'fl-builder' ), $type );
}
return $file;
}
/**
* Default HTML for no image.
* @since 1.10.8
* @return string
*/
static public function default_image_html( $classes ) {
return sprintf( '
', FL_BUILDER_URL . 'img/no-image.png', $classes );
}
/**
* Check if debug is enabled.
* @since 1.10.8.2
* @return bool
*/
static public function is_debug() {
$debug = false;
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
$debug = true;
}
return apply_filters( 'fl_is_debug', $debug );
}
/**
* Get the fa5 url.
* @since 2.2
* @return string url
*/
static public function get_fa5_url() {
return ( apply_filters( 'fl_enable_fa5_pro', false ) ) ? self::$fa5_pro_url : plugins_url( '/fonts/fontawesome/css/all.min.css', FL_BUILDER_FILE );
}
/**
* @since 1.0
* @deprecated 1.7.4
*/
static public function layout_styles_scripts( $post_id ) {
_deprecated_function( __METHOD__, '1.7.4', __CLASS__ . '::enqueue_layout_styles_scripts()' );
self::enqueue_layout_styles_scripts();
}
/**
* @since 1.0
* @deprecated 1.7.4
*/
static public function styles_scripts() {
_deprecated_function( __METHOD__, '1.7.4', __CLASS__ . '::enqueue_ui_styles_scripts()' );
self::enqueue_ui_styles_scripts();
}
/**
* @since 1.0
* @deprecated 1.8
*/
static public function register_templates_post_type() {
_deprecated_function( __METHOD__, '1.8', 'FLBuilderUserTemplates::register_post_type()' );
if ( class_exists( 'FLBuilderUserTemplates' ) ) {
FLBuilderUserTemplates::register_post_type();
}
}
/**
* @since 1.0
* @deprecated 1.8
*/
static public function render_template( $template ) {
_deprecated_function( __METHOD__, '1.8', 'FLBuilderUserTemplates::template_include()' );
if ( class_exists( 'FLBuilderUserTemplates' ) ) {
FLBuilderUserTemplates::template_include();
}
}
/**
* @since 1.6.3
* @deprecated 1.8
*/
static public function render_ui_panel_node_templates() {
_deprecated_function( __METHOD__, '1.8', 'FLBuilderUserTemplates::render_ui_panel_node_templates()' );
if ( class_exists( 'FLBuilderUserTemplates' ) ) {
FLBuilderUserTemplates::render_ui_panel_node_templates();
}
}
/**
* @since 1.0
* @deprecated 1.8
*/
static public function render_user_template_settings() {
_deprecated_function( __METHOD__, '1.8', 'FLBuilderUserTemplates::render_settings()' );
if ( class_exists( 'FLBuilderUserTemplates' ) ) {
FLBuilderUserTemplates::render_settings();
}
}
/**
* @since 1.6.3
* @deprecated 1.8
*/
static public function render_node_template_settings( $node_id = null ) {
_deprecated_function( __METHOD__, '1.8', 'FLBuilderUserTemplates::render_node_settings()' );
if ( class_exists( 'FLBuilderUserTemplates' ) ) {
FLBuilderUserTemplates::render_node_settings( $node_id );
}
}
/**
* @since 1.0
* @deprecated 2.0
*/
static public function render_template_selector() {
_deprecated_function( __METHOD__, '2.0' );
return array(
'html' => '',
);
}
/**
* @since 1.8
* @deprecated 2.0
*/
static public function render_ui_panel_row_templates() {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.8
* @deprecated 2.0
*/
static public function render_ui_panel_modules_templates() {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.8
* @deprecated 2.0
*/
static public function render_layout_settings() {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.0
* @deprecated 2.0
*/
static public function render_global_settings() {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.0
* @deprecated 2.0
*/
static public function render_row_settings( $node_id = null ) {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.0
* @deprecated 2.0
*/
static public function render_column_settings( $node_id = null ) {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 1.0
* @deprecated 2.0
*/
static public function render_module_settings( $node_id = null, $type = null, $parent_id = null, $render_state = true ) {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* @since 2.0.1
* @deprecated 2.0.7
*/
static public function render_settings_config() {
_deprecated_function( __METHOD__, '2.0.7', 'FLBuilderUISettingsForms::render_settings_config()' );
FLBuilderUISettingsForms::render_settings_config();
}
}
FLBuilder::init();