Current File : /home/escuelai/public_html/wp-content/plugins/w3-total-cache/Minify_Page.php
<?php
/**
 * File: Minify_Page.php
 *
 * @package W3TC
 */

namespace W3TC;

/**
 * Class Minify_Page
 *
 * phpcs:disable PSR2.Classes.PropertyDeclaration.Underscore
 * phpcs:disable WordPress.PHP.NoSilencedErrors.Discouraged
 */
class Minify_Page extends Base_Page_Settings {
	/**
	 * Current page
	 *
	 * @var string
	 */
	protected $_page = 'w3tc_minify';

	/**
	 * Renders the minify settings page.
	 *
	 * @return void
	 */
	public function view() {
		$minify_enabled = $this->_config->get_boolean( 'minify.enabled' );
		$config_state   = Dispatcher::config_state();

		$minify_rewrite_disabled = ! Util_Rule::can_check_rules() || $this->_config->is_sealed( 'minify.rewrite' );
		$themes                  = Util_Theme::get_themes_by_key();
		$templates               = array();

		$current_theme     = Util_Theme::get_current_theme_name();
		$current_theme_key = '';

		foreach ( $themes as $theme_key => $theme_name ) {
			if ( $theme_name === $current_theme ) {
				$current_theme_key = $theme_key;
			}

			$templates[ $theme_key ] = Util_Theme::get_theme_templates( $theme_name );
		}

		$css_imports_values = array(
			''        => 'None',
			'bubble'  => 'Bubble',
			'process' => 'Process',
		);

		$auto = $this->_config->get_boolean( 'minify.auto' );

		$js_theme  = Util_Request::get_string( 'js_theme', $current_theme_key );
		$js_groups = $this->_config->get_array( 'minify.js.groups' );

		$css_theme  = Util_Request::get_string( 'css_theme', $current_theme_key );
		$css_groups = $this->_config->get_array( 'minify.css.groups' );

		$js_engine   = $this->_config->get_string( 'minify.js.engine' );
		$css_engine  = $this->_config->get_string( 'minify.css.engine' );
		$html_engine = $this->_config->get_string( 'minify.html.engine' );

		$css_imports = $this->_config->get_string( 'minify.css.imports' );

		// Required for Update Media Query String button.
		$browsercache_enabled         = $this->_config->get_boolean( 'browsercache.enabled' );
		$browsercache_update_media_qs = $this->_config->get_boolean( 'browsercache.cssjs.replace' );

		include W3TC_INC_DIR . '/options/minify.php';
	}

	/**
	 * Displays the minify recommendations for a theme.
	 *
	 * @return void
	 */
	public function recommendations() {
		$themes = Util_Theme::get_themes_by_key();

		$current_theme     = Util_Theme::get_current_theme_name();
		$current_theme_key = array_search( $current_theme, $themes, true );

		$theme_key  = Util_Request::get_string( 'theme_key', $current_theme_key );
		$theme_name = ( isset( $themes[ $theme_key ] ) ? $themes[ $theme_key ] : $current_theme );

		$templates       = Util_Theme::get_theme_templates( $theme_name );
		$recommendations = $this->get_theme_recommendations( $theme_name );

		list ( $js_groups, $css_groups ) = $recommendations;

		$minify_js_groups  = $this->_config->get_array( 'minify.js.groups' );
		$minify_css_groups = $this->_config->get_array( 'minify.css.groups' );

		$checked_js   = array();
		$checked_css  = array();
		$locations_js = array();

		if ( isset( $minify_js_groups[ $theme_key ] ) ) {
			foreach ( (array) $minify_js_groups[ $theme_key ] as $template => $locations ) {
				foreach ( (array) $locations as $location => $config ) {
					if ( isset( $config['files'] ) ) {
						foreach ( (array) $config['files'] as $file ) {
							if ( ! isset( $js_groups[ $template ] ) || ! in_array( $file, $js_groups[ $template ], true ) ) {
								$js_groups[ $template ][] = $file;
							}

							$checked_js[ $template ][ $file ]   = true;
							$locations_js[ $template ][ $file ] = $location;
						}
					}
				}
			}
		}

		if ( isset( $minify_css_groups[ $theme_key ] ) ) {
			foreach ( (array) $minify_css_groups [ $theme_key ] as $template => $locations ) {
				foreach ( (array) $locations as $location => $config ) {
					if ( isset( $config['files'] ) ) {
						foreach ( (array) $config['files'] as $file ) {
							if ( ! isset( $css_groups[ $template ] ) || ! in_array( $file, $css_groups[ $template ], true ) ) {
								$css_groups[ $template ][] = $file;
							}

							$checked_css[ $template ][ $file ] = true;
						}
					}
				}
			}
		}

		include W3TC_INC_DIR . '/lightbox/minify_recommendations.php';
	}

