b0y-101 Mini Shell


Current Path : E:/www3/chiangrai/wp-content/plugins/envira-gallery-lite/includes/global/
File Upload :
Current File : E:/www3/chiangrai/wp-content/plugins/envira-gallery-lite/includes/global/shortcode.php

<?php
/**
 * Shortcode class.
 *
 * @since 1.0.0
 *
 * @package Envira_Gallery
 * @author  Envira Gallery Team
 */

// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

/**
 * Envira Gallery Shortcode
 *
 * @since 1.0.0
 */
class Envira_Gallery_Shortcode {

	/**
	 * Holds the class object.
	 *
	 * @since 1.0.0
	 *
	 * @var object
	 */
	public static $instance;

	/**
	 * Path to the file.
	 *
	 * @since 1.0.0
	 *
	 * @var string
	 */
	public $file = __FILE__;

	/**
	 * Holds the base class object.
	 *
	 * @since 1.0.0
	 *
	 * @var object
	 */
	public $base;

	/**
	 * Holds the gallery data.
	 *
	 * @since 1.0.0
	 *
	 * @var array
	 */
	public $data;

	/**
	 * Holds gallery IDs for init firing checks.
	 *
	 * @since 1.0.0
	 *
	 * @var array
	 */
	public $done = [];

	/**
	 * Iterator for galleries on the page.
	 *
	 * @since 1.0.0
	 *
	 * @var int
	 */
	public $counter = 1;

	/**
	 * Holds image URLs for indexing.
	 *
	 * @since 1.0.0
	 *
	 * @var array
	 */
	public $index = [];

	/**
	 * Helper for gallery data
	 *
	 * @var array
	 */
	public $gallery_data = [];

	/**
	 * Holds Common Class
	 *
	 * @var object
	 */
	public $common;

	/**
	 * Mobile check
	 *
	 * @var boolean
	 */
	public $is_mobile;

	/**
	 * Allowed Html
	 *
	 * @var array
	 */
	public $wp_kses_allowed_html = [
		'a'      => [
			'href'                => [],
			'target'              => [],
			'class'               => [],
			'title'               => [],
			'data-status'         => [],
			'data-envira-tooltip' => [],
			'data-id'             => [],
		],
		'br'     => [],
		'img'    => [
			'src'   => [],
			'class' => [],
			'alt'   => [],
		],
		'div'    => [
			'class' => [],
		],
		'li'     => [
			'id'                              => [],
			'class'                           => [],
			'data-envira-gallery-image'       => [],
			'data-envira-gallery-image-model' => [],
		],
		'em'     => [],
		'span'   => [
			'class' => [],
		],
		'strong' => [],
	];

	/**
	 * Primary class constructor.
	 *
	 * @since 1.0.0
	 */
	public function __construct() {

		// Load the base class object.
		$this->base      = Envira_Gallery_Lite::get_instance();
		$this->common    = Envira_Gallery_Common::get_instance();
		$this->is_mobile = envira_lite_mobile_detect()->isMobile();

		// Register main gallery style.
		wp_register_style( $this->base->plugin_slug . '-style', plugins_url( 'assets/css/envira.css', $this->base->file ), [], $this->base->version );
		wp_register_style( $this->base->plugin_slug . '-fancybox', plugins_url( 'assets/css/fancybox.css', $this->base->file ), [], $this->base->version );
		wp_register_style( $this->base->plugin_slug . '-lazyload', plugins_url( 'assets/css/responsivelyLazy.css', $this->base->file ), [], $this->base->version );
		wp_register_style( $this->base->plugin_slug . '-jgallery', plugins_url( 'assets/css/justifiedGallery.css', $this->base->file ), [], $this->base->version );

		// Register main gallery script.
		wp_register_script( $this->base->plugin_slug . '-script', plugins_url( 'assets/js/min/envira-min.js', $this->base->file ), [ 'jquery', 'jquery-masonry' ], $this->base->version, true );

		// Load hooks and filters.
		add_shortcode( 'envira-gallery', [ $this, 'shortcode' ] );
		add_filter( 'widget_text', 'do_shortcode' );

		add_filter( 'style_loader_tag', [ $this, 'add_stylesheet_property_attribute' ] );
	}

	/**
	 * Creates the shortcode for the plugin.
	 *
	 * @since 1.0.0
	 *
	 * @global object $post The current post object.
	 *
	 * @param array $atts Array of shortcode attributes.
	 * @return string     The gallery output.
	 */
	public function shortcode( $atts ) {

		if ( is_admin() ) {
			return;
		}
		global $post;

		// If no attributes have been passed, the gallery should be pulled from the current post.
		$gallery_id = false;
		if ( empty( $atts ) ) {
			$gallery_id = $post->ID;
			$data       = is_preview() ? $this->base->_get_gallery( $gallery_id ) : $this->base->get_gallery( $gallery_id );
		} elseif ( isset( $atts['id'] ) ) {
			$gallery_id = (int) $atts['id'];
			$data       = is_preview() ? $this->base->_get_gallery( $gallery_id ) : $this->base->get_gallery( $gallery_id );
		} elseif ( isset( $atts['slug'] ) ) {
			$gallery_id = $atts['slug'];
			$data       = is_preview() ? $this->base->_get_gallery_by_slug( $gallery_id ) : $this->base->get_gallery_by_slug( $gallery_id );
		} else {
			// A custom attribute must have been passed. Allow it to be filtered to grab data from a custom source.
			$data       = apply_filters( 'envira_gallery_custom_gallery_data', false, $atts, $post );
			$gallery_id = $data['config']['id'];
		}

		// Change the gallery order, if specified.
		$data = $this->maybe_sort_gallery( $data, $gallery_id );

		// Limit the number of images returned, if specified.
		// [envira-gallery id="123" limit="10"] would only display 10 images.
		if ( isset( $atts['limit'] ) && is_numeric( $atts['limit'] ) ) {
			$images          = $data['gallery'];
			$images          = array_slice( $data['gallery'], 0, absint( $atts['limit'] ), true );
			$data['gallery'] = $images;
		}

		// Allow the data to be filtered before it is stored and used to create the gallery output.
		$data = apply_filters( 'envira_gallery_pre_data', $data, $gallery_id );

		// If there is no data to output or the gallery is inactive, do nothing.
		if ( ! $data || empty( $data['gallery'] ) || isset( $data['status'] ) && 'inactive' === $data['status'] && ! is_preview() ) {
			return;
		}

		$this->gallery_data = $data;

		// Get rid of any external plugins trying to jack up our stuff where a gallery is present.
		$this->plugin_humility();

		// Prepare variables.
		$this->data[ $data['id'] ]  = $this->gallery_data;
		$this->index[ $data['id'] ] = [];
		$gallery                    = '';
		$i                          = 1;

		// If this is a feed view, customize the output and return early.
		if ( is_feed() ) {
			return $this->do_feed_output( $data );
		}

		$lazy_loading_delay = isset( $this->gallery_data['config']['lazy_loading_delay'] ) ? intval( $this->gallery_data['config']['lazy_loading_delay'] ) : 500;

		// Load scripts and styles.
		wp_enqueue_style( $this->base->plugin_slug . '-style' );
		wp_enqueue_style( $this->base->plugin_slug . '-lazyload' );
		wp_enqueue_style( $this->base->plugin_slug . '-fancybox' );
		wp_enqueue_style( $this->base->plugin_slug . '-jgallery' );
		wp_enqueue_script( 'jquery-masonry' );
		wp_enqueue_script( $this->base->plugin_slug . '-script' );
		// If lazy load is active, load the lazy load script.
		if ( $this->get_config( 'lazy_loading', $data ) === 1 ) {
			wp_localize_script( $this->base->plugin_slug . '-script', 'envira_lazy_load', [ 'true' ] );
			wp_localize_script( $this->base->plugin_slug . '-script', 'envira_lazy_load_initial', [ 'false' ] );
			wp_localize_script( $this->base->plugin_slug . '-script', 'envira_lazy_load_delay', [ (string) $lazy_loading_delay ] );
		}

		// Load custom gallery themes if necessary.
		if ( 'base' !== $this->get_config( 'gallery_theme', $this->gallery_data ) && $this->get_config( 'columns', $this->gallery_data ) > 0 ) {
			// if columns is zero, then it's automattic which means we do not load gallery themes because it will mess up the new javascript layout.
			$this->load_gallery_theme( $this->get_config( 'gallery_theme', $this->gallery_data ) );
		}

		// Load custom lightbox themes if necessary.
		if ( 'base' !== $this->get_config( 'lightbox_theme', $this->gallery_data ) ) {
			$this->load_lightbox_theme( $this->get_config( 'lightbox_theme', $this->gallery_data ) );
		}

		// Load gallery init code in the footer.
		add_action( 'wp_footer', [ $this, 'gallery_init' ], 1000 );

		// Run a hook before the gallery output begins but after scripts and inits have been set.
		do_action( 'envira_gallery_before_output', $this->gallery_data );

		// Apply a filter before starting the gallery HTML.
		$gallery = apply_filters( 'envira_gallery_output_start', $gallery, $this->gallery_data );

		// Build out the gallery HTML.
		$gallery    .= '<div id="envira-gallery-wrap-' . sanitize_html_class( $this->gallery_data['id'] ) . '" class="' . $this->get_gallery_classes( $this->gallery_data ) . '" itemscope itemtype="https://schema.org/ImageGallery">';
			$gallery = apply_filters( 'envira_gallery_output_before_container', $gallery, $this->gallery_data );

		// Description.
		if ( isset( $this->gallery_data['config']['description_position'] ) && 'above' === $this->gallery_data['config']['description_position'] ) {
			$gallery = $this->description( $gallery, $this->gallery_data );
		}

			$opacity_insert = false;
		if ( $this->get_config( 'columns', $this->gallery_data ) === 0 ) {
			$opacity_insert = ' style="opacity: 0.0" ';
		}

			// add justified CSS?
			$extra_css               = 'envira-gallery-justified-public';
			$row_height              = false;
			$justified_gallery_theme = false;
		if ( $this->get_config( 'columns', $this->gallery_data ) > 0 ) {
			$extra_css = false;
		} else {
			$row_height              = $this->get_config( 'justified_row_height', $this->gallery_data );
			$justified_gallery_theme = $this->get_config( 'justified_gallery_theme', $this->gallery_data );
		}

			$gallery .= '<div' . $opacity_insert . ' data-row-height="' . $row_height . '" data-gallery-theme="' . $justified_gallery_theme . '" id="envira-gallery-' . sanitize_html_class( $this->gallery_data['id'] ) . '" class="envira-gallery-public ' . $extra_css . ' envira-gallery-' . sanitize_html_class( $this->get_config( 'columns', $this->gallery_data ) ) . '-columns envira-clear' . ( $this->get_config( 'isotope', $this->gallery_data ) ? ' enviratope' : '' ) . ( $this->get_config( 'css_animations', $this->gallery_data ) ? ' envira-gallery-css-animations' : '' ) . '" data-envira-columns="' . $this->get_config( 'columns', $this->gallery_data ) . '">';

				// Start image loop.
		foreach ( $data['gallery'] as $id => $item ) {

			// Add the gallery item to the markup.
			$gallery = $this->generate_gallery_item_markup( $gallery, $this->gallery_data, $item, $id, $i );

			// Increment the iterator.
			++$i;

		}
				// End image loop.

			$gallery .= '</div>';
			// Description.
		if ( isset( $this->gallery_data['config']['description_position'] ) && 'below' === $this->gallery_data['config']['description_position'] ) {
			$gallery = $this->description( $gallery, $this->gallery_data );
		}

			$gallery = apply_filters( 'envira_gallery_output_after_container', $gallery, $this->gallery_data );

		$gallery .= '</div>';
		$gallery  = apply_filters( 'envira_gallery_output_end', $gallery, $this->gallery_data );

		// Increment the counter.
		++$this->counter;

		// Remove any contextual filters so they don't affect other galleries on the page.
		if ( $this->get_config( 'mobile', $this->gallery_data ) ) {
			remove_filter( 'envira_gallery_output_image_attr', [ $this, 'mobile_image' ], 999, 4 );
		}

		// Add no JS fallback support.
		$no_js    = '<noscript>';
		$no_js   .= $this->get_indexable_images( $this->gallery_data['id'] );
		$no_js   .= '</noscript>';
		$gallery .= $no_js;

		// Return the gallery HTML.
		return apply_filters( 'envira_gallery_output', $gallery, $this->gallery_data );
	}


