use Elementor\Core\Files\Fonts\Google_Font;
use Elementor\Core\Utils\Collection;
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly.
* Elementor utils handler class is responsible for different utility methods
const DEPRECATION_RANGE = 0.4;
const EDITOR_BREAK_LINES_OPTION_KEY = 'elementor_editor_break_lines';
* A list of safe tags for `validate_html_tag` method.
const ALLOWED_HTML_WRAPPER_TAGS = [
const EXTENDED_ALLOWED_HTML_TAGS = [
'allowfullscreen' => true,
'referrerpolicy' => true,
'aria-labelledby' => true,
* Variables for free to pro upsale modal promotions
const ANIMATED_HEADLINE = 'animated_headline';
const VIDEO_PLAYLIST = 'video_playlist';
const TESTIMONIAL_WIDGET = 'testimonial_widget';
const IMAGE_CAROUSEL = 'image_carousel';
* Whether WordPress CLI mode is enabled or not.
public static function is_wp_cli() {
return defined( 'WP_CLI' ) && WP_CLI;
* Whether script debug is enabled or not.
public static function is_script_debug() {
return defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG;
* Whether Elementor debug is enabled or not.
public static function is_elementor_debug() {
return defined( 'ELEMENTOR_DEBUG' ) && ELEMENTOR_DEBUG;
* Whether Elementor test mode is enabled or not.
public static function is_elementor_tests() {
return defined( 'ELEMENTOR_TESTS' ) && ELEMENTOR_TESTS;
* Retrieve the link to Elementor Pro.
* @param string $link URL to Elementor pro.
* @return string Elementor pro link.
public static function get_pro_link( $link ) {
static $theme_name = false;
$theme_obj = wp_get_theme();
if ( $theme_obj->parent() ) {
$theme_name = $theme_obj->parent()->get( 'Name' );
$theme_name = $theme_obj->get( 'Name' );
$theme_name = sanitize_key( $theme_name );
$link = add_query_arg( 'utm_term', $theme_name, $link );
* Replace old URLs to new URLs. This method also updates all the Elementor data.
* @throws \Exception Replace URL exception.
public static function replace_urls( $from, $to ) {
throw new \Exception( 'Couldn’t replace your address because the old URL was not provided. Try again by entering the old URL.' );
throw new \Exception( 'Couldn’t replace your address because the new URL was not provided. Try again by entering the new URL.' );
throw new \Exception( 'Couldn’t replace your address because both of the URLs provided are identical. Try again by entering different URLs.' );
$is_valid_urls = ( filter_var( $from, FILTER_VALIDATE_URL ) && filter_var( $to, FILTER_VALIDATE_URL ) );
if ( ! $is_valid_urls ) {
throw new \Exception( 'Couldn’t replace your address because at least one of the URLs provided are invalid. Try again by entering valid URLs.' );
$escaped_from = str_replace( '/', '\\/', $from );
$escaped_to = str_replace( '/', '\\/', $to );
$meta_value_like = '[%'; // meta_value LIKE '[%' are json formatted
$rows_affected = $wpdb->query(
"UPDATE {$wpdb->postmeta} " .
'SET `meta_value` = REPLACE(`meta_value`, %s, %s) ' .
"WHERE `meta_key` = '_elementor_data' AND `meta_value` LIKE %s;",
if ( false === $rows_affected ) {
throw new \Exception( 'An error occurred while replacing URL\'s.' );
// Allow externals to replace-urls, when they have to.
$rows_affected += (int) apply_filters( 'elementor/tools/replace-urls', 0, $from, $to );
Plugin::$instance->files_manager->clear_cache();
Google_Font::clear_cache();
/* translators: %d: Number of rows. */
_n( '%d database row affected.', '%d database rows affected.', $rows_affected, 'elementor' ),
* Is post supports Elementor.
* Whether the post supports editing with Elementor.
* @param int $post_id Optional. Post ID. Default is `0`.
* @return string True if post supports editing with Elementor, false otherwise.
public static function is_post_support( $post_id = 0 ) {
$post_type = get_post_type( $post_id );
$is_supported = self::is_post_type_support( $post_type );
* Filters whether the post type supports editing with Elementor.
* @deprecated 2.2.0 Use `elementor/utils/is_post_support` hook Instead.
* @param bool $is_supported Whether the post type supports editing with Elementor.
* @param int $post_id Post ID.
* @param string $post_type Post type.
$is_supported = apply_filters( 'elementor/utils/is_post_type_support', $is_supported, $post_id, $post_type );
* Filters whether the post supports editing with Elementor.
* @param bool $is_supported Whether the post type supports editing with Elementor.
* @param int $post_id Post ID.
* @param string $post_type Post type.
$is_supported = apply_filters( 'elementor/utils/is_post_support', $is_supported, $post_id, $post_type );
* Is post type supports Elementor.
* Whether the post type supports editing with Elementor.
* @param string $post_type Post Type.
* @return string True if post type supports editing with Elementor, false otherwise.
public static function is_post_type_support( $post_type ) {
if ( ! post_type_exists( $post_type ) ) {
if ( ! post_type_supports( $post_type, 'elementor' ) ) {
* Get placeholder image source.
* Retrieve the source of the placeholder image.
* @return string The source of the default placeholder image used by Elementor.
public static function get_placeholder_image_src() {
$placeholder_image = ELEMENTOR_ASSETS_URL . 'images/placeholder.png';
* Get placeholder image source.
* Filters the source of the default placeholder image used by Elementor.
* @param string $placeholder_image The source of the default placeholder image.
$placeholder_image = apply_filters( 'elementor/utils/get_placeholder_image_src', $placeholder_image );
return $placeholder_image;
* Generate random string.
* Returns a string containing a hexadecimal representation of random number.
* @return string Random string.
public static function generate_random_string() {
* Tell WordPress cache plugins not to cache this request.
public static function do_not_cache() {
if ( ! defined( 'DONOTCACHEPAGE' ) ) {
define( 'DONOTCACHEPAGE', true );
if ( ! defined( 'DONOTCACHEDB' ) ) {
define( 'DONOTCACHEDB', true );
if ( ! defined( 'DONOTMINIFY' ) ) {
define( 'DONOTMINIFY', true );
if ( ! defined( 'DONOTCDN' ) ) {
define( 'DONOTCDN', true );
if ( ! defined( 'DONOTCACHEOBJECT' ) ) {
define( 'DONOTCACHEOBJECT', true );
// Set the headers to prevent caching for the different browsers.
* Retrieve timezone string from the WordPress database.
* @return string Timezone string.
public static function get_timezone_string() {
$current_offset = (float) get_option( 'gmt_offset' );
$timezone_string = get_option( 'timezone_string' );
// Create a UTC+- zone if no timezone string exists.
if ( empty( $timezone_string ) ) {
if ( $current_offset < 0 ) {
$timezone_string = 'UTC' . $current_offset;
$timezone_string = 'UTC+' . $current_offset;
* Get create new post URL.
* Retrieve a custom URL for creating a new post/page using Elementor.
* @deprecated 3.3.0 Use `Plugin::$instance->documents->get_create_new_post_url()` instead.
* @param string $post_type Optional. Post type slug. Default is 'page'.
* @param string|null $template_type Optional. Query arg 'template_type'. Default is null.
* @return string A URL for creating new post using Elementor.
public static function get_create_new_post_url( $post_type = 'page', $template_type = null ) {
Plugin::$instance->modules_manager->get_modules( 'dev-tools' )->deprecation->deprecated_function( __FUNCTION__, '3.3.0', 'Plugin::$instance->documents->get_create_new_post_url()' );
return Plugin::$instance->documents->get_create_new_post_url( $post_type, $template_type );
* Retrieve an autosave for any given post.
* @param int $post_id Post ID.
* @param int $user_id Optional. User ID. Default is `0`.
* @return \WP_Post|false Post autosave or false.
public static function get_post_autosave( $post_id, $user_id = 0 ) {
$post = get_post( $post_id );
$where = $wpdb->prepare( 'post_parent = %d AND post_name LIKE %s AND post_modified_gmt > %s', [ $post_id, "{$post_id}-autosave%", $post->post_modified_gmt ] );
$where .= $wpdb->prepare( ' AND post_author = %d', $user_id );
$revision = $wpdb->get_row( "SELECT * FROM $wpdb->posts WHERE $where AND post_type = 'revision'" ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared
$revision = new \WP_Post( $revision );
* Is CPT supports custom templates.
* Whether the Custom Post Type supports templates.