	/**
	 * Retrieves the theme URLs for a given theme name.
	 *
	 * @param string $theme_name The name of the theme to retrieve URLs for.
	 *
	 * @return array An associative array where the keys are template names and the values are their respective URLs.
	 */
	public function get_theme_urls( $theme_name ) {
		$urls  = array();
		$theme = Util_Theme::get( $theme_name );

		if ( $theme && isset( $theme['Template Files'] ) ) {
			$front_page_template = false;

			if ( 'page' === get_option( 'show_on_front' ) ) {
				$front_page_id = get_option( 'page_on_front' );

				if ( $front_page_id ) {
					$front_page_template_file = get_post_meta( $front_page_id, '_wp_page_template', true );

					if ( $front_page_template_file ) {
						$front_page_template = basename( $front_page_template_file, '.php' );
					}
				}
			}

			$home_url       = get_home_url();
			$template_files = (array) $theme['Template Files'];

			$mime_types        = get_allowed_mime_types();
			$custom_mime_types = array();

			foreach ( $mime_types as $mime_type ) {
				list ( $type1, $type2 ) = explode( '/', $mime_type );
				$custom_mime_types      = array_merge(
					$custom_mime_types,
					array(
						$type1,
						$type2,
						$type1 . '_' . $type2,
					)
				);
			}

			foreach ( $template_files as $template_file ) {
				$link     = false;
				$template = basename( $template_file, '.php' );

				// Check common templates.
				switch ( true ) {
					// Handle home.php or index.php or front-page.php or custom home page.
					case ( ! $front_page_template && 'home' === $template ):
					case ( ! $front_page_template && 'index' === $template ):
					case ( ! $front_page_template && 'front-page' === $template ):
					case ( $template === $front_page_template ):
						$link = $home_url . '/';
						break;

					// Handle 404.php.
					case ( '404' === $template ):
						$permalink = get_option( 'permalink_structure' );
						if ( $permalink ) {
							$link = sprintf( '%s/%s/', $home_url, '404_test' );
						} else {
							$link = sprintf( '%s/?p=%d', $home_url, 999999999 );
						}
						break;

					// Handle search.php.
					case ( 'search' === $template ):
						$link = sprintf( '%s/?s=%s', $home_url, 'search_test' );
						break;

					// Handle date.php or archive.php.
					case ( 'date' === $template ):
					case ( 'archive' === $template ):
						$posts = get_posts(
							array(
								'numberposts' => 1,
								'orderby'     => 'rand',
							)
						);

						if ( is_array( $posts ) && count( $posts ) ) {
							$time = strtotime( $posts[0]->post_date );
							$link = get_day_link( gmdate( 'Y', $time ), gmdate( 'm', $time ), gmdate( 'd', $time ) );
						}
						break;

					// Handle author.php.
					case ( 'author' === $template ):
						$author_id = false;
						if ( function_exists( 'get_users' ) ) {
							$users = get_users();
							if ( is_array( $users ) && count( $users ) ) {
								$user      = current( $users );
								$author_id = $user->ID;
							}
						} else {
							$author_ids = get_users(
								array(
									'role'   => 'author',
									'fields' => 'ID', // Only fetch user IDs.
								)
							);
							if ( is_array( $author_ids ) && count( $author_ids ) ) {
								$author_id = $author_ids[0];
							}
						}

						if ( $author_id ) {
							$link = get_author_posts_url( $author_id );
						}
						break;

					// Handle category.php.
					case ( 'category' === $template ):
						$category_ids = get_terms(
							array(
								'taxonomy' => 'category',
								'orderby'  => 'id',
								'number'   => 1,
								'fields'   => 'ids',
							)
						);

						if ( is_array( $category_ids ) && count( $category_ids ) ) {
							$link = get_category_link( $category_ids[0] );
						}
						break;

					// Handle tag.php.
					case ( 'tag' === $template ):
						$term_ids = get_terms(
							array(
								'taxonomy' => 'post_tag',
								'fields'   => 'ids',
							)
						);

						if ( is_array( $term_ids ) && count( $term_ids ) ) {
							$link = get_term_link( $term_ids[0], 'post_tag' );
						}

						break;

					// Handle taxonomy.php.
					case ( 'taxonomy' === $template ):
						$taxonomy = '';
						if ( isset( $GLOBALS['wp_taxonomies'] ) && is_array( $GLOBALS['wp_taxonomies'] ) ) {
							foreach ( $GLOBALS['wp_taxonomies'] as $wp_taxonomy ) {
								if (
									! in_array(
										$wp_taxonomy->name,
										array(
											'category',
											'post_tag',
											'link_category',
										),
										true
									)
								) {
									$taxonomy = $wp_taxonomy->name;
									break;
								}
							}
						}

						if ( $taxonomy ) {
							$terms = get_terms(
								array(
									'taxonomy' => $taxonomy,
									'number'   => 1,
								)
							);

							if ( is_array( $terms ) && count( $terms ) ) {
								$link = get_term_link( $terms[0], $taxonomy );
							}
						}
						break;

					// Handle attachment.php.
					case ( 'attachment' === $template ):
						$attachments = get_posts(
							array(
								'post_type'   => 'attachment',
								'numberposts' => 1,
								'orderby'     => 'rand',
							)
						);
						if ( is_array( $attachments ) && count( $attachments ) ) {
							$link = get_attachment_link( $attachments[0]->ID );
						}
						break;

					// Handle single.php.
					case ( 'single' === $template ):
						$posts = get_posts(
							array(
								'numberposts' => 1,
								'orderby'     => 'rand',
							)
						);
						if ( is_array( $posts ) && count( $posts ) ) {
							$link = get_permalink( $posts[0]->ID );
						}
						break;

					// Handle page.php.
					case ( 'page' === $template ):
						$pages_ids = get_all_page_ids();
						if ( is_array( $pages_ids ) && count( $pages_ids ) ) {
							$link = get_page_link( $pages_ids[0] );
						}
						break;

					// Handle comments-popup.php.
					case ( 'comments-popup' === $template ):
						$posts = get_posts(
							array(
								'numberposts' => 1,
								'orderby'     => 'rand',
							)
						);
						if ( is_array( $posts ) && count( $posts ) ) {
							$link = sprintf( '%s/?comments_popup=%d', $home_url, $posts[0]->ID );
						}
						break;

					// Handle paged.php.
					case ( 'paged' === $template ):
						global $wp_rewrite;
						if ( $wp_rewrite->using_permalinks() ) {
							$link = sprintf( '%s/page/%d/', $home_url, 1 );
						} else {
							$link = sprintf( '%s/?paged=%d', 1 );
						}
						break;

					// Handle author-id.php or author-nicename.php.
					case preg_match( '~^author-(.+)$~', $template, $matches ):
						if ( is_numeric( $matches[1] ) ) {
							$link = get_author_posts_url( $matches[1] );
						} else {
							$link = get_author_posts_url( null, $matches[1] );
						}
						break;

					// Handle category-id.php or category-slug.php.
					case preg_match( '~^category-(.+)$~', $template, $matches ):
						if ( is_numeric( $matches[1] ) ) {
							$link = get_category_link( $matches[1] );
						} else {
							$term = get_term_by( 'slug', $matches[1], 'category' );
							if ( is_object( $term ) ) {
								$link = get_category_link( $term->term_id );
							}
						}
						break;

					// Handle tag-id.php or tag-slug.php.
					case preg_match( '~^tag-(.+)$~', $template, $matches ):
						if ( is_numeric( $matches[1] ) ) {
							$link = get_tag_link( $matches[1] );
						} else {
							$term = get_term_by( 'slug', $matches[1], 'post_tag' );
							if ( is_object( $term ) ) {
								$link = get_tag_link( $term->term_id );
							}
						}
						break;

					// Handle taxonomy-taxonomy-term.php.
					case preg_match( '~^taxonomy-(.+)-(.+)$~', $template, $matches ):
						$link = get_term_link( $matches[2], $matches[1] );
						break;

					// Handle taxonomy-taxonomy.php.
					case preg_match( '~^taxonomy-(.+)$~', $template, $matches ):
						$terms = get_terms(
							array(
								'taxonomy' => $matches[1],
								'number'   => 1,
							)
						);

						if ( is_array( $terms ) && count( $terms ) ) {
							$link = get_term_link( $terms[0], $matches[1] );
						}

						break;

					// Handle MIME_type.php.
					case in_array( $template, $custom_mime_types, true ):
						$posts = get_posts(
							array(
								'post_mime_type' => '%' . $template . '%',
								'post_type'      => 'attachment',
								'numberposts'    => 1,
								'orderby'        => 'rand',
							)
						);
						if ( is_array( $posts ) && count( $posts ) ) {
							$link = get_permalink( $posts[0]->ID );
						}
						break;

					// Handle single-posttype.php.
					case preg_match( '~^single-(.+)$~', $template, $matches ):
						$posts = get_posts(
							array(
								'post_type'   => $matches[1],
								'numberposts' => 1,
								'orderby'     => 'rand',
							)
						);

						if ( is_array( $posts ) && count( $posts ) ) {
							$link = get_permalink( $posts[0]->ID );
						}
						break;

					// Handle page-id.php or page-slug.php.
					case preg_match( '~^page-(.+)$~', $template, $matches ):
						if ( is_numeric( $matches[1] ) ) {
							$link = get_permalink( $matches[1] );
						} else {
							$posts = get_posts(
								array(
									'pagename'    => $matches[1],
									'post_type'   => 'page',
									'numberposts' => 1,
								)
							);

							if ( is_array( $posts ) && count( $posts ) ) {
								$link = get_permalink( $posts[0]->ID );
							}
						}
						break;

					// Try to handle custom template.
					default:
						$posts = get_posts(
							array(
								'pagename'    => $template,
								'post_type'   => 'page',
								'numberposts' => 1,
							)
						);

						if ( is_array( $posts ) && count( $posts ) ) {
							$link = get_permalink( $posts[0]->ID );
						}
						break;
				}

				if ( $link && ! is_wp_error( $link ) ) {
					$urls[ $template ] = $link;
				}
			}
		}

		return $urls;
	}

