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 ) { $path_str = jetpack_boost_page_optimize_generate_concat_path( $js_array['paths'], $this->dependency_path_mapping ); $href = $siteurl . jetpack_boost_get_static_prefix() . '??' . $path_str; } 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; } }