345 lines
8.5 KiB
PHP
345 lines
8.5 KiB
PHP
<?php
|
|
namespace W3TC;
|
|
|
|
class Minify_AutoJs {
|
|
/**
|
|
* Config
|
|
*/
|
|
private $config;
|
|
|
|
/**
|
|
* Processed buffer
|
|
*
|
|
* @var string
|
|
*/
|
|
private $buffer;
|
|
|
|
/**
|
|
* JS files to ignore
|
|
*
|
|
* @var array
|
|
*/
|
|
private $ignore_js_files;
|
|
|
|
/**
|
|
* Embed type
|
|
*
|
|
* @var string
|
|
*/
|
|
private $embed_type;
|
|
|
|
/**
|
|
* Helper object to use
|
|
*
|
|
* @var _W3_MinifyHelpers
|
|
*/
|
|
private $minify_helpers;
|
|
|
|
/**
|
|
* Array of processed scripts
|
|
*
|
|
* @var array
|
|
*/
|
|
private $debug_minified_urls = array();
|
|
|
|
/**
|
|
* Current list of script files to minify
|
|
*/
|
|
private $files_to_minify;
|
|
|
|
/**
|
|
* Current group type
|
|
*
|
|
* @var string
|
|
*/
|
|
private $group_type = 'head';
|
|
|
|
private $debug = false;
|
|
|
|
/**
|
|
* 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_js_files = $this->config->get_array( 'minify.reject.files.js' );
|
|
$this->ignore_js_files = array_map( array( '\W3TC\Util_Environment', 'normalize_file' ), $this->ignore_js_files );
|
|
|
|
// define embed type
|
|
$this->embed_type = array(
|
|
'head' => $this->config->get_string( 'minify.js.header.embed_type' ),
|
|
'body' => $this->config->get_string( 'minify.js.body.embed_type' )
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Does auto-minification
|
|
*
|
|
* @return string buffer of minified content
|
|
*/
|
|
public function execute() {
|
|
// find all script tags
|
|
$buffer_nocomments = preg_replace( '~<!--.*?-->\s*~s', '', $this->buffer );
|
|
$matches = null;
|
|
|
|
// end of <head> means another group of scripts, cannt be combined
|
|
if ( !preg_match_all( '~(<script\s*[^>]*>.*?</script>|</head>)~is',
|
|
$buffer_nocomments, $matches ) ) {
|
|
$matches = null;
|
|
}
|
|
|
|
if ( is_null( $matches ) ) {
|
|
return $this->buffer;
|
|
}
|
|
|
|
$script_tags = $matches[1];
|
|
$script_tags = apply_filters( 'w3tc_minify_js_script_tags',
|
|
$script_tags );
|
|
|
|
// pass scripts
|
|
$this->files_to_minify = array(
|
|
'sync' => array(
|
|
'embed_pos' => 0,
|
|
'files' => array()
|
|
),
|
|
'async' => array(
|
|
'embed_pos' => 0,
|
|
'files' => array()
|
|
),
|
|
'defer' => array(
|
|
'embed_pos' => 0,
|
|
'files' => array()
|
|
)
|
|
);
|
|
|
|
for ( $n = 0; $n < count( $script_tags ); $n++ ) {
|
|
$this->process_script_tag( $script_tags[$n], $n );
|
|
}
|
|
|
|
$this->flush_collected( 'sync', '' );
|
|
$this->flush_collected( 'async', '' );
|
|
$this->flush_collected( 'defer', '' );
|
|
|
|
return $this->buffer;
|
|
}
|
|
|
|
/**
|
|
* Returns list of minified scripts
|
|
*
|
|
* @return array
|
|
*/
|
|
public function get_debug_minified_urls() {
|
|
return $this->debug_minified_urls;
|
|
}
|
|
|
|
/**
|
|
* Processes script tag
|
|
*
|
|
* @param unknown $script_tag
|
|
* @return void
|
|
*/
|
|
private function process_script_tag( $script_tag, $script_tag_number ) {
|
|
if ( $this->debug ) {
|
|
Minify_Core::log( 'processing tag ' . substr( $script_tag, 0, 150 ) );
|
|
}
|
|
|
|
$tag_pos = strpos( $this->buffer, $script_tag );
|
|
if ( $tag_pos === false ) {
|
|
// script is external but not found, skip processing it
|
|
if ( $this->debug ) {
|
|
Minify_Core::log( 'script not found:' . $script_tag );
|
|
}
|
|
return;
|
|
}
|
|
|
|
$match = null;
|
|
if ( !preg_match( '~<script\s+[^<>]*src=["\']?([^"\'> ]+)["\'> ]~is',
|
|
$script_tag, $match ) ) {
|
|
$match = null;
|
|
}
|
|
if ( is_null( $match ) ) {
|
|
$data = array(
|
|
'script_tag_original' => $script_tag,
|
|
'script_tag_new' => $script_tag,
|
|
'script_tag_number' => $script_tag_number,
|
|
'script_tag_pos' => $tag_pos,
|
|
'should_replace' => false,
|
|
'buffer' => $this->buffer
|
|
);
|
|
|
|
$data = apply_filters( 'w3tc_minify_js_do_local_script_minification',
|
|
$data );
|
|
$this->buffer = $data['buffer'];
|
|
|
|
if ( $data['should_replace'] ) {
|
|
$this->buffer = substr_replace( $this->buffer,
|
|
$data['script_tag_new'], $tag_pos,
|
|
strlen( $script_tag ) );
|
|
}
|
|
|
|
// it's not external script, have to flush what we have before it
|
|
if ( $this->debug ) {
|
|
Minify_Core::log( 'its not src=, flushing' );
|
|
}
|
|
|
|
$this->flush_collected( 'sync', $script_tag );
|
|
|
|
if ( preg_match( '~</head>~is', $script_tag, $match ) )
|
|
$this->group_type = 'body';
|
|
|
|
return;
|
|
}
|
|
|
|
$script_src = $match[1];
|
|
$script_src = Util_Environment::url_relative_to_full( $script_src );
|
|
$file = Util_Environment::url_to_docroot_filename( $script_src );
|
|
|
|
$step1_result = $this->minify_helpers->is_file_for_minification(
|
|
$script_src, $file );
|
|
if ( $step1_result == 'url' )
|
|
$file = $script_src;
|
|
|
|
$step1 = !empty( $step1_result );
|
|
$step2 = !in_array( $file, $this->ignore_js_files );
|
|
|
|
$do_tag_minification = $step1 && $step2;
|
|
$do_tag_minification = apply_filters( 'w3tc_minify_js_do_tag_minification',
|
|
$do_tag_minification, $script_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_js_files: ' . ( $step2 ? 'true' : 'false' ) );
|
|
}
|
|
|
|
$data = array(
|
|
'script_tag_original' => $script_tag,
|
|
'script_tag_new' => $script_tag,
|
|
'script_tag_number' => $script_tag_number,
|
|
'script_tag_pos' => $tag_pos,
|
|
'script_src' => $script_src,
|
|
'should_replace' => false,
|
|
'buffer' => $this->buffer
|
|
);
|
|
|
|
$data = apply_filters( 'w3tc_minify_js_do_excluded_tag_script_minification',
|
|
$data );
|
|
$this->buffer = $data['buffer'];
|
|
|
|
if ( $data['should_replace'] ) {
|
|
$this->buffer = substr_replace( $this->buffer,
|
|
$data['script_tag_new'], $tag_pos,
|
|
strlen( $script_tag ) );
|
|
}
|
|
|
|
$this->flush_collected( 'sync', $script_tag );
|
|
return;
|
|
}
|
|
|
|
$this->debug_minified_urls[] = $file;
|
|
$this->buffer = substr_replace( $this->buffer, '',
|
|
$tag_pos, strlen( $script_tag ) );
|
|
|
|
$m = null;
|
|
if ( !preg_match( '~\s+(async|defer)[> ]~is', $script_tag, $m ) ) {
|
|
$sync_type = 'sync';
|
|
|
|
// for head group - put minified file at the place of first script
|
|
// for body - put at the place of last script, to make as more DOM
|
|
// objects available as possible
|
|
if ( count( $this->files_to_minify[$sync_type]['files'] ) <= 0 ||
|
|
$this->group_type == 'body' ) {
|
|
$this->files_to_minify[$sync_type]['embed_pos'] = $tag_pos;
|
|
}
|
|
} else {
|
|
$sync_type = strtolower( $m[1] );
|
|
$this->files_to_minify[$sync_type]['embed_pos'] = $tag_pos;
|
|
}
|
|
|
|
$this->files_to_minify[$sync_type]['files'][] = $file;
|
|
|
|
if ( $this->config->get_string( 'minify.js.method' ) == 'minify' )
|
|
$this->flush_collected( $sync_type, '' );
|
|
}
|
|
|
|
/**
|
|
* Minifies collected scripts
|
|
*/
|
|
private function flush_collected( $sync_type, $last_script_tag ) {
|
|
if ( count( $this->files_to_minify[$sync_type]['files'] ) <= 0 ) {
|
|
return;
|
|
}
|
|
$do_flush_collected = apply_filters( 'w3tc_minify_js_do_flush_collected',
|
|
true, $last_script_tag, $this, $sync_type );
|
|
if ( !$do_flush_collected ) {
|
|
return;
|
|
}
|
|
|
|
// build minified script tag
|
|
if ( $sync_type == 'sync' ) {
|
|
$embed_type = $this->embed_type[$this->group_type];
|
|
} elseif ( $sync_type == 'async' ) {
|
|
$embed_type = 'nb-async';
|
|
} elseif ( $sync_type == 'defer' ) {
|
|
$embed_type = 'nb-defer';
|
|
}
|
|
|
|
$data = array(
|
|
'files_to_minify' => $this->files_to_minify[$sync_type]['files'],
|
|
'embed_pos' => $this->files_to_minify[$sync_type]['embed_pos'],
|
|
'embed_type' => $embed_type,
|
|
'buffer' => $this->buffer
|
|
);
|
|
|
|
$data = apply_filters( 'w3tc_minify_js_step', $data );
|
|
$this->buffer = $data['buffer'];
|
|
|
|
if ( !empty( $data['files_to_minify'] ) ) {
|
|
$url = $this->minify_helpers->get_minify_url_for_files(
|
|
$data['files_to_minify'], 'js' );
|
|
|
|
$script = '';
|
|
if ( !is_null( $url ) ) {
|
|
$script .= $this->minify_helpers->generate_script_tag( $url,
|
|
$data['embed_type'] );
|
|
}
|
|
|
|
$data['script_to_embed_url'] = $url;
|
|
$data['script_to_embed_body'] = $script;
|
|
$data = apply_filters( 'w3tc_minify_js_step_script_to_embed',
|
|
$data );
|
|
$this->buffer = $data['buffer'];
|
|
|
|
if ( $this->config->getf_boolean( 'minify.js.http2push' ) ) {
|
|
$this->minify_helpers->http2_header_add(
|
|
$data['script_to_embed_url'], 'script' );
|
|
}
|
|
|
|
// replace
|
|
$this->buffer = substr_replace( $this->buffer,
|
|
$data['script_to_embed_body'], $data['embed_pos'], 0 );
|
|
|
|
foreach ( $this->files_to_minify as $key => $i ) {
|
|
if ( $key != $sync_type && $i['embed_pos'] >= $data['embed_pos'] ) {
|
|
$this->files_to_minify[$key]['embed_pos'] += strlen( $data['script_to_embed_body'] );
|
|
}
|
|
}
|
|
}
|
|
|
|
$this->files_to_minify[$sync_type] = array(
|
|
'embed_pos' => 0,
|
|
'files' => array()
|
|
);
|
|
}
|
|
}
|