	/**
	 * Retrieves the theme recommendations (JS and CSS files) based on the specified theme.
	 *
	 * @param string $theme_name The name of the theme for which to get recommendations.
	 *
	 * @return array An array containing the theme recommendations, categorized by JS and CSS files.
	 *
	 * @throws \Exception If an error occurs while processing the recommendations.
	 */
	public function get_theme_recommendations( $theme_name ) {
		$urls = $this->get_theme_urls( $theme_name );

		$js_groups  = array();
		$css_groups = array();

		@set_time_limit( $this->_config->get_integer( 'timelimit.minify_recommendations' ) );

		foreach ( $urls as $template => $url ) {
			// Append theme identifier.
			$url .= ( strstr( $url, '?' ) !== false ? '&' : '?' ) . 'w3tc_theme=' . rawurlencode( $theme_name );

			// If preview mode enabled append w3tc_preview.
			if ( $this->_config->is_preview() ) {
				$url .= '&w3tc_preview=1';
			}

			// Get page contents.
			$response = Util_Http::get( $url );

			if (
				! is_wp_error( $response ) &&
				(
					200 === $response['response']['code'] ||
					(
						404 === $response['response']['code'] &&
						'404' === $template
					)
				)
			) {
				$js_files  = $this->get_recommendations_js( $response['body'] );
				$css_files = $this->get_recommendations_css( $response['body'] );

				$js_groups[ $template ]  = $js_files;
				$css_groups[ $template ] = $css_files;
			}
		}

		$js_groups  = $this->get_theme_recommendations_by_groups( $js_groups );
		$css_groups = $this->get_theme_recommendations_by_groups( $css_groups );

		$recommendations = array(
			$js_groups,
			$css_groups,
		);

		return $recommendations;
	}

