HEX
Server: Apache/2.4.41 (Amazon) OpenSSL/1.0.2k-fips PHP/5.6.40
System: Linux ip-172-31-40-18 4.14.146-93.123.amzn1.x86_64 #1 SMP Tue Sep 24 00:45:23 UTC 2019 x86_64
User: apache (48)
PHP: 5.6.40
Disabled: NONE
Upload Files
File: /var/www/html/qcr24/wp-content/plugins/rocket-lazy-load/src/Subscriber/LazyloadSubscriber.php
<?php

namespace RocketLazyLoadPlugin\Subscriber;

use RocketLazyLoadPlugin\EventManagement\SubscriberInterface;
use RocketLazyLoadPlugin\Options\OptionArray;
use RocketLazyLoadPlugin\Dependencies\RocketLazyload\Assets;
use RocketLazyLoadPlugin\Dependencies\RocketLazyload\Image;
use RocketLazyLoadPlugin\Dependencies\RocketLazyload\Iframe;

/**
 * Lazyload Subscriber
 *
 * @since 2.0
 * @author Remy Perona
 */
class LazyloadSubscriber implements SubscriberInterface {

	/**
	 * OptionArray instance
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @var OptionArray
	 */
	private $option_array;

	/**
	 * Assets instance
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @var Assets
	 */
	private $assets;

	/**
	 * Image instance
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @var Image
	 */
	private $image;

	/**
	 * Iframe instance
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @var Iframe
	 */
	private $iframe;

	/**
	 * Constructor
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @param OptionArray $option_array OptionArray instance.
	 * @param Assets      $assets Assets instance.
	 * @param Image       $image Image instance.
	 * @param Iframe      $iframe Iframe instance.
	 */
	public function __construct( OptionArray $option_array, Assets $assets, Image $image, Iframe $iframe ) {
		$this->option_array = $option_array;
		$this->assets       = $assets;
		$this->image        = $image;
		$this->iframe       = $iframe;
	}

	/**
	 * Returns an array of events that this subscriber wants to listen to.
	 *
	 * @return array
	 */
	public function getSubscribedEvents() {
		return [
			'wp_footer'            => [
				[ 'insertLazyloadScript', \ROCKET_LL_INT_MAX ],
				[ 'insertYoutubeThumbnailScript', \ROCKET_LL_INT_MAX ],
			],
			'wp_head'              => [ 'insertNoJSStyle', \ROCKET_LL_INT_MAX ],
			'wp_enqueue_scripts'   => [ 'insertYoutubeThumbnailStyle', \ROCKET_LL_INT_MAX ],
			'template_redirect'    => [ 'lazyload', 2 ],
			'rocket_lazyload_html' => 'lazyloadResponsive',
			'init'                 => 'lazyloadSmilies',
		];
	}

	/**
	 * Inserts the lazyload script in the footer
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function insertLazyloadScript() {
		if ( ! $this->option_array->get( 'images' ) && ! $this->option_array->get( 'iframes' ) ) {
			return;
		}

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

		/**
		 * Filters the threshold at which lazyload is triggered
		 *
		 * @since 1.2
		 * @author Remy Perona
		 *
		 * @param int $threshold Threshold value.
		 */
		$threshold = apply_filters( 'rocket_lazyload_threshold', 300 );

		/**
		 * Filters the use of the polyfill for intersectionObserver
		 *
		 * @since 3.3
		 * @author Remy Perona
		 *
		 * @param bool $polyfill True to use the polyfill, false otherwise.
		 */
		$polyfill = apply_filters( 'rocket_lazyload_polyfill', false );

		$script_args = [
			'base_url' => ROCKET_LL_FRONT_JS_URL,
			'version'  => '16.1',
			'polyfill' => $polyfill,
		];

		$inline_args = [
			'threshold' => $threshold,
		];

		/**
		 * Filters the use of native lazyload
		 *
		 * @since 2.3.3
		 * @param bool $use_native True to enable native lazyload usage.
		 */
		if ( apply_filters( 'rocket_use_native_lazyload', false ) ) { // phpcs:ignore WordPress.NamingConventions.PrefixAllGlobals
			$inline_args['options'] = [
				'use_native' => 'true',
			];
		}

		if ( $this->option_array->get( 'images' ) || $this->option_array->get( 'iframes' ) ) {
			// This filter is documented in src/Subscriber/LazyloadSubscriber.php.
			if ( apply_filters( 'rocket_use_native_lazyload', false ) ) { // phpcs:ignore WordPress.NamingConventions.PrefixAllGlobals
				$inline_args['elements']            = isset( $inline_args['elements'] ) ? $inline_args['elements'] : [];
				$inline_args['elements']['loading'] = '[loading=lazy]';
			}
		}

		if ( $this->option_array->get( 'images' ) ) {
			$inline_args['elements']                     = isset( $inline_args['elements'] ) ? $inline_args['elements'] : [];
			$inline_args['elements']['image']            = 'img[data-lazy-src]';
			$inline_args['elements']['background_image'] = '.rocket-lazyload';
		}

		if ( $this->option_array->get( 'iframes' ) ) {
			$inline_args['elements']           = isset( $inline_args['elements'] ) ? $inline_args['elements'] : [];
			$inline_args['elements']['iframe'] = 'iframe[data-lazy-src]';
		}

		/**
		 * Filters the arguments array for the lazyload script options
		 *
		 * @since 2.0
		 * @author Remy Perona
		 *
		 * @param array $inline_args Arguments used for the lazyload script options.
		 */
		$inline_args = apply_filters( 'rocket_lazyload_script_args', $inline_args );

