Uname: Linux premium294.web-hosting.com 4.18.0-553.45.1.lve.el8.x86_64 #1 SMP Wed Mar 26 12:08:09 UTC 2025 x86_64
Software: LiteSpeed
PHP version: 8.1.32 [ PHP INFO ] PHP os: Linux
Server Ip: 104.21.112.1
Your Ip: 216.73.216.223
User: mjbynoyq (1574) | Group: mjbynoyq (1570)
Safe Mode: OFF
Disable Function:
NONE

name : class-responsive-images.php
<?php
if( ! defined( 'AVIA_FW' ) ) {  exit;  }    // Exit if accessed directly

/**
 * Provides support for responsive images.
 *
 * As WP already offers support for this feature this class mainly provides wrappers to be used
 * to prepare html code for WP to be able to add needed attributes.
 *
 * With WP 5.5 loading="lazy" attribute was added. This is also supported in this class.
 *
 * @since 4.7.5.1
 * @added_by Günter
 */

if( ! class_exists( 'av_responsive_images' ) )
{

	class av_responsive_images extends aviaFramework\base\object_properties
	{

		/**
		 * Holds the instance of this class
		 *
		 * @since 4.7.5.1
		 * @var av_responsive_images
		 */
		static private $_instance = null;

		/**
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $config;

		/**
		 * Default WP thumbnails, can be changed with filter
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $wp_default_names;

		/**
		 * Stores all WP default image sizes (original sizes, no responsive scaled down)
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $wp_default_images;

		/**
		 * Stores all theme image sizes (original sizes, no responsive scaled down)
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $theme_images;

		/**
		 * Stores all image sizes defined by plugins (original sizes, no responsive scaled down)
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $plugin_images;

		/**
		 * For performance this are the merged basic image sized (WP, theme, plugin)
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $base_images;

		/**
		 * Array of image sizes and their human readable string
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $readable_img_sizes;

		/**
		 * Array of image sizes grouped by aspect ratio
		 *
		 * @since 4.7.5.1
		 * @var array
		 */
		protected $size_groups;

		/**
		 * Holds an array of id's of images that must not get the loading="lazy" attribute
		 *
		 * @since 4.7.6.2
		 * @var array
		 */
		protected $no_lazy_loading_ids;

		/**
		 * Option key for relationship attachment URL to attachment ID
		 *
		 * @since 4.8.4
		 * @var string
		 */
		protected $opt_key_attachment_urls;

		/**
		 * ALB elements can temporary disable usage of responsive images and override theme option setting (only when enabled)
		 *
		 * @since 4.8.6.3
		 * @var boolean
		 */
		protected $temporary_disabled;

		/**
		 * Return the instance of this class
		 *
		 * @since 4.7.5.1
		 * @param array|null $config
		 * @return av_responsive_images
		 */
		static public function instance( $config = array() )
		{
			if( is_null( av_responsive_images::$_instance ) )
			{
				av_responsive_images::$_instance = new av_responsive_images( $config );
			}

			return av_responsive_images::$_instance;
		}

		/**
		 * @since 4.7.5.1
		 * @param array|null $config
		 */
		protected function __construct( $config = array() )
		{
			global $avia;

			$this->config = apply_filters( 'avf_responsive_images_defaults', array(
						'default_jpeg_quality'	=> 100,			//	used by WP filter
						'theme_images'			=> array(),
						'readableImgSizes'		=> array(),
						'no_lazy_loading_ids'	=> array()
					) );

			$this->reinit( $config );

			/**
			 * @since 4.7.5.1
			 * @param array
			 * @return array
			 */
			$this->wp_default_names = apply_filters( 'avf_wp_default_thumbnail_names', array( 'thumb', 'thumbnail', 'medium', 'medium_large', 'large', 'post-thumbnail', '1536x1536', '2048x2048' ) );

			$this->wp_default_images = array();
			$this->theme_images = array();
			$this->plugin_images = array();
			$this->base_images = array();
			$this->readable_img_sizes = array();
			$this->size_groups = array();
			$this->no_lazy_loading_ids = array();
			$this->opt_key_attachment_urls = avia_backend_safe_string( $avia->base_data['prefix'] ) . '-attachment_urls';
			$this->temporary_disabled = false;

			add_filter( 'avf_modify_thumb_size',  array( $this, 'handler_avf_modify_thumb_size'), 10, 1 );
			add_filter( 'avf_modify_readable_image_sizes',  array( $this, 'handler_avf_modify_readable_image_sizes'), 10, 2 );

			add_action( 'init', array( $this, 'handler_wp_init'), 999999 );
			add_filter( 'body_class', array( $this, 'handler_body_class' ), 10, 2 );

			add_filter( 'jpeg_quality', array( $this, 'handler_wp_jpeg_quality' ), 99, 2 );
			add_filter( 'wp_editor_set_quality', array( $this, 'handler_wp_editor_set_quality'), 99, 2 );

			add_filter( 'post_thumbnail_html', array( $this, 'handler_wp_post_thumbnail_html' ), 10, 5 );
			add_filter( 'wp_get_attachment_image_attributes', array( $this, 'handler_wp_get_attachment_image_attributes' ), 99, 3 );

			add_filter( 'wp_lazy_loading_enabled', array( $this, 'handler_wp_lazy_loading_enabled' ), 99, 3 );
			add_filter( 'wp_img_tag_add_loading_attr', array( $this, 'handler_wp_img_tag_add_loading_attr' ), 99, 3 );

			//	reset postmeta and delete all css files
			add_action( 'ava_after_theme_update', array( $this, 'handler_ava_reset_db_options' ), 100, 1 );
			add_action( 'ava_after_import_demo_settings', array( $this, 'handler_ava_reset_db_options'), 100, 1 );
			add_action( 'avia_ajax_after_save_options_page', array( $this, 'handler_ava_reset_db_options'), 100, 1 );
		}

		/**
		 * @since 4.7.5.1
		 */
		public function __destruct()
		{
			unset( $this->config );
			unset( $this->wp_default_names );
			unset( $this->wp_default_images );
			unset( $this->theme_images );
			unset( $this->plugin_images );
			unset( $this->base_images );
			unset( $this->readable_img_sizes );
			unset( $this->size_groups );
			unset( $this->no_lazy_loading_ids );
		}

		/**
		 * Allows a reinitialisation of config array
		 *
		 * @since 4.7.5.1
		 * @param array $config
		 */
		public function reinit( array $config )
		{
			if( empty( $config ) )
			{
				return;
			}

			$default_jpeg_quality = isset( $config['default_jpeg_quality'] ) ? $config['default_jpeg_quality'] : $this->config['default_jpeg_quality'];

			$this->config = array_merge_recursive( $this->config, $config );

			//	array_merge_recursive creates array !!!
			$this->config['default_jpeg_quality'] = $default_jpeg_quality;
		}


		/**
		 * Add additional image sizes (prepared in case we allow to add custom image sizes by theme)
		 *
		 * @since 4.7.5.1
		 * @param array $imgSizes
		 * @return array
		 */
		public function handler_avf_modify_thumb_size( array $imgSizes )
		{
			return $imgSizes;
		}

		/**
		 * Add additional human readable image sizes (prepared in case we allow to add custom image sizes by theme)
		 *
		 * @since 4.7.5.1
		 * @param array $readableImgSizes
		 * @param array $imgSizes
		 * @return array
		 */
		public function handler_avf_modify_readable_image_sizes( array $readableImgSizes, array $imgSizes )
		{
			return $readableImgSizes;
		}

		/**
		 * Loads all defined image sizes and inits local variables
		 * All plugins must have registered their images before
		 *
		 * @since 4.7.5.1
		 */
		public function handler_wp_init()
		{
			global $_wp_additional_image_sizes;

			/**
			 * @since 4.7.5.1
			 * @param array
			 * @return array
			 */
			$this->no_lazy_loading_ids = (array) apply_filters( 'avf_init_no_lazy_loading_ids', $this->config['no_lazy_loading_ids'] );


			if( ! is_admin() )
			{
				return;
			}

			$this->theme_images = $this->config['theme_images'];

			foreach( get_intermediate_image_sizes() as $_size )
			{
				$img_size = array();

				if( in_array( $_size, $this->wp_default_names ) )
				{
					if( isset( $_wp_additional_image_sizes[ $_size ] ) )
					{
						$img_size['width'] = $_wp_additional_image_sizes[ $_size ]['width'];
						$img_size['height'] = $_wp_additional_image_sizes[ $_size ]['height'];
						$img_size['crop'] = (bool) $_wp_additional_image_sizes[ $_size ]['crop'];
					}
					else
					{
						$img_size['width'] = get_option( "{$_size}_size_w", 0 );
						$img_size['height'] = get_option( "{$_size}_size_h", 0 );
						$img_size['crop'] = (bool) get_option( "{$_size}_crop", false );
					}

					$this->wp_default_images[ $_size ] = $img_size;
				}
				else if( array_key_exists( $_size, $this->theme_images ) )
				{
					$this->theme_images[ $_size ]['width'] = is_numeric( $this->theme_images[ $_size ]['width'] ) ? (int) $this->theme_images[ $_size ]['width'] : 0;
					$this->theme_images[ $_size ]['height'] = is_numeric( $this->theme_images[ $_size ]['height'] ) ? (int) $this->theme_images[ $_size ]['height'] : 0;
					$this->theme_images[ $_size ]['crop'] = empty( $this->theme_images[ $_size ]['crop'] ) ? false : true;
				}
				else if ( isset( $_wp_additional_image_sizes[ $_size ] ) )
				{
					$img_size['width'] = $_wp_additional_image_sizes[ $_size ]['width'];
					$img_size['height'] = $_wp_additional_image_sizes[ $_size ]['height'];
					$img_size['crop'] = (bool) $_wp_additional_image_sizes[ $_size ]['crop'];
					$this->plugin_images[ $_size ] = $img_size;
				}
			}

			$this->base_images = array_merge( $this->wp_default_images, $this->theme_images, $this->plugin_images );

			/**
			 * Allows to translate WP and plugin thumbnail names to human readable
			 *
			 * @since 4.7.5.1
			 * @param array $this->config['readableImgSizes']
			 * @param array $this->base_images
			 * @param av_responsive_images $this
			 * @return array
			 */
			$this->readable_img_sizes = apply_filters( 'avf_resp_images_readable_sizes', $this->config['readableImgSizes'], $this->base_images, $this );

			$this->group_image_sizes();

			/**
			 *
			 * @since 4.7.5.1
			 * @param array $this->size_groups
			 * @param av_responsive_images $this
			 */
			do_action( 'ava_responsive_image_sizes_grouped', $this->size_groups, $this );
		}

		/**
		 * Add extra classes
		 *
		 * @since 4.8.2
		 * @param array $classes
		 * @param array $class
		 * @return string
		 */
		public function handler_body_class( array $classes, array $class )
		{
			if( ! $this->responsive_images_active() )
			{
				return $classes;
			}

			$classes[] = 'avia-responsive-images-support';

			if( $this->responsive_images_lightbox_active() )
			{
				$classes[] = 'responsive-images-lightbox-support';
			}

			return $classes;
		}

		/**
		 * Prepares images for WP to recognise for scrset and sizes and for lazy loading attr.
		 *
		 * @since 4.7.5.1
		 * @param string $html
		 * @param int $post_id
		 * @param int $post_thumbnail_id
		 * @param string|array $size
		 * @param string $attr
		 * @return string
		 */
		public function handler_wp_post_thumbnail_html( $html, $post_id, $post_thumbnail_id, $size, $attr )
		{
			$lazy_loading = $this->is_attachment_id_not_lazy_loading( $post_thumbnail_id ) ? 'disabled' : 'enabled';

			return $this->prepare_single_image( $html, $post_thumbnail_id, $lazy_loading );
		}

		/**
		 * Checks for theme responsive image setting and removes the attributes if necessary.
		 * Also removes default "loading" attribute for a specific attachment depending
		 * on theme or global settings.
		 *
		 * @since 4.7.5.1
		 * @param array $attr
		 * @param WP_Post $attachment
		 * @param string|array $size
		 * @return array
		 */
		public function handler_wp_get_attachment_image_attributes( $attr, $attachment, $size )
		{
			global $avia_config;

			if( ! $this->responsive_images_active() )
			{
				unset( $attr['srcset'] );
				unset( $attr['sizes'] );
			}

			if( ! array_key_exists( 'loading', $attr ) )
			{
				return $attr;
			}

			if( isset( $avia_config['alb_html_lazy_loading'] ) && ( $avia_config['alb_html_lazy_loading'] != 'enabled' ) )
			{
				$this->add_attachment_id_to_not_lazy_loading( $attachment->ID );
			}

			if( $this->is_attachment_id_not_lazy_loading( $attachment->ID ) )
			{
				unset( $attr['loading'] );

				$class = 'avia-img-lazy-loading-not-' . $attachment->ID;
			}
			else
			{
				$class = 'avia-img-lazy-loading-' . $attachment->ID;
			}

			if( array_key_exists( 'class', $attr ) && false === strpos( $attr['class'], $class ) )
			{
				$class .= ' ' . $attr['class'];
			}

			$attr['class'] = $class;

			return $attr;
		}

		/**
		 * Sets the default image to our default quality (100%) for more beautiful images when used in conjunction with img optimization plugins
		 *
		 * @since 4.7.5.1
		 * @since 4.3 in functions-enfold added_by Kriesi
		 * @param int $quality
		 * @param string $mime_type
		 * @return int
		 */
		public function handler_wp_editor_set_quality( $quality, $mime_type = '' )
		{
			return apply_filters( 'avf_wp_editor_set_quality', $this->config['default_jpeg_quality'], $quality, $mime_type );
		}


		/**
		 * Sets the default image to our default quality (100%) for more beautiful images when used in conjunction with img optimization plugins
		 *
		 * @since 4.7.5.1
		 * @since 4.3 in functions-enfold added_by Kriesi
		 * @param int $quality
		 * @param string $context				edit_image | image_resize
		 * @return int
		 */
		public function handler_wp_jpeg_quality( $quality, $context = '' )
		{
			return apply_filters( 'avf_jpeg_quality', $this->config['default_jpeg_quality'], $quality, $context );
		}

		/**
		 * Reset option to store attachment URL -> attachment ID relationship to speed up frontend
		 *
		 * @since 4.8.4
		 */
		public function handler_ava_reset_db_options()
		{
			update_option( $this->opt_key_attachment_urls, array() );
		}

		/**
		 * General WP handler to disable lazy loading
		 * WP 5.5 does not give any information about the image - so we can only
		 * make a global check here. This might change in a future version which might allow
		 * to change our checks. At the moment we only can remove the loading attribute in
		 * av_responsive_images::handler_wp_img_tag_add_loading_attr
		 *
		 * @since 4.7.6.2
		 * @param boolean $default
		 * @param string $tag_name
		 * @param string $context
		 * @return boolean
		 */
		public function handler_wp_lazy_loading_enabled( $default, $tag_name = 'img', $context = '' )
		{
			if( $tag_name != 'img' )
			{
				return $default;
			}

			/**
			 * Disable for images loaded in an ajax call
			 */
			if( defined( 'DOING_AJAX' ) && DOING_AJAX )
			{
				return false;
			}

			if( 'none' == $this->lazy_loading_active() )
			{
				return false;
			}

			return $default;
		}

		/**
		 * Checks if an image must not be lazy loaded.
		 * Individual immage settings overrules global settings
		 *
		 * @since 4.7.6.2
		 * @param string|bool $attr_value			false | 'lazy' | 'eager'
		 * @param string $image_tag
		 * @param string $context
		 * @return type
		 */
		public function handler_wp_img_tag_add_loading_attr( $attr_value, $image_tag = '', $context = '' )
		{
			if( false === $attr_value )
			{
				return $attr_value;
			}

			if( 'none' == $this->lazy_loading_active() )
			{
				return false;
			}

			if( false !== strpos( $image_tag, 'avia-img-lazy-loading-not-' ) )
			{
				return false;
			}

			if( false !== strpos( $image_tag, 'avia-img-lazy-loading-' ) )
			{
				return 'lazy';
			}

			$attachment_id = $this->get_attachment_id( $image_tag );

			if( false === $attachment_id )
			{
				return $attr_value;
			}

			if( $this->is_attachment_id_not_lazy_loading( $attachment_id ) )
			{
				return false;
			}

			return $attr_value;
		}

		/**
		 * Allow to disable usage of responsive images for a period of code execution - e.g. during a single ALB element.
		 * Has to be reset when finished !!!
		 * Anything different from 'disabled' will reset this flag.
		 *
		 * @since 4.8.6.3
		 * @param string $action			'disabled' | mixed
		 */
		public function force_disable( $action = '' )
		{
			if( current_theme_supports( 'avia_show_alb_responsive_image_option' ) )
			{
				$this->temporary_disabled = 'disabled' === $action;
			}
			else
			{
				$this->temporary_disabled = false;
			}
		}

		/**
		 * Adds an attachment id or array of ids to be recognized for not lazy loading.
		 *
		 * @since 4.7.6.2
		 * @param int|array|mixed $ids
		 */
		public function add_attachment_id_to_not_lazy_loading( $ids )
		{
			if( ! is_array( $ids ) )
			{
				if( ! is_numeric( $ids ) || ( $ids <= 0 ) )
				{
					return;
				}

				$ids = array( $ids );
			}

			$this->no_lazy_loading_ids = array_unique( array_merge( $this->no_lazy_loading_ids, $ids ) );
		}

		/**
		 * Checks a given attachment id if lazy loading is prohibited
		 *
		 * @since 4.7.6.2
		 * @param int $id
		 * @return boolean
		 */
		public function is_attachment_id_not_lazy_loading( $id )
		{
			return in_array( $id, $this->no_lazy_loading_ids );
		}

		/**
		 * Return true, if option is selected
		 *
		 * @since 4.7.5.1
		 * @return boolean
		 */
		public function responsive_images_active()
		{
			$opt_active = 'responsive_images' == avia_get_option( 'responsive_images', '' );

			//	check if temporary disabled
			if( $opt_active )
			{
				$active = ! $this->temporary_disabled;
			}
			else
			{
				$active = false;
			}

			/**
			 * @since 4.8.2
			 * @since 4.8.6.3				added $opt_active, $this->temporary_disabled
			 * @param boolean $active
			 * @param boolean $opt_active
			 * @param boolean $this->temporary_disabled
			 * @return boolean
			 */
			return apply_filters( 'avf_responsive_images_active', $active, $opt_active, $this->temporary_disabled );
		}

		/**
		 * Return true, if option is selected
		 *
		 * @since 4.8.2
		 * @return boolean
		 */
		public function responsive_images_lightbox_active()
		{
			$opt_active = 'responsive_images_lightbox' == avia_get_option( 'responsive_images_lightbox', '' );

			//	check if temporary disabled
			if( $opt_active )
			{
				$active = ! $this->temporary_disabled;
			}
			else
			{
				$active = false;
			}

			/**
			 * @since 4.8.2
			 * @since 4.8.6.3				added $opt_active, $this->temporary_disabled
			 * @param boolean $active
			 * @param boolean $opt_active
			 * @param boolean $this->temporary_disabled
			 * @return boolean
			 */
			return apply_filters( 'avf_responsive_images_lightbox_active', $active, $opt_active, $this->temporary_disabled );
		}


		/**
		 * Return info about theme options setting for Lazy Loading.
		 * Currently only applies for images, but might be extended for iframe in future.
		 *
		 * @since 4.7.6.2
		 * @param string $what				'image' | .....
		 * @return string					'none' | 'all'
		 */
		public function lazy_loading_active( $what = 'image' )
		{
			$option = avia_get_option( 'lazy_loading', '' );

			switch( $option )
			{
				case 'no_lazy_loading_all':
					$return = 'none';
					break;
				case '':
				default:
					$return = 'all';
			}

			return $return;
		}

		/**
		 * Adds the class "wp-image-{$attachment_ID}" to <img ...> tag.
		 * Needed by WP to add scrset and sizes attributes.
		 * Class will be added to all img tags in content
		 *
		 * @since 4.7.5.1
		 * @deprecated since version 4.7.6.2
		 * @param string $html
		 * @param int $attachment_id
		 * @return string
		 */
		public function add_attachment_id( $html, $attachment_id )
		{
			_deprecated_function( 'av_responsive_images::add_attachment_id', '4.7.6.2', 'av_responsive_images::prepare_images' );

			return $this->prepare_single_image( $html, $attachment_id );
		}

		/**
		 * Adds the class "wp-image-{$attachment_ID}" to <img ...> tag.
		 * Needed by WP to add scrset and sizes attributes.
		 *
		 * Adds class "avia-lazy-loading-{$attachment_ID}" or "avia-lazy-loading-not-{$attachment_ID}"
		 * to identify image as not allowed for lazy loading
		 * Attachment ID is saved for not lazy loading.
		 *
		 * Classes will be added to all img tags in content !!!
		 *
		 * @since 4.7.6.2
		 * @param string $html
		 * @param int $attachment_id
		 * @param string $lazy_loading					'' | 'enabled' | 'disabled'
		 * @return string
		 */
		public function prepare_single_image( $html, $attachment_id, $lazy_loading = 'disabled' )
		{
			$lazy_loading = $this->validate_lazy_loading_alb_option( $lazy_loading );

			if( ! $this->responsive_images_active() && ( 'none' == $this->lazy_loading_active() ) )
			{
				return $html;
			}

			$matches = array();
			if ( ! preg_match_all( '/<img [^>]+>/', $html, $matches ) )
			{
				return $html;
			}

			foreach ( $matches[0] as $image )
			{
				$new_img = $this->add_lazy_loading_to_img( $image, $attachment_id, $lazy_loading );

				if( is_numeric( $attachment_id ) || 0 != $attachment_id )
				{
					$new_img = $this->add_attachment_id_to_img( $new_img, $attachment_id );
				}

				$pos = strpos( $html, $image );
				if( false !== $pos )
				{
					$html = substr_replace( $html, $new_img, $pos, strlen( $image ) );
				}
			}

			return $html;
		}

		/**
		 * Returns a "valid" string.
		 *
		 * @since 4.7.6.2
		 * @param string $lazy_loading
		 * @return string					'enabled' | 'disabled'
		 */
		public function validate_lazy_loading_alb_option( $lazy_loading = '' )
		{
			return in_array( $lazy_loading, array( 'enabled', 'disabled' ) ) ? $lazy_loading : 'disabled';
		}


		/**
		 * Gets the attachment id from image tag
		 *
		 * @since 4.7.5.1
		 * @param string $image_tag
		 * @return int|false
		 */
		public function get_attachment_id( $image_tag )
		{
			$attachment_id = false;
			$match = array();
			if ( preg_match( '/wp-image-([0-9]+)/i', $image_tag, $match ) )
			{
				$attachment_id = absint( $match[1] );
			}

			return $attachment_id;
		}


		/**
		 * Gets final image HTML with scrset and sizes added if necessary
		 * and attribute loading. Is prepared to handle multiple img tags, but
		 * it only makes sense to call with a single img as we add the same $attachment_id
		 * to all img
		 *
		 * @since 4.7.5.1
		 * @param string $html
		 * @param int $attachment_id
		 * @param string $lazy_loading					'' | 'enabled' | 'disabled'
		 * @return string
		 */
		public function make_image_responsive( $html, $attachment_id, $lazy_loading = '' )
		{
			$img = $this->prepare_single_image( $html, $attachment_id, $lazy_loading );
			return $this->make_content_images_responsive( $img );
		}

		/**
		 * Wrapper function to prepare HTML attributes that standard WP function
		 * wp_make_content_images_responsive resp. wp_filter_content_tags can handle
		 * images properly to add scrset, sizes, and loading attributes
		 * (callback handler_wp_lazy_loading_enabled and handler_wp_img_tag_add_loading_attr)
		 *
		 * Returns final HTML for output
		 *
		 * @since 4.7.5.1
		 * @param string $content
		 * @return string
		 */
		public function make_content_images_responsive( $content )
		{
			global $wp_version;

			if( ! $this->responsive_images_active() )
			{
				return $content;
			}

			//	Stay backwards comp.
			if( version_compare( $wp_version, '5.4.99999', '<' ) && ! function_exists( 'wp_make_content_images_responsive' ) )
			{
				return $content;
			}

			$matches = array();
			if ( ! preg_match_all( '/<img [^>]+>/', $content, $matches ) )
			{
				return $content;
			}

			foreach ( $matches[0] as $image )
			{
				$new_image = $this->ensure_attr_enclosure( $image );
				if( $new_image != $image )
				{
					$pos = strpos( $content, $image );

					if( false !== $pos )
					{
						$content = substr_replace( $content, $new_image, $pos, strlen( $image ) );
					}
				}
			}

			if( version_compare( $wp_version, '5.4.99999', '<' ) )
			{
				$return = wp_make_content_images_responsive( $content );
			}
			else
			{
				$return = wp_filter_content_tags( $content );
			}

			return $return;
		}

		/**
		 * Remove the loading attribute from image tags in content.
		 * This function is a fallback to scan content and remove all loading='lazy' attributes.
		 *
		 * @since 4.7.6.2
		 * @param string $content
		 * @return string
		 */
		public function remove_loading_lazy_attributes( $content )
		{
			$matches = array();
			if ( ! preg_match_all( '/<img [^>]+>/', $content, $matches ) )
			{
				return $content;
			}

			foreach ( $matches[0] as $image )
			{
				$attachment_id = $this->get_attachment_id( $image );
				if( false !== $attachment_id )
				{
					$this->add_attachment_id_to_not_lazy_loading( $attachment_id );
				}

				$count = 0;
				$new_image = str_ireplace( array( 'loading="lazy"', "loading='lazy'" ), '', $image, $count );
				if( $count != 0 )
				{
					$pos = strpos( $content, $image );

					if( false !== $pos )
					{
						$content = substr_replace( $content, $new_image, $pos, strlen( $image ) );
					}
				}
			}

			return $content;
		}

		/**
		 * Wrapper for WP attachment_url_to_postid().
		 * Tries to convert an URL to an attachment id. If not exist, returns URL
		 * Uses DB option to store already converted URL's. Option deleted when theme options change.
		 *
		 * @since 4.8.4
		 * @param string|int $attachment
		 * @return string|int
		 */
		public function attachment_url_to_postid( $attachment )
		{
			if( is_numeric( $attachment ) )
			{
				return $attachment;
			}

			$cache = (array) get_option( $this->opt_key_attachment_urls, array() );

			if( isset( $cache[ $attachment ] ) && is_numeric( $cache[ $attachment ] ) && $cache[ $attachment ] > 0 )
			{
				return $cache[ $attachment ];
			}

			$id = attachment_url_to_postid ($attachment );

			if( $id <= 0 )
			{
				return $attachment;
			}

			$cache[ $attachment ] = $id;

			update_option( $this->opt_key_attachment_urls, $cache );

			return $id;
		}

		/**
		 * Returns wp_get_attachment_image_src() extended by scrset and sizes
		 *
		 * @since 4.8.2
		 * @param int $attachment_id
		 * @param string $image_size
		 * @return array|false
		 */
		public function responsive_image_src( $attachment_id, $image_size = 'large' )
		{
			$img_src = wp_get_attachment_image_src( $attachment_id, $image_size );

			if( false === $img_src )
			{
				return $img_src;
			}

			$img_src['srcset'] = '';
			$img_src['sizes'] = '';

			if( ! $this->responsive_images_active() )
			{
				return $img_src;
			}

			$img = "<img src='{$img_src[0]}' width='{$img_src[1]}' height='{$img_src[2]}' />";

			$img = $this->make_image_responsive( $img, $attachment_id );

			$attrs = array( 'srcset', 'sizes' );

			foreach( $attrs as $attr )
			{
				$match = array();
				if( preg_match( "/{$attr}='([^']+)'/", $img, $match ) )
				{
					$img_src[ $attr ] = $match[1];
					continue;
				}
				if( preg_match( "/{$attr}=\"([^\"]+)\"/", $img, $match ) )
				{
					$img_src[ $attr ] = $match[1];
				}
			}

			return $img_src;
		}

		/**
		 * Returns the attribute string created from extended array of wp_get_attachment_image_src and scrset and sizes
		 * or from AviaHelper::get_url()
		 *
		 * Replaces the attribute keys according to <img> or <a> tag
		 *
		 * @since 4.8.2
		 * @param array|string $image_src
		 * @param boolean $image_link
		 * @return string
		 */
		public function html_attr_image_src( $image_src, $image_link = true )
		{
			if( ! is_array( $image_src ) )
			{
				return $image_link ? 'src="' . esc_attr( $image_src ) . '"' : 'href="' . esc_attr( $image_src ) . '"';
			}

			$atts = array();

			foreach( $image_src as $key => $value )
			{
				//	PHP fix < 8.0: case does not compare string and int -> force to string and compare strings !!!
				$key .= '';

				switch( $key )
				{
					case '0':				//	= url
						$a = $image_link ? 'src' : 'href';
						break;
					case 'srcset':
					case 'sizes':
						$a = $image_link ? $key : 'data-' . $key;
						break;
					default:
						$a = null;
						break;
				}

				if( is_null( $a ) || empty( $value ) )
				{
					continue;
				}

				$atts[] = $a . '="' . esc_attr( $value ) . '"';
			}

			return implode( ' ', $atts );
		}

		/**
		 * @since 4.7.6.2
		 * @param string $image
		 * @param int $attachment_id
		 * @param string $lazy_loading				'enabled' | 'disabled'
		 * @return string
		 */
		protected function add_lazy_loading_to_img( $image, $attachment_id, $lazy_loading )
		{
			if( 'disabled' == $lazy_loading )
			{
				$this->add_attachment_id_to_not_lazy_loading( $attachment_id );
			}

			$prefix = 'avia-img-lazy-loading-';

			if( false !== strpos( $image, $prefix ) )
			{
				return $image;
			}

			if( 'none' == $this->lazy_loading_active() )
			{
				$lazy_loading = 'disabled';
			}

			$class = ( 'disabled' == $lazy_loading ) ? $prefix . 'not-' : $prefix;
			$class .= $attachment_id;

			$image = $this->add_class_to_img_tag( $image, $class );

			return $image;
		}

		/**
		 * Adds the WP class "wp-image-{$attachment_ID}" to <img> tag
		 *
		 * @since 4.7.5.1
		 * @param string $image
		 * @param int $attachment_id
		 * @return string
		 */
		protected function add_attachment_id_to_img( $image, $attachment_id )
		{
			$prefix = 'wp-image-';

			if( false !== strpos( $image, $prefix ) )
			{
				return $image;
			}

			$class = $prefix . $attachment_id;

			$image = $this->add_class_to_img_tag( $image, $class );

			return $image;
		}

		/**
		 * @since 4.7.6.2
		 * @param string $image
		 * @param string $class
		 * @return string
		 */
		protected function add_class_to_img_tag( $image, $class )
		{
			if( false === strpos( $image, 'class=' ) )
			{
				$image = str_replace( '<img', '<img class="' . $class . '" ', $image );
			}
			else if( false !== strpos( $image, 'class="' ) )
			{
				$image = str_replace( 'class="', 'class="' . $class . ' ', $image );
			}
			else if( false !== strpos( $image, "class='" ) )
			{
				$image = str_replace( "class='", "class='" . $class . ' ', $image );
			}

			return $image;
		}

		/**
		 * Make sure that HTML is xxx="...." and mot xxx='.....' which is not recognized by WP:
		 *		- src
		 *		- height
		 *		- width
		 *
		 * @since 4.7.5.1
		 * @param string $image
		 * @return string
		 */
		protected function ensure_attr_enclosure( $image )
		{
			$attrs = array( 'src', 'height', 'width' );

			foreach( $attrs as $attr )
			{
				$match_attr = array();

				if( preg_match( "/{$attr}='([^']+)'/", $image, $match_attr ) )
				{
					$new_attr = $attr . '="' . $match_attr[1] . '"';

					$pos = strpos( $image, $match_attr[0] );

					if( false !== $pos )
					{
						$image = substr_replace( $image, $new_attr, $pos, strlen( $match_attr[0] ) );
					}
				}
			}

			return $image;
		}

		/**
		 * Build internal responsive groups
		 * Needed to display in backend only for user information
		 *
		 * @since 4.7.5.1
		 */
		protected function group_image_sizes()
		{
			$widths = array();
			$all = $this->base_images;

			foreach( $all as $sizes )
			{
				if( ! in_array( $sizes['width'], $widths ) )
				{
					$widths[] = $sizes['width'];
				}
			}

			sort( $widths );

			$groups = array();

			while( count( $widths ) > 0 )
			{
				$current_width = array_shift( $widths );

				do
				{
					$found = false;
					$height_key = '';

					foreach( $all as $name => $sizes )
					{
						if( $sizes['width'] == $current_width )
						{
							$height_key = $sizes['height'];
							$group_key = $current_width . '*' . $height_key;
							$groups[ $group_key ][ $name ] = $sizes;

							unset( $all[ $name ] );
							$found = true;
							break;
						}
					}

					if( ! $found )
					{
						break;
					}

					/**
					 * Check remaining for same aspect ratio
					 */
					foreach( $all as $name => $sizes )
					{
						if( wp_image_matches_ratio( $current_width, $height_key, $sizes['width'], $sizes['height'] ) )
						{
							$groups[ $group_key ][ $name ] = $sizes;
							unset( $all[ $name ] );
						}
					}

				} while ( $found );
			}

			$this->size_groups = array();

			foreach( $groups as $group => $images )
			{
				$widths = array();

				foreach( $images as $image )
				{
					if( ! in_array( $image['width'], $widths ) )
					{
						$widths[] = $image['width'];
					}
				}

				sort( $widths );

				while( count( $widths ) > 0 )
				{
					$current_width = array_shift( $widths );

					foreach( $images as $name => $sizes )
					{
						if( $sizes['width'] == $current_width )
						{
							$this->size_groups[ $group ][ $name ] = $sizes;
							unset( $images[ $name ] );
						}
					}
				}

			}

		}

		/**
		 * Prepare overview of used image sizes for theme options page
		 *
		 * @since 4.7.5.1
		 * @return string
		 */
		public function options_page_overview()
		{
			$html = '';

			foreach( $this->size_groups as $size_group => $sizes )
			{
				$html .= '<h3>' . $this->get_group_headline( $size_group ) . '</h3>';

				$html .= '<ul>';

				foreach( $sizes as $key => $image )
				{
					$info = $image['width'] . '*' . $image['height'];
					if( isset( $image['crop'] ) && true === $image['crop'] )
					{
						$info .= '  ' . __( '(cropped)', 'avia_framework' );
					}

					$info .= ' - ' . $this->get_image_key_info( $key );

					$html .= '<li>' . $info . '</li>';
				}

				$html .= '</ul>';
			}

			return $html;
		}

		/**
		 * Returns the string for the group headline.
		 * Calculates the aspect ratio or only width/height if one value is 0
		 *
		 * @since 4.7.5.1
		 * @param string $group
		 * @return string
		 */
		protected function get_group_headline( $group )
		{
			$headline = '';

			$sizes = explode( '*', $group );

			$w = isset( $sizes[0] ) ? (int) $sizes[0] : 0;
			$h = isset( $sizes[1] ) ? (int) $sizes[1] : 0;

			if( 0 == $h )
			{
				$headline .= __( 'Images keeping original aspect ratio', 'avia_framework' );
			}
			else if ( 0 == $w )
			{
				$headline .= __( 'Images keeping original aspect ratio', 'avia_framework' );
			}
			else
			{
				$gcd = $this->greatest_common_divisor( $w, $h );
				$w = (int) ( $w / $gcd );
				$h = (int) ( $h / $gcd );

				$headline .= sprintf( __( 'Images aspect ratio: %d : %d', 'avia_framework' ), $w, $h );
			}

			/**
			 *
			 * @since 4.7.5.1
			 * @param string $headline
			 * @param string $group
			 * @return string
			 */
			return apply_filters( 'avf_admin_image_group_headline', $headline, $group );
		}

		/**
		 * Return readable info for an image size key for options page
		 *
		 * @since 4.7.5.1
		 * @param string $image_key
		 * @return string
		 */
		protected function get_image_key_info( $image_key )
		{
			$info = '';

			$info .= ( array_key_exists( $image_key, $this->readable_img_sizes ) ) ? $this->readable_img_sizes[ $image_key ] : $image_key;

			$info .= '  (';

			if( array_key_exists( $image_key, $this->theme_images ) )
			{
				$info .= __( 'added by theme', 'avia_framework' );
			}
			else if( array_key_exists( $image_key, $this->plugin_images ) )
			{
				$info .= __( 'added by a plugin', 'avia_framework' );
			}
			else if( array_key_exists( $image_key, $this->wp_default_images ) )
			{
				$info .= __( 'WP default size', 'avia_framework' );
			}
			else
			{
				$info .= __( 'unknown', 'avia_framework' );
			}

			$info .= ')';

			/**
			 *
			 * @since 4.7.5.1
			 * @param string $info
			 * @param string $image_key
			 * @return string
			 */
			return apply_filters( 'avf_admin_image_key_info', $info, $image_key );
		}


		/**
		 * Calculates the value based on https://en.wikipedia.org/wiki/Greatest_common_divisor - euclid's algorithm
		 *
		 * @since 4.7.5.1
		 * @param int $a
		 * @param int $b
		 * @return int
		 */
		protected function greatest_common_divisor( $a, $b )
		{
			if( 0 == $a )
			{
				return abs( $b );
			}

			if( 0 == $b )
			{
				return abs( $a );
			}

			if( $a < $b )
			{
				$h = $a;
				$a = $b;
				$b = $h;
			}

			do
			{
				$h = $a % $b;
				$a = $b;
				$b = $h;
			} while ( $b != 0 );

			return abs( $a );
		}
	}

	/**
	 * Returns the main instance of av_responsive_images to prevent the need to use globals
	 *
	 * @since 4.7.5.1
	 * @param array|null $config
	 * @return av_responsive_images
	 */
	function Av_Responsive_Images( $config = array() )
	{
		return av_responsive_images::instance( $config );
	}

}
© 2025 XylotrechusZ