Current File : /home/kelaby89/sergio-cuchi.tattoo/wp-content/plugins/so-widgets-bundle/base/base.php |
<?php
require plugin_dir_path( __FILE__ ) . 'inc/fields/siteorigin-widget-field-class-loader.class.php';
require plugin_dir_path( __FILE__ ) . 'siteorigin-widget.class.php';
require plugin_dir_path( __FILE__ ) . 'inc/widget-manager.class.php';
require plugin_dir_path( __FILE__ ) . 'inc/meta-box-manager.php';
require plugin_dir_path( __FILE__ ) . 'inc/post-selector.php';
require plugin_dir_path( __FILE__ ) . 'inc/string-utils.php';
require plugin_dir_path( __FILE__ ) . 'inc/array-utils.php';
require plugin_dir_path( __FILE__ ) . 'inc/attachments.php';
require plugin_dir_path( __FILE__ ) . 'inc/actions.php';
require plugin_dir_path( __FILE__ ) . 'inc/shortcode.php';
require plugin_dir_path( __FILE__ ) . 'inc/video.php';
require plugin_dir_path( __FILE__ ) . 'inc/routes/sowb-rest-routes.php';
require plugin_dir_path( __FILE__ ) . 'inc/shapes/shapes.php';
// Load the Installer if it's not already active.
if ( is_admin() && ! class_exists( 'SiteOrigin_Installer' ) ) {
include plugin_dir_path( __FILE__ ) . 'inc/installer/siteorigin-installer.php';
}
function siteorigin_widget_add_inline_css( $css ) {
global $siteorigin_widgets_inline_styles;
if ( empty( $siteorigin_widgets_inline_styles ) ) {
$siteorigin_widgets_inline_styles = array();
}
$siteorigin_widgets_inline_styles[] = $css;
}
/**
* Print any inline styles that have been added with siteorigin_widget_add_inline_css
*/
function siteorigin_widget_print_styles() {
global $siteorigin_widgets_inline_styles;
if ( ! empty( $siteorigin_widgets_inline_styles ) ) {
foreach ( $siteorigin_widgets_inline_styles as $widget_css ) {
if ( ! empty( $widget_css ) ) {
?>
<style<?php echo current_theme_supports( 'html5', 'style' ) ? '' : ' type="text/css"'; ?>><?php echo $widget_css; ?></style>
<?php
}
}
}
$siteorigin_widgets_inline_styles = array();
}
add_action( 'wp_head', 'siteorigin_widget_print_styles' );
add_action( 'wp_footer', 'siteorigin_widget_print_styles' );
/**
* The Ajax handler for getting a list of available icons.
*/
function siteorigin_widget_get_icon_list() {
if ( empty( $_REQUEST['_widgets_nonce'] ) || ! wp_verify_nonce( $_REQUEST['_widgets_nonce'], 'widgets_action' ) ) {
wp_die( __( 'Invalid request.', 'so-widgets-bundle' ), 403 );
}
if ( empty( $_GET['family'] ) ) {
wp_die( __( 'Invalid request.', 'so-widgets-bundle' ), 400 );
}
$widget_icon_families = apply_filters( 'siteorigin_widgets_icon_families', array() );
$icons = ! empty( $widget_icon_families[ $_GET['family'] ] ) ? $widget_icon_families[ $_GET['family'] ] : array();
wp_send_json( $icons );
}
add_action( 'wp_ajax_siteorigin_widgets_get_icons', 'siteorigin_widget_get_icon_list' );
/**
* @param bool $icon_styles
* @param string $title
*
* @return bool|string
*/
function siteorigin_widget_get_icon( $icon_value, $icon_styles = false, $title = null ) {
if ( empty( $icon_value ) ) {
return false;
}
static $widget_icon_families;
static $widget_icons_enqueued = array();
if ( empty( $widget_icon_families ) ) {
$widget_icon_families = apply_filters( 'siteorigin_widgets_icon_families', array() );
}
// Get an array of available icon families styles to pass to SiteOrigin_Widget_Field_Icon::get_value_parts()
$icon_families_styles = SiteOrigin_Widget_Field_Icon::get_icon_families_styles( $widget_icon_families );
$value_parts = SiteOrigin_Widget_Field_Icon::get_value_parts( $icon_value, $icon_families_styles );
$family = $value_parts['family'];
$style = empty( $value_parts['style'] ) ? null : $value_parts['style'];
$icon = $value_parts['icon'];
if ( empty( $family ) || ! isset( $icon ) ) {
return false;
}
if ( empty( $widget_icon_families[ $family ] ) ||
empty( $widget_icon_families[ $family ]['icons'][ $icon ] ) ) {
return false;
}
if ( empty( $widget_icons_enqueued[ $family ] ) &&
! empty( $widget_icon_families[ $family ]['style_uri'] ) ) {
if ( ! wp_style_is( 'siteorigin-widget-icon-font-' . $family ) ) {
wp_enqueue_style( 'siteorigin-widget-icon-font-' . $family, $widget_icon_families[ $family ]['style_uri'] );
}
$family_style = 'sow-icon-' . $family . ( empty( $style ) ? '' : ' ' . $style );
$icon_data = $widget_icon_families[ $family ]['icons'][ $icon ];
$unicode = '';
if ( is_array( $icon_data ) && ! empty( $icon_data['unicode'] ) ) {
$unicode = $icon_data['unicode'];
} elseif ( is_string( $icon_data ) ) {
$unicode = $icon_data;
}
return '<span class="' . esc_attr( $family_style ) . '" data-sow-icon="' . $unicode . '"
' . ( is_array( $icon_styles ) ? 'style="' . implode( '; ', $icon_styles ) . '"' : '' ) . ' ' .
( ! empty( $title ) ? 'title="' . esc_attr( $title ) . '"' : '' ) . '
aria-hidden="true"></span>';
} else {
return false;
}
}
/**
* @param $font_value
*
* @return array
*/
$sow_registered_fonts = array();
function siteorigin_widget_get_font( $font_value ) {
$web_safe = array(
'Arial' => 'Arial, Helvetica Neue, Helvetica, sans-serif',
'Courier New' => 'Courier New, Courier, Lucida Sans Typewriter, Lucida Typewriter, monospace',
'Georgia' => 'Georgia, Times, Times New Roman, serif',
'Helvetica Neue' => 'Helvetica Neue, Helvetica, Arial, sans-serif',
'Lucida Grande' => 'Lucida Grande, Lucida Sans Unicode, Lucida Sans, Geneva, Verdana, sans-serif',
'Times New Roman' => 'Times New Roman, Times, Baskerville, Georgia, serif',
'default' => 'default',
);
$font = array();
if ( isset( $web_safe[ $font_value ] ) ) {
$font['family'] = $web_safe[ $font_value ];
} elseif ( siteorigin_widgets_is_google_webfont( $font_value ) ) {
global $sow_registered_fonts;
$font_parts = explode( ':', $font_value );
$font['family'] = sanitize_text_field( $font_parts[0] );
$font_url_param = urlencode( $font_parts[0] );
if ( count( $font_parts ) > 1 ) {
$font['weight'] = $font_parts[1];
$font_url_param .= ':' . $font_parts[1];
$font['weight_raw'] = filter_var( $font['weight'], FILTER_SANITIZE_NUMBER_INT );
$font['style'] = ! is_numeric( $font['weight'] ) || $font['weight'] == 'italic' ? 'italic' : '';
}
$font['url'] = esc_url( apply_filters( 'siteorigin_web_font_url', 'https://fonts.googleapis.com/css' ) . "?family=$font_url_param" );
$style_name = 'sow-google-font-' . strtolower( $font['family'] );
if ( ! empty( $font['weight'] ) ) {
$font_slug = $font['weight_raw'] . ( ! empty( $font['style'] ) ? 'i' : '' );
} else {
// Default to 400 if no weight is set.
$font_slug = 400;
}
$sow_registered_fonts[ $font['family'] ][ $font_slug ] = true;
// Check if WB (or something else has) has already enqueued the font.
if ( ! wp_style_is( $style_name ) ) {
wp_enqueue_style( $style_name, $font['url'] . '&display=swap' );
} elseif ( ! empty( $sow_registered_fonts[ $font['family'] ] ) ) {
// Font already present. Update URL.
global $wp_styles;
global $sow_registered_fonts;
$font_weight_styles = array_keys( $sow_registered_fonts[ $font['family'] ] );
$wp_styles->registered[ $style_name ]->src = esc_url(
apply_filters(
'siteorigin_web_font_url_processed',
apply_filters(
'siteorigin_web_font_url',
'https://fonts.googleapis.com/css'
) . '?family=' . urlencode(
$font['family'] . ':' . implode( ',', $font_weight_styles )
)
)
);
}
} else {
$font['family'] = sanitize_text_field( $font_value );
$font = apply_filters( 'siteorigin_widget_get_custom_font_family', $font );
}
return $font;
}
/**
* Compatibility with Page Builder, add the groups and icons.
*
* @return mixed
*/
function siteorigin_widget_add_bundle_groups( $widgets ) {
foreach ( $widgets as $class => &$widget ) {
if ( preg_match( '/SiteOrigin_Widgets?_(.*)_Widget/i', $class, $matches ) ) {
$widget['icon'] = 'so-widget-icon so-widget-icon-' . strtolower( $matches[1] );
$widget['groups'] = array( 'so-widgets-bundle' );
}
}
return $widgets;
}
add_filter( 'siteorigin_panels_widgets', 'siteorigin_widget_add_bundle_groups', 11 );
/**
* Get the allowed protocols for escaping URLs.
*
* @return array
*/
function sow_get_allowed_esc_url_protocols() {
$protocols = wp_allowed_protocols();
$protocols[] = 'skype';
$protocols[] = 'steam';
return apply_filters( 'siteorigin_esc_url_protocols', $protocols );
}
/**
* Escape a URL
*
* @return string
*/
function sow_esc_url( $url ) {
if ( preg_match( '/^post: *([0-9]+)/', $url, $matches ) ) {
// Convert the special post URL into a permalink
$url = get_the_permalink( (int) $matches[1] );
if ( empty( $url ) ) {
return '';
}
}
$protocols = sow_get_allowed_esc_url_protocols();
return esc_url( $url, $protocols );
}
/**
* A special URL escaping function that handles additional protocols
*
* @return string
*/
function sow_esc_url_raw( $url ) {
if ( preg_match( '/^post: *([0-9]+)/', $url, $matches ) ) {
// Convert the special post URL into a permalink
$url = get_the_permalink( (int) $matches[1] );
}
$protocols = sow_get_allowed_esc_url_protocols();
return esc_url_raw( $url, $protocols );
}
/**
* Escape an HTML attribute
*
* This is a copy of the WP core `esc_attr` function, but modified to allow specifying arguments to the
* `_wp_specialchars` function for a bit more control. This was specifically necessary to allow double-encoding for
* the layout builder field.
*
* @param int $quote_style
* @param bool $charset
* @param bool $double_encode
*
* @return string
*/
function sow_esc_attr( $text, $quote_style = ENT_QUOTES, $charset = false, $double_encode = false ) {
$safe_text = wp_check_invalid_utf8( $text );
$safe_text = _wp_specialchars( $safe_text, $quote_style, $charset, $double_encode );
return apply_filters( 'attribute_escape', $safe_text, $text );
}
/**
* Get all the Google Web Fonts.
*
* @return mixed|void
*/
function siteorigin_widgets_fonts_google_webfonts() {
$fonts = include plugin_dir_path( __FILE__ ) . 'inc/fonts.php';
$fonts = apply_filters( 'siteorigin_widgets_google_webfonts', $fonts );
return ! empty( $fonts ) ? $fonts : array();
}
function siteorigin_widgets_is_google_webfont( $font_value ) {
$google_webfonts = siteorigin_widgets_fonts_google_webfonts();
$font_family = explode( ':', $font_value );
$font_family = $font_family[0];
return isset( $google_webfonts[ $font_family ] );
}
function siteorigin_widgets_font_families() {
// Add the default fonts
$font_families = array(
'Arial' => 'Arial',
'Courier New' => 'Courier New',
'Georgia' => 'Georgia',
'Helvetica Neue' => 'Helvetica Neue',
'Lucida Grande' => 'Lucida Grande',
'Times New Roman' => 'Times New Roman',
);
// Add in all the Google font families
foreach ( siteorigin_widgets_fonts_google_webfonts() as $font => $variants ) {
foreach ( $variants as $variant ) {
if ( $variant == 'regular' || $variant == 400 ) {
$font_families[ $font ] = $font;
} else {
$label_variant = is_numeric( $variant ) || $variant == 'italic' ? $variant : filter_var( $variant, FILTER_SANITIZE_NUMBER_INT ) . ' italic';
$font_families[ $font . ':' . $variant ] = $font . ' (' . $label_variant . ')';
}
}
}
return apply_filters( 'siteorigin_widgets_font_families', $font_families );
}
/**
* Get list of supported measurements
*
* @return array
*/
function siteorigin_widgets_get_measurements_list() {
$measurements = array(
'px', '%', 'in', 'cm', 'mm', 'em', 'rem', 'pt', 'pc', 'ex', 'ch', 'vw', 'vh', 'vmin', 'vmax',
);
// Allow themes and plugins to trim or enhance the list.
return apply_filters( 'siteorigin_widgets_get_measurements_list', $measurements );
}
/**
* Returns the base URL of our widget with `$path` appended.
*
* @param string $path Extra path to append to the end of the URL.
*
* @return string Base URL of the widget, with $path appended.
*/
function siteorigin_widgets_url( $path = '' ) {
return plugins_url( 'so-widgets-bundle/' . $path );
}
/**
* Check if the Page Builder can render.
*
* This method checks if the necessary conditions are met for Page Builder to
* render. It first verifies that Page Builder is active. It then checks
* if either:
* - The request is in the admin area, OR
* - If this is a REST request and the block editor is active.
*
* If none of these conditions are met, Page Builder can't render.
*
* @return bool True if Page Builder builder can render, false otherwise.
*/
function siteorigin_widgets_can_render_builder_field() {
if ( ! defined( 'SITEORIGIN_PANELS_VERSION' ) ) {
return false;
}
if ( is_admin() ) {
return true;
}
// Is this field being rendered inside one of our blocks?
if ( defined( 'REST_REQUEST' ) && function_exists( 'register_block_type' ) ) {
return true;
}
return false;
}
function siteorigin_loading_optimization_attributes( $attr, $widget, $instance, $class ) {
// Allow other plugins to override whether this widget is lazy loaded or not.
if (
! empty(
apply_filters(
'siteorigin_widgets_' . $widget . '_lazy_load',
'lazy',
$instance,
$class
)
)
) {
if ( function_exists( 'wp_get_loading_optimization_attributes' ) ) {
// WP 6.3.
$attr = array_merge(
$attr,
wp_get_loading_optimization_attributes( 'img', $attr, 'wp_get_attachment_image' )
);
} elseif (
function_exists( 'wp_lazy_loading_enabled' ) &&
wp_lazy_loading_enabled( 'img', 'sow-image' )
) {
// < WP 6.3.
$attr['loading'] = function_exists( 'wp_get_loading_attr_default' ) ? wp_get_loading_attr_default( 'the_content' ) : 'lazy';
}
}
return $attr;
}
/**
* The ajax handler for the links field using the the post: ID format without a title set.
*/
function siteorigin_widgets_links_get_title() {
if (
empty( $_REQUEST['_widgets_nonce'] ) ||
! wp_verify_nonce( $_REQUEST['_widgets_nonce'], 'widgets_action' )
) {
wp_die( __( 'Invalid request.', 'so-widgets-bundle' ), 403 );
}
if (
empty( $_GET['postId'] ) ||
! is_numeric( $_GET['postId'] )
) {
wp_die( __( 'Invalid request.', 'so-widgets-bundle' ), 400 );
}
// Don't allow users to link to posts they can't view.
if ( ! current_user_can( 'read_post', $_GET['postId'] ) ) {
wp_die( __( 'Invalid request.', 'so-widgets-bundle' ), 403 );
}
$postTitle = get_the_title( $_GET['postId'] );
echo ! empty( $postTitle ) ? esc_attr( $postTitle ) : esc_html__( '(No Title)', 'so-widgets-bundle' );
die();
}
add_action( 'wp_ajax_so_widgets_links_get_title', 'siteorigin_widgets_links_get_title' );
/**
* Strips escape sequences & HTML entities from a given value.
*
* Accounts for:
* - Unicode escape sequences.
* - Hexadecimal escape sequences.
* - Octal escape sequences.
* - Control characters.
*
* @param string $value The value to strip escape sequences from.
* @param bool $html Optional. Whether to remove HTML entities. Default false.
* @return string The value with escape sequences stripped.
*/
function siteorigin_widgets_strip_escape_sequences( $value, $html = false ) {
// Remove escape sequences.
$value = preg_replace( '/\\\\u[0-9a-fA-F]{4}|\\\\x[0-9a-fA-F]{2}|\\\\[0-7]{3}|[\p{C}&&[^\r\n]]+/u', '', $value );
// HTML entities.
if ( $html ) {
$value = preg_replace( '/&[^;]+;/', '', $value );
}
return $value;
}
/**
* Filters onclick attributes to remove disallowed code.
*
* @param string $onclick The onclick attribute value.
* @param bool $recursive Whether to recursively filter the onclick attribute.
* @return string The filtered onclick attribute value.
*/
function siteorigin_widget_onclick( $onclick = null, $recursive = true ) {
if ( empty( $onclick ) ) {
return;
}
$stripped_onclick = siteorigin_widgets_strip_escape_sequences( $onclick );
if ( $stripped_onclick !== $onclick ) {
// There was some escape sequences removed.
// To play it safe, return nothing.
return;
}
if ( apply_filters( 'siteorigin_widgets_onclick_disallowlist', true ) ) {
// It's possible for allowed functions to contain disallowed functions, so we need to loop through and remove.
$disallowed_functions = array( 'alert', 'eval', 'execScript', 'setTimeout', 'setInterval', 'function', 'document', 'Object', 'window', 'innerHTML', 'outerHTML', 'onload', 'onerror', 'onclick', 'storage', 'fetch', 'XMLHttpRequest', 'jQuery', '$.', 'prototype', '__proto__', 'constructor', 'decode', 'encode', 'atob', 'btoa', 'Promise', 'setImmediate', 'unescape', 'escape', 'captureEvents', 'proxy', 'Reflect', 'Array', 'String', 'Math', 'Date', 'property', 'Properties', 'Error', 'Map', 'Set', 'Generator', 'Web', 'dataview', 'Blob', 'javascript', 'Text', 'Intl', 'JSON', 'RegExp', 'console', 'history', 'location', 'navigator', 'screen', 'worker', 'FinalizationRegistry', 'weak', 'top', 'self', 'parent', 'frame', 'import', 'fragment', 'globalThis', 'frames', 'import', 'this', 'escape', 'watch', 'element', 'file', 'db', 'worker', 'EventSource', 'join', 'upper' );
if ( preg_match( '/\b(' . implode( '|', array_map( 'preg_quote', $disallowed_functions ) ) . ')\b/i', $onclick ) ) {
return;
}
// If string contains 'open' or 'close', check if there's a
// preceding dot. If not, disallow.
if ( preg_match( '/\b(open|close)\b/i', $onclick ) ) {
if ( ! preg_match( '/\.\s*(open|close)\s*\(/i', $onclick ) ) {
return;
}
}
// Case sensitive disallow.
$case_sensitive_disallow = array(
'URL',
);
if ( preg_match( '/\b(' . implode( '|', array_map( 'preg_quote', $case_sensitive_disallow ) ) . ')\b/', $onclick ) ) {
return;
}
}
if ( apply_filters( 'siteorigin_widgets_onclick_allowlist', true ) ) {
// Ensure $onclick ends with a semicolon to prevent syntax errors.
if ( substr( $onclick, -1 ) !== ';' ) {
$onclick .= ';';
}
$onclick_parts = explode( ');', $onclick );
$adjusted_onclick = '';
$allowed_functions = array_flip(
apply_filters(
'siteorigin_widgets_onclick_allowlist_functions',
array(
'_km',
'_paq',
'_qevents',
'_vis_opt',
'amplitude',
'ce',
'chartbeat',
'clarity',
'clicky',
'crazyegg',
'datalayer.push',
'fathom',
'fbq',
'fullstory',
'ga',
'google_optimize',
'gosquared',
'gtag',
'heap',
'hj',
'hubspot',
'Intercom',
'linkedin_data_partner_id',
'logrocket',
'mixpanel',
'mouseflow',
'optimizely',
'parsely',
'pinterest',
'piwik',
'plausible',
's.omtr',
'snaptr',
'statcounter',
'tealium',
'twttr',
'woopra',
'ym',
'ml_account', // MailerLite.
'calendly.initpopupwidget', // Calendly.
'pum.open', // Popup Maker.
)
)
);
// Remove anything not inside of an allowed function.
foreach ( $onclick_parts as $part ) {
$part = trim( $part );
// Allow Buttons to prevent the default action.
if (
$part === 'return false;' ||
$part === 'return;'
) {
$adjusted_onclick .= $part;
continue;
}
$function_name = substr( $part, 0, strpos( $part, '(' ) );
$function_name = strtolower( trim( $function_name ) );
if ( ! isset( $allowed_functions[ $function_name ] ) ) {
// Not an allowed function name, skip this part
continue;
}
$adjusted_onclick .= $part . ');';
}
$onclick = $adjusted_onclick;
}
$onclick = siteorigin_widgets_strip_escape_sequences( $onclick, true );
if ( $recursive ) {
// Keep filtering the $onclick value until it's safe as the script allows.
$current_value = $onclick;
$recursive_value = siteorigin_widget_onclick( $current_value, false );
while ( $current_value !== $recursive_value ) {
$current_value = $recursive_value;
$recursive_value = siteorigin_widget_onclick( $current_value, false );
}
}
return wp_unslash( esc_js( sanitize_text_field( $onclick ) ) );
}
/**
* Ensure the tag is valid before output. If it's not, return the fallback.
*
* @param string $field The field to check in the 'design' array.
* @param string $fallback The fallback value if the field is empty or invalid.
* @param array $valid_tags An array containing valid tags.
* @return string A valid HTML tag for the widget.
*/
function siteorigin_widget_valid_tag( $tag, $fallback = null, $valid_tags = array() ) {
if ( empty( $valid_tags ) || ! is_array( $valid_tags ) ) {
$valid_tags = array( 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p' );
}
if ( ! in_array( $tag, $valid_tags ) ) {
return $fallback;
}
return $tag;
}
/**
* Sanitize HTML attribute names.
*
* This function sanitizes the given attribute name by converting it to lowercase,
* running it through sanitize_key, and then specifically rejecting any
* attribute name that starts with "on".
*
* @param string $attr The attribute name to be sanitized.
* @return string Sanitized attribute name or 'invalid-attribute' if it starts with "on".
*/
function siteorigin_sanitize_attribute_key( $attr ) {
$attr = sanitize_key( strtolower( $attr ) );
// "On" prefixed attributes are too risky to allow.
if (
empty( $attr ) ||
strpos( $attr, 'on' ) === 0
) {
return 'invalid-attribute';
}
return $attr;
}
/**
* Sanitize a JSON string by validating structure and cleaning values.
*
* Decodes JSON, validates the structure, and sanitizes all string values.
* Removes any values with disallowed types. Allowed types are:
* - string.
* - integer.
* - double.
* - boolean.
* - NULL.
*
* @param string $value The JSON string to sanitize.
* @param int $max_depth Maximum recursion depth when decoding.
*
* @return string Sanitized JSON string, or '[]' if invalid.
*/
function siteorigin_sanitize_json( $value, $max_depth = 10 ) {
if ( empty( $value ) ) {
return '[]';
}
// Decode the JSON to ensure it's valid.
$decoded = json_decode( $value, true, $max_depth, JSON_BIGINT_AS_STRING );
if (
$decoded === null ||
! is_array( $decoded ) ||
json_last_error() !== JSON_ERROR_NONE
) {
return '[]';
}
$allowed_types = array(
'string',
'integer',
'double',
'boolean',
'NULL',
);
// Sanitize the JSON values.
array_walk_recursive(
$decoded,
function ( &$item ) use ( $allowed_types ) {
if ( ! in_array( gettype( $item ), $allowed_types, true ) ) {
$item = '';
return;
}
if ( is_string( $item ) ) {
$item = sanitize_text_field( $item );
}
}
);
$json = wp_json_encode( $decoded, JSON_UNESCAPED_SLASHES );
return $json ? $json : '[]';
}