	/**
	 * Groups the theme recommendations by their usage across different templates.
	 *
	 * @param array $groups An array of theme recommendations categorized by template.
	 *
	 * @return array An array of grouped recommendations, with a 'default' group for common files.
	 */
	public function get_theme_recommendations_by_groups( $groups ) {
		// First calculate file usage count.
		$all_files = array();

		foreach ( $groups as $template => $files ) {
			foreach ( $files as $file ) {
				if ( ! isset( $all_files[ $file ] ) ) {
					$all_files[ $file ] = 0;
				}

				++$all_files[ $file ];
			}
		}

		// Determine default group files.
		$default_files = array();
		$count         = count( $groups );

		foreach ( $all_files as $all_file => $all_file_count ) {
			// If file usage count == groups count then file is common.
			if ( $count === $all_file_count ) {
				$default_files[] = $all_file;

				// If common file found unset it from all groups.
				foreach ( $groups as $template => $files ) {
					foreach ( $files as $index => $file ) {
						if ( $file === $all_file ) {
							array_splice( $groups[ $template ], $index, 1 );
							if ( ! count( $groups[ $template ] ) ) {
								unset( $groups[ $template ] );
							}
							break;
						}
					}
				}
			}
		}

		// If there are common files append add them into default group.
		if ( count( $default_files ) ) {
			$new_groups            = array();
			$new_groups['default'] = $default_files;

			foreach ( $groups as $template => $files ) {
				$new_groups[ $template ] = $files;
			}

			$groups = $new_groups;
		}

		// Unset empty templates.
		foreach ( $groups as $template => $files ) {
			if ( ! count( $files ) ) {
				unset( $groups[ $template ] );
			}
		}

		return $groups;
	}

