341 lines
12 KiB
PHP
Raw Normal View History

2024-05-20 15:37:46 +03:00
<?php
namespace WpAssetCleanUp\OptimiseAssets;
use WpAssetCleanUp\Main;
use WpAssetCleanUp\Menu;
use WpAssetCleanUp\MetaBoxes;
use WpAssetCleanUp\Misc;
/**
* Class MinifyCss
* @package WpAssetCleanUp\OptimiseAssets
*/
class MinifyCss
{
/**
* @param $cssContent
* @param bool $forInlineStyle
*
* @return string
*/
public static function applyMinification($cssContent, $forInlineStyle = false
)
{
if (class_exists('\MatthiasMullie\Minify\CSS')) {
$sha1OriginalContent = sha1($cssContent);
$checkForAlreadyMinifiedShaOne = mb_strlen($cssContent) > 40000;
// Let's check if the content is already minified
// Save resources as the minify process can take time if the content is very large
// Limit the total number of entries tp 100: if it's more than that, it's likely because there's dynamic JS altering on every page load
if ($checkForAlreadyMinifiedShaOne && OptimizeCommon::originalContentIsAlreadyMarkedAsMinified($sha1OriginalContent, 'styles')) {
return $cssContent;
}
$cssContentBeforeAnyBugChanges = $cssContent;
// [CUSTOM BUG FIX]
// Encode the special matched content to avoid any wrong minification from the minifier
$hasVarWithZeroUnit = false;
preg_match_all('#--([a-zA-Z0-9_-]+):(\s+)0(em|ex|%|px|cm|mm|in|pt|pc|ch|rem|vh|vw|vmin|vmax|vm)#', $cssContent, $cssVariablesMatches);
if (isset($cssVariablesMatches[0]) && ! empty($cssVariablesMatches[0])) {
$hasVarWithZeroUnit = true;
foreach ($cssVariablesMatches[0] as $zeroUnitMatch) {
$cssContent = str_replace( $zeroUnitMatch, '[wpacu]' . base64_encode( $zeroUnitMatch ) . '[/wpacu]', $cssContent );
}
}
// Fix: If the content is something like "calc(50% - 22px) calc(50% - 22px);" then leave it as it is
preg_match_all('#calc(|\s+)\((.*?)(;|})#si', $cssContent, $cssCalcMatches);
$multipleOrSpecificCalcMatches = array(); // with multiple calc() or with at least one calc() that contains new lines
if (isset($cssCalcMatches[0]) && ! empty($cssCalcMatches[0])) {
foreach ($cssCalcMatches[0] as $cssCalcMatch) {
if (substr_count($cssCalcMatch, 'calc') > 1 || strpos($cssCalcMatch, "\n") !== false) {
$cssContent = str_replace( $cssCalcMatch, '[wpacu]' . base64_encode( $cssCalcMatch ) . '[/wpacu]', $cssContent );
$multipleOrSpecificCalcMatches[] = $cssCalcMatch;
}
}
}
// [/CUSTOM BUG FIX]
$minifier = new \MatthiasMullie\Minify\CSS( $cssContent );
if ( $forInlineStyle ) {
// If the minification is applied for inlined CSS (within STYLE) leave the background URLs unchanged as it sometimes lead to issues
$minifier->setImportExtensions( array() );
}
$minifiedContent = trim( $minifier->minify() );
// [CUSTOM BUG FIX]
// Restore the original content
if ($hasVarWithZeroUnit) {
foreach ( $cssVariablesMatches[0] as $zeroUnitMatch ) {
$zeroUnitMatchAlt = str_replace(': 0', ':0', $zeroUnitMatch); // remove the space
$minifiedContent = str_replace( '[wpacu]' . base64_encode( $zeroUnitMatch ) . '[/wpacu]', $zeroUnitMatchAlt, $minifiedContent );
}
}
if ( ! empty($multipleOrSpecificCalcMatches) ) {
foreach ( $multipleOrSpecificCalcMatches as $cssCalcMatch ) {
$originalCssCalcMatch = $cssCalcMatch;
$cssCalcMatch = preg_replace(array('#calc\(\s+#', '#\s+\);#'), array('calc(', ');'), $originalCssCalcMatch);
$cssCalcMatch = str_replace(' ) calc(', ') calc(', $cssCalcMatch);
$minifiedContent = str_replace( '[wpacu]' . base64_encode( $originalCssCalcMatch ) . '[/wpacu]', $cssCalcMatch, $minifiedContent );
}
}
// [/CUSTOM BUG FIX]
// Is there any [wpacu] left? Hmm, the replacement wasn't alright. Make sure to use the original minified version
if (strpos($minifiedContent, '[wpacu]') !== false && strpos($minifiedContent, '[/wpacu]') !== false) {
$minifier = new \MatthiasMullie\Minify\CSS( $cssContentBeforeAnyBugChanges );
if ( $forInlineStyle ) {
// If the minification is applied for inlined CSS (within STYLE) leave the background URLs unchanged as it sometimes leads to issues
$minifier->setImportExtensions( array() );
}
$minifiedContent = trim( $minifier->minify() );
}
if ($checkForAlreadyMinifiedShaOne && $minifiedContent === $cssContent) {
// If the resulting content is the same, mark it as minified to avoid the minify process next time
OptimizeCommon::originalContentMarkAsAlreadyMinified( $sha1OriginalContent, 'styles' );
}
return $minifiedContent;
}
return $cssContent;
}
/**
* @param $href
* @param string $handle
*
* @return bool
*/
public static function skipMinify($href, $handle = '')
{
// Things like WP Fastest Cache Toolbar CSS shouldn't be minified and take up space on the server
if ($handle !== '' && in_array($handle, Main::instance()->skipAssets['styles'])) {
return true;
}
// Some of these files (e.g. from Oxygen, WooCommerce) are already minified
$regExps = array(
'#/wp-content/plugins/wp-asset-clean-up(.*?).min.css#',
// Formidable Forms
'#/wp-content/plugins/formidable/css/formidableforms.css#',
// Oxygen
//'#/wp-content/plugins/oxygen/component-framework/oxygen.css#',
// WooCommerce
'#/wp-content/plugins/woocommerce/assets/css/woocommerce-layout.css#',
'#/wp-content/plugins/woocommerce/assets/css/woocommerce.css#',
'#/wp-content/plugins/woocommerce/assets/css/woocommerce-smallscreen.css#',
'#/wp-content/plugins/woocommerce/assets/css/blocks/style.css#',
'#/wp-content/plugins/woocommerce/packages/woocommerce-blocks/build/style.css#',
// Google Site Kit: the files are already optimized
'#/wp-content/plugins/google-site-kit/#',
// Other libraries from the core that end in .min.css
'#/wp-includes/css/(.*?).min.css#',
// Files within /wp-content/uploads/ or /wp-content/cache/
// Could belong to plugins such as "Elementor", "Oxygen" etc.
'#/wp-content/uploads/elementor/(.*?).css#',
'#/wp-content/uploads/oxygen/css/(.*?)-(.*?).css#',
'#/wp-content/cache/(.*?).css#',
// Already minified, and it also has a random name making the cache folder make bigger
'#/wp-content/bs-booster-cache/#',
);
$regExps = Misc::replaceRelPluginPath($regExps);
if (Main::instance()->settings['minify_loaded_css_exceptions'] !== '') {
$loadedCssExceptionsPatterns = trim(Main::instance()->settings['minify_loaded_css_exceptions']);
if (strpos($loadedCssExceptionsPatterns, "\n")) {
// Multiple values (one per line)
foreach (explode("\n", $loadedCssExceptionsPatterns) as $loadedCssExceptionPattern) {
$regExps[] = '#'.trim($loadedCssExceptionPattern).'#';
}
} else {
// Only one value?
$regExps[] = '#'.trim($loadedCssExceptionsPatterns).'#';
}
}
foreach ($regExps as $regExp) {
if ( preg_match( $regExp, $href ) || ( strpos($href, $regExp) !== false ) ) {
return true;
}
}
return false;
}
/**
* @param $htmlSource
*
* @return mixed|string
*/
public static function minifyInlineStyleTags($htmlSource)
{
if (stripos($htmlSource, '<style') === false) {
return $htmlSource; // no STYLE tags
}
$skipTagsContaining = array(
'data-wpacu-skip',
'astra-theme-css-inline-css',
'astra-edd-inline-css',
'et-builder-module-design-cached-inline-styles',
'fusion-stylesheet-inline-css',
'woocommerce-general-inline-css',
'woocommerce-inline-inline-css',
'data-wpacu-own-inline-style',
// Only shown to the admin, irrelevant for any optimization (save resources)
'data-wpacu-inline-css-file'
// already minified/optimized since the INLINE was generated from the cached file
);
$fetchType = 'regex';
if ( $fetchType === 'regex' ) {
preg_match_all( '@(<style[^>]*?>).*?</style>@si', $htmlSource, $matchesStyleTags, PREG_SET_ORDER );
if ( $matchesStyleTags === null ) {
return $htmlSource;
}
foreach ($matchesStyleTags as $matchedStyle) {
if ( ! (isset($matchedStyle[0]) && $matchedStyle[0]) ) {
continue;
}
$originalTag = $matchedStyle[0];
if (substr($originalTag, -strlen('></style>')) === strtolower('></style>')) {
// No empty STYLE tags
continue;
}
// No need to use extra resources as the tag is already minified
if ( preg_match( '(' . implode( '|', $skipTagsContaining ) . ')', $originalTag ) ) {
continue;
}
$tagOpen = $matchedStyle[1];
$withTagOpenStripped = substr($originalTag, strlen($tagOpen));
$originalTagContents = substr($withTagOpenStripped, 0, -strlen('</style>'));
if ( $originalTagContents ) {
$newTagContents = OptimizeCss::maybeAlterContentForInlineStyleTag( $originalTagContents, true, array( 'just_minify' ) );
// Only comments or no content added to the inline STYLE tag? Strip it completely to reduce the number of DOM elements
if ( $newTagContents === '/**/' || ! $newTagContents ) {
$htmlSource = str_replace( '>' . $originalTagContents . '</', '></', $htmlSource );
preg_match( '#<style.*?>#si', $originalTag, $matchFromStyle );
if ( isset( $matchFromStyle[0] ) && $styleTagWithoutContent = $matchFromStyle[0] ) {
$styleTagWithoutContentAlt = str_ireplace( '"', '\'', $styleTagWithoutContent );
$htmlSource = str_ireplace( array(
$styleTagWithoutContent . '</style>',
$styleTagWithoutContentAlt . '</style>'
), '', $htmlSource );
}
} else {
// It has content; do the replacement
$htmlSource = str_replace(
'>' . $originalTagContents . '</style>',
'>' . $newTagContents . '</style>',
$htmlSource
);
}
}
}
}
return $htmlSource;
}
/**
* @return bool
*/
public static function isMinifyCssEnabled()
{
if (defined('WPACU_IS_MINIFY_CSS_ENABLED')) {
return WPACU_IS_MINIFY_CSS_ENABLED;
}
// Request Minify On The Fly
// It will preview the page with CSS minified
// Only if the admin is logged-in as it uses more resources (CPU / Memory)
if ( isset($_GET['wpacu_css_minify']) && Menu::userCanManageAssets() ) {
self::isMinifyCssEnabledChecked('true');
return true;
}
if ( isset($_REQUEST['wpacu_no_css_minify']) || // not on query string request (debugging purposes)
is_admin() || // not for Dashboard view
(! Main::instance()->settings['minify_loaded_css']) || // Minify CSS has to be Enabled
(Main::instance()->settings['test_mode'] && ! Menu::userCanManageAssets()) ) { // Does not trigger if "Test Mode" is Enabled
self::isMinifyCssEnabledChecked('false');
return false;
}
$isSingularPage = defined('WPACU_CURRENT_PAGE_ID') && WPACU_CURRENT_PAGE_ID > 0 && is_singular();
if ($isSingularPage || Misc::isHomePage()) {
// If "Do not minify CSS on this page" is checked in "Asset CleanUp: Options" side meta box
if ($isSingularPage) {
$pageOptions = MetaBoxes::getPageOptions( WPACU_CURRENT_PAGE_ID ); // Singular page
} else {
$pageOptions = MetaBoxes::getPageOptions(0, 'front_page'); // Home page
}
if ( isset( $pageOptions['no_css_minify'] ) && $pageOptions['no_css_minify'] ) {
self::isMinifyCssEnabledChecked('false');
return false;
}
}
if (OptimizeCss::isOptimizeCssEnabledByOtherParty('if_enabled')) {
self::isMinifyCssEnabledChecked('false');
return false;
}
self::isMinifyCssEnabledChecked('true');
return true;
}
/**
* @param $value
*/
public static function isMinifyCssEnabledChecked($value)
{
if (! defined('WPACU_IS_MINIFY_CSS_ENABLED')) {
if ($value === 'true') {
define( 'WPACU_IS_MINIFY_CSS_ENABLED', true );
} elseif ($value === 'false') {
define( 'WPACU_IS_MINIFY_CSS_ENABLED', false );
}
}
}
}