	/**
	 * Outputs an individual gallery item in the grid
	 *
	 * @since 1.4.2.1
	 *
	 * @param    string $gallery    Gallery HTML.
	 * @param    array  $data       Gallery Config.
	 * @param    array  $item       Gallery Item (Image).
	 * @param    int    $id         Gallery Image ID.
	 * @param    int    $i          Index.
	 * @return   string              Gallery HTML.
	 */
	public function generate_gallery_item_markup( $gallery, $data, $item, $id, $i ) {

		// Get some config values that we'll reuse for each image.
		$padding             = absint( round( $this->get_config( 'gutter', $data ) / 2 ) );
		$html5_attribute     = ( ( $this->get_config( 'html5', $data ) === '1' ) ? 'data-envirabox-group' : 'rel' );
		$thumbnail_start_url = get_bloginfo( 'url' ) . '/' . get_bloginfo( 'url' );

		// Skip over images that are pending (ignore if in Preview mode).
		if ( isset( $item['status'] ) && 'pending' === $item['status'] && ! is_preview() ) {
			return $gallery;
		}

		$item = apply_filters( 'envira_gallery_output_item_data', $item, $id, $data, $i );

		// Get image and image retina URLs
		// These calls will generate thumbnails as necessary for us.
		$imagesrc         = $this->get_image_src( $id, $item, $data );
		$image_src_retina = apply_filters( 'envira_gallery_output_item_src_retina', $this->get_image_src( $id, $item, $data, false, true ), $item, $id, $data, $i, $this->is_mobile );
		$placeholder      = wp_get_attachment_image_src( $id, 'medium' ); // $placeholder is null because $id is 0 for instagram?
		// Filter output before starting this gallery item.
		$gallery = apply_filters( 'envira_gallery_output_before_item', $gallery, $id, $item, $data, $i );

		// Maybe change the item's link if it is an image and we have an image size defined for the Lightbox.
		$item = $this->maybe_change_link( $id, $item, $data );

		// Non-ASCII filenames fail when FILTER_VALIDATE_URL is applied to them when saving a gallery to generate thumbs.
		// This resulted in the blog URL being prepended to the URL, therefore breaking the thumbnail URL.
		// This reverts that change for the few edge cases where this happened.
		if ( ! empty( $item['thumb'] ) && strpos( $item['thumb'], $thumbnail_start_url ) !== false ) {
			$item['thumb'] = str_replace( $thumbnail_start_url, get_bloginfo( 'url' ) . '/', $item['thumb'] );
		} elseif ( ! isset( $item['thumb'] ) ) {
			// this must be at least defined.
			$item['thumb'] = false;
		}

		$output = '<div id="envira-gallery-item-' . sanitize_html_class( $id ) . '" class="' . $this->get_gallery_item_classes( $item, $i, $data ) . '" style="padding-left: ' . $padding . 'px; padding-bottom: ' . $this->get_config( 'margin', $data ) . 'px; padding-right: ' . $padding . 'px;" ' . apply_filters( 'envira_gallery_output_item_attr', '', $id, $item, $data, $i ) . ' itemscope itemtype="https://schema.org/ImageObject">';

		$output .= '<div class="envira-gallery-item-inner">';
		$output  = apply_filters( 'envira_gallery_output_before_link', $output, $id, $item, $data, $i );

		// Top Left box.

		$css_class = false; // no css classes yet.
		$css_class = apply_filters( 'envira_gallery_output_dynamic_position_css', $css_class, $output, $id, $item, $data, $i, 'top-left' );

		$output .= '<div class="envira-gallery-position-overlay ' . $css_class . ' envira-gallery-top-left">';
		$output  = apply_filters( 'envira_gallery_output_dynamic_position', $output, $id, $item, $data, $i, 'top-left' );
		$output .= '</div>';

		// Top Right box.

		$css_class = false; // no css classes yet.
		$css_class = apply_filters( 'envira_gallery_output_dynamic_position_css', $css_class, $output, $id, $item, $data, $i, 'top-right' );

		$output .= '<div class="envira-gallery-position-overlay ' . $css_class . ' envira-gallery-top-right">';
		$output  = apply_filters( 'envira_gallery_output_dynamic_position', $output, $id, $item, $data, $i, 'top-right' );
		$output .= '</div>';

		// Bottom Left box.

		$css_class = false; // no css classes yet.
		$css_class = apply_filters( 'envira_gallery_output_dynamic_position_css', $css_class, $output, $id, $item, $data, $i, 'bottom-left' );

		$output .= '<div class="envira-gallery-position-overlay ' . $css_class . ' envira-gallery-bottom-left">';
		$output  = apply_filters( 'envira_gallery_output_dynamic_position', $output, $id, $item, $data, $i, 'bottom-left' );
		$output .= '</div>';

		// Bottom Right box.
		$css_class = false; // no css classes yet.
		$css_class = apply_filters( 'envira_gallery_output_dynamic_position_css', $css_class, $output, $id, $item, $data, $i, 'bottom-right' );

		$output .= '<div class="envira-gallery-position-overlay ' . $css_class . ' envira-gallery-bottom-right">';
		$output  = apply_filters( 'envira_gallery_output_dynamic_position', $output, $id, $item, $data, $i, 'bottom-right' );
		$output .= '</div>';

		$title = esc_html( wp_strip_all_tags( htmlspecialchars( $item['title'], ENT_QUOTES, 'UTF-8' ) ) );
		$title = htmlentities( $title, ENT_QUOTES, 'UTF-8' );

		// Caption.
		// Lite doesn't support captions, so we fallback to the title.
		$caption = $title;

		// Determine if we create a link.
		// If the user has disabled lightbox, there should not be a link.
		// "Turn off the lightbox then the image shouldn't be clicked".
		$create_link = ! empty( $item['link'] ) && (
			( $this->get_config( 'gallery_link_enabled', $data ) || $this->get_config( 'lightbox_enabled', $data ) )
		) ? true : false;

		// Filter the ability to create a link.
		$create_link = apply_filters( 'envira_gallery_create_link', $create_link, $data, $id, $item, $i, $this->is_mobile );

		if ( $create_link ) {
			$output .= '<a href="' . esc_url( $item['link'] ) . '" class="' . ( ( isset( $item['link_new_window'] ) && true === boolval( $item['link_new_window'] ) ) ? '' : 'envira-gallery-' . sanitize_html_class( $data['id'] ) . ' ' ) . 'envira-gallery-link" ' . $html5_attribute . '="enviragallery' . sanitize_html_class( $data['id'] ) . '" title="' . $title . '" data-envira-caption="' . $caption . '" data-envira-retina="' . ( isset( $item['lightbox_retina_image'] ) ? $item['lightbox_retina_image'] : '' ) . '" data-thumbnail="' . esc_url( $item['thumb'] ) . '"' . ( ( isset( $item['link_new_window'] ) && true === boolval( $item['link_new_window'] ) ) ? ' target="_blank"' : '' ) . ' ' . apply_filters( 'envira_gallery_output_link_attr', '', $id, $item, $data, $i ) . ' itemprop="contentUrl">';
		}

		$output = apply_filters( 'envira_gallery_output_before_image', $output, $id, $item, $data, $i );

		$gallery_theme = intval( $this->get_config( 'columns', $data ) ) === 0 ? ' envira-' . $this->get_config( 'justified_gallery_theme', $data ) : '';

		// Build the image and allow filtering.
		// How we build the html depends on the lazy load script.
		// Check if user has lazy loading on - if so, we add the css class.
		$envira_lazy_load = intval( $this->get_config( 'lazy_loading', $data ) ) === 1 ? 'envira-lazy' : '';

		// Determine/confirm the width/height of the image.
		// $placeholder should hold it but not for instagram.
		if ( 'mobile' === $this->is_mobile && ! $this->get_config( 'mobile', $data ) ) { // if the user is viewing on mobile AND user unchecked 'Create Mobile Gallery Images?' in mobile tab.
			$output_src = $item['src'];
		} elseif ( $this->get_config( 'crop', $data ) ) { // the user has selected the image to be cropped.
			$output_src = $imagesrc;
		} elseif ( $this->get_config( 'image_size', $data ) && $imagesrc ) { // use the image being provided thanks to the user selecting a unique image size.
			$output_src = $imagesrc;
		} elseif ( ! empty( $item['width'] ) ) {
			$output_src = $item['src'];
		} elseif ( ! empty( $placeholder[0] ) ) {
			$output_src = $placeholder[0];
		} else {
			$output_src = false;
		}

		if ( $this->get_config( 'crop', $data ) && $this->get_config( 'crop_width', $data ) && $this->get_config( 'image_size', $data ) !== 'full' ) {
			$output_width = $this->get_config( 'crop_width', $data );
		} elseif ( intval( $this->get_config( 'columns', $data ) ) !== 0 && $this->get_config( 'image_size', $data ) && $this->get_config( 'image_size', $data ) !== 'full' && $this->get_config( 'crop_width', $data ) && $this->get_config( 'crop_height', $data ) ) {
			$output_width = $this->get_config( 'crop_width', $data );
		} elseif ( ! empty( $item['width'] ) ) {
			$output_width = $item['width'];
		} elseif ( ! empty( $placeholder[1] ) ) {
			$output_width = $placeholder[1];
		} else {
			$output_width = false;
		}

		if ( $this->get_config( 'crop', $data ) && $this->get_config( 'crop_width', $data ) && $this->get_config( 'image_size', $data ) !== 'full' ) {
			$output_height = $this->get_config( 'crop_height', $data );
		} elseif ( intval( $this->get_config( 'columns', $data ) ) !== 0 && $this->get_config( 'image_size', $data ) && $this->get_config( 'image_size', $data ) !== 'full' && $this->get_config( 'crop_width', $data ) && $this->get_config( 'crop_height', $data ) ) {
			$output_height = $this->get_config( 'crop_height', $data );
		} elseif ( ! empty( $item['height'] ) ) {
			$output_height = $item['height'];
		} elseif ( ! empty( $placeholder[2] ) ) {
			$output_height = $placeholder[2];
		} else {
			$output_height = false;
		}

		if ( intval( $this->get_config( 'columns', $data ) ) === 0 ) {

			// Automatic.

			$output_item = '<img id="envira-gallery-image-' . sanitize_html_class( $id ) . '" class="envira-gallery-image envira-gallery-image-' . $i . $gallery_theme . ' ' . $envira_lazy_load . '" data-envira-index="' . $i . '" src="' . esc_url( $output_src ) . '"' . ( $this->get_config( 'dimensions', $data ) ? ' width="' . $this->get_config( 'crop_width', $data ) . '" height="' . $this->get_config( 'crop_height', $data ) . '"' : '' ) . ' data-envira-src="' . esc_url( $output_src ) . '" data-envira-gallery-id="' . $data['id'] . '" data-envira-item-id="' . $id . '" data-envira-caption="' . $caption . '" alt="' . esc_attr( $item['alt'] ) . '" title="' . wp_strip_all_tags( htmlspecialchars( $item['title'] ) ) . '" ' . apply_filters( 'envira_gallery_output_image_attr', '', $id, $item, $data, $i ) . ' itemprop="thumbnailUrl" data-envira-srcset="' . esc_url( $output_src ) . ' 400w,' . esc_url( $output_src ) . ' 2x" data-envira-width="' . $output_width . '" data-envira-height="' . $output_height . '" srcset="' . ( ( $envira_lazy_load ) ? 'data:image/gif;base64,R0lGODlhAQABAIAAAP///////yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==' : esc_url( $image_src_retina ) . ' 2x' ) . '" data-safe-src="' . ( ( $envira_lazy_load ) ? 'data:image/gif;base64,R0lGODlhAQABAIAAAP///////yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==' : esc_url( $output_src ) ) . '" />';

		} else {

			// Legacy.

			$output_item = false;

			if ( $envira_lazy_load ) {

				if ( $output_height > 0 && $output_width > 0 ) {
					$padding_bottom = ( $output_height / $output_width ) * 100;
				} else {
					// this shouldn't be happening, but this avoids a debug message.
					$padding_bottom = 100;
				}
				$output_item .= '<div class="envira-lazy" data-test-width="' . $output_width . '" data-test-height="' . $output_height . '" style="padding-bottom:' . $padding_bottom . '%;">';

			}

			$output_item .= '<img id="envira-gallery-image-' . sanitize_html_class( $id ) . '" class="envira-gallery-image envira-gallery-image-' . $i . $gallery_theme . '" data-envira-index="' . $i . '" src="' . esc_url( $output_src ) . '"' . ( $this->get_config( 'dimensions', $data ) ? ' width="' . $this->get_config( 'crop_width', $data ) . '" height="' . $this->get_config( 'crop_height', $data ) . '"' : '' ) . ' data-envira-src="' . esc_url( $output_src ) . '" data-envira-gallery-id="' . $data['id'] . '" data-envira-item-id="' . $id . '" data-envira-caption="' . $caption . '" alt="' . esc_attr( $item['alt'] ) . '" title="' . wp_strip_all_tags( htmlspecialchars( $title ) ) . '" ' . apply_filters( 'envira_gallery_output_image_attr', '', $id, $item, $data, $i ) . ' itemprop="thumbnailUrl" data-envira-srcset="' . esc_url( $output_src ) . ' 400w,' . esc_url( $output_src ) . ' 2x" srcset="' . ( ( $envira_lazy_load ) ? 'data:image/gif;base64,R0lGODlhAQABAIAAAP///////yH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==' : esc_url( $image_src_retina ) . ' 2x' ) . '" />';

			if ( $envira_lazy_load ) {

				$output_item .= '</div>';

			}
		}

				$output_item = apply_filters( 'envira_gallery_output_image', $output_item, $id, $item, $data, $i );

				// Add image to output.
				$output .= $output_item;
				$output  = apply_filters( 'envira_gallery_output_after_image', $output, $id, $item, $data, $i );

		if ( $create_link ) {
			$output .= '</a>';
		}

			$output  = apply_filters( 'envira_gallery_output_after_link', $output, $id, $item, $data, $i );
			$output .= '</div>';

		$output .= '</div>';
		$output  = apply_filters( 'envira_gallery_output_single_item', $output, $id, $item, $data, $i );

		// Append the image to the gallery output.
		$gallery .= $output;

		// Filter the output before returning.
		$gallery = apply_filters( 'envira_gallery_output_after_item', $gallery, $id, $item, $data, $i );

		return $gallery;
	}

