'ElementorPro\Modules\QueryControl\Module', 'ElementorPro\Modules\PanelPostsControl\Controls\Group_Control_Posts' => 'ElementorPro\Modules\QueryControl\Controls\Group_Control_Posts', 'ElementorPro\Modules\PanelPostsControl\Controls\Query' => 'ElementorPro\Modules\QueryControl\Controls\Query', ]; /** * @var \ElementorPro\License\Updater */ public $updater; /** * @var PHP_Api */ public $php_api; /** * Throw error on object clone * * The whole idea of the singleton design pattern is that there is a single * object therefore, we don't want the object to be cloned. * * @since 1.0.0 * @return void */ public function __clone() { _doing_it_wrong( __FUNCTION__, sprintf( 'Cloning instances of the singleton "%s" class is forbidden.', get_class( $this ) ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped '1.0.0' ); } /** * Disable unserializing of the class * * @since 1.0.0 * @return void */ public function __wakeup() { _doing_it_wrong( __FUNCTION__, sprintf( 'Unserializing instances of the singleton "%s" class is forbidden.', get_class( $this ) ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped '1.0.0' ); } /** * @return \Elementor\Plugin */ public static function elementor() { return \Elementor\Plugin::$instance; } /** * @return Plugin */ public static function instance() { if ( is_null( self::$_instance ) ) { self::$_instance = new self(); } return self::$_instance; } public function autoload( $class ) { if ( 0 !== strpos( $class, __NAMESPACE__ ) ) { return; } $has_class_alias = isset( $this->classes_aliases[ $class ] ); // Backward Compatibility: Save old class name for set an alias after the new class is loaded if ( $has_class_alias ) { $class_alias_name = $this->classes_aliases[ $class ]; $class_to_load = $class_alias_name; } else { $class_to_load = $class; } if ( ! class_exists( $class_to_load ) ) { $filename = strtolower( preg_replace( [ '/^' . __NAMESPACE__ . '\\\/', '/([a-z])([A-Z])/', '/_/', '/\\\/' ], [ '', '$1-$2', '-', DIRECTORY_SEPARATOR ], $class_to_load ) ); $filename = ELEMENTOR_PRO_PATH . $filename . '.php'; if ( is_readable( $filename ) ) { include( $filename ); } } if ( $has_class_alias ) { class_alias( $class_alias_name, $class ); } } public function enqueue_styles() { $suffix = $this->get_assets_suffix(); $direction_suffix = is_rtl() ? '-rtl' : ''; $frontend_file_name_base = $this->is_optimized_css_mode() ? 'frontend-lite' : 'frontend'; $frontend_file_name = $frontend_file_name_base . $direction_suffix . $suffix . '.css'; $has_custom_file = self::elementor()->breakpoints->has_custom_breakpoints(); $frontend_file_url = $this->get_frontend_file_url( $frontend_file_name, $has_custom_file ); wp_enqueue_style( 'elementor-pro', $frontend_file_url, [], $has_custom_file ? null : ELEMENTOR_PRO_VERSION ); } public function get_frontend_file_url( $frontend_file_name, $custom_file ) { if ( $custom_file ) { $frontend_file = $this->get_frontend_file( $frontend_file_name ); $frontend_file_url = $frontend_file->get_url(); } else { $frontend_file_url = ELEMENTOR_PRO_ASSETS_URL . 'css/' . $frontend_file_name; } return $frontend_file_url; } public function get_frontend_file_path( $frontend_file_name, $custom_file ) { if ( $custom_file ) { $frontend_file = $this->get_frontend_file( $frontend_file_name ); $frontend_file_path = $frontend_file->get_path(); } else { $frontend_file_path = ELEMENTOR_PRO_ASSETS_PATH . 'css/' . $frontend_file_name; } return $frontend_file_path; } public function enqueue_frontend_scripts() { $suffix = $this->get_assets_suffix(); wp_enqueue_script( 'elementor-pro-frontend', ELEMENTOR_PRO_URL . 'assets/js/frontend' . $suffix . '.js', $this->get_frontend_depends(), ELEMENTOR_PRO_VERSION, true ); wp_set_script_translations( 'elementor-pro-frontend', 'elementor-pro', ELEMENTOR_PRO_PATH . 'languages' ); wp_enqueue_script( 'pro-elements-handlers' ); $assets_url = ELEMENTOR_PRO_ASSETS_URL; /** * Elementor Pro assets URL. * * Filters the assets URL used by Elementor Pro. * * By default Elementor Pro assets URL is set by the ELEMENTOR_PRO_ASSETS_URL * constant. This hook allows developers to change this URL. * * @param string $assets_url Elementor Pro assets URL. */ $assets_url = apply_filters( 'elementor_pro/frontend/assets_url', $assets_url ); $locale_settings = [ 'ajaxurl' => admin_url( 'admin-ajax.php' ), 'nonce' => wp_create_nonce( 'elementor-pro-frontend' ), 'urls' => [ 'assets' => $assets_url, 'rest' => get_rest_url(), ], ]; /** * Localized frontend settings. * * Filters the localized settings used in the frontend as JavaScript variables. * * By default Elementor Pro passes some frontend settings to be consumed as JavaScript * variables. This hook allows developers to add extra settings values to be consumed * using JavaScript in the frontend. * * @since 1.0.0 * * @param array $locale_settings Localized frontend settings. */ $locale_settings = apply_filters( 'elementor_pro/frontend/localize_settings', $locale_settings ); Utils::print_js_config( 'elementor-pro-frontend', 'ElementorProFrontendConfig', $locale_settings ); if ( $this->is_assets_loader_exist() ) { $this->register_assets(); } } public function register_frontend_scripts() { $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; wp_register_script( 'elementor-pro-webpack-runtime', ELEMENTOR_PRO_URL . 'assets/js/webpack-pro.runtime' . $suffix . '.js', [], ELEMENTOR_PRO_VERSION, true ); wp_register_script( 'pro-elements-handlers', ELEMENTOR_PRO_URL . 'assets/js/elements-handlers' . $suffix . '.js', [ 'elementor-frontend', ], ELEMENTOR_PRO_VERSION, true ); wp_register_script( 'smartmenus', ELEMENTOR_PRO_URL . 'assets/lib/smartmenus/jquery.smartmenus' . $suffix . '.js', [ 'jquery', ], '1.2.1', true ); if ( ! $this->is_assets_loader_exist() ) { wp_register_script( 'elementor-sticky', ELEMENTOR_PRO_URL . 'assets/lib/sticky/jquery.sticky' . $suffix . '.js', [ 'jquery', ], ELEMENTOR_PRO_VERSION, true ); } } public function register_preview_scripts() { $suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; wp_enqueue_script( 'elementor-pro-preview', ELEMENTOR_PRO_URL . 'assets/js/preview' . $suffix . '.js', [ 'wp-i18n', 'elementor-frontend', ], ELEMENTOR_PRO_VERSION, true ); } public function get_responsive_stylesheet_templates( $templates ) { $templates_paths = glob( $this->get_responsive_templates_path() . '*.css' ); foreach ( $templates_paths as $template_path ) { $file_name = 'custom-pro-' . basename( $template_path ); $templates[ $file_name ] = $template_path; } return $templates; } public function on_elementor_init() { $this->modules_manager = new Modules_Manager(); /** TODO: BC for Elementor v2.4.0 */ if ( class_exists( '\Elementor\Core\Upgrade\Manager' ) ) { $this->upgrade = UpgradeManager::instance(); } /** * Elementor Pro init. * * Fires on Elementor Pro initiation, after Elementor has finished loading * but before any headers are sent. * * @since 1.0.0 */ do_action( 'elementor_pro/init' ); } /** * @param \Elementor\Core\Base\Document $document */ public function on_document_save_version( $document ) { $document->update_meta( '_elementor_pro_version', ELEMENTOR_PRO_VERSION ); } private function get_frontend_depends() { $frontend_depends = [ 'elementor-pro-webpack-runtime', 'elementor-frontend-modules', ]; if ( ! $this->is_assets_loader_exist() ) { $frontend_depends[] = 'elementor-sticky'; } return $frontend_depends; } private function get_responsive_templates_path() { return ELEMENTOR_PRO_ASSETS_PATH . 'css/templates/'; } private function add_subscription_template_access_level_to_settings( $settings ) { // Core >= 3.2.0 if ( isset( $settings['library_connect']['current_access_level'] ) ) { $settings['library_connect']['current_access_level'] = API::get_library_access_level(); } // Core >= 3.18.0 if ( isset( $settings['library_connect']['current_access_tier'] ) ) { $settings['library_connect']['current_access_tier'] = API::get_access_tier(); } return $settings; } private function setup_hooks() { add_action( 'elementor/init', [ $this, 'on_elementor_init' ] ); add_action( 'elementor/frontend/before_register_scripts', [ $this, 'register_frontend_scripts' ] ); add_action( 'elementor/preview/enqueue_scripts', [ $this, 'register_preview_scripts' ] ); add_action( 'elementor/frontend/before_enqueue_scripts', [ $this, 'enqueue_frontend_scripts' ] ); add_action( 'elementor/frontend/after_enqueue_styles', [ $this, 'enqueue_styles' ] ); add_filter( 'elementor/core/breakpoints/get_stylesheet_template', [ $this, 'get_responsive_stylesheet_templates' ] ); add_action( 'elementor/document/save_version', [ $this, 'on_document_save_version' ] ); add_filter( 'elementor/editor/localize_settings', function ( $settings ) { return $this->add_subscription_template_access_level_to_settings( $settings ); }, 11 /** After Elementor Core (Library) */ ); add_filter( 'elementor/common/localize_settings', function ( $settings ) { return $this->add_subscription_template_access_level_to_settings( $settings ); }, 11 /** After Elementor Core (Library) */ ); } private function is_optimized_css_mode() { $is_optimized_css_loading = self::elementor()->experiments->is_feature_active( 'e_optimized_css_loading' ); return ! Utils::is_script_debug() && $is_optimized_css_loading && ! self::elementor()->preview->is_preview_mode(); } private function get_assets() { $suffix = $this->get_assets_suffix(); return [ 'scripts' => [ 'e-sticky' => [ 'src' => ELEMENTOR_PRO_URL . 'assets/lib/sticky/jquery.sticky' . $suffix . '.js', 'version' => ELEMENTOR_PRO_VERSION, 'dependencies' => [ 'jquery', ], ], ], ]; } private function register_assets() { $assets = $this->get_assets(); if ( $assets ) { self::elementor()->assets_loader->add_assets( $assets ); } } private function is_assets_loader_exist() { return ! ! self::elementor()->assets_loader; } /** * Plugin constructor. */ private function __construct() { spl_autoload_register( [ $this, 'autoload' ] ); Compatibility::register_actions(); new Connect\Manager(); $this->setup_hooks(); $this->editor = new Editor(); $this->preview = new Preview(); $this->app = new App(); $this->license_admin = new License\Admin(); $this->php_api = new PHP_Api(); if ( is_user_logged_in() ) { $this->integrations = new Integrations_Manager(); // TODO: This one is safe to move out of the condition. $this->notifications = new Notifications_Manager(); } if ( is_admin() ) { $this->admin = new Admin(); $this->license_admin->register_actions(); } // The `Updater` class is responsible for adding some updates related filters, including auto updates, and since // WP crons don't run on admin mode, it should not depend on it. $this->updater = new Updater(); } private function get_assets_suffix() { return defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min'; } private function get_frontend_file( $frontend_file_name ) { $template_file_path = self::get_responsive_templates_path() . $frontend_file_name; return self::elementor()->frontend->get_frontend_file( $frontend_file_name, 'custom-pro-', $template_file_path ); } final public static function get_title() { return esc_html__( 'Elementor Pro', 'elementor-pro' ); } } if ( ! defined( 'ELEMENTOR_PRO_TESTS' ) ) { // In tests we run the instance manually. Plugin::instance(); }