Файловый менеджер - Редактировать - /var/www/portal.bdu.ac.bd/public_html/wp-content/plugins/magazine-blocks/includes/WebFontLoader.php
Назад
<?php /** * WebFontLoader class. * * @since 2.0.7.3 */ namespace MagazineBlocks ; defined( 'ABSPATH' ) || exit; use WP_Filesystem_Base; /** * Class WebFontLoader. */ class WebFontLoader { /** * The font-format. * * Use "woff" or "woff2". * This will change the user-agent user to make the request. * * @since 2.0.7.3 * @var string */ protected $font_format = 'woff2'; /** * The remote URL. * * @since 2.0.7.3 * @var string */ protected $remote_url; /** * Base path. * * @since 2.0.7.3 * @var string */ protected $base_path; /** * Base URL. * * @since 2.0.7.3 * @var string */ protected $base_url; /** * Subfolder name. * * @since 2.0.7.3 * @var string */ protected $subfolder_name; /** * The fonts folder. * * @since 2.0.7.3 * @var string */ protected $fonts_folder; /** * The local stylesheet's path. * * @since 2.0.7.3 * @var string */ protected $local_stylesheet_path; /** * The local stylesheet's URL. * * @since 2.0.7.3 * @var string */ protected $local_stylesheet_url; /** * The remote CSS. * * @since 2.0.7.3 * @var string */ protected $remote_styles; /** * The final CSS. * * @since 2.0.7.3 * @var string */ protected $css; /** * Cleanup routine frequency. */ const CLEANUP_FREQUENCY = 'monthly'; /** * Preload fonts flag. * * @var boolean */ protected $preload = false; /** * Constructor. * * Get a new instance of the object for a new URL. * * @access public * @since 2.0.7.3 * @param string $url The remote URL. */ public function __construct( $url = '', $preload = false ) { $this->remote_url = $url; $this->preload = $preload; $upload = wp_upload_dir(); $this->base_path = $upload['basedir'] . '/magazine-blocks'; $this->base_url = $upload['baseurl'] . '/magazine-blocks'; $this->subfolder_name = 'fonts'; // Add a cleanup routine. $this->schedule_cleanup(); add_action( 'magazine_blocks_delete_fonts_folder', array( $this, 'magazine_blocks_delete_fonts_folder' ) ); } /** * Get the local URL which contains the styles. * * Fallback to the remote URL if we were unable to write the file locally. * * @access public * @since 2.0.7.3 * @return string */ public function get_url() { // Check if the local stylesheet exists. if ( $this->local_file_exists() ) { // Attempt to update the stylesheet. Return the local URL on success. if ( $this->write_stylesheet() ) { $this->preload_fonts(); return $this->get_local_stylesheet_url(); } } $this->preload_fonts(); // If the local file exists, return its URL, with a fallback to the remote URL. return file_exists( $this->get_local_stylesheet_path() ) ? $this->get_local_stylesheet_url() : $this->remote_url; } /** * Preload fonts. * * @return void */ protected function preload_fonts() { if ( ! $this->preload ) { return; } $styles = $this->get_styles(); if ( empty( $styles ) ) { return; } $css_doc = ( new \Sabberworm\CSS\Parser( $styles ) )->parse(); $rule_sets = array_filter( $css_doc->getAllRuleSets(), function( $rule_set ) { $rules = $rule_set->getRulesAssoc(); if ( isset( $rules['unicode-range'] ) && $rules['unicode-range'] instanceof \Sabberworm\CSS\Rule\Rule ) { return false !== strpos( (string) $rules['unicode-range']->getValue(), 'U+0000-00FF' ); } return false; } ); $data = array(); foreach ( $rule_sets as $rule_set ) { $family = null; $url = null; $rules = $rule_set->getRulesAssoc(); if ( isset( $rules['font-family'] ) && $rules['font-family'] instanceof \Sabberworm\CSS\Rule\Rule ) { $family = (string) $rules['font-family']->getValue(); } if ( isset( $rules['src'] ) && $rules['src'] instanceof \Sabberworm\CSS\Rule\Rule ) { $rule = $rules['src']->getValue(); if ( $rule instanceof \Sabberworm\CSS\Value\RuleValueList ) { foreach ( $rule->getListComponents() as $component ) { if ( $component instanceof \Sabberworm\CSS\Value\URL ) { $url = (string) $component->getURL(); break; } } } } if ( ! $family || ! $url ) { continue; } if ( isset( $data[ $family ] ) && ! in_array( $url, $data[ $family ], true ) ) { $data[ $family ][] = $url; } else { $data[ $family ] = array( $url ); } } if ( ! empty( $data ) ) { foreach ( magazine_blocks_array_flatten( array_values( $data ) ) as $local_url ) { echo '<link rel="preload" href="' . esc_url( $local_url ) . '" as="font" type="font/' . esc_attr( $this->font_format ) . '" crossorigin>'; } } } /** * Get the local stylesheet URL. * * @access public * @since 2.0.7.3 * @return string */ public function get_local_stylesheet_url() { if ( ! $this->local_stylesheet_url ) { $this->local_stylesheet_url = str_replace( $this->get_base_path(), $this->get_base_url(), $this->get_local_stylesheet_path() ); } return $this->local_stylesheet_url; } /** * Get styles with fonts downloaded locally. * * @access public * @since 2.0.7.3 * @return string */ public function get_styles() { // If we already have the local file, return its contents. $local_stylesheet_contents = $this->get_local_stylesheet_contents(); if ( $local_stylesheet_contents ) { return $local_stylesheet_contents; } // Get the remote URL contents. $this->remote_styles = $this->get_remote_url_contents(); // Get an array of locally-hosted files. $files = $this->get_local_files_from_css(); // Convert paths to URLs. foreach ( $files as $remote => $local ) { $files[ $remote ] = str_replace( $this->get_base_path(), $this->get_base_url(), $local ); } $this->css = str_replace( array_keys( $files ), array_values( $files ), $this->remote_styles ); $this->write_stylesheet(); return $this->css; } /** * Get local stylesheet contents. * * @access public * @since 2.0.7.3 * @return string|false Returns the remote URL contents. */ public function get_local_stylesheet_contents() { $local_path = $this->get_local_stylesheet_path(); // Check if the local stylesheet exists. if ( $this->local_file_exists() ) { // Attempt to update the stylesheet. Return false on fail. if ( ! $this->write_stylesheet() ) { return false; } } ob_start(); include $local_path; return ob_get_clean(); } /** * Get remote file contents. * * @access public * @since 2.0.7.3 * @return string Returns the remote URL contents. */ public function get_remote_url_contents() { /** * The user-agent we want to use. * * The default user-agent is the only one compatible with woff (not woff2) * which also supports unicode ranges. */ $user_agent = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_5) AppleWebKit/603.3.8 (KHTML, like Gecko) Version/10.1.2 Safari/603.3.8'; // Switch to a user-agent supporting woff2 if we don't need to support IE. if ( 'woff2' === $this->font_format ) { $user_agent = 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:73.0) Gecko/20100101 Firefox/73.0'; } // Get the response. $response = wp_remote_get( $this->remote_url, array( 'user-agent' => $user_agent ) ); // Early exit if there was an error. if ( is_wp_error( $response ) ) { return ''; } // Get the CSS from our response. $contents = wp_remote_retrieve_body( $response ); return $contents; } /** * Download files mentioned in our CSS locally. * * @access public * @since 2.0.7.3 * @return array Returns an array of remote URLs and their local counterparts. */ public function get_local_files_from_css() { $font_files = $this->get_remote_files_from_css(); $stored = get_site_option( 'downloaded_font_files', array() ); $change = false; // If in the end this is true, we need to update the cache option. if ( ! defined( 'FS_CHMOD_DIR' ) ) { define( 'FS_CHMOD_DIR', ( 0755 & ~ umask() ) ); } // If the fonts folder don't exist, create it. if ( ! file_exists( $this->get_fonts_folder() ) ) { $this->get_filesystem()->mkdir( $this->get_fonts_folder(), FS_CHMOD_DIR ); } foreach ( $font_files as $font_family => $files ) { // The folder path for this font-family. $folder_path = $this->get_fonts_folder() . '/' . $font_family; // If the folder doesn't exist, create it. if ( ! file_exists( $folder_path ) ) { $this->get_filesystem()->mkdir( $folder_path, FS_CHMOD_DIR ); } foreach ( $files as $url ) { // Get the filename. $filename = basename( wp_parse_url( $url, PHP_URL_PATH ) ); $font_path = $folder_path . '/' . $filename; /** * In Typekit, the filename will always be the same. We also need to check for query vars in their URLs. * They provide this font variation description that we can use https://github.com/typekit/fvd */ $queries = wp_parse_url( $url, PHP_URL_QUERY ); if ( ! empty( $queries ) ) { $query_args = array(); parse_str( $queries, $query_args ); if ( array_key_exists( 'fvd', $query_args ) ) { $font_path .= $query_args['fvd']; } } // Check if the file already exists. if ( file_exists( $font_path ) ) { // Skip if already cached. if ( isset( $stored[ $url ] ) ) { continue; } // Add file to the cache and change the $changed var to indicate we need to update the option. $stored[ $url ] = $font_path; $change = true; // Since the file exists we don't need to proceed with downloading it. continue; } /** * If we got this far, we need to download the file. */ // require file.php if the download_url function doesn't exist. if ( ! function_exists( 'download_url' ) ) { require_once wp_normalize_path( ABSPATH . '/wp-admin/includes/file.php' ); } // Download file to temporary location. $tmp_path = download_url( $url ); // Make sure there were no errors. if ( is_wp_error( $tmp_path ) ) { continue; } // Move temp file to final destination. $success = $this->get_filesystem()->move( $tmp_path, $font_path, true ); if ( $success ) { $stored[ $url ] = $font_path; $change = true; } } } // If there were changes, update the option. if ( $change ) { // Cleanup the option and then save it. foreach ( $stored as $url => $path ) { if ( ! file_exists( $path ) ) { unset( $stored[ $url ] ); } } update_site_option( 'downloaded_font_files', $stored ); } return $stored; } /** * Get font files from the CSS. * * @access public * @since 2.0.7.3 * @return array Returns an array of font-families and the font-files used. */ public function get_remote_files_from_css() { $font_faces = explode( '@font-face', $this->remote_styles ); $result = array(); // Loop all our font-face declarations. foreach ( $font_faces as $font_face ) { // Make sure we only process styles inside this declaration. $style = explode( '}', $font_face )[0]; // Sanity check. if ( false === strpos( $style, 'font-family' ) ) { continue; } // Get an array of our font-families. preg_match_all( '/font-family.*?\;/', $style, $matched_font_families ); // Get an array of our font-files. preg_match_all( '/url\(.*?\)/i', $style, $matched_font_files ); // Get the font-family name. $font_family = 'unknown'; if ( isset( $matched_font_families[0] ) && isset( $matched_font_families[0][0] ) ) { $font_family = rtrim( ltrim( $matched_font_families[0][0], 'font-family:' ), ';' ); $font_family = trim( str_replace( array( "'", ';' ), '', $font_family ) ); $font_family = sanitize_key( strtolower( str_replace( ' ', '-', $font_family ) ) ); } // Make sure the font-family is set in our array. if ( ! isset( $result[ $font_family ] ) ) { $result[ $font_family ] = array(); } // Get files for this font-family and add them to the array. foreach ( $matched_font_files as $match ) { // Sanity check. if ( ! isset( $match[0] ) ) { continue; } // Add the file URL. $font_family_url = rtrim( ltrim( $match[0], 'url(' ), ')' ); $font_family_url = str_replace( '"', '', $font_family_url ); // Make sure to convert relative URLs to absolute. $font_family_url = $this->get_absolute_path( $font_family_url ); $result[ $font_family ][] = $font_family_url; } // Make sure we have unique items. // We're using array_flip here instead of array_unique for improved performance. $result[ $font_family ] = array_flip( array_flip( $result[ $font_family ] ) ); } return $result; } /** * Write the CSS to the filesystem. * * @since 2.0.7.3 * @return string|false Returns the absolute path of the file on success, or false on fail. */ protected function write_stylesheet() { $file_path = $this->get_local_stylesheet_path(); $filesystem = $this->get_filesystem(); if ( ! defined( 'FS_CHMOD_DIR' ) ) { define( 'FS_CHMOD_DIR', ( 0755 & ~ umask() ) ); } // If the folder doesn't exist, create it. if ( ! file_exists( $this->get_fonts_folder() ) ) { $this->get_filesystem()->mkdir( $this->get_fonts_folder(), FS_CHMOD_DIR ); } // If the file doesn't exist, create it. Return false if it can not be created. if ( ! $filesystem->exists( $file_path ) && ! $filesystem->touch( $file_path ) ) { return false; } // If we got this far, we need to write the file. // Get the CSS. if ( ! $this->css ) { $this->get_styles(); } // Put the contents in the file. Return false if that fails. if ( ! $filesystem->put_contents( $file_path, $this->css ) ) { return false; } return $file_path; } /** * Get the stylesheet path. * * @access public * @since 2.0.7.3 * @return string */ public function get_local_stylesheet_path() { if ( ! $this->local_stylesheet_path ) { $this->local_stylesheet_path = $this->get_fonts_folder() . '/' . $this->get_local_stylesheet_filename() . '.css'; } return $this->local_stylesheet_path; } /** * Get the local stylesheet filename. * * This is a hash, generated from the site-URL, the wp-content path and the URL. * This way we can avoid issues with sites changing their URL, or the wp-content path etc. * * @access public * @since 2.0.7.3 * @return string */ public function get_local_stylesheet_filename() { return md5( $this->get_base_url() . $this->get_base_path() . $this->remote_url . $this->font_format ); } /** * Set the font-format to be used. * * @access public * @since 2.0.7.3 * @param string $format The format to be used. Use "woff" or "woff2". * @return void */ public function set_font_format( $format = 'woff2' ) { $this->font_format = $format; } /** * Check if the local stylesheet exists. * * @access public * @since 2.0.7.3 * @return bool */ public function local_file_exists() { return ( ! file_exists( $this->get_local_stylesheet_path() ) ); } /** * Get the base path. * * @access public * @since 2.0.7.3 * @return string */ public function get_base_path() { return $this->base_path; } /** * Get the base URL. * * @access public * @since 2.0.7.3 * @return string */ public function get_base_url() { return $this->base_url; } /** * Get the subfolder name. * * @access public * @since 2.0.7.3 * @return string */ public function get_subfolder_name() { return $this->subfolder_name; } /** * Get the folder for fonts. * * @access public * @return string */ public function get_fonts_folder() { if ( ! $this->fonts_folder ) { $this->fonts_folder = $this->get_base_path(); if ( $this->get_subfolder_name() ) { $this->fonts_folder .= '/' . $this->get_subfolder_name(); } } return $this->fonts_folder; } /** * Schedule a cleanup. * * Deletes the fonts files on a regular basis. * This way font files will get updated regularly, * and we avoid edge cases where unused files remain in the server. * * @access public * @since 2.0.7.3 * @return void */ public function schedule_cleanup() { if ( ! is_multisite() || ( is_multisite() && is_main_site() ) ) { if ( ! wp_next_scheduled( 'magazine_blocks_delete_fonts_folder' ) && ! wp_installing() ) { wp_schedule_event( time(), self::CLEANUP_FREQUENCY, 'delete_fonts_folder' ); } } } /** * Delete the fonts folder. * * This runs as part of a cleanup routine. * * @access public * @since 2.0.7.3 * @return bool */ public function delete_fonts_folder() { return $this->get_filesystem()->delete( $this->get_fonts_folder(), true ); } /** * Get the filesystem. * * @since 2.0.7.3 * @return \WP_Filesystem_Base */ protected function get_filesystem() { global $wp_filesystem; // If the filesystem has not been instantiated yet, do it here. if ( ! $wp_filesystem ) { if ( ! function_exists( 'WP_Filesystem' ) ) { require_once wp_normalize_path( ABSPATH . '/wp-admin/includes/file.php' ); } WP_Filesystem(); } return $wp_filesystem; } /** * Get an absolute URL from a relative URL. * * * @param string $url The URL. * * @return string */ protected function get_absolute_path( $url ) { // If dealing with a root-relative URL. if ( 0 === stripos( $url, '/' ) ) { $parsed_url = wp_parse_url( $this->remote_url ); return $parsed_url['scheme'] . '://' . $parsed_url['hostname'] . $url; } return $url; } }
| ver. 1.4 |
Github
|
.
| PHP 7.4.3-4ubuntu2.28 | Генерация страницы: 0.01 |
proxy
|
phpinfo
|
Настройка