old_scripts = new WP_Scripts();
} else {
$this->old_scripts = $scripts;
}
// Unset all the object properties except our private copy of the scripts object.
// We have to unset everything so that the overload methods talk to $this->old_scripts->whatever
// instead of $this->whatever.
foreach ( array_keys( get_object_vars( $this ) ) as $key ) {
if ( 'old_scripts' === $key ) {
continue;
}
unset( $this->$key );
}
$this->dependency_path_mapping = new Dependency_Path_Mapping(
/**
* Filter the URL of the site the plugin will be concatenating CSS or JS on
*
* @param bool $url URL of the page with CSS or JS to concatonate.
*
* @since 1.0.0
*/
apply_filters( 'page_optimize_site_url', $this->base_url )
);
}
protected function has_inline_content( $handle ) {
$before_output = $this->get_data( $handle, 'before' );
if ( ! empty( $before_output ) ) {
return true;
}
$after_output = $this->get_data( $handle, 'after' );
if ( ! empty( $after_output ) ) {
return true;
}
// JavaScript translations
$has_translations = ! empty( $this->registered[ $handle ]->textdomain );
if ( $has_translations ) {
return true;
}
return false;
}
/**
* Override for WP_Scripts::do_item() - this is the method that actually outputs the scripts.
*/
public function do_items( $handles = false, $group = false ) {
$handles = false === $handles ? $this->queue : (array) $handles;
$javascripts = array();
/**
* Filter the URL of the site the plugin will be concatenating CSS or JS on
*
* @param bool $url URL of the page with CSS or JS to concatonate.
*
* @since 1.0.0
*/
$siteurl = apply_filters( 'page_optimize_site_url', $this->base_url );
$this->all_deps( $handles );
$level = 0;
$using_strict = false;
foreach ( $this->to_do as $key => $handle ) {
$script_is_strict = false;
if ( in_array( $handle, $this->done, true ) || ! isset( $this->registered[ $handle ] ) ) {
continue;
}
if ( 0 === $group && $this->groups[ $handle ] > 0 ) {
$this->in_footer[] = $handle;
unset( $this->to_do[ $key ] );
continue;
}
if ( ! $this->registered[ $handle ]->src ) { // Defines a group.
if ( $this->has_inline_content( $handle ) ) {
++$level;
$javascripts[ $level ]['type'] = 'do_item';
$javascripts[ $level ]['handle'] = $handle;
++$level;
unset( $this->to_do[ $key ] );
} else {
// if there are localized items, echo them
$this->print_extra_script( $handle );
$this->done[] = $handle;
}
continue;
}
if ( false === $group && in_array( $handle, $this->in_footer, true ) ) {
$this->in_footer = array_diff( $this->in_footer, (array) $handle );
}
$obj = $this->registered[ $handle ];
$js_url = jetpack_boost_enqueued_to_absolute_url( $obj->src );
$js_url_parsed = wp_parse_url( $js_url );
// Don't concat by default
$do_concat = false;
// Only try to concat static js files
if ( str_contains( $js_url_parsed['path'], '.js' ) ) {
// Previously, the value of this variable was determined by a function.
// Now, since concatenation is always enabled when the module is active,
// the value will always be true for static files.
$do_concat = true;
} elseif ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ), esc_html( $obj->src ) );
}
// Don't try to concat externally hosted scripts
$is_internal_uri = $this->dependency_path_mapping->is_internal_uri( $js_url );
if ( $do_concat && ! $is_internal_uri ) {
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ), esc_url( $js_url ) );
}
$do_concat = false;
}
if ( $do_concat ) {
// Resolve paths and concat scripts that exist in the filesystem
$js_realpath = $this->dependency_path_mapping->dependency_src_to_fs_path( $js_url );
if ( false === $js_realpath ) {
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ), esc_html( $js_realpath ) );
}
$do_concat = false;
}
}
if ( $do_concat && $this->has_inline_content( $handle ) ) {
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ) );
}
$do_concat = false;
}
// Skip core scripts that use Strict Mode
if ( $do_concat && ( 'react' === $handle || 'react-dom' === $handle ) ) {
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ) );
}
$do_concat = false;
$script_is_strict = true;
// phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents
} elseif ( $do_concat && preg_match_all( '/^[\',"]use strict[\',"];/Uims', file_get_contents( $js_realpath ), $matches ) ) {
// Skip third-party scripts that use Strict Mode
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ) );
}
$do_concat = false;
$script_is_strict = true;
} else {
$script_is_strict = false;
}
// Skip concating scripts from exclusion list
$exclude_list = jetpack_boost_page_optimize_js_exclude_list();
foreach ( $exclude_list as $exclude ) {
if ( $do_concat && $handle === $exclude ) {
$do_concat = false;
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ) );
}
}
}
/**
* Filter that allows plugins to disable concatenation of certain scripts.
*
* @param bool $do_concat if true, then perform concatenation
* @param string $handle handle to JS file
*
* @since 1.0.0
*/
if ( $do_concat && ! apply_filters( 'js_do_concat', $do_concat, $handle ) ) {
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
printf( "\n\n", esc_html( $handle ) );
}
}
/**
* Filter that allows plugins to disable concatenation of certain scripts.
*
* @param bool $do_concat if true, then perform concatenation
* @param string $handle handle to JS file
*
* @since 1.0.0
*/
$do_concat = apply_filters( 'js_do_concat', $do_concat, $handle );
if ( true === $do_concat ) {
if ( ! isset( $javascripts[ $level ] ) ) {
$javascripts[ $level ]['type'] = 'concat';
}
$javascripts[ $level ]['paths'][] = $js_url_parsed['path'];
$javascripts[ $level ]['handles'][] = $handle;
} else {
++$level;
$javascripts[ $level ]['type'] = 'do_item';
$javascripts[ $level ]['handle'] = $handle;
++$level;
}
unset( $this->to_do[ $key ] );
if ( $using_strict !== $script_is_strict ) {
if ( $script_is_strict ) {
$using_strict = true;
$strict_count = 0;
} else {
$using_strict = false;
}
}
if ( $script_is_strict ) {
++$strict_count;
}
}
if ( empty( $javascripts ) ) {
return $this->done;
}
foreach ( $javascripts as $js_array ) {
if ( 'do_item' === $js_array['type'] ) {
if ( $this->do_item( $js_array['handle'], $group ) ) {
$this->done[] = $js_array['handle'];
}
} elseif ( 'concat' === $js_array['type'] ) {
array_map( array( $this, 'print_extra_script' ), $js_array['handles'] );
if ( isset( $js_array['paths'] ) && count( $js_array['paths'] ) > 1 ) {
$file_name = jetpack_boost_page_optimize_generate_concat_path( $js_array['paths'], $this->dependency_path_mapping );
if ( get_site_option( 'jetpack_boost_static_minification' ) ) {
$href = jetpack_boost_get_minify_url( $file_name . '.min.js' );
} else {
$href = $siteurl . jetpack_boost_get_static_prefix() . '??' . $file_name;
}
} elseif ( isset( $js_array['paths'] ) && is_array( $js_array['paths'] ) ) {
$href = jetpack_boost_page_optimize_cache_bust_mtime( $js_array['paths'][0], $siteurl );
}
$this->done = array_merge( $this->done, $js_array['handles'] );
// Print before/after scripts from wp_inline_scripts() and concatenated script tag
if ( isset( $js_array['extras']['before'] ) ) {
foreach ( $js_array['extras']['before'] as $inline_before ) {
// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
echo $inline_before;
}
}
if ( isset( $href ) ) {
$handles = implode( ',', $js_array['handles'] );
if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
$tag = "\n";
} else {
$tag = "\n";
}
if ( is_array( $js_array['handles'] ) && count( $js_array['handles'] ) === 1 ) {
/**
* Filters the HTML script tag of an enqueued script
* A copy of the core filter of the same name. https://developer.wordpress.org/reference/hooks/script_loader_tag/
* Because we have a single script, let's apply the `script_loader_tag` filter as core does in `do_item()`.
* That way, we interfere less with plugin and theme script filtering. For example, without this filter,
* there is a case where we block the TwentyTwenty theme from adding async/defer attributes.
* https://github.com/Automattic/page-optimize/pull/44
*
* @param string $tag Script tag for the enqueued script.
* @param string $handle The script's registered handle.
* @param string $href URL of the script.
*
* @since 1.0.0
*/
$tag = apply_filters( 'script_loader_tag', $tag, $js_array['handles'][0], $href );
}
// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
echo $tag;
}
if ( isset( $js_array['extras']['after'] ) ) {
foreach ( $js_array['extras']['after'] as $inline_after ) {
// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
echo $inline_after;
}
}
}
}
do_action( 'js_concat_did_items', $javascripts );
return $this->done;
}
public function __isset( $key ) {
return isset( $this->old_scripts->$key );
}
public function __unset( $key ) {
unset( $this->old_scripts->$key );
}
public function &__get( $key ) {
return $this->old_scripts->$key;
}
public function __set( $key, $value ) {
$this->old_scripts->$key = $value;
}
}