	/**
	 * Adds a srcset attribute to an image if wp_get_attachment_image_srcset() exists in WordPress (4.4.+)
	 *
	 * This provides responsive compatibility and allows browsers to choose which image to download
	 *
	 * @since 1.4.0.3
	 *
	 * @param string $atts   Image Attributes.
	 * @param int    $id     ID.
	 * @param array  $item   Image.
	 * @param array  $data   Gallery Config.
	 * @param int    $i      Image Index.
	 * @return string           Image Attributes.
	 */
	public function add_image_srcset_attributes( $atts, $id, $item, $data, $i ) {

		// Check if wp_get_attachment_image_srcset() exists.
		if ( ! function_exists( 'wp_get_attachment_image_srcset' ) ) {
			return $atts;
		}

		// Check if item is a Media Library item.
		if ( ! is_numeric( $id ) ) {
			return $atts;
		}

		// Check if the gallery has cropping enabled. If so, don't apply srcset attribute, as it'll output.
		// images that don't conform to the user's requirements.
		if ( $this->get_config( 'crop', $data ) ) {
			return $atts;
		}

		// Iterate through WordPress' registered sizes to find the largest available size.
		// that doesn't exceed this Galleries image dimensions.
		$sizes     = get_intermediate_image_sizes();
		$max_width = 0;
		foreach ( $sizes as $size ) {
			// Get the width.
			$width = get_option( $size . '_size_w' );

			// Check if the width is smaller or equal to the gallery image width option.
			if ( $width <= $this->get_config( 'crop_width', $data ) && $width > $max_width ) {
				// This size can be used in the srcset.
				$max_width = $width;
				$max_size  = $size;
			}
		}

		// If no size meets our criteria, bail.
		if ( ! isset( $max_size ) ) {
			return $atts;
		}

		// Add the srcset and sizes based on $max_size.
		$atts .= ' srcset="' . wp_get_attachment_image_srcset( $id, $max_size ) . '"';
		$atts .= ' sizes="' . wp_calculate_image_sizes( [ $this->get_config( 'crop_width', $data ), $this->get_config( 'crop_height', $data ) ], $item['src'] ) . '"';

		// Return.
		return $atts;
	}

	/**
	 * Add the 'property' tag to stylesheets enqueued in the body.
	 *
	 * @param string $tag Stylesheet Tag.
	 * @since 1.4.1.1
	 */
	public function add_stylesheet_property_attribute( $tag ) {

		// If the <link> stylesheet is any Envira-based stylesheet, add the property attribute.
		if ( strpos( $tag, "id='envira-" ) !== false ) {
			$tag = str_replace( '/>', 'property="stylesheet" />', $tag );
		}

		return $tag;
	}

	/**
	 * Maybe sort the gallery images, if specified in the config
	 *
	 * Note: To ensure backward compat with the previous 'random' config
	 * key, the sorting parameter is still stored in the 'random' config
	 * key.
	 *
	 * @since 1.3.8
	 *
	 * @param   array $data       Gallery Config.
	 * @param   int   $gallery_id Gallery ID.
	 * @return  array               Gallery Config.
	 */
	public function maybe_sort_gallery( $data, $gallery_id ) {

		// Get sorting method.
		$sorting_method    = (string) $this->get_config( 'random', $data );
		$sorting_direction = $this->get_config( 'sorting_direction', $data );

		// Sort images based on method.
		switch ( $sorting_method ) {
			/**
			* Random.
			* - Again, by design, to ensure backward compat when upgrading from 1.3.7.x or older.
			* where we had a 'random' key = 0 or 1. Sorting was introduced in 1.3.8.
			*/
			case '1':
				// Shuffle keys.
				$keys = array_keys( $data['gallery'] );
				shuffle( $keys );

				// Rebuild array in new order.
				$new = [];
				foreach ( $keys as $key ) {
					$new[ $key ] = $data['gallery'][ $key ];
				}

				// Assign back to gallery.
				$data['gallery'] = $new;
				break;

			/**
			* Image Meta.
			*/
			case 'src':
			case 'title':
			case 'caption':
			case 'alt':
			case 'link':
				// Get metadata.
				$keys = [];
				foreach ( $data['gallery'] as $id => $item ) {
					$keys[ $id ] = wp_strip_all_tags( $item[ $sorting_method ] );
				}

				// Sort titles / captions.
				if ( 'ASC' === $sorting_direction ) {
					asort( $keys );
				} else {
					arsort( $keys );
				}

				// Iterate through sorted items, rebuilding gallery.
				$new = [];
				foreach ( $keys as $key => $title ) {
					$new[ $key ] = $data['gallery'][ $key ];
				}

				// Assign back to gallery.
				$data['gallery'] = $new;
				break;

			/**
			* Published Date.
			*/
			case 'date':
				// Get published date for each.
				$keys = [];
				foreach ( $data['gallery'] as $id => $item ) {
					// If the attachment isn't in the Media Library, we can't get a post date - assume now.
					$attachment = get_post( $id );
					if ( ! is_numeric( $id ) || ( false === $attachment ) ) {
						$keys[ $id ] = gmdate( 'Y-m-d H:i:s' );
					} else {
						$keys[ $id ] = $attachment->post_date;
					}
				}

				// Sort titles / captions.
				if ( 'ASC' === $sorting_direction ) {
					asort( $keys );
				} else {
					arsort( $keys );
				}

				// Iterate through sorted items, rebuilding gallery.
				$new = [];
				foreach ( $keys as $key => $title ) {
					$new[ $key ] = $data['gallery'][ $key ];
				}

				// Assign back to gallery.
				$data['gallery'] = $new;
				break;

			/**
			* None.
			* - Do nothing.
			*/
			case '0':
			case '':
				break;

			/**
			* If developers have added their own sort options, let them run them here
			*/
			default:
				$data = apply_filters( 'envira_gallery_sort_gallery', $data, $sorting_method, $gallery_id );
				break;

		}

		return $data;
	}

