all-in-one-event-calendar/lib/less/lessphp.php

488 lines
16 KiB
PHP
Raw Normal View History

2017-03-16 16:59:53 +01:00
<?php
/**
* Class that handles less related functions.
*
* @author Time.ly Network Inc.
* @since 2.0
*
* @package AI1EC
* @subpackage AI1EC.Less
*/
class Ai1ec_Less_Lessphp extends Ai1ec_Base {
2017-11-09 17:36:04 +01:00
/**
*
* @var string
*/
const DB_KEY_FOR_LESS_VARIABLES = "ai1ec_less_variables";
/**
*
* @var lessc
*/
private $lessc;
/**
*
* @var array
*/
private $files = array();
/**
*
* @var string
*/
private $unparsed_variable_file;
/**
*
* @var string
*/
private $parsed_css;
/**
*
* @var string
*/
private $default_theme_url;
/**
*
* @var Ai1ec_File_Less
*/
private $variable_file;
/**
* Variables used for compilation.
*
* @var array
*/
private $variables;
public function __construct(
Ai1ec_Registry_Object $registry,
$default_theme_url = AI1EC_DEFAULT_THEME_URL
) {
parent::__construct( $registry );
$this->lessc = $this->_registry->get( 'lessc' );
$this->lessc->setFormatter( 'compressed' );
$this->default_theme_url = $this->sanitize_default_theme_url( $default_theme_url );
$this->parsed_css = '';
$this->variables = array();
$this->files = array(
'style.less',
'event.less',
'calendar.less',
);
}
/**
*
* @param Ai1ec_File_Less $file
*/
public function set_variable_file( Ai1ec_File_Less $file ) {
$this->variable_file = $file;
}
/**
*
* @param Ai1ec_File_Less $file
*/
public function add_file( $file ) {
$this->files[] = $file;
}
/**
* Parse all the less files resolving the dependencies.
*
* @param array $variables
* @param bool $compile_core If set to true, it forces compilation of core CSS only, suitable for shipping.
* @throws Ai1ec_File_Not_Found_Exception|Exception
* @throws Exception
* @return string
*/
public function parse_less_files( array $variables = null, $compile_core = false ) {
// If no variables are passed, initialize from DB, config file, and
// extension injections in one call.
if ( empty( $variables ) ) {
$variables = $this->get_saved_variables( false );
}
// convert the variables to key / value
$variables = $this->convert_less_variables_for_parsing( $variables );
// Inject additional constants from extensions
$variables = apply_filters( 'ai1ec_less_constants', $variables );
// Use this variables for hashmap purposes.
$this->variables = $variables;
// Load the static variables defined in the theme's variables.less file.
$this->load_static_theme_variables();
$loader = $this->_registry->get( 'theme.loader' );
//Allow extensions to add their own LESS files.
$this->files = apply_filters( 'ai1ec_less_files', $this->files );
$this->files[] = 'override.less';
// Find out the active theme URL.
$option = $this->_registry->get( 'model.option' );
$theme = $option->get( 'ai1ec_current_theme' );
$this->lessc->addImportDir(
$theme['theme_dir'] . DIRECTORY_SEPARATOR . 'less'
);
$import_dirs = array();
foreach ( $this->files as $file ) {
$file_to_parse = null;
try {
// Get the filename following our fallback convention
$file_to_parse = $loader->get_file( $file );
} catch ( Ai1ec_Exception $e ) {
// We let child themes override styles of Vortex.
// So there is no fallback for override and we can continue.
if ( $file !== 'override.less' ) {
throw $e;
} else {
// It's an override, skip it.
continue;
}
}
// We prepend the unparsed variables.less file we got earlier.
// We do this as we do not import that anymore in the less files.
$this->unparsed_variable_file .= $file_to_parse->get_content();
// Set the import directories for the file. Includes current directory of
// file as well as theme directory in core. This is important for
// dependencies to be resolved correctly.
$dir = dirname( $file_to_parse->get_name() );
if ( ! isset( $import_dirs[$dir] ) ) {
$import_dirs[$dir] = true;
$this->lessc->addImportDir( $dir );
}
}
$variables['fontdir'] = '~"' .
Ai1ec_Http_Response_Helper::remove_protocols(
$theme['theme_url']
) . '/font"';
$variables['fontdir_default'] = '~"' .
Ai1ec_Http_Response_Helper::remove_protocols(
$this->default_theme_url
) . 'font"';
$variables['imgdir'] = '~"' .
Ai1ec_Http_Response_Helper::remove_protocols(
$theme['theme_url']
) . '/img"';
$variables['imgdir_default'] = '~"' .
Ai1ec_Http_Response_Helper::remove_protocols(
$this->default_theme_url
) . 'img"';
if ( true === $compile_core ) {
$variables['fontdir'] = '~"../font"';
$variables['fontdir_default'] = '~"../font"';
$variables['imgdir'] = '~"../img"';
$variables['imgdir_default'] = '~"../img"';
}
try {
$this->parsed_css = $this->lessc->parse(
$this->unparsed_variable_file,
$variables
);
} catch ( Exception $e ) {
throw $e;
}
// Replace font placeholders
$this->parsed_css = preg_replace_callback(
'/__BASE64_FONT_([a-zA-Z0-9]+)_(\S+)__/m',
array( $this, 'load_font_base64' ),
$this->parsed_css
);
return $this->parsed_css;
}
/**
* Check LESS variables are stored in the options table; if not, initialize
* with defaults from config file and extensions.
*/
public function initialize_less_variables_if_not_set() {
$variables = $this->_registry->get( 'model.option' )->get(
self::DB_KEY_FOR_LESS_VARIABLES,
array()
);
if ( empty( $variables ) ) {
// Initialize variables with defaults from config file and extensions,
// omitting descriptions.
$variables = $this->get_saved_variables( false );
// Save the new/updated variable array back to the database.
$this->_registry->get( 'model.option' )->set(
self::DB_KEY_FOR_LESS_VARIABLES,
$variables
);
}
}
/**
* Invalidates CSS cache if ai1ec_invalidate_css_cache option was flagged.
* Deletes flag afterwards.
*/
public function invalidate_css_cache_if_requested() {
$option = $this->_registry->get( 'model.option' );
if (
$option->get( 'ai1ec_invalidate_css_cache' ) ||
Ai1ec_Css_Frontend::PARSE_LESS_FILES_AT_EVERY_REQUEST
) {
$css_controller = $this->_registry->get( 'css.frontend' );
$css_controller->invalidate_cache( null, true );
$option->delete( 'ai1ec_invalidate_css_cache' );
}
}
/**
* After updating core themes, we also need to update the LESS variables with
* the new ones as they may have changed. This function assumes that the
* user_variables.php file in the active theme and/or parent theme has just
* been updated.
*/
public function update_less_variables_on_theme_update() {
// Get old variables from the DB.
$saved_variables = $this->get_saved_variables( false );
// Get the new variables from file.
$new_variables = $this->get_less_variable_data_from_config_file();
foreach ( $new_variables as $name => $attributes ) {
// If the variable already exists, keep the old value.
if ( isset( $saved_variables[$name] ) ) {
$new_variables[$name]['value'] = $saved_variables[$name]['value'];
}
}
// Save the new variables to the DB.
$this->_registry->get( 'model.option' )->set(
self::DB_KEY_FOR_LESS_VARIABLES,
$new_variables
);
}
/**
* Get the theme variables from the theme user_variables.php file; also inject
* any other variables provided by extensions.
*
* @return array
*/
public function get_less_variable_data_from_config_file() {
// Load the file to parse using the theme loader to select the right file.
$loader = $this->_registry->get( 'theme.loader' );
$file = $loader->get_file( 'less/user_variables.php', array(), false );
// This variables are returned by evaluating the PHP file.
$variables = $file->get_content();
// Inject extension variables into this array.
return apply_filters( 'ai1ec_less_variables', $variables );
}
/**
* Returns compilation specific hashmap.
*
* @return array Hashmap.
*/
public function get_less_hashmap() {
foreach ( $this->variables as $key => $value ) {
if ( 'fontdir_' === substr( $key, 0, 8 ) ) {
unset( $this->variables[$key] );
}
}
$hashmap = $this->_registry->get(
'filesystem.misc'
)->build_current_theme_hashmap();
$variables = $this->variables;
ksort( $variables );
return array(
'variables' => $variables,
'files' => $hashmap,
);
}
/**
* Returns whether LESS compilation should be performed or not.
*
* @param array|null $variables LESS variables.
*
* @return bool Result.
*
* @throws Ai1ec_Bootstrap_Exception
*/
public function is_compilation_needed( $variables = array() ) {
if (
apply_filters( 'ai1ec_always_recompile_less', false ) ||
(
defined( 'AI1EC_DEBUG' ) &&
AI1EC_DEBUG
)
) {
return true;
}
if ( null === $variables ) {
$variables = array();
}
/* @var $misc Ai1ec_Filesystem_Misc */
$misc = $this->_registry->get( 'filesystem.misc' );
$cur_hashmap = $misc->get_current_theme_hashmap();
if ( empty( $variables ) ) {
$variables = $this->get_saved_variables( false );
}
$variables = $this->convert_less_variables_for_parsing( $variables );
$variables = apply_filters( 'ai1ec_less_constants', $variables );
$variables = $this->_compilation_check_clear_variables( $variables );
ksort( $variables );
if (
null === $cur_hashmap ||
$variables !== $cur_hashmap['variables']
) {
return true;
}
$file_hashmap = $misc->build_current_theme_hashmap();
return ! $misc->compare_hashmaps( $file_hashmap, $cur_hashmap['files'] );
}
/**
* Gets the saved variables from the database, and make sure all variables
* are set correctly as required by config file and any extensions. Also
* adds translations of variable descriptions as required at runtime.
*
* @param $with_description bool Whether to return variables with translated descriptions
* @return array
*/
public function get_saved_variables( $with_description = true ) {
// We don't store description in options table, so find it in current config
// file. Variables from extensions are already injected during this call.
$variables_from_config = $this->get_less_variable_data_from_config_file();
// Fetch current variable settings from options table.
$variables = $this->_registry->get( 'model.option' )->get(
self::DB_KEY_FOR_LESS_VARIABLES,
array()
);
// Generate default variable array from the config file, and union these
// with any saved variables to make sure all required variables are set.
$variables += $variables_from_config;
// Add the description at runtime so that it can be translated.
foreach ( $variables as $name => $attrs ) {
// Also filter out any legacy variables that are no longer found in
// current config file (exceptions thrown if this is not handled here).
if ( ! isset( $variables_from_config[$name] ) ) {
unset( $variables[$name] );
}
else {
// If description is requested and is available in config file, use it.
if (
$with_description &&
isset( $variables_from_config[$name]['description'] )
) {
$variables[$name]['description'] =
$variables_from_config[$name]['description'];
} else {
unset( $variables[$name]['description'] );
}
}
}
return $variables;
}
/**
* Tries to fix the double url as of AIOEC-882
*
* @param string $url
* @return string
*/
public function sanitize_default_theme_url( $url ) {
$pos_http = strrpos( $url, 'http://');
$pos_https = strrpos( $url, 'https://');
// if there are two http
if( 0 !== $pos_http ) {
// cut of the first one
$url = substr( $url, $pos_http );
} else if ( 0 !== $pos_https ) {
$url = substr( $url, $pos_https );
}
return $url;
}
/**
* Drop extraneous attributes from variable array and convert to simple
* key-value pairs required by the LESS parser.
*
* @param array $variables
* @return array
*/
private function convert_less_variables_for_parsing( array $variables ) {
$converted_variables = array();
foreach ( $variables as $variable_name => $variable_params ) {
$converted_variables[$variable_name] = $variable_params['value'];
}
return $converted_variables;
}
/**
* Different themes need different variables.less files. This uses the theme
* loader (searches active theme first, then default) to load it unparsed.
*/
private function load_static_theme_variables() {
$loader = $this->_registry->get( 'theme.loader' );
$file = $loader->get_file( 'variables.less', array(), false );
$this->unparsed_variable_file = $file->get_content();
}
/**
* Load font as base 64 encoded
*
* @param array $matches
* @return string
*/
private function load_font_base64( $matches ) {
// Find out the active theme URL.
$option = $this->_registry->get( 'model.option' );
$theme = $option->get( 'ai1ec_current_theme' );
$dirs = apply_filters(
'ai1ec_font_dirs',
array(
'AI1EC' => array(
$theme['theme_dir'] . DIRECTORY_SEPARATOR . 'font',
AI1EC_DEFAULT_THEME_PATH . DIRECTORY_SEPARATOR . 'font',
)
)
);
$directories = $dirs[$matches[1]];
foreach ( $directories as $dir ) {
$font_file = $dir . DIRECTORY_SEPARATOR . $matches[2];
if ( file_exists( $font_file ) ) {
return base64_encode( file_get_contents( $font_file ) );
}
}
return '';
}
/**
* Removes fontdir variables added by add-ons.
*
* @param array $variables Input variables array.
*
* @return array Modified variables.
*/
protected function _compilation_check_clear_variables( array $variables ) {
foreach ( $variables as $key => $value ) {
if ( 'fontdir_' === substr( $key, 0, 8 ) ) {
unset( $variables[$key] );
}
}
return $variables;
}
2017-03-16 16:59:53 +01:00
}