		echo '<script>' . $this->assets->getInlineLazyloadScript( $inline_args ) . '</script>'; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
		$this->assets->insertLazyloadScript( $script_args );
	}

	/**
	 * Inserts the Youtube thumbnail script in the footer
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function insertYoutubeThumbnailScript() {
		if ( ! $this->option_array->get( 'youtube' ) ) {
			return;
		}

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

		/**
		 * Filters the resolution of the YouTube thumbnail
		 *
		 * @since 1.4.8
		 * @author Arun Basil Lal
		 *
		 * @param string $thumbnail_resolution The resolution of the thumbnail. Accepted values: default, mqdefault, sddefault, hqdefault, maxresdefault
		 */
		$thumbnail_resolution = apply_filters( 'rocket_lazyload_youtube_thumbnail_resolution', 'hqdefault' );

		$this->assets->insertYoutubeThumbnailScript(
			[
				'resolution' => $thumbnail_resolution,
				'lazy_image' => (bool) $this->option_array->get( 'images' ),
			]
		);
	}

	/**
	 * Inserts the no JS CSS compatibility in the header
	 *
	 * @since 2.0.3
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function insertNoJSStyle() {
		if ( ! $this->shouldLazyload() ) {
			return;
		}

		$this->assets->insertNoJSCSS();
	}

	/**
	 * Inserts the Youtube thumbnail CSS in the header
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function insertYoutubeThumbnailStyle() {
		if ( ! $this->option_array->get( 'youtube' ) ) {
			return;
		}

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

		$this->assets->insertYoutubeThumbnailCSS(
			[
				'base_url'          => ROCKET_LL_ASSETS_URL,
				'responsive_embeds' => current_theme_supports( 'responsive-embeds' ),
			]
		);
	}

	/**
	 * Checks if lazyload should be applied
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return bool
	 */
	private function shouldLazyload() {
		if ( is_admin() || is_feed() || is_preview() || ( defined( 'REST_REQUEST' ) && REST_REQUEST ) || ( defined( 'DONOTLAZYLOAD' ) && DONOTLAZYLOAD ) ) {
			return false;
		}

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

		/**
		 * Filters the lazyload application
		 *
		 * @since 2.0
		 * @author Remy Perona
		 *
		 * @param bool $do_rocket_lazyload True to apply lazyload, false otherwise.
		 */
		if ( ! apply_filters( 'do_rocket_lazyload', true ) ) { // phpcs:ignore WordPress.NamingConventions.PrefixAllGlobals
			return false;
		}

		return true;
	}

	/**
	 * Checks if current page is a page builder editor.
	 *
	 * @since 2.2.2
	 * @author Remy Perona
	 *
	 * @return bool
	 */
	private function isPageBuilder() {
		// Exclude Page Builders editors.
		$excluded_parameters = [
			'fl_builder',
			'et_fb',
			'ct_builder',
		];

		foreach ( $excluded_parameters as $excluded ) {
			if ( isset( $_GET[ $excluded ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
				return true;
			}
		}

		return false;
	}

	/**
	 * Gets the content to lazyload
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function lazyload() {
		if ( ! $this->shouldLazyload() ) {
			return;
		}

		ob_start( [ $this, 'lazyloadBuffer' ] );
	}

	/**
	 * Applies lazyload on the provided content
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @param string $html HTML content.
	 * @return string
	 */
	public function lazyloadBuffer( $html ) {
		$buffer = $this->ignoreScripts( $html );
		$buffer = $this->ignoreNoscripts( $buffer );

		if ( $this->option_array->get( 'images' ) ) {
			$html = $this->image->lazyloadImages( $html, $buffer );
			$html = $this->image->lazyloadPictures( $html, $buffer );
			$html = $this->image->lazyloadBackgroundImages( $html, $buffer );
		}

		if ( $this->option_array->get( 'iframes' ) ) {
			$args = [
				'youtube' => $this->option_array->get( 'youtube' ),
			];

			$html = $this->iframe->lazyloadIframes( $html, $buffer, $args );
		}

		return $html;
	}

	/**
	 * Applies lazyload on responsive images attributes srcset and sizes
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @param string $html Image HTML.
	 * @return string
	 */
	public function lazyloadResponsive( $html ) {
		return $this->image->lazyloadResponsiveAttributes( $html );
	}

	/**
	 * Applies lazyload on WordPress smilies
	 *
	 * @since 2.0
	 * @author Remy Perona
	 *
	 * @return void
	 */
	public function lazyloadSmilies() {
		if ( ! $this->shouldLazyload() ) {
			return;
		}

		if ( ! $this->option_array->get( 'images' ) ) {
			return;
		}

		$filters = [
			'the_content'  => 10,
			'the_excerpt'  => 10,
			'comment_text' => 20,
		];

		foreach ( $filters as $filter => $prio ) {
			if ( ! has_filter( $filter ) ) {
				continue;
			}

			remove_filter( $filter, 'convert_smilies', $prio );
			add_filter( $filter, [ $this->image, 'convertSmilies' ], $prio );
		}
	}

	/**
	 * Remove inline scripts from the HTML to parse
	 *
	 * @param string $html HTML content.
	 * @return string
	 */
	private function ignoreScripts( $html ) {
		return preg_replace( '/<script\b(?:[^>]*)>(?:.+)?<\/script>/Umsi', '', $html );
	}

	/**
	 * Remove noscript tags from the HTML to parse
	 *
	 * @param string $html HTML content.
	 * @return string
	 */
	private function ignoreNoscripts( $html ) {
		return preg_replace( '#<noscript>(?:.+)</noscript>#Umsi', '', $html );
	}
}