	/**
	 * Builds HTML for the Gallery Description
	 *
	 * @since 1.3.0.2
	 *
	 * @param string $gallery Gallery HTML.
	 * @param array  $data Data.
	 * @return string Gallery HTML.
	 */
	public function description( $gallery, $data ) {

		$gallery    .= '<div class="envira-gallery-description envira-gallery-description-above" style="padding-bottom: ' . $this->get_config( 'margin', $data ) . 'px;">';
			$gallery = apply_filters( 'envira_gallery_output_before_description', $gallery, $data );

			// Get description.
			$description = $data['config']['description'];

			// If the WP_Embed class is available, use that to parse the content using registered oEmbed providers.
		if ( isset( $GLOBALS['wp_embed'] ) ) {
			$description = $GLOBALS['wp_embed']->autoembed( $description );
		}

			// Get the description and apply most of the filters that apply_filters( 'the_content' ) would use
			// We don't use apply_filters( 'the_content' ) as this would result in a nested loop and a failure.
			$description = wptexturize( $description );
			$description = convert_smilies( $description );
			$description = wpautop( $description );
			$description = prepend_attachment( $description );

			// Requires WordPress 4.4+.
		if ( function_exists( 'wp_make_content_images_responsive' ) ) {
			$description = wp_filter_content_tags( $description );
		}

			// Append the description to the gallery output.
			$gallery .= $description;

			// Filter the gallery HTML.
			$gallery = apply_filters( 'envira_gallery_output_after_description', $gallery, $data );
		$gallery    .= '</div>';

		return $gallery;
	}


	/**
	 * Outputs the gallery init script in the footer.
	 *
	 * @since 1.0.0
	 */
	public function gallery_init() {

		// envira_galleries stores all Fancybox instances.
		// envira_isotopes stores all Isotope instances.
		// envira_isotopes_config stores Isotope configs for each Gallery.
		?>

		<script type="text/javascript">
			<?php ob_start(); ?>

			var envira_galleries = [],
				envira_gallery_images = [],
				envira_isotopes = [],
				envira_isotopes_config = [];

			jQuery(document).ready(function($){

				<?php
				do_action( 'envira_gallery_api_start_global' );
				foreach ( $this->data as $data ) {
					// Prevent multiple init scripts for the same gallery ID.
					if ( in_array( $data['id'], $this->done, true ) ) {
						continue;
					}
					$this->done[] = $data['id'];

					do_action( 'envira_gallery_api_start', $data );

					// Define container.
					?>
					var envira_container_<?php echo intval( $data['id'] ); ?> = '';

				function envira_album_lazy_load_image( $id ) {

					<?php if ( $this->get_config( 'lazy_loading', $data ) ) { ?>

				var responsivelyLazy = window.responsivelyLazy;

					responsivelyLazy.run('#envira-gallery-'+ $id);

					<?php } else { ?>
						/* to do: enable ENVIRA DEBUG to display this and other errors */
						/* console.log ('load_images was pinged, but lazy load turned off'); */
					<?php } ?>

				}

					<?php if ( intval( $this->get_config( 'columns', $data ) ) === 0 ) : ?>

						<?php

						// if the user has selected a custom theme, only output the needed JS.
						$gallery_theme = $this->get_config( 'justified_gallery_theme', $data );

						// in some cases, previous gallery using the old automattic layout aren't showing a row height, so just in case...
						$justified_row_height = $this->get_config( 'justified_row_height', $data ) ? $this->get_config( 'justified_row_height', $data ) : 150;

						?>

						$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').enviraJustifiedGallery({
							rowHeight : <?php echo esc_js( $justified_row_height ); ?>,
							maxRowHeight: -1,
							waitThumbnailsLoad: true,
							selector: '> div > div',
							lastRow: 'nojustify',
							border: 0,
							margins: <?php echo $this->get_config( 'justified_margins', $data ) ? intval( $this->get_config( 'justified_margins', $data ) ) : 1; ?>,

						});

						$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').justifiedGallery().on('jg.complete', function (e) {

							envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
							$(window).scroll(function(event){
								envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
							});

						});

