309 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			309 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
namespace W3TC;
 | 
						|
 | 
						|
class Minify_AutoCss {
 | 
						|
	private $config;
 | 
						|
	private $buffer;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * CSS files to ignore
 | 
						|
	 *
 | 
						|
	 * @var array
 | 
						|
	 */
 | 
						|
	private $ignore_css_files;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Helper object to use
 | 
						|
	 *
 | 
						|
	 * @var _W3_MinifyHelpers
 | 
						|
	 */
 | 
						|
	private $minify_helpers;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Array of processed styles
 | 
						|
	 *
 | 
						|
	 * @var array
 | 
						|
	 */
 | 
						|
	private $debug_minified_urls = array();
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Current position to embed minified style
 | 
						|
	 *
 | 
						|
	 * @var integer
 | 
						|
	 */
 | 
						|
	private $embed_pos;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Current list of files to minify
 | 
						|
	 *
 | 
						|
	 * @var array
 | 
						|
	 */
 | 
						|
	private $files_to_minify;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Current number of minification group
 | 
						|
	 *
 | 
						|
	 * @var integer
 | 
						|
	 */
 | 
						|
	private $debug = false;
 | 
						|
	private $embed_to_html;
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Constructor
 | 
						|
	 *
 | 
						|
	 * @param unknown $config
 | 
						|
	 * @param unknown $buffer
 | 
						|
	 * @param unknown $minify_helpers
 | 
						|
	 */
 | 
						|
	function __construct( $config, $buffer, $minify_helpers ) {
 | 
						|
		$this->config = $config;
 | 
						|
		$this->debug = $config->get_boolean( 'minify.debug' );
 | 
						|
		$this->buffer = $buffer;
 | 
						|
		$this->minify_helpers = $minify_helpers;
 | 
						|
 | 
						|
		// ignored files
 | 
						|
		$this->ignore_css_files = $this->config->get_array(
 | 
						|
			'minify.reject.files.css' );
 | 
						|
		$this->ignore_css_files = array_map( array( '\W3TC\Util_Environment',
 | 
						|
			'normalize_file' ), $this->ignore_css_files );
 | 
						|
 | 
						|
		$this->embed_to_html = $this->config->get_boolean( 'minify.css.embed' );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Does auto-minification
 | 
						|
	 *
 | 
						|
	 * @return string buffer of minified content
 | 
						|
	 */
 | 
						|
	public function execute() {
 | 
						|
		// find all style tags
 | 
						|
		$buffer_nocomments = preg_replace( '~<!--.*?-->\s*~s', '', $this->buffer );
 | 
						|
		$matches = null;
 | 
						|
 | 
						|
		// end of <head> means another group of styles, cannt be combined
 | 
						|
		if ( !preg_match_all( '~((<style\s*[^>]*>.*?</style>)|(<link\s+([^>]+)/?>(.*</link>)?))~is',
 | 
						|
				$buffer_nocomments, $matches ) ) {
 | 
						|
			$matches = null;
 | 
						|
		}
 | 
						|
 | 
						|
		if ( is_null( $matches ) ) {
 | 
						|
			return $this->buffer;
 | 
						|
		}
 | 
						|
 | 
						|
		$style_tags = $matches[1];
 | 
						|
		$style_tags = apply_filters( 'w3tc_minify_css_style_tags',
 | 
						|
			$style_tags );
 | 
						|
 | 
						|
		// pass styles
 | 
						|
		$this->embed_pos = null;
 | 
						|
		$this->files_to_minify = array();
 | 
						|
 | 
						|
		for ( $n = 0; $n < count( $style_tags ); $n++ ) {
 | 
						|
			$this->process_style_tag( $style_tags[$n], $n );
 | 
						|
		}
 | 
						|
 | 
						|
		$this->flush_collected( '' );
 | 
						|
 | 
						|
		return $this->buffer;
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Returns list of minified styles
 | 
						|
	 *
 | 
						|
	 * @return array
 | 
						|
	 */
 | 
						|
	public function get_debug_minified_urls() {
 | 
						|
		return $this->debug_minified_urls;
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Processes style tag
 | 
						|
	 *
 | 
						|
	 * @param unknown $style_tag
 | 
						|
	 * @return void
 | 
						|
	 */
 | 
						|
	private function process_style_tag( $style_tag, $style_tag_number ) {
 | 
						|
		if ( $this->debug ) {
 | 
						|
			Minify_Core::log( 'processing tag ' . substr( $style_tag, 0, 150 ) );
 | 
						|
		}
 | 
						|
 | 
						|
		$tag_pos = strpos( $this->buffer, $style_tag );
 | 
						|
		if ( $tag_pos === false ) {
 | 
						|
			// style is external but not found, skip processing it
 | 
						|
			if ( $this->debug ) {
 | 
						|
				Minify_Core::log( 'style not found:' . $style_tag );
 | 
						|
			}
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$style_href = null;
 | 
						|
		$causes_flush = true;
 | 
						|
		if ( preg_match( '~<link\s+([^>]+)/?>(.*</link>)?~Uis', $style_tag, $match ) ) {
 | 
						|
			// all link tags dont cause automatic flush since
 | 
						|
			// its minified or its not style <link> tag
 | 
						|
			$causes_flush = false;
 | 
						|
 | 
						|
			$attrs = array();
 | 
						|
			$attr_matches = null;
 | 
						|
			if ( preg_match_all( '~(\w+)=["\']([^"\']*)["\']~', $match[1],
 | 
						|
					$attr_matches, PREG_SET_ORDER ) ) {
 | 
						|
				foreach ( $attr_matches as $attr_match ) {
 | 
						|
					$name = strtolower( $attr_match[1] );
 | 
						|
					$attrs[$name] = trim( $attr_match[2] );
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			if ( isset( $attrs['href'] ) && isset( $attrs['rel'] ) &&
 | 
						|
				stristr( $attrs['rel'], 'stylesheet' ) !== false &&
 | 
						|
				( !isset( $attrs['media'] ) || stristr( $attrs['media'], 'print' ) === false ) ) {
 | 
						|
				$style_href = $attrs['href'];
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if ( $causes_flush ) {
 | 
						|
			$data = array(
 | 
						|
				'style_tag_original' => $style_tag,
 | 
						|
				'style_tag_new' => $style_tag,
 | 
						|
				'style_tag_number' => $style_tag_number,
 | 
						|
				'style_tag_pos' => $tag_pos,
 | 
						|
				'should_replace' => false,
 | 
						|
				'buffer' => $this->buffer
 | 
						|
			);
 | 
						|
 | 
						|
			$data = apply_filters( 'w3tc_minify_css_do_local_style_minification',
 | 
						|
				$data );
 | 
						|
			$this->buffer = $data['buffer'];
 | 
						|
 | 
						|
			if ( $data['should_replace'] ) {
 | 
						|
				$this->buffer = substr_replace( $this->buffer,
 | 
						|
					$data['style_tag_new'], $tag_pos,
 | 
						|
					strlen( $style_tag ) );
 | 
						|
			}
 | 
						|
 | 
						|
			// it's not external style, have to flush what we have before it
 | 
						|
			if ( $this->debug ) {
 | 
						|
				Minify_Core::log( 'its not link tag, flushing' );
 | 
						|
			}
 | 
						|
 | 
						|
			$this->flush_collected( $style_tag );
 | 
						|
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		if ( empty( $style_href ) ) {
 | 
						|
			if ( $this->debug ) {
 | 
						|
				Minify_Core::log( 'its not style link tag' );
 | 
						|
			}
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$style_href = Util_Environment::url_relative_to_full( $style_href );
 | 
						|
		$file = Util_Environment::url_to_docroot_filename( $style_href );
 | 
						|
 | 
						|
		$step1_result = $this->minify_helpers->is_file_for_minification(
 | 
						|
			$style_href, $file );
 | 
						|
		if ( $step1_result == 'url' )
 | 
						|
			$file = $style_href;
 | 
						|
 | 
						|
		$step1 = !empty( $step1_result );
 | 
						|
		$step2 = !in_array( $file, $this->ignore_css_files );
 | 
						|
 | 
						|
		$do_tag_minification = $step1 && $step2;
 | 
						|
		$do_tag_minification = apply_filters(
 | 
						|
			'w3tc_minify_css_do_tag_minification',
 | 
						|
			$do_tag_minification, $style_tag, $file );
 | 
						|
 | 
						|
		if ( !$do_tag_minification ) {
 | 
						|
			if ( $this->debug ) {
 | 
						|
				Minify_Core::log( 'file ' . $file .
 | 
						|
					' didnt pass minification check:' .
 | 
						|
					' file_for_min: ' . ( $step1 ? 'true' : 'false' ) .
 | 
						|
					' ignore_css_files: ' . ( $step2 ? 'true' : 'false' ) );
 | 
						|
			}
 | 
						|
 | 
						|
			$data = array(
 | 
						|
				'style_tag_original' => $style_tag,
 | 
						|
				'style_tag_new' => $style_tag,
 | 
						|
				'style_tag_number' => $style_tag_number,
 | 
						|
				'style_tag_pos' => $tag_pos,
 | 
						|
				'style_href' => $style_href,
 | 
						|
				'should_replace' => false,
 | 
						|
				'buffer' => $this->buffer
 | 
						|
			);
 | 
						|
 | 
						|
			$data = apply_filters( 'w3tc_minify_css_do_excluded_tag_style_minification',
 | 
						|
				$data );
 | 
						|
			$this->buffer = $data['buffer'];
 | 
						|
 | 
						|
			if ( $data['should_replace'] ) {
 | 
						|
				$this->buffer = substr_replace( $this->buffer,
 | 
						|
					$data['style_tag_new'], $tag_pos,
 | 
						|
					strlen( $style_tag ) );
 | 
						|
			}
 | 
						|
 | 
						|
			$this->flush_collected( $style_tag );
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->debug_minified_urls[] = $file;
 | 
						|
		$this->buffer = substr_replace( $this->buffer, '',
 | 
						|
			$tag_pos, strlen( $style_tag ) );
 | 
						|
 | 
						|
		// put minified file at the place of first tag
 | 
						|
		if ( count( $this->files_to_minify ) <= 0 )
 | 
						|
			$this->embed_pos = $tag_pos;
 | 
						|
		$this->files_to_minify[] = $file;
 | 
						|
 | 
						|
		if ( $this->config->get_string( 'minify.css.method' ) == 'minify' )
 | 
						|
			$this->flush_collected( '' );
 | 
						|
	}
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Minifies collected styles
 | 
						|
	 */
 | 
						|
	private function flush_collected( $last_style_tag ) {
 | 
						|
		if ( count( $this->files_to_minify ) <= 0 )
 | 
						|
			return;
 | 
						|
		$do_flush_collected = apply_filters( 'w3tc_minify_css_do_flush_collected',
 | 
						|
			true, $last_style_tag, $this );
 | 
						|
		if ( !$do_flush_collected )
 | 
						|
			return;
 | 
						|
 | 
						|
		// find embed position
 | 
						|
		$embed_pos = $this->embed_pos;
 | 
						|
 | 
						|
		// build minified style tag
 | 
						|
		$data = array(
 | 
						|
			'files_to_minify' => $this->files_to_minify,
 | 
						|
			'embed_pos' => $embed_pos,
 | 
						|
			'buffer' => $this->buffer,
 | 
						|
			'embed_to_html' => $this->embed_to_html
 | 
						|
		);
 | 
						|
 | 
						|
		$data = apply_filters( 'w3tc_minify_css_step', $data );
 | 
						|
		$this->buffer = $data['buffer'];
 | 
						|
 | 
						|
		if ( !empty( $data['files_to_minify'] ) ) {
 | 
						|
			$style_data = $this->minify_helpers->generate_css_style_tag(
 | 
						|
				$data['files_to_minify'],
 | 
						|
				$data['embed_to_html'] );
 | 
						|
 | 
						|
			$data['style_to_embed_url'] = $style_data['url'];
 | 
						|
			$data['style_to_embed_body'] = $style_data['body'];
 | 
						|
			$data = apply_filters( 'w3tc_minify_css_step_style_to_embed',
 | 
						|
				$data );
 | 
						|
			$this->buffer = $data['buffer'];
 | 
						|
 | 
						|
			if ( $this->config->getf_boolean( 'minify.css.http2push' ) ) {
 | 
						|
				$this->minify_helpers->http2_header_add(
 | 
						|
					$data['style_to_embed_url'], 'style' );
 | 
						|
			}
 | 
						|
 | 
						|
			// replace
 | 
						|
			$this->buffer = substr_replace( $this->buffer,
 | 
						|
				$data['style_to_embed_body'], $data['embed_pos'], 0 );
 | 
						|
		}
 | 
						|
 | 
						|
		$this->files_to_minify = array();
 | 
						|
	}
 | 
						|
}
 |