File "CSSOutput.php"
Full Path: /home/pumpbmko/public_html/themes/momota/lib/colibriwp/src/Components/CSSOutput.php
File size: 7.19 KB
MIME-type: text/x-php
Charset: utf-8
<?php
namespace ColibriWP\Theme\Components;
use ColibriWP\Theme\ActiveCallback;
use ColibriWP\Theme\Core\ComponentBase;
use ColibriWP\Theme\Core\ComponentInterface;
use ColibriWP\Theme\Core\ConfigurableInterface;
use ColibriWP\Theme\Core\Hooks;
use ColibriWP\Theme\Core\Utils;
use ColibriWP\Theme\Customizer\Formatter;
use ColibriWP\Theme\Defaults;
use ColibriWP\Theme\Theme;
use Exception;
use function get_theme_mod;
class CSSOutput implements ComponentInterface {
const NO_MEDIA = '__colibri__no__media__';
// changed from gutentag
const GRADIENT_VALUE_PATTERN = 'linear-gradient(#angle#deg,#steps.0.color# #steps.0.position#% ,#steps.1.color# #steps.1.position#%)';
public function render() {
Hooks::prefixed_add_filter(
'customizer_additional_js_data',
function ( $data ) {
$data['css_selectors_prefix'] = $this->themePrefix();
return $data;
}
);
?>
<style data-kubio-theme-style="true">
<?php echo $this->getCSSOutput(); ?>
</style>
<?php
}
public function themePrefix() {
return 'html.' . Theme::slug() . '-theme #colibri ';
}
public function getCSSOutput() {
return $this->generateCSSOutput();
}
private function generateCSSOutput() {
$content = '';
$data = $this->getCSSData();
$medias = $this->groupDataByMedia( $data );
if ( array_key_exists( self::NO_MEDIA, $medias ) ) {
$content .= $this->generateCSSOutputForMedia( '', $medias[ self::NO_MEDIA ] );
}
if ( array_key_exists( self::mobileMedia(), $medias ) ) {
$content .= $this->generateCSSOutputForMedia(
self::mobileMedia(),
$medias[ self::mobileMedia() ]
);
}
if ( array_key_exists( self::tabletMedia(), $medias ) ) {
$content .= $this->generateCSSOutputForMedia(
self::tabletMedia(),
$medias[ self::tabletMedia() ]
);
}
if ( array_key_exists( self::desktopMedia(), $medias ) ) {
$content .= $this->generateCSSOutputForMedia(
self::desktopMedia(),
$medias[ self::desktopMedia() ]
);
}
return $content;
}
/**
* @return array
* @throws Exception
*/
private function getCSSData() {
$data = array();
$components = Theme::getInstance()->getRepository()->getAllDefinitions();
foreach ( $components as $key => $component ) {
$interfaces = class_implements( $component );
if ( array_key_exists( ConfigurableInterface::class, $interfaces ) ) {
/** @var ConfigurableInterface $component */
$opts = (array) $component::options();
if ( array_key_exists( 'settings', $opts ) && is_array( $opts['settings'] ) ) {
foreach ( $opts['settings'] as $mod => $setting ) {
if ( array_key_exists( 'css_output', $setting ) && is_array( $setting['css_output'] ) ) {
if ( $this->activeRulesAreMet( $setting, $component ) ) {
$default = isset( $setting['default'] ) ? $setting['default'] : '';
$control_type = Utils::pathGet( $setting, 'control.type' );
foreach ( $setting['css_output'] as $css_output ) {
$rule = $this->prepareOutputDataForMod(
$css_output,
$mod,
$default,
$control_type
);
if ( ! ( empty( $rule['selector'] ) || empty( $rule['property'] ) ) ) {
$data[] = $rule;
}
}
}
}
}
}
}
}
return $data;
}
/**
* @param $settings
* @param $component
*
* @return bool
* @throws Exception
*/
private function activeRulesAreMet( $settings, $component ) {
if ( ! array_key_exists( 'active_rules', $settings ) ) {
return true;
}
$activeCallback = ( new ActiveCallback() )->setComponent( $component )->setRules( $settings['active_rules'] );
return $activeCallback->applyRules();
}
private function prepareOutputDataForMod( $output, $mod, $default = '', $control_type = '' ) {
$output = static::normalizeOutput( $output );
$mod_value = get_theme_mod( $mod, $default );
$mod_value = Formatter::sanitizeControlValue( $control_type, $mod_value );
if ( $control_type !== 'switch' && empty( $mod_value ) && $mod_value !== 0 ) {
return '';
}
if ( isset( $output['value'] ) && is_array( $output['value'] ) ) {
if ( isset( $output['value'][ $mod_value ] ) ) {
$output['value'] = $output['value'][ $mod_value ];
}
} else {
$output['value'] = ComponentBase::mabyDeserializeModValue( $mod_value );
}
return $output;
}
public static function normalizeOutput( $output ) {
return array_replace(
array(
'selector' => '',
'media' => self::NO_MEDIA,
'property' => '',
'value_pattern' => '%s',
),
$output
);
}
private function groupDataByMedia( $data ) {
$medias = array();
foreach ( $data as $item ) {
if ( ! array_key_exists( $item['media'], $medias ) ) {
$medias[ $item['media'] ] = array();
}
$medias[ $item['media'] ][] = $item;
}
return $medias;
}
private function generateCSSOutputForMedia( $media, $data ) {
$selectors = array();
foreach ( $data as $item ) {
$selector = $item['selector'];
if ( is_array( $selector ) ) {
$selector = implode( ',', $selector );
}
if ( ! array_key_exists( $selector, $selectors ) ) {
$selectors[ $selector ] = array();
}
$value = $this->getValue( $item );
if ( $value !== null ) {
$selectors[ $selector ][ $item['property'] ] = $value;
}
}
$content = '';
foreach ( $selectors as $selector => $rules ) {
$rules_parts = array();
foreach ( $rules as $prop => $value ) {
$rules_parts[] = "{$prop}:{$value}";
}
$rules = implode( ';', $rules_parts );
$content .= $this->themePrefix() . "{$selector}{{$rules}}";
}
if ( ! empty( $media ) ) {
$content = "{$media}{{$content}}";
}
return $content;
}
private function getValue( $item ) {
if ( is_array( $item['value'] ) ) {
$that = $this;
$item_value = $item['value'];
$value = preg_replace_callback(
'/#([\w\.]+)#/',
function ( $matches ) use ( $item_value, $that ) {
return $that->getValueInTree( $item_value, $matches[1] );
},
$item['value_pattern']
);
return $value;
} else {
if ( is_bool( $item['value'] ) ) {
if ( $item['value'] ) {
if ( isset( $item['true_value'] ) ) {
return $item['true_value'];
} else {
return null;
}
} else {
if ( isset( $item['false_value'] ) ) {
return $item['false_value'];
} else {
return null;
}
}
}
}
if ( $item['value'] || $item['value'] === 0 ) {
return sprintf( $item['value_pattern'], $item['value'] );
}
}
private function getValueInTree( $tree, $path ) {
$path = explode( '.', $path );
$result = $tree;
while ( count( $path ) && is_array( $tree ) ) {
$next_key = array_shift( $path );
if ( array_key_exists( $next_key, $result ) ) {
$result = $result[ $next_key ];
} else {
$result = '';
break;
}
}
if ( is_array( $result ) ) {
$result = '';
}
return $result;
}
public static function mobileMedia() {
return Defaults::get( 'mobile_media', '@media (max-width: 767px)' );
}
public static function tabletMedia() {
return Defaults::get( 'tablet_media', '@media (min-width: 768px) and (max-width: 1023px)' );
}
public static function desktopMedia() {
return Defaults::get( 'tablet_media', '@media (min-width: 1024px)' );
}
}