						$( document ).on( "envira_pagination_ajax_load_completed", function() {
							$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').justifiedGallery().on('jg.complete', function (e) {

								envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
								$(window).scroll(function(event){
									envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
								});

							});
						});

						<?php if ( 'js-desaturate' === $gallery_theme || 'js-threshold' === $gallery_theme || 'js-blur' === $gallery_theme || 'js-vintage' === $gallery_theme ) : ?>

						$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').on('jg.complete', function (e) {
							if( navigator.userAgent.match(/msie/i) || $.browser.msie || navigator.appVersion.indexOf('Trident/') > 0 ) {
								$('#envira-gallery-<?php echo intval( $data['id'] ); ?> img').each(function() {
									var keep_id = $(this).attr('id');
									$(this).attr('id', keep_id + '-effects' );
									$(this).wrap('<div class="effect-wrapper" style="display:inline-block;width:' + this.width + 'px;height:' + this.height + 'px;">').clone().addClass('gotcolors').css({'position': 'absolute', 'opacity' : 0, 'z-index' : 1 }).attr('id', keep_id).insertBefore(this);
									<?php

									switch ( $gallery_theme ) {
										case 'js-desaturate':
											echo 'this.src = jg_effect_desaturate($(this).attr("src"));';
											break;
										case 'js-threshold':
											echo 'this.src = jg_effect_threshold(this.src);';
											break;
										case 'js-blur':
											echo 'this.src = jg_effect_blur(this.src);';
											break;
										case 'js-vintage':
											echo 'jg_effect_vintage( this );';
											break;
									}

									?>
								});
								$('#envira-gallery-<?php echo intval( $data['id'] ); ?> img').hover(
									function() {
										$(this).stop().animate({opacity: 1}, 200);
									},
									function() {
										$(this).stop().animate({opacity: 0}, 200);
									}
								);
							}
							else {

								$('#envira-gallery-<?php echo intval( $data['id'] ); ?> img').hover(
									function() {
										$(this).removeClass('envira-<?php echo esc_attr( $gallery_theme ); ?>');
									},
									function() {
										$(this).addClass('envira-<?php echo esc_attr( $gallery_theme ); ?>');
									}
								);
							}


						});

						<?php endif; ?>

						$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').css('opacity', '1');

					<?php endif; ?>

					<?php
					// Isotope: Start.
					if ( intval( $this->get_config( 'columns', $data ) ) > 0 && $this->get_config( 'isotope', $data ) ) {
						// Define config for this Isotope Gallery.
						?>
						envira_isotopes_config['<?php echo intval( $data['id'] ); ?>'] = {
							<?php do_action( 'envira_gallery_api_enviratope_config', $data ); ?>
							itemSelector: '.envira-gallery-item',
								masonry: {
									columnWidth: '.envira-gallery-item'
								}
						};
						<?php
						// Initialize Isotope.
						?>
						envira_isotopes['<?php echo intval( $data['id'] ); ?>'] = envira_container_<?php echo intval( $data['id'] ); ?>
																	= $('#envira-gallery-<?php echo intval( $data['id'] ); ?>').masonry(envira_isotopes_config['<?php echo intval( $data['id'] ); ?>']);

						$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').on( 'layoutComplete',
						function( event, laidOutItems ) {
							envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
							$(window).scroll(function(event){
								envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
							});
						}
						);

						$( document ).on( "envira_pagination_ajax_load_completed", function() {
							$('#envira-gallery-<?php echo intval( $data['id'] ); ?>').on( 'layoutComplete',
							function( event, laidOutItems ) {
								envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
								$(window).scroll(function(event){
									envira_album_lazy_load_image(<?php echo intval( $data['id'] ); ?>);
								});
							}
							);
						});

						<?php
						// Re-layout Isotope when each image loads.
						?>
						envira_isotopes['<?php echo intval( $data['id'] ); ?>'].enviraImagesLoaded()
							.done(function() {
								envira_isotopes['<?php echo intval( $data['id'] ); ?>'].masonry('layout');
							})
							.progress(function() {
								envira_isotopes['<?php echo intval( $data['id'] ); ?>'].masonry('layout');
							});
						<?php
						do_action( 'envira_gallery_api_enviratope', $data );
					}
					// Isotope: End.

					// CSS Animations: Start.
					if ( $this->get_config( 'css_animations', $data ) ) {
						$opacity = $this->get_config( 'css_opacity', $data );

						// Defaults Addon Gallery may not have been saved since opacity introduction, so force a value if one doesn't exist.
						if ( empty( $opacity ) ) {
							$opacity = 100;
						}

						// Reduce to factor of 1.
						$opacity = ( $opacity / 100 );
						?>
						envira_container_<?php echo intval( $data['id'] ); ?> = $('#envira-gallery-<?php echo intval( $data['id'] ); ?>').enviraImagesLoaded( function() {
							$('.envira-gallery-item img').fadeTo( 'slow', <?php echo intval( $opacity ); ?> );
						});
						<?php
					}
					// CSS Animations: End.

					// Fancybox: Start.
					if ( $this->get_config( 'lightbox_enabled', $data ) ) {
						// By default, we'll use the images in the Gallery DOM to populate the lightbox.
						// However, Addons (e.g. Pagination) may require us to give access to all images
						// in a Gallery, not just the paginated subset on screen.
						// Those Addons can populate this array now which will tell envirabox which images to use.
						$lightbox_images = apply_filters( 'envira_gallery_lightbox_images', false, $data );
						$title_fb1       = apply_filters( 'envira_gallery_title_display', $this->get_config( 'title_display', $data ), $data );
						if ( 'float_wrap' === $title_fb1 || 'float' === $title_fb1 ) {
							$title_fb1 = 'float';
						}

						?>
						envira_gallery_options = {
							// All of these options are for Envira Lite with Fancybox 1 to try and match Fancybox 2.
							padding: <?php echo in_array( $this->get_config( 'lightbox_theme', $data ), [ 'base_dark' ], true ) ? '4' : '15'; ?>,
							cyclic: true,
							titlePosition: '<?php echo esc_js( $title_fb1 ); ?>',
							// End options for Envira Lite with Fancybox 1.

							<?php do_action( 'envira_gallery_api_config', $data ); // Depreciated. ?>
							<?php do_action( 'envira_gallery_api_envirabox_config', $data ); ?>
							<?php if ( ! $this->get_config( 'keyboard', $data ) ) : ?>
							keys: 0,
							<?php endif; ?>
							margin: <?php echo intval( apply_filters( 'envirabox_margin', 60, $data ) ); ?>,
							arrows: <?php echo in_array( $this->get_config( 'lightbox_theme', $data ), [ 'base_dark' ], true ) ? 'true' : esc_attr( $this->get_config( 'arrows', $data ) ); ?>,
							aspectRatio: <?php echo esc_js( apply_filters( 'envirabox_aspect', $this->get_config( 'aspect', $data ), $data ) ); ?>,
							loop: <?php echo esc_js( apply_filters( 'envirabox_loop', $this->get_config( 'loop', $data ), $data ) ); ?>,
							mouseWheel: <?php echo esc_js( apply_filters( 'mousewheel', $this->get_config( 'mousewheel', $data ), $data ) ); ?>,
							preload: 1,

							<?php
							/* Get open and transition effects */
							$lightbox_open_close_effect = apply_filters( 'lightbox_open_close_effect', $this->get_config( 'lightbox_open_close_effect', $data ), $data );
							$lightbox_transition_effect = apply_filters( 'effect', $this->get_config( 'effect', $data ), $data );

							/* Get standard effects */
							$lightbox_standard_effects = $this->common->get_transition_effects_values();

							/* If open/close is standard, use openEffect, closeEffect */
							if ( in_array( $lightbox_open_close_effect, $lightbox_standard_effects, true ) ) {
								?>
								openEffect: '<?php echo esc_attr( $lightbox_open_close_effect ); ?>',
								closeEffect: '<?php echo esc_attr( $lightbox_open_close_effect ); ?>',
								<?php
							} else {

								// easing effects have been depreciated, and will default back to 'swing'.
								?>
								openEffect: 'elastic', // FancyBox default is fade, should be elastic for the openEffect/closeEffect functions
								closeEffect: 'elastic',
								openEasing: 'swing', // <?php echo ( 'Swing' === $lightbox_open_close_effect ? 'swing' : 'easeIn' ); ?>',
								closeEasing: 'swing', // <?php echo ( 'Swing' === $lightbox_open_close_effect ? 'swing' : 'easeOut' ); ?>',
								openSpeed: <?php echo intval( apply_filters( 'envirabox_open_speed', 500, $data ) ); ?>,
								closeSpeed: <?php echo intval( apply_filters( 'envirabox_close_speed', 500, $data ) ); ?>,
								<?php
							}

							/* If transition effect is standard, use nextEffect, prevEffect */
							if ( in_array( $lightbox_transition_effect, $lightbox_standard_effects, true ) ) {
								?>
								nextEffect: '<?php echo esc_attr( $lightbox_transition_effect ); ?>',
								prevEffect: '<?php echo esc_attr( $lightbox_transition_effect ); ?>',
								<?php
							} else {

								// easing effects have been depreciated, and will default back to 'swing'.
								?>
								nextEasing: 'swing', // '<?php echo ( 'Swing' === $lightbox_transition_effect ? 'swing' : 'easeIn' ); ?>',
								prevEasing: 'swing', // '<?php echo ( 'Swing' === $lightbox_transition_effect ? 'swing' : 'easeOut' ); ?>',
								nextSpeed: <?php echo intval( apply_filters( 'envirabox_next_speed', 600, $data ) ); ?>,
								prevSpeed: <?php echo intval( apply_filters( 'envirabox_prev_speed', 600, $data ) ); ?>,
								<?php
							}
							?>
							tpl: {
								wrap     : '<?php echo $this->get_lightbox_template( $data ); // @codingStandardsIgnoreLine - Markup escapped via function ?>',
								image    : '<img class="envirabox-image" src="{href}" alt="" data-envira-title="" data-envira-caption="" data-envira-index="" data-envira-data="" />',
								iframe   : '<iframe id="envirabox-frame{rnd}" name="envirabox-frame{rnd}" class="envirabox-iframe" frameborder="0" vspace="0" hspace="0" allowtransparency="true" wekitallowfullscreen mozallowfullscreen allowfullscreen></iframe>',
								error    : '<p class="envirabox-error"><?php echo esc_attr__( 'The requested content cannot be loaded.<br/>Please try again later.</p>', 'envira-gallery-lite' ); ?>',
								closeBtn : '<a title="<?php echo esc_attr__( 'Close', 'envira-gallery-lite' ); ?>" class="envirabox-item envirabox-close" href="#"></a>',
								next     : '<a title="<?php echo esc_attr__( 'Next', 'envira-gallery-lite' ); ?>" class="envirabox-nav envirabox-next envirabox-arrows-<?php echo esc_attr( $this->get_config( 'arrows_position', $data ) ); ?>" href="#"><span></span></a>',
								prev     : '<a title="<?php echo esc_attr__( 'Previous', 'envira-gallery-lite' ); ?>" class="envirabox-nav envirabox-prev envirabox-arrows-<?php echo esc_attr( $this->get_config( 'arrows_position', $data ) ); ?>" href="#"><span></span></a>'
								<?php do_action( 'envira_gallery_api_templates', $data ); ?>
							},
							helpers: {
								<?php
								do_action( 'envira_gallery_api_helper_config', $data );
								// Grab title display.
								$title_display = $this->get_config( 'title_display', $data );
								if ( 'float_wrap' === $title_display ) {
									$title_display = 'float';
								}
								?>
								title: {
									<?php do_action( 'envira_gallery_api_title_config', $data ); ?>
									type: '<?php echo esc_attr( $title_display ); ?>'
								},
								<?php if ( $this->get_config( 'thumbnails', $data ) ) : ?>
								thumbs: {
									width: <?php echo esc_attr( $this->get_config( 'thumbnails_width', $data ) ); ?>,
									height: <?php echo esc_attr( $this->get_config( 'thumbnails_height', $data ) ); ?>,
									source: function(current) {
										if ( typeof current.element == 'undefined' ) {
											return current.thumbnail;
										} else {
											return $(current.element).data('thumbnail');
										}
									},
									<?php if ( in_array( $this->get_config( 'lightbox_theme', $data ), [ 'base_dark' ], true ) ) : ?>
									dynamicMargin: true,
									<?php endif; ?>
									position: '
									<?php
									echo in_array( $this->get_config( 'lightbox_theme', $data ), [ 'base_dark' ], true )
													? 'bottom'
													: esc_attr( $this->get_config( 'thumbnails_position', $data ) );
									?>
													'
								},
								<?php endif; ?>
								<?php if ( $this->get_config( 'toolbar', $data ) && ! in_array( $this->get_config( 'lightbox_theme', $data ), [ 'base_dark' ], true ) ) : ?>
								buttons: {
									tpl: '<?php echo $this->get_toolbar_template( $data ); // @codingStandardsIgnoreLine ?>',
									position: '<?php echo esc_attr( $this->get_config( 'toolbar_position', $data ) ); ?>',
									padding: '<?php echo ( ( 'bottom' === $this->get_config( 'toolbar_position', $data ) && $this->get_config( 'thumbnails', $data ) && $this->get_config( 'thumbnails_position', $data ) === 'bottom' ) ? true : false ); ?>'
								},
								<?php endif; ?>
							},
							<?php do_action( 'envira_gallery_api_config_callback', $data ); ?>
							beforeLoad: function(){

								<?php

								if ( ! $lightbox_images ) {

									?>
								this.title = $(this.element).attr('data-envira-caption');
									<?php

								} else {

									?>
								this.title = this.group[ this.index ].caption;
									<?php

								}

								?>

								<?php do_action( 'envira_gallery_api_before_load', $data ); ?>
							},
							afterLoad: function(){
								$('envirabox-overlay-fixed').on({
									'touchmove' : function(e){
										e.preventDefault();
									}
								});

								<?php do_action( 'envira_gallery_api_after_load', $data ); ?>
							},
							beforeShow: function(){
								$(window).on({
									'resize.envirabox' : function(){
										$.envirabox.update();
									}
								});

								<?php
								// Set data attributes on the lightbox image, based on either.
								// the image in the DOM or (if $lightbox_images defined) the image.
								// from $lightbox_images.
								?>
								if ( typeof this.element === 'undefined' ) {
									<?php
									// Using $lightbox_images.
									?>
									var gallery_id = this.group[ this.index ].gallery_id;
									var gallery_item_id = this.group[ this.index ].id;
									var alt = this.group[ this.index ].alt;
									var title = this.group[ this.index ].title;
									var caption = this.group[ this.index ].caption;
									var index = this.index;
								} else {
									<?php
									// Using image from DOM.
									// Get a bunch of data attributes from clicked image link.
									?>
									var gallery_id = this.element.find('img').data('envira-gallery-id');
									var gallery_item_id = this.element.find('img').data('envira-item-id');
									var alt = this.element.find('img').attr('alt');
									var title = this.element.find('img').parent().attr('title');
									var caption = this.element.find('img').parent().data('envira-caption');
									var retina_image = this.element.find('img').parent().data('envira-retina');
									var index = this.element.find('img').data('envira-index');
								}

								<?php
								// Set alt, data-envira-title, data-envira-caption and data-envira-index attributes on Lightbox image.
								?>
								this.inner.find('img').attr('alt', alt)
													.attr('data-envira-gallery-id', gallery_id)
													.attr('data-envira-item-id', gallery_item_id)
													.attr('data-envira-title', title)
													.attr('data-envira-caption', caption)
													.attr('data-envira-index', index);

								<?php
								// Set retina image srcset if specified.
								?>
								if ( typeof retina_image !== 'undefined' && retina_image !== '' ) {
									this.inner.find('img').attr('srcset', retina_image + ' 2x');
								}

								<?php do_action( 'envira_gallery_api_before_show', $data ); ?>
							},
							onStart: function(){
																$('#envirabox-wrap, #envirabox-wrap #envirabox-left, #envirabox-wrap #envirabox-right').swipe( {
									excludedElements:"label, button, input, select, textarea, .noSwipe",
									swipe: function(event, direction, distance, duration, fingerCount, fingerData) {
										if (direction === 'left') {
											$.envirabox.next(direction);
										} else if (direction === 'right') {
											$.envirabox.prev(direction);
										} else if (direction === 'up') {
											$.envirabox.close();
										}
									}
								} );
								<?php

								// If title helper = float_wrap, add a CSS class so we can disable word-wrap.
								if ( $this->get_config( 'title_display', $data ) === 'float_wrap' ) {
									?>
									if ( typeof this.helpers.title !== 'undefined' ) {
										if ( ! $( 'div.envirabox-title' ).hasClass( 'envirabox-title-text-wrap' ) ) {
											$( 'div.envirabox-title' ).addClass( 'envirabox-title-text-wrap' );
										}
									}
									<?php
								}

								do_action( 'envira_gallery_api_after_show', $data );
								?>
							},
							beforeClose: function(){
								<?php do_action( 'envira_gallery_api_before_close', $data ); ?>
							},
							afterClose: function(){
								$(window).off('resize.envirabox');
								<?php do_action( 'envira_gallery_api_after_close', $data ); ?>
							},
							onUpdate: function(){
								<?php
								if ( $this->get_config( 'toolbar', $data ) ) :
									?>
								var envira_buttons_<?php echo intval( $data['id'] ); ?> = $('#envirabox-buttons li').map(function(){
									return $(this).width();
								}).get(),
									envira_buttons_total_<?php echo intval( $data['id'] ); ?> = 0;
								$.each(envira_buttons_<?php echo intval( $data['id'] ); ?>, function(i, val){
									envira_buttons_total_<?php echo intval( $data['id'] ); ?> += parseInt(val, 10);
								});

								envira_buttons_total_<?php echo intval( $data['id'] ); ?> += 1;

								$('#envirabox-buttons ul').width(envira_buttons_total_<?php echo intval( $data['id'] ); ?>);
								$('#envirabox-buttons').width(envira_buttons_total_<?php echo intval( $data['id'] ); ?>).css('left', ($(window).width() - envira_buttons_total_<?php echo intval( $data['id'] ); ?>)/2);
								<?php endif; ?>

								<?php do_action( 'envira_gallery_api_on_update', $data ); ?>
							},
							onCancel: function(){
								<?php do_action( 'envira_gallery_api_on_cancel', $data ); ?>
							},
							onPlayStart: function(){
								<?php do_action( 'envira_gallery_api_on_play_start', $data ); ?>
							},
							onPlayEnd: function(){
								<?php do_action( 'envira_gallery_api_on_play_end', $data ); ?>
							}
						};
						<?php

						if ( ! $lightbox_images ) {
							// No lightbox images specified, so use images from DOM.
							?>
							envira_galleries['<?php echo intval( $data['id'] ); ?>'] = $('.envira-gallery-<?php echo intval( $data['id'] ); ?>').envirabox( envira_gallery_options );
							<?php

						} else {
							// Use images from $lightbox_images.
							add_filter( 'envira_minify_strip_double_forward_slashes', '__return_false' );
							?>
							envira_gallery_images[<?php echo intval( $data['id'] ); ?>] = [];
							<?php
							// Build a JS array of all images.
							$count = 0;
							foreach ( $lightbox_images as $image_id => $image ) {
								// If no image ID exists, skip.
								if ( empty( $image_id ) ) {
									continue;
								}

								$title = wp_strip_all_tags( htmlspecialchars( $image['title'], ENT_QUOTES, 'UTF-8' ) );
								$title = htmlentities( $title, ENT_QUOTES, 'UTF-8' );

								?>
								envira_gallery_images[<?php echo intval( $data['id'] ); ?>].push({
									href: '<?php echo esc_url( $image['src'] ); ?>',
									gallery_id: <?php echo intval( $data['id'] ); ?>,
									id: <?php echo intval( $image_id ); ?>,
									alt: '<?php echo esc_attr( addslashes( str_replace( "\n", '<br />', $image['alt'] ) ) ); ?>',
									caption: '<?php echo esc_html( $title ); ?>',
									title: '<?php echo esc_html( $title ); ?>',
									index: <?php echo intval( $count ); ?>,
									thumbnail: '<?php echo ( isset( $image['thumb'] ) ? esc_js( $image['thumb'] ) : '' ); ?>'
									<?php do_action( 'envira_gallery_api_lightbox_image_attributes', $image, $image_id, $lightbox_images, $data ); ?>
								});
								<?php
								++$count;
							}

							// Open envirabox when an image is clicked, telling envirabox which images are available to it.
							?>
							envira_galleries['<?php echo intval( $data['id'] ); ?>'] = $('.envira-gallery-<?php echo intval( $data['id'] ); ?>').envirabox( envira_gallery_options, envira_gallery_images[<?php echo intval( $data['id'] ); ?>] );
							$('#envira-gallery-wrap-<?php echo intval( $data['id'] ); ?>').on('click', 'a.envira-gallery-link', function(e) {
								e.preventDefault();
								e.stopPropagation();
								$.envirabox.close();
								if ( $('#envira-gallery-wrap-<?php echo intval( $data['id'] ); ?> div.envira-pagination').length > 0 ) { /* this exists, perhaps pagination doesn't display nav because it's only one page? */
									var envirabox_page = ( $('#envira-gallery-wrap-<?php echo intval( $data['id'] ); ?> div.envira-pagination').data('page') - 1 );
								} else {
									var envirabox_page = 0;
								}

								if ( $('#envira-gallery-wrap-<?php echo intval( $data['id'] ); ?> div.envira-pagination').length > 0 ) { /* this exists, perhaps pagination doesn't display nav because it's only one page? */
									var envirabox_per_page = $('#envira-gallery-wrap-<?php echo intval( $data['id'] ); ?> div.envira-pagination').data('per-page');
								} else {
									var envirabox_per_page = $('.envira-gallery-image').length;
								}

								var envirabox_index = ( Number($('img', $(this)).data('envira-index')) - 1 );
								envira_gallery_options.index = ((envirabox_page * envirabox_per_page) + envirabox_index);
								envira_galleries['<?php echo intval( $data['id'] ); ?>'] = $.envirabox( envira_gallery_images[<?php echo intval( $data['id'] ); ?>], envira_gallery_options );
							});
							<?php
						}

						do_action( 'envira_gallery_api_lightbox', $data );
						// Fancybox: End.
					}

					do_action( 'envira_gallery_api_end', $data );
				} // foreach

				do_action( 'envira_gallery_api_end_global', $this->data );

				?>
			});

			<?php

			if ( defined( 'SCRIPT_DEBUG' ) && ! SCRIPT_DEBUG ) {

				// Minify before outputting to improve page load time.
				echo $this->minify( ob_get_clean() ); // @codingStandardsIgnoreLine

			} else {

				echo ob_get_clean(); // @codingStandardsIgnoreLine

			}

			?>
		</script>
		<?php
	}

	/**
	 * Loads a custom gallery display theme.
	 *
	 * @since 1.0.0
	 *
	 * @param string $theme The custom theme slug to load.
	 */
	public function load_gallery_theme( $theme ) {

		// Loop through the available themes and enqueue the one called.
		foreach ( $this->common->get_gallery_themes() as $array => $data ) {
			if ( $theme !== $data['value'] ) {
				continue;
			}
			if ( file_exists( plugin_dir_path( $data['file'] ) . 'themes/' . $theme . '/style.css' ) ) {
				wp_enqueue_style( $this->base->plugin_slug . $theme . '-theme', plugins_url( 'themes/' . $theme . '/style.css', $data['file'] ), [ $this->base->plugin_slug . '-style' ], $this->base->version );
			} else {
				wp_enqueue_style( $this->base->plugin_slug . $theme . '-theme', plugins_url( 'themes/' . $theme . '/css/style.css', $data['file'] ), [ $this->base->plugin_slug . '-style' ], $this->base->version );
			}
			break;
		}
	}

	/**
	 * Loads a custom gallery lightbox theme.
	 *
	 * @since 1.0.0
	 *
	 * @param string $theme The custom theme slug to load.
	 */
	public function load_lightbox_theme( $theme ) {

		// Loop through the available themes and enqueue the one called.
		foreach ( $this->common->get_lightbox_themes() as $array => $data ) {
			if ( $theme !== $data['value'] ) {
				continue;
			}

			if ( file_exists( plugin_dir_path( $data['file'] ) . 'themes/' . $theme . '/style.css' ) ) {
				wp_enqueue_style( $this->base->plugin_slug . $theme . '-theme', plugins_url( 'themes/' . $theme . '/style.css', $data['file'] ), [ $this->base->plugin_slug . '-style' ], $this->base->version );
			} else {
				wp_enqueue_style( $this->base->plugin_slug . $theme . '-theme', plugins_url( 'themes/' . $theme . '/css/style.css', $data['file'] ), [ $this->base->plugin_slug . '-style' ], $this->base->version );
			}
			break;
		}
	}

	/**
	 * Helper method for adding custom gallery classes.
	 *
	 * @since 1.0.0
	 *
	 * @param array $data The gallery data to use for retrieval.
	 * @return string     String of space separated gallery classes.
	 */
	public function get_gallery_classes( $data ) {

		// Set default class.
		$classes   = [];
		$classes[] = 'envira-gallery-wrap';

		// Add custom class based on data provided.
		$classes[] = 'envira-gallery-theme-' . $this->get_config( 'gallery_theme', $data );
		$classes[] = 'envira-lightbox-theme-' . $this->get_config( 'lightbox_theme', $data );

		// If we have custom classes defined for this gallery, output them now.
		foreach ( (array) $this->get_config( 'classes', $data ) as $class ) {
			$classes[] = sanitize_html_class( wp_unslash( $class ) );
		}

		// If the gallery has RTL support, add a class for it.
		if ( $this->get_config( 'rtl', $data ) ) {
			$classes[] = 'envira-gallery-rtl';
		}

		// Allow filtering of classes and then return what's left.
		$classes = apply_filters( 'envira_gallery_output_classes', $classes, $data );
		return trim( implode( ' ', array_map( 'trim', array_map( 'sanitize_html_class', array_unique( $classes ) ) ) ) );
	}

	/**
	 * Helper method for adding custom gallery classes.
	 *
	 * @since 1.0.4
	 *
	 * @param array $item Array of item data.
	 * @param int   $i      The current position in the gallery.
	 * @param array $data The gallery data to use for retrieval.
	 * @return string     String of space separated gallery item classes.
	 */
	public function get_gallery_item_classes( $item, $i, $data ) {

		// Set default class.
		$classes   = [];
		$classes[] = 'envira-gallery-item';
		$classes[] = 'enviratope-item';
		$classes[] = 'envira-gallery-item-' . $i;

		// If lazy load exists, add that.
		if ( isset( $data['config']['lazy_loading'] ) && $data['config']['lazy_loading'] ) {
			$classes[] = 'envira-lazy-load';
		}

		// Allow filtering of classes and then return what's left.
		$classes = apply_filters( 'envira_gallery_output_item_classes', $classes, $item, $i, $data );
		return trim( implode( ' ', array_map( 'trim', array_map( 'sanitize_html_class', array_unique( $classes ) ) ) ) );
	}

	/**
	 * Changes the link attribute of an image, if the Lightbox config
	 * requires a different sized image to be displayed.
	 *
	 * @since 1.3.6
	 *
	 * @param int   $id      The image attachment ID to use.
	 * @param array $item  Gallery item data.
	 * @param array $data  The gallery data to use for retrieval.
	 * @return array       Image array
	 */
	public function maybe_change_link( $id, $item, $data ) {

		// Check gallery config.
		$image_size = $this->get_config( 'lightbox_image_size', $data );

		// check if the url is a valid image if not return it.
		if ( ! $this->is_image( $item['link'] ) ) {
			return $item;
		}

		// Return if we are serving a full size image.
		if ( 'default' === $image_size || 'full_width' === $image_size ) {
			return $item;
		}

		// Get media library attachment at requested size.
		$image = wp_get_attachment_image_src( $id, $image_size );

		if ( ! is_array( $image ) ) {
			return $item;
		}

		// Inject new image size into $item.
		$item['link'] = $image[0];

		// Return.
		return $item;
	}

	/**
	 * Helper method to retrieve the proper image src attribute based on gallery settings.
	 *
	 * @since 1.0.0
	 *
	 * @param int   $id            The image attachment ID to use.
	 * @param array $item          Gallery item data.
	 * @param array $data          The gallery data to use for retrieval.
	 * @param bool  $mobile        Whether or not to retrieve the mobile image.
	 * @param bool  $retina        Whether to return a retina sized image.
	 * @return string              The proper image src attribute for the image.
	 */
	public function get_image_src( $id, $item, $data, $mobile = false, $retina = false ) {

		// Define variable.
		$src = false;

		// If this image is an instagram, we grab the src and don't use the $id.
		// otherwise using the $id refers to a postID in the database and has been known.
		// at times to pull up the wrong thumbnail instead of the instagram one.

		$instagram = false;

		if ( ! empty( $item['src'] ) && strpos( $item['src'], 'cdninstagram' ) !== false ) :
			// using 'cdninstagram' because it seems all urls contain it - but be watchful in the future.
			$instagram = true;
			$src       = $item['src'];
			$image     = $item['src'];
		endif;

		$image_size = $this->get_config( 'image_size', $data );

		if ( ! $src ) :

			// Check if this Gallery uses a WordPress defined image size.
			if ( 'default' !== $image_size && ! $retina ) {
				// If image size is envira_gallery_random, get a random image size.
				if ( 'envira_gallery_random' === $image_size ) {

					// Get random image sizes that have been chosen for this Gallery.
					$image_sizes_random = (array) $this->get_config( 'image_sizes_random', $data );

					if ( count( $image_sizes_random ) === 0 ) {
						// The user didn't choose any image sizes - use them all.
						$wordpress_image_sizes           = $this->common->get_image_sizes( true );
						$wordpress_image_size_random_key = array_rand( $wordpress_image_sizes, 1 );
						$image_size                      = $wordpress_image_sizes[ $wordpress_image_size_random_key ]['value'];
					} else {
						$wordpress_image_size_random_key = array_rand( $image_sizes_random, 1 );
						$image_size                      = $image_sizes_random[ $wordpress_image_size_random_key ];
					}

					// Get the random WordPress defined image size.
					$src = wp_get_attachment_image_src( $id, $image_size );
				} else {
					// Get the requested WordPress defined image size.
					$src = wp_get_attachment_image_src( $id, $image_size );
				}
			} elseif ( ! $retina ) {

					$isize = $this->find_clostest_size( $data ) !== '' ? $this->find_clostest_size( $data ) : 'full';
					$isize = ( 'full' === $image_size ) ? 'full' : $isize;
					$src   = apply_filters( 'envira_gallery_retina_image_src', wp_get_attachment_image_src( $id, $isize ), $id, $item, $data, $this->is_mobile );

			} else {
				$src = apply_filters( 'envira_gallery_retina_image_src', wp_get_attachment_image_src( $id, 'full' ), $id, $item, $data, $this->is_mobile );
			}

		endif;

		// Check if this returned an image.
		if ( ! $src ) {
			// Fallback to the $item's image source.
			$image = $item['src'];
		} elseif ( ! $instagram ) {
			$image = $src[0];
		}

		// If we still don't have an image at this point, something went wrong.
		if ( ! isset( $image ) ) {
			return apply_filters( 'envira_gallery_no_image_src', $item['link'], $id, $item, $data );
		}

		// Prep our indexable images.
		if ( $image && ! $this->is_mobile ) {
			$this->index[ $data['id'] ][ $id ] = [
				'src' => $image,
				'alt' => ! empty( $item['alt'] ) ? $item['alt'] : '',
			];
		}

		// If the current layout is justified/automatic.
		// if the image size is a WordPress size and we're not requesting a retina image we don't need to resize or crop anything.
		if ( 'default' !== $image_size && ! $retina ) {
			return apply_filters( 'envira_gallery_image_src', $image, $id, $item, $data );
		}

		// If the image size is default (i.e. the user has input their own custom dimensions in the Gallery),
		// we may need to resize the image now
		// This is safe to call every time, as resize_image() will check if the image already exists, preventing thumbnails
		// from being generated every single time.
		$type = $this->is_mobile ? 'mobile' : 'crop'; // 'crop' is misleading here - it's the key that stores the thumbnail width + height
		$args = [
			'position' => 'c',
			'width'    => $this->get_config( $type . '_width', $data ),
			'height'   => $this->get_config( $type . '_height', $data ),
			'quality'  => 100,
			'retina'   => $retina,
		];

		// If we're requesting a retina image, and the gallery uses a registered WordPress image size,
		// we need use the width and height of that registered WordPress image size - not the gallery's
		// image width and height, which are hidden settings.
		if ( 'full' !== $image_size && ( 'default' !== $image_size && $retina ) ) {
			// Find WordPress registered image size.
			$wordpress_image_sizes = $this->common->get_image_sizes( true ); // true = WordPress only image sizes (excludes random).

			foreach ( $wordpress_image_sizes as $size ) {
				if ( $size['value'] !== $image_size ) {
					continue;
				}

				// Define the $args so at least they exist.
				$args['width']  = false;
				$args['height'] = false;

				// We found the image size. Use its dimensions.
				if ( ! empty( $size['width'] ) ) {
					$args['width'] = $size['width'];
				}
				if ( ! empty( $size['height'] ) ) {
					$args['height'] = $size['height'];
				}
				break;

			}
		}

		// Filter.
		$args          = apply_filters( 'envira_gallery_crop_image_args', $args );
		$resized_image = $this->common->resize_image( $image, $args['width'], $args['height'], $this->get_config( 'crop', $data ), $args['position'], $args['quality'], $args['retina'], $data );

		// If there is an error, possibly output error message and return the default image src.
		if ( is_wp_error( $resized_image ) ) {
			// If WP_DEBUG is enabled, and we're logged in, output an error to the user.
			if ( defined( 'WP_DEBUG' ) && WP_DEBUG && is_user_logged_in() ) {
				echo '<pre>Envira: Error occured resizing image (these messages are only displayed to logged in WordPress users):<br />';
				echo 'Error: ' . esc_html( $resized_image->get_error_message() ) . '<br />';
				echo 'Image: ' . esc_html( $image ) . '<br />';
				echo 'Args: ' . var_export( $args, true ) . '</pre>'; // @codingStandardsIgnoreLine
			}

			// Return the non-cropped image as a fallback.
			return apply_filters( 'envira_gallery_image_src', $image, $id, $item, $data );
		} else {
			return apply_filters( 'envira_gallery_image_src', $resized_image, $id, $item, $data );
		}
	}

	/**
	 * Helper Method to find closest size
	 *
	 * @param array $data Gallery Data.
	 * @return string
	 */
	public function find_clostest_size( $data ) {

		$image_sizes = $this->get_image_sizes();
		$dimensions  = $this->get_config( 'dimensions', $data );
		$width       = $this->get_config( 'crop_width', $data );
		$height      = $this->get_config( 'crop_height', $data );
		$match       = false;

		usort( $image_sizes, [ $this, 'usort_callback' ] );

		foreach ( $image_sizes as $num ) {

			$num['width']  = (int) $num['width'];
			$num['height'] = (int) $num['height'];

			// skip over sizes that are smaller.
			if ( $num['height'] < $height || $num['width'] < $width ) {
				continue;
			}
			if ( $num['width'] > $width && $num['height'] > $height ) {

				if ( false === $match ) {

					$match = true;
					$size  = $num['name'];

					return $size;
				}
			}
		}
		return '';
	}

	/**
	 * Helper function for usort and php 5.3 >.
	 *
	 * @access public
	 * @param mixed $a First sort.
	 * @param mixed $b Second sort.
	 * @return int
	 */
	public function usort_callback( $a, $b ) {

		return intval( $a['width'] ) - intval( $b['width'] );
	}

	/**
	 * Helper Method to get image sizes.
	 *
	 * @access public
	 * @return array
	 */
	public function get_image_sizes() {

		global $_wp_additional_image_sizes;

		$sizes = [];
		foreach ( get_intermediate_image_sizes() as $_size ) {

			if ( in_array( $_size, [ 'thumbnail', 'medium', 'medium_large', 'large' ], true ) ) {

				if ( (bool) get_option( "{$_size}_crop" ) === true ) {

					continue;

				}
				$sizes[ $_size ]['name']   = $_size;
				$sizes[ $_size ]['width']  = get_option( "{$_size}_size_w" );
				$sizes[ $_size ]['height'] = get_option( "{$_size}_size_h" );
				$sizes[ $_size ]['crop']   = (bool) get_option( "{$_size}_crop" );

			} elseif ( isset( $_wp_additional_image_sizes[ $_size ] ) ) {

				if ( true === $_wp_additional_image_sizes[ $_size ]['crop'] ) {

					continue;

				}

				$sizes[ $_size ] = [
					'name'   => $_size,
					'width'  => $_wp_additional_image_sizes[ $_size ]['width'],
					'height' => $_wp_additional_image_sizes[ $_size ]['height'],
					'crop'   => $_wp_additional_image_sizes[ $_size ]['crop'],
				];
			}
		}
		return $sizes;
	}


	/**
	 * Helper method to retrieve the proper gallery toolbar template.
	 *
	 * @since 1.0.0
	 *
	 * @param array $data Array of gallery data.
	 * @return string     String template for the gallery toolbar.
	 */
	public function get_toolbar_template( $data ) {

		// Build out the custom template based on options chosen.
		$template         = '<div id="envirabox-buttons">';
			$template    .= '<ul>';
				$template = apply_filters( 'envira_gallery_toolbar_start', $template, $data );

				// Prev.
				$template .= '<li><a class="btnPrev" title="' . __( 'Previous', 'envira-gallery-lite' ) . '" href="javascript:;"></a></li>';
				$template  = apply_filters( 'envira_gallery_toolbar_after_prev', $template, $data );

				// Next.
				$template .= '<li><a class="btnNext" title="' . __( 'Next', 'envira-gallery-lite' ) . '" href="javascript:;"></a></li>';
				$template  = apply_filters( 'envira_gallery_toolbar_after_next', $template, $data );

				// Title.
		if ( $this->get_config( 'toolbar_title', $data ) ) {
			$template .= '<li id="envirabox-buttons-title"><span>' . $this->get_config( 'title', $data ) . '</span></li>';
			$template  = apply_filters( 'envira_gallery_toolbar_after_title', $template, $data );
		}

				// Close.
				$template .= '<li><a class="btnClose" title="' . __( 'Close', 'envira-gallery-lite' ) . '" href="javascript:;"></a></li>';
				$template  = apply_filters( 'envira_gallery_toolbar_after_close', $template, $data );

				$template = apply_filters( 'envira_gallery_toolbar_end', $template, $data );
			$template    .= '</ul>';
		$template        .= '</div>';

		// Return the template, filters applied and all.
		return apply_filters( 'envira_gallery_toolbar', $template, $data );
	}

	/**
	 * Helper method to retrieve the gallery lightbox template.
	 *
	 * @since 1.3.1.4
	 *
	 * @param array $data Array of gallery data.
	 * @return string String template for the gallery lightbox.
	 */
	public function get_lightbox_template( $data ) {

		// Build out the lightbox template.
		$envirabox_wrap_css_classes = apply_filters( 'envirabox_wrap_css_classes', 'envirabox-wrap', $data );

		$envirabox_theme = apply_filters( 'envirabox_theme', 'envirabox-theme-' . $this->get_config( 'lightbox_theme', $data ), $data );

		$template = '<div class="' . $envirabox_wrap_css_classes . '" tabIndex="-1"><div class="envirabox-skin ' . $envirabox_theme . '"><div class="envirabox-outer"><div class="envirabox-inner">';

		// Top Left box.
		$template .= '<div class="envirabox-position-overlay envira-gallery-top-left">';
		$template  = apply_filters( 'envirabox_output_dynamic_position', $template, $data, 'top-left' );
		$template .= '</div>';

		// Top Right box.
		$template .= '<div class="envirabox-position-overlay envira-gallery-top-right">';
		$template  = apply_filters( 'envirabox_output_dynamic_position', $template, $data, 'top-right' );
		$template .= '</div>';

		// Bottom Left box.
		$template .= '<div class="envirabox-position-overlay envira-gallery-bottom-left">';
		$template  = apply_filters( 'envirabox_output_dynamic_position', $template, $data, 'bottom-left' );
		$template .= '</div>';

		// Bottom Right box.
		$template .= '<div class="envirabox-position-overlay envira-gallery-bottom-right">';
		$template  = apply_filters( 'envirabox_output_dynamic_position', $template, $data, 'bottom-right' );
		$template .= '</div>';

		$template .= '</div></div></div></div>';

		// Return the template, filters applied.
		return apply_filters( 'envira_gallery_lightbox_template', str_replace( "\n", '', $template ), $data );
	}

	/**
	 * Helper method for retrieving config values.
	 *
	 * @since 1.0.0
	 *
	 * @param string $key The config key to retrieve.
	 * @param array  $data The gallery data to use for retrieval.
	 * @return string     Key value on success, default if not set.
	 */
	public function get_config( $key, $data ) {

		return isset( $data['config'][ $key ] ) ? $data['config'][ $key ] : $this->common->get_config_default( $key );
	}

	/**
	 * Helper method to minify a string of data.
	 *
	 * @since 1.0.4
	 *
	 * @param string  $string_data String of data to minify.
	 * @param boolean $strip_double_forward_slashes stirpe forward slashes.
	 *
	 * @return string $string Minified string of data.
	 */
	public function minify( $string_data, $strip_double_forward_slashes = true ) {

		// Added a switch for stripping double forwardslashes.
		// This can be disabled when using URLs in JS, to ensure http:// doesn't get removed.
		// All other comment removal and minification will take place.
		$strip_double_forward_slashes = apply_filters( 'envira_minify_strip_double_forward_slashes', $strip_double_forward_slashes );

		if ( $strip_double_forward_slashes ) {
			$clean = preg_replace( '/((?:\/\*(?:[^*]|(?:\*+[^*\/]))*\*+\/)|(?:\/\/.*))/', '', $string_data );
		} else {
			// Use less aggressive method.
			$clean = preg_replace( '!/\*.*?\*/!s', '', $string_data );
			$clean = preg_replace( '/\n\s*\n/', "\n", $clean );
		}

		$clean = str_replace( [ "\r\n", "\r", "\t", "\n", '  ', '    ', '     ' ], '', $clean );

		return apply_filters( 'envira_gallery_minified_string', $clean, $string_data );
	}

	/**
	 * I'm sure some plugins mean well, but they go a bit too far trying to reduce
	 * conflicts without thinking of the consequences.
	 *
	 * 1. Prevents Foobox from completely borking envirabox as if Foobox rules the world.
	 *
	 * @since 1.0.0
	 */
	public function plugin_humility() {

		if ( class_exists( 'fooboxV2' ) ) {
			remove_action( 'wp_footer', [ $GLOBALS['foobox'], 'disable_other_lightboxes' ], 200 );
		}

		if ( function_exists( 'is_plugin_active' ) && is_plugin_active( 'enable-media-replace/enable-media-replace.php' ) ) {
			add_action( 'enable-media-replace-upload-done', [ $this, 'emr_clear_envira_cache' ], 10, 3 );
		}
	}

	/**
	 * Specific function that clears gallery cache when a photo gets replaced by other third-party plugins.
	 *
	 * @since 1.8.4.0
	 *
	 * @param string $target_url Target URL.
	 * @param string $source_url Source URL.
	 * @param int    $post_id Post ID.
	 */
	public function emr_clear_envira_cache( $target_url = false, $source_url = false, $post_id = false ) {

		$has_gallery = get_post_meta( $post_id, '_eg_has_gallery', true );

		if ( function_exists( 'flush_gallery_caches' ) && count( $has_gallery ) > 0 ) :

			foreach ( $has_gallery as $gallery_id ) {
				flush_gallery_caches( $gallery_id );
			}

		endif;
	}

	/**
	 * Outputs only the first image of the gallery inside a regular <div> tag
	 * to avoid styling issues with feeds.
	 *
	 * @since 1.0.5
	 *
	 * @param array $data      Array of gallery data.
	 * @return string $gallery Custom gallery output for feeds.
	 */
	public function do_feed_output( $data ) {

		$gallery = '<div class="envira-gallery-feed-output">';
		foreach ( $data['gallery'] as $id => $item ) {
			// Skip over images that are pending (ignore if in Preview mode).
			if ( isset( $item['status'] ) && 'pending' === $item['status'] && ! is_preview() ) {
				continue;
			}

			$imagesrc = $this->get_image_src( $id, $item, $data );
			$gallery .= '<img class="envira-gallery-feed-image" src="' . esc_url( $imagesrc ) . '" title="' . trim( esc_html( $item['title'] ) ) . '" alt="' . trim( esc_html( $item['alt'] ) ) . '" />';
			break;
		}
		$gallery .= '</div>';

		return apply_filters( 'envira_gallery_feed_output', $gallery, $data );
	}

	/**
	 * Returns a set of indexable image links to allow SEO indexing for preloaded images.
	 *
	 * @since 1.0.0
	 *
	 * @param mixed $id       The slider ID to target.
	 * @return string $images String of indexable image HTML.
	 */
	public function get_indexable_images( $id ) {

		// If there are no images, don't do anything.
		$images = '';
		$i      = 1;
		if ( empty( $this->index[ $id ] ) ) {
			return $images;
		}

		foreach ( (array) $this->index[ $id ] as $attach_id => $data ) {
			$images .= '<img src="' . esc_url( $data['src'] ) . '" alt="' . esc_attr( $data['alt'] ) . '" />';
			++$i;
		}

		return apply_filters( 'envira_gallery_indexable_images', $images, $this->index, $id );
	}

	/**
	 * Helper Method to check if url is an image.
	 *
	 * @access public
	 * @param mixed $url Url of unknown.
	 * @return boolean
	 */
	public function is_image( $url ) {

		$p = strrpos( $url, '.' );

		if ( false === $p ) {

			return false;

		}

		$extension = strtolower( trim( substr( $url, $p ) ) );

		$img_extensions = [ '.gif', '.jpg', '.jpeg', '.png', '.tiff', '.tif', '.webp' ];

		if ( in_array( $extension, $img_extensions, true ) ) {

			return true;

		}

		return false;
	}

	/**
	 * Returns the singleton instance of the class.
	 *
	 * @since 1.0.0
	 *
	 * @return object The Envira_Gallery_Shortcode object.
	 */
	public static function get_instance() {

		if ( ! isset( self::$instance ) && ! ( self::$instance instanceof Envira_Gallery_Shortcode ) ) {
			self::$instance = new Envira_Gallery_Shortcode();
		}

		return self::$instance;
	}
}

// Load the shortcode class.
$envira_gallery_shortcode = Envira_Gallery_Shortcode::get_instance();

Copyright © 2019 by b0y-101