namespace Elementor\Core\Utils;
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly.
use Elementor\Core\Admin\Admin_Notices;
const SUCCESS = 'success';
const WARNING = 'warning';
const DEFINED = 'defined';
const NOT_DEFINED = 'not_defined';
const DISMISSED = 'dismissed';
const CAPABILITY = 'capability';
const PLUGIN_INSTALLED = 'plugin_installed';
const PLUGIN_ACTIVE = 'plugin_active';
const NOT_HAS_OPTION = 'not_has_option';
const INSTALL = 'install';
const ACTIVATE = 'activate';
const CONNECT = 'connect';
const CUSTOMIZE = 'customize';
public static function get_notice_types(): array {
* @return array|string[]|\string[][]
public static function get_hints( $hint_key = null ): array {
'image-optimization-once' => [
self::DISMISSED => 'image-optimization-once',
self::CAPABILITY => 'install_plugins',
self::DEFINED => 'IMAGE_OPTIMIZATION_VERSION',
'image-optimization-once-media-modal' => [
self::DISMISSED => 'image-optimization-once-media-modal',
self::CAPABILITY => 'install_plugins',
self::DEFINED => 'IMAGE_OPTIMIZATION_VERSION',
'image-optimization' => [
self::DISMISSED => 'image_optimizer_hint',
self::CAPABILITY => 'install_plugins',
self::DEFINED => 'IMAGE_OPTIMIZATION_VERSION',
'image-optimization-connect' => [
self::DISMISSED => 'image_optimizer_hint',
self::CAPABILITY => 'manage_options',
self::NOT_DEFINED => 'IMAGE_OPTIMIZATION_VERSION',
self::NOT_HAS_OPTION => 'image_optimizer_access_token',
'image-optimization-media-modal' => [
self::DISMISSED => 'image-optimization-media-modal',
self::CAPABILITY => 'install_plugins',
self::DEFINED => 'IMAGE_OPTIMIZATION_VERSION',
'ally_heading_notice' => [
self::DISMISSED => 'ally_heading_notice',
self::CAPABILITY => 'install_plugins',
self::NOT_HAS_OPTION => 'ea11y_access_token',
return $hints[ $hint_key ] ?? [];
public static function get_notice_icon(): string {
return '<div class="elementor-control-notice-icon">
<svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M2.25 9H3M9 2.25V3M15 9H15.75M4.2 4.2L4.725 4.725M13.8 4.2L13.275 4.725M7.27496 12.75H10.725M6.75 12C6.12035 11.5278 5.65525 10.8694 5.42057 10.1181C5.1859 9.36687 5.19355 8.56082 5.44244 7.81415C5.69133 7.06748 6.16884 6.41804 6.80734 5.95784C7.44583 5.49764 8.21294 5.25 9 5.25C9.78706 5.25 10.5542 5.49764 11.1927 5.95784C11.8312 6.41804 12.3087 7.06748 12.5576 7.81415C12.8065 8.56082 12.8141 9.36687 12.5794 10.1181C12.3448 10.8694 11.8796 11.5278 11.25 12C10.9572 12.2899 10.7367 12.6446 10.6064 13.0355C10.4761 13.4264 10.4397 13.8424 10.5 14.25C10.5 14.6478 10.342 15.0294 10.0607 15.3107C9.77936 15.592 9.39782 15.75 9 15.75C8.60218 15.75 8.22064 15.592 7.93934 15.3107C7.65804 15.0294 7.5 14.6478 7.5 14.25C7.56034 13.8424 7.52389 13.4264 7.3936 13.0355C7.2633 12.6446 7.04282 12.2899 6.75 12Z" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round"/>
* Print or Retrieve the notice template.
* @param bool $should_return
public static function get_notice_template( array $notice, bool $should_return = false ) {
$notice_settings = array_merge( $default_settings, $notice );
if ( empty( $notice_settings['heading'] ) && empty( $notice_settings['content'] ) ) {
if ( ! in_array( $notice_settings['type'], self::get_notice_types(), true ) ) {
$notice_settings['type'] = 'info';
if ( $notice_settings['icon'] ) {
$icon = self::get_notice_icon();
if ( ! empty( $notice_settings['heading'] ) ) {
$heading = '<div class="elementor-control-notice-main-heading">' . $notice_settings['heading'] . '</div>';
if ( ! empty( $notice_settings['content'] ) ) {
$content = '<div class="elementor-control-notice-main-content">' . $notice_settings['content'] . '</div>';
if ( ! empty( $notice_settings['button_text'] ) ) {
$button_settings = ( ! empty( $notice_settings['button_data'] ) ) ? ' data-settings="' . esc_attr( wp_json_encode( $notice_settings['button_data'] ) ) . '"' : '';
$button = '<div class="elementor-control-notice-main-actions">
<button type="button" class="e-btn e-' . $notice_settings['type'] . ' e-btn-1" data-event="' . $notice_settings['button_event'] . '"' . $button_settings . '>
' . $notice_settings['button_text'] . '
if ( $notice_settings['dismissible'] ) {
$dismissible = '<button class="elementor-control-notice-dismiss tooltip-target" data-event="' . $notice_settings['dismissible'] . '" data-tooltip="' . esc_attr__( 'Don’t show again.', 'elementor' ) . '" aria-label="' . esc_attr__( 'Don’t show again.', 'elementor' ) . '">
<i class="eicon eicon-close" aria-hidden="true"></i>
$notice_template = sprintf( '<div class="elementor-control-notice elementor-control-notice-type-%1$s" data-display="%7$s">
<div class="elementor-control-notice-main">
$notice_settings['type'],
$notice_settings['display']
echo wp_kses( $notice_template, self::get_notice_allowed_html() );
public static function get_plugin_install_url( $plugin_slug ): string {
$action = 'install-plugin';
'plugin' => $plugin_slug,
admin_url( 'update.php' )
$action . '_' . $plugin_slug
* Get_plugin_activate_url
public static function get_plugin_activate_url( $plugin_slug ): string {
$path = "$plugin_slug/$plugin_slug.php";
admin_url( 'plugins.php?action=activate&plugin=' . $path ),
'activate-plugin_' . $path
public static function is_dismissed( $key ): bool {
$dismissed = User::get_dismissed_editor_notices();
return in_array( $key, $dismissed, true );
public static function should_display_hint( $hint_key ): bool {
$hint = self::get_hints( $hint_key );
foreach ( $hint as $key => $value ) {
if ( self::is_dismissed( $value ) ) {
if ( ! current_user_can( $value ) ) {
if ( defined( $value ) ) {
if ( ! defined( $value ) ) {
case self::PLUGIN_INSTALLED:
if ( ! self::is_plugin_installed( $value ) ) {
case self::PLUGIN_ACTIVE:
if ( ! self::is_plugin_active( $value ) ) {
case self::NOT_HAS_OPTION:
$option = get_option( $value );
if ( ! empty( $option ) ) {
private static function is_conflict_plugin_installed(): bool {
if ( ! Utils::has_pro() ) {
'optimole-wp/optimole-wp.php',
'ewww-image-optimizer/ewww-image-optimizer.php',
'ewww-image-optimizer-cloud/ewww-image-optimizer-cloud.php',
'kraken-image-optimizer/kraken.php',
'shortpixel-image-optimiser/wp-shortpixel.php',
'wp-smushit/wp-smush.php',
'wp-smush-pro/wp-smush.php',
'tiny-compress-images/tiny-compress-images.php',
foreach ( $conflicting_plugins as $plugin ) {
if ( self::is_plugin_active( $plugin ) ) {
public static function is_plugin_installed( $plugin ): bool {
if ( ! function_exists( 'get_plugins' ) ) {
require_once ABSPATH . 'wp-admin/includes/plugin.php';
$plugins = get_plugins();
$plugin = self::ensure_plugin_folder( $plugin );
return ! empty( $plugins[ $plugin ] );
public static function is_plugin_active( $plugin ): bool {
$plugin = self::ensure_plugin_folder( $plugin );
return is_plugin_active( $plugin );
public static function get_plugin_action_url( $plugin ): string {
if ( ! self::is_plugin_installed( $plugin ) ) {
return self::get_plugin_install_url( $plugin );
if ( ! self::is_plugin_active( $plugin ) ) {
return self::get_plugin_activate_url( $plugin );
private static function ensure_plugin_folder( $plugin ): string {
if ( false === strpos( $plugin, '/' ) ) {
$plugin = $plugin . '/' . $plugin . '.php';
* Get_notice_allowed_html
public static function get_notice_allowed_html(): array {
public static function is_plugin_connected( $option_prefix ): bool {
return ! empty( get_option( $option_prefix . '_access_token' ) );
public static function is_plugin_connected_to_one_subscription(): bool {
$has_one_subscription = Plugin::$instance->experiments->is_feature_active( 'e_editor_one' );
if ( ! $has_one_subscription ) {
return self::is_plugin_connected( 'elementor_one' );
private static function get_all_widget_content( $step, $one_subscription = false ) {
if ( $one_subscription ) {
self::INSTALL => esc_html__( 'Want to create an inclusive experience? Install Ally, included in ONE, and add an accessibility widget to your site.', 'elementor' ),
self::ACTIVATE => esc_html__( 'Your ONE subscription includes Ally. Activate it to place an accessibility widget on your site.', 'elementor' ),
self::CONNECT => esc_html__( "Connect the Ally plugin to your account to access all of it's accessibility features.", 'elementor' ),
self::CUSTOMIZE => esc_html__( "Customize the widget's look, position and the capabilities available for your visitors.", 'elementor' ),
self::INSTALL => esc_html__( 'Install Ally to add an accessibility widget visitors can use to navigate your site.', 'elementor' ),
self::ACTIVATE => esc_html__( 'Activate the Ally plugin to turn its accessibility features on across your site.', 'elementor' ),
self::CONNECT => esc_html__( "Connect the Ally plugin to your account to access all of it's accessibility features.", 'elementor' ),
self::CUSTOMIZE => esc_html__( "Customize the widget's look, position and the capabilities available for your visitors.", 'elementor' ),
private static function get_all_widget_action_button( $step ) {
self::INSTALL => esc_html__( 'Install now', 'elementor' ),
self::ACTIVATE => esc_html__( 'Activate now', 'elementor' ),
self::CONNECT => esc_html__( 'Connect now', 'elementor' ),
self::CUSTOMIZE => esc_html__( 'Customize', 'elementor' ),
private static function get_all_widget_action_url( $step, $one_subscription = false ) {
if ( in_array( $step, [ self::INSTALL, self::ACTIVATE ], true ) ) {
$action = ( self::INSTALL === $step ? 'install' : 'activate' );
if ( $one_subscription ) {
'name' => 'elementor_ea11y_campaign',
'campaign' => 'acc-usability-widget-plg-ally-one-' . $action,
'source' => 'editor-ally-widget-one-' . $action,
'medium' => 'editor-one',