	/**
	 * Extracts and normalizes the JavaScript files from the provided content.
	 *
	 * @param string $content The HTML content from which to extract JavaScript files.
	 *
	 * @return array A list of unique JavaScript file URLs after normalization.
	 */
	public function get_recommendations_js( $content ) {
		$files = Minify_Extract::extract_js( $content );

		$files        = array_map( array( '\W3TC\Util_Environment', 'normalize_file_minify' ), $files );
		$files        = array_unique( $files );
		$ignore_files = $this->_config->get_array( 'minify.reject.files.js' );
		$files        = array_diff( $files, $ignore_files );

		return $files;
	}

	/**
	 * Extracts and normalizes the CSS files from the provided content.
	 *
	 * @param string $content The HTML content from which to extract CSS files.
	 *
	 * @return array A list of unique CSS file URLs after normalization.
	 */
	public function get_recommendations_css( $content ) {
		$tag_files = Minify_Extract::extract_css( $content );
		$files     = array();
		foreach ( $tag_files as $tag_file ) {
			$files[] = $tag_file[1];
		}

		$files        = array_map( array( '\W3TC\Util_Environment', 'normalize_file_minify' ), $files );
		$files        = array_unique( $files );
		$ignore_files = $this->_config->get_array( 'minify.reject.files.css' );
		$files        = array_diff( $files, $ignore_files );

		return $files;
	}
}