\d+)/i', $agent, $B) && $B['v'] <= 10 ) || // Chrome lte 25 ( preg_match('/Chrome\/(?P\d+)/i', $agent, $B) && $B['v'] <= 25 ) || // Firefox lte 21 ( preg_match('/Firefox\/(?P\d+)/i', $agent, $B) && $B['v'] <= 21 ) || // Safari lte 7 ( preg_match('/Version\/(?P\d+).*?Safari\/\d+/i', $agent, $B) && $B['v'] <= 6 ); } /** * Autoload Page Builder specific classses. * * @param $class */ public static function autoloader( $class ) { $filename = false; if ( strpos( $class, 'SiteOrigin_Panels_Widgets_' ) === 0 ) { $filename = str_replace( 'SiteOrigin_Panels_Widgets_', '', $class ); $filename = str_replace( '_', '-', $filename ); $filename = strtolower( preg_replace( '/([a-z])([A-Z])/', '$1-$2', $filename ) ); $filename = plugin_dir_path( __FILE__ ) . 'inc/widgets/' . $filename . '.php'; } else if ( strpos( $class, 'SiteOrigin_Panels_Compat_' ) === 0 ) { $filename = str_replace( array( 'SiteOrigin_Panels_Compat_', '_' ), array( '', '-' ), $class ); $filename = plugin_dir_path( __FILE__ ) . 'compat/' . strtolower( $filename ) . '.php'; } else if ( strpos( $class, 'SiteOrigin_Panels_' ) === 0 ) { $filename = str_replace( array( 'SiteOrigin_Panels_', '_' ), array( '', '-' ), $class ); $filename = plugin_dir_path( __FILE__ ) . 'inc/' . strtolower( $filename ) . '.php'; } if ( ! empty( $filename ) && file_exists( $filename ) ) { include $filename; } } public static function activate() { add_option( 'siteorigin_panels_initial_version', SITEORIGIN_PANELS_VERSION, '', 'no' ); } /** * Initialize SiteOrigin Page Builder * * @action plugins_loaded */ public function init() { if ( ! is_admin() && siteorigin_panels_setting( 'sidebars-emulator' ) && ( ! get_option( 'permalink_structure' ) || get_option( 'rewrite_rules' ) ) ) { // Initialize the sidebars emulator SiteOrigin_Panels_Sidebars_Emulator::single(); } // Initialize the language load_plugin_textdomain( 'siteorigin-panels', false, dirname( plugin_basename( __FILE__ ) ) . '/lang/' ); // Initialize all the extra classes SiteOrigin_Panels_Home::single(); // Check if we need to initialize the admin class. if ( is_admin() ) { SiteOrigin_Panels_Admin::single(); } } /** * @return mixed|void Are we currently viewing the home page */ public static function is_home() { $home = ( is_front_page() && is_page() && get_option( 'show_on_front' ) == 'page' && get_option( 'page_on_front' ) == get_the_ID() && get_post_meta( get_the_ID(), 'panels_data' ) ); return apply_filters( 'siteorigin_panels_is_home', $home ); } /** * Check if we're currently viewing a page builder page. * * @param bool $can_edit Also check if the user can edit this page * * @return bool */ public static function is_panel( $can_edit = false ) { // Check if this is a panel $is_panel = ( siteorigin_panels_is_home() || ( is_singular() && get_post_meta( get_the_ID(), 'panels_data', false ) ) ); return $is_panel && ( ! $can_edit || ( ( is_singular() && current_user_can( 'edit_post', get_the_ID() ) ) || ( siteorigin_panels_is_home() && current_user_can( 'edit_theme_options' ) ) ) ); } /** * Check if we're in the Live Editor in the frontend. * * @return bool */ static function is_live_editor(){ return ! empty( $_GET['siteorigin_panels_live_editor'] ); } public static function preview_url() { global $post, $wp_post_types; if ( empty( $post ) || empty( $wp_post_types ) || empty( $wp_post_types[ $post->post_type ] ) || ! $wp_post_types[ $post->post_type ]->public ) { $preview_url = add_query_arg( 'siteorigin_panels_live_editor', 'true', admin_url( 'admin-ajax.php?action=so_panels_live_editor_preview' ) ); $preview_url = wp_nonce_url( $preview_url, 'live-editor-preview', '_panelsnonce' ); } else { $preview_url = add_query_arg( 'siteorigin_panels_live_editor', 'true', set_url_scheme( get_permalink() ) ); } return $preview_url; } /** * Get the Page Builder data for the home page. * * @return bool|mixed */ public function get_home_page_data() { $page_id = get_option( 'page_on_front' ); if ( empty( $page_id ) ) { $page_id = get_option( 'siteorigin_panels_home_page_id' ); } if ( empty( $page_id ) ) { return false; } $panels_data = get_post_meta( $page_id, 'panels_data', true ); if ( is_null( $panels_data ) ) { // Load the default layout $layouts = apply_filters( 'siteorigin_panels_prebuilt_layouts', array() ); $panels_data = ! empty( $layouts['default_home'] ) ? $layouts['default_home'] : current( $layouts ); } return $panels_data; } /** * Generate post content for WooCommerce shop page if it's using a PB layout. * * @param $content * * @return string * * @filter woocommerce_format_content */ public function generate_woocommerce_content( $content ) { if ( class_exists( 'WooCommerce' ) && is_shop() ) { return $this->generate_post_content( $content ); } return $content; } /** * Generate post content for the current post. * * @param $content * * @return string * * @filter the_content */ public function generate_post_content( $content ) { global $post, $preview; if ( empty( $post ) && ! in_the_loop() ) { return $content; } if ( ! apply_filters( 'siteorigin_panels_filter_content_enabled', true ) ) { return $content; } $post_id = $this->get_post_id(); // Check if this post has panels_data if ( get_post_meta( $post_id, 'panels_data', true ) ) { $panel_content = SiteOrigin_Panels::renderer()->render( $post_id, // Add CSS if this is not the main single post, this is handled by add_single_css $preview || $post_id !== get_queried_object_id() ); if ( ! empty( $panel_content ) ) { $content = $panel_content; if ( ! is_singular() ) { // This is an archive page, so try strip out anything after the more text if ( preg_match( '//', $content, $matches ) ) { $content = explode( $matches[0], $content, 2 ); $content = $content[0]; $content = force_balance_tags( $content ); if ( ! empty( $matches[1] ) && ! empty( $more_link_text ) ) { $more_link_text = strip_tags( wp_kses_no_null( trim( $matches[1] ) ) ); } else { $more_link_text = __( 'Read More', 'siteorigin-panels' ); } $more_link = apply_filters( 'the_content_more_link', ' ID}\" class=\"more-link\">$more_link_text", $more_link_text ); $content .= '

' . $more_link . '

'; } } } } return $content; } /** * Generate an excerpt for the current post, if possible. * * @param $text * * @return mixed|string */ public function generate_post_excerpt( $text ) { global $post; if ( ( empty( $post ) && ! in_the_loop() ) || $text !== '' ) { return $text; } $post_id = $this->get_post_id(); // Check if this post has panels_data $panels_data = get_post_meta( $post_id, 'panels_data', true ); if ( $panels_data && ! empty( $panels_data['widgets'] ) ) { $raw_excerpt = ''; $excerpt_length = apply_filters( 'excerpt_length', 55 ); foreach ( $panels_data['widgets'] as $widget ) { $panels_info = $widget['panels_info']; if ( $panels_info['grid'] > 1 ) { // Limiting search for a text type widget to the first two PB rows to avoid having excerpt content // that's very far down in a post. break; } if ( $panels_info['class'] == 'SiteOrigin_Widget_Editor_Widget' || $panels_info['class'] == 'WP_Widget_Text' ) { $raw_excerpt .= ' ' . $widget['text']; // This is all effectively default behavior for excerpts, copied from the `wp_trim_excerpt` function. // We're just applying it to text type widgets content in the first two rows. $text = strip_shortcodes( $raw_excerpt ); $text = str_replace( ']]>', ']]>', $text ); if ( $this->get_localized_word_count( $text ) >= $excerpt_length ) { break; } } } $text = strip_shortcodes( $raw_excerpt ); $text = str_replace( ']]>', ']]>', $text ); $excerpt_more = apply_filters( 'excerpt_more', ' ' . '[…]' ); $text = wp_trim_words( $text, $excerpt_length, $excerpt_more ); } return $text; } private function get_localized_word_count( $text ) { // From the core `wp_trim_words` function to get localized word count. $text = wp_strip_all_tags( $text ); if ( strpos( _x( 'words', 'Word count type. Do not translate!' ), 'characters' ) === 0 && preg_match( '/^utf\-?8$/i', get_option( 'blog_charset' ) ) ) { $text = trim( preg_replace( "/[\n\r\t ]+/", ' ', $text ), ' ' ); preg_match_all( '/./u', $text, $words_array ); $words_array = $words_array[0]; } else { $words_array = preg_split( "/[\n\r\t ]+/", $text, -1, PREG_SPLIT_NO_EMPTY ); } return count( $words_array ); } /** * Generate CSS for the current post */ public function generate_post_css() { $post_id = $this->get_post_id(); if( is_singular() && get_post_meta( $post_id, 'panels_data', true ) ) { $renderer = SiteOrigin_Panels::renderer(); $renderer->add_inline_css( $post_id, $renderer->generate_css( $post_id ) ); } } /** * Get the post id for the current post. */ function get_post_id() { $post_id = get_the_ID(); if ( class_exists( 'WooCommerce' ) && is_shop() ) { $post_id = wc_get_page_id( 'shop' ); } global $preview; // If we're viewing a preview make sure we load and render the autosave post's meta. if ( $preview ) { $preview_post = wp_get_post_autosave( $post_id, get_current_user_id() ); if ( ! empty( $preview_post ) ) { $post_id = $preview_post->ID; } } return $post_id; } /** * Add all the necessary body classes. * * @param $classes * * @return array */ function body_class( $classes ) { if( self::is_panel() ) { $classes[] = 'siteorigin-panels'; $classes[] = 'siteorigin-panels-before-js'; add_action( 'wp_footer', array( $this, 'strip_before_js' ), 99 ); } if( self::is_home() ) $classes[] = 'siteorigin-panels-home'; if( self::is_live_editor() ) $classes[] = 'siteorigin-panels-live-editor'; return $classes; } /** * Add the Edit Home Page item to the admin bar. * * @param WP_Admin_Bar $admin_bar * * @return WP_Admin_Bar */ function admin_bar_menu( $admin_bar ) { // Add the edit home page link if ( siteorigin_panels_setting( 'home-page' ) && current_user_can( 'edit_theme_options' ) && ( is_home() || is_front_page() ) ) { if ( ( is_page() && get_post_meta( get_the_ID(), 'panels_data', true ) !== '' ) || ! is_page() ) { $admin_bar->add_node( array( 'id' => 'edit-home-page', 'title' => __( 'Edit Home Page', 'siteorigin-panels' ), 'href' => admin_url( 'themes.php?page=so_panels_home_page' ) ) ); if ( is_page() ) { // Remove the standard edit button $admin_bar->remove_node( 'edit' ); } } } // Add a Live Edit link if this is a Page Builder page that the user can edit if ( siteorigin_panels_setting( 'live-editor-quick-link' ) && is_singular() && current_user_can( 'edit_post', get_the_ID() ) && get_post_meta( get_the_ID(), 'panels_data', true ) ) { $admin_bar->add_node( array( 'id' => 'so_live_editor', 'title' => __( 'Live Editor', 'siteorigin-panels' ), 'href' => add_query_arg( 'so_live_editor', 1, get_edit_post_link( get_the_ID() ) ), 'meta' => array( 'class' => 'live-edit-page' ) ) ); add_action( 'wp_enqueue_scripts', array( $this, 'live_edit_link_style' ) ); } return $admin_bar; } function widgets_init(){ register_widget( 'SiteOrigin_Panels_Widgets_PostContent' ); register_widget( 'SiteOrigin_Panels_Widgets_PostLoop' ); register_widget( 'SiteOrigin_Panels_Widgets_Layout' ); } function live_edit_link_style() { if ( is_singular() && current_user_can( 'edit_post', get_the_ID() ) && get_post_meta( get_the_ID(), 'panels_data', true ) ) { // Add the style for the eye icon before the Live Editor link $css = '#wpadminbar #wp-admin-bar-so_live_editor > .ab-item:before { content: "\f177"; top: 2px; }'; wp_add_inline_style( 'siteorigin-panels-front', $css ); } } /** * Process panels data to make sure everything is properly formatted * * @param array $panels_data * * @return array */ function process_panels_data( $panels_data ) { // Process all widgets to make sure that panels_info is properly represented if ( ! empty( $panels_data['widgets'] ) && is_array( $panels_data['widgets'] ) ) { $last_gi = 0; $last_ci = 0; $last_wi = 0; foreach ( $panels_data['widgets'] as &$widget ) { // Transfer legacy content if ( empty( $widget['panels_info'] ) && ! empty( $widget['info'] ) ) { $widget['panels_info'] = $widget['info']; unset( $widget['info'] ); } // Filter the widgets to add indexes if ( $widget['panels_info']['grid'] != $last_gi ) { $last_gi = $widget['panels_info']['grid']; $last_ci = $widget['panels_info']['cell']; $last_wi = 0; } elseif ( $widget['panels_info']['cell'] != $last_ci ) { $last_ci = $widget['panels_info']['cell']; $last_wi = 0; } $widget['panels_info']['cell_index'] = $last_wi ++; } foreach ( $panels_data['grids'] as &$grid ) { if ( ! empty( $grid['style'] ) && is_string( $grid['style'] ) ) { $grid['style'] = array(); } } } return $panels_data; } /** * Fix class names that have been incorrectly escaped * * @param $class * * @return mixed */ public function fix_namespace_escaping( $class ){ return preg_replace( '/\\\\+/', '\\', $class ); } public static function front_css_url(){ return self::renderer()->front_css_url(); } /** * Trigger a siteorigin_panels_version_changed action if the version has changed */ public function version_check(){ $active_version = get_option( 'siteorigin_panels_active_version', false ); if( empty( $active_version ) || $active_version !== SITEORIGIN_PANELS_VERSION ) { do_action( 'siteorigin_panels_version_changed' ); update_option( 'siteorigin_panels_active_version', SITEORIGIN_PANELS_VERSION ); } } /** * Script that removes the siteorigin-panels-before-js class from the body. */ public function strip_before_js(){ ?>widgets[ $class_or_hash ] ) ) { return $wp_widget_factory->widgets[ $class_or_hash ]; } else { foreach ( $wp_widget_factory->widgets as $widget_instance ) { if ( $widget_instance instanceof $class_or_hash ) { return $widget_instance; } } } return null; } /** * Redirect to a welcome page after activation. * * @param $plugin */ public function activation_redirect( $plugin ){ if( $plugin == plugin_basename( __FILE__ ) ) { exit( wp_redirect( admin_url( 'options-general.php?page=siteorigin_panels#welcome' ) ) ); } } } SiteOrigin_Panels::single();