2021-03-19 20:00:11 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
*/
|
|
|
|
|
|
|
|
declare( strict_types=1 );
|
|
|
|
|
|
|
|
namespace MediaWiki\Extension\TemplateStylesExtender;
|
|
|
|
|
2021-03-19 21:48:47 +00:00
|
|
|
use ConfigException;
|
2021-03-19 20:00:11 +00:00
|
|
|
use InvalidArgumentException;
|
|
|
|
use MediaWiki\Extension\TemplateStylesExtender\Matcher\VarNameMatcher;
|
2021-03-19 21:48:47 +00:00
|
|
|
use MediaWiki\MediaWikiServices;
|
2021-03-19 20:00:11 +00:00
|
|
|
use Wikimedia\CSS\Grammar\Alternative;
|
|
|
|
use Wikimedia\CSS\Grammar\FunctionMatcher;
|
|
|
|
use Wikimedia\CSS\Grammar\Juxtaposition;
|
|
|
|
use Wikimedia\CSS\Grammar\KeywordMatcher;
|
|
|
|
use Wikimedia\CSS\Grammar\MatcherFactory;
|
2021-03-19 20:57:10 +00:00
|
|
|
use Wikimedia\CSS\Grammar\Quantifier;
|
2021-03-19 20:00:11 +00:00
|
|
|
use Wikimedia\CSS\Grammar\WhitespaceMatcher;
|
|
|
|
use Wikimedia\CSS\Sanitizer\StylePropertySanitizer;
|
|
|
|
|
|
|
|
class TemplateStylesExtender {
|
|
|
|
|
2021-03-19 20:41:33 +00:00
|
|
|
/**
|
2021-03-20 08:03:23 +00:00
|
|
|
* Adds a css wide keyword matcher for css variables
|
|
|
|
* Matches 0-INF preceding css declarations at least one var( --content ) and 0-INF following declarations
|
2021-03-19 20:41:33 +00:00
|
|
|
*
|
|
|
|
* @param StylePropertySanitizer $propertySanitizer
|
2021-03-20 08:03:23 +00:00
|
|
|
* @param MatcherFactory $factory
|
2021-03-19 20:41:33 +00:00
|
|
|
*/
|
2021-03-20 08:03:23 +00:00
|
|
|
public function addVarSelector( StylePropertySanitizer $propertySanitizer, MatcherFactory $factory ): void {
|
2021-03-24 08:25:26 +00:00
|
|
|
$var = new FunctionMatcher(
|
|
|
|
'var',
|
|
|
|
new Juxtaposition( [
|
|
|
|
new WhitespaceMatcher( [ 'significant' => false ] ),
|
|
|
|
new VarNameMatcher(),
|
|
|
|
new WhitespaceMatcher( [ 'significant' => false ] ),
|
|
|
|
] )
|
|
|
|
);
|
|
|
|
|
2021-03-20 08:03:23 +00:00
|
|
|
$anyProperty = Quantifier::star(
|
|
|
|
new Alternative( [
|
2021-03-24 08:25:26 +00:00
|
|
|
$var,
|
2021-03-20 08:03:23 +00:00
|
|
|
$factory->color(),
|
|
|
|
$factory->image(),
|
|
|
|
$factory->length(),
|
|
|
|
$factory->integer(),
|
|
|
|
$factory->percentage(),
|
|
|
|
$factory->number(),
|
|
|
|
$factory->angle(),
|
|
|
|
$factory->frequency(),
|
|
|
|
$factory->resolution(),
|
|
|
|
$factory->position(),
|
|
|
|
$factory->cssSingleTimingFunction(),
|
|
|
|
$factory->comma(),
|
|
|
|
$factory->cssWideKeywords(),
|
2021-03-24 08:25:26 +00:00
|
|
|
new KeywordMatcher( [
|
|
|
|
'solid', 'double', 'dotted', 'dashed', 'wavy'
|
|
|
|
] )
|
2021-03-20 08:03:23 +00:00
|
|
|
] )
|
|
|
|
);
|
|
|
|
|
|
|
|
// Match anything*\s?[var anything|anything var]+\s?anything*(!important)?
|
2021-03-20 08:24:29 +00:00
|
|
|
// The problem is, that var() can be used more or less anywhere
|
|
|
|
// Setting ONLY var as a CssWideKeywordMatcher would limit the matching to one property
|
|
|
|
// E.g.: color: var( --color-base ); would work
|
|
|
|
// border: 1px var( --border-type ) black; would not
|
2021-03-20 08:03:23 +00:00
|
|
|
// So we need to construct a matcher that matches anything + var somewhere
|
2021-03-19 20:00:11 +00:00
|
|
|
$propertySanitizer->setCssWideKeywordsMatcher(
|
2021-03-19 21:48:47 +00:00
|
|
|
new Juxtaposition( [
|
2021-03-20 08:03:23 +00:00
|
|
|
$anyProperty,
|
2021-03-19 21:48:47 +00:00
|
|
|
new WhitespaceMatcher( [ 'significant' => false ] ),
|
|
|
|
Quantifier::plus(
|
2021-03-20 08:03:23 +00:00
|
|
|
new Alternative( [
|
|
|
|
new Juxtaposition( [ $var, $anyProperty ] ),
|
|
|
|
new Juxtaposition( [ $anyProperty, $var ] ),
|
|
|
|
] )
|
2021-03-19 21:48:47 +00:00
|
|
|
),
|
|
|
|
new WhitespaceMatcher( [ 'significant' => false ] ),
|
2021-03-20 08:03:23 +00:00
|
|
|
$anyProperty,
|
2021-03-19 21:48:47 +00:00
|
|
|
Quantifier::optional(
|
|
|
|
new KeywordMatcher( [ '!important' ] )
|
|
|
|
)
|
|
|
|
] ),
|
2021-03-19 20:00:11 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-03-19 20:41:33 +00:00
|
|
|
/**
|
|
|
|
* Adds the image-rendering matcher
|
|
|
|
* T222678
|
|
|
|
*
|
|
|
|
* @param StylePropertySanitizer $propertySanitizer
|
|
|
|
*/
|
2021-03-19 20:00:11 +00:00
|
|
|
public function addImageRendering( StylePropertySanitizer $propertySanitizer ): void {
|
|
|
|
try {
|
|
|
|
$propertySanitizer->addKnownProperties( [
|
|
|
|
'image-rendering' => new KeywordMatcher( [
|
|
|
|
'auto',
|
|
|
|
'crisp-edges',
|
|
|
|
'pixelated',
|
|
|
|
] )
|
|
|
|
] );
|
|
|
|
} catch ( InvalidArgumentException $e ) {
|
|
|
|
// Fail silently
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-19 20:41:33 +00:00
|
|
|
/**
|
|
|
|
* Adds the ruby-position and ruby-align matcher
|
|
|
|
* T277755
|
|
|
|
*
|
|
|
|
* @param StylePropertySanitizer $propertySanitizer
|
|
|
|
*/
|
2021-03-19 20:00:11 +00:00
|
|
|
public function addRuby( StylePropertySanitizer $propertySanitizer ): void {
|
|
|
|
try {
|
|
|
|
$propertySanitizer->addKnownProperties( [
|
|
|
|
'ruby-position' => new KeywordMatcher( [
|
|
|
|
'start',
|
|
|
|
'center',
|
|
|
|
'space-between',
|
|
|
|
'space-around',
|
|
|
|
] )
|
|
|
|
] );
|
|
|
|
|
|
|
|
$propertySanitizer->addKnownProperties( [
|
|
|
|
'ruby-align' => new KeywordMatcher( [
|
|
|
|
'over',
|
|
|
|
'under',
|
|
|
|
'inter-character',
|
|
|
|
] )
|
|
|
|
] );
|
|
|
|
} catch ( InvalidArgumentException $e ) {
|
|
|
|
// Fail silently
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-19 20:41:33 +00:00
|
|
|
/**
|
|
|
|
* Adds scroll-margin-* and scroll-padding-* matcher
|
|
|
|
* TODO: This is not well tested
|
|
|
|
* T271598
|
|
|
|
*
|
|
|
|
* @param StylePropertySanitizer $propertySanitizer
|
|
|
|
* @param MatcherFactory $factory
|
|
|
|
*/
|
2021-03-19 21:48:47 +00:00
|
|
|
public function addScrollMarginProperties( $propertySanitizer, $factory ): void {
|
2021-03-19 20:00:11 +00:00
|
|
|
$suffixes = [
|
|
|
|
'margin-block-end',
|
|
|
|
'margin-block-start',
|
|
|
|
'margin-block',
|
|
|
|
'margin-bottom',
|
|
|
|
'margin-inline-end',
|
|
|
|
'margin-inline-start',
|
|
|
|
'margin-inline',
|
|
|
|
'margin-left',
|
|
|
|
'margin-right',
|
|
|
|
'margin-top',
|
|
|
|
'margin',
|
|
|
|
'padding-block-end',
|
|
|
|
'padding-block-start',
|
|
|
|
'padding-block',
|
|
|
|
'padding-bottom',
|
|
|
|
'padding-inline-end',
|
|
|
|
'padding-inline-start',
|
|
|
|
'padding-inline',
|
|
|
|
'padding-left',
|
|
|
|
'padding-right',
|
|
|
|
'padding-top',
|
|
|
|
'padding',
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ( $suffixes as $suffix ) {
|
|
|
|
try {
|
|
|
|
$propertySanitizer->addKnownProperties( [
|
|
|
|
sprintf( 'scroll-%s', $suffix ) => new Alternative( [
|
|
|
|
$factory->length()
|
|
|
|
] )
|
|
|
|
] );
|
|
|
|
} catch ( InvalidArgumentException $e ) {
|
|
|
|
// Fail silently
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-03-19 21:48:47 +00:00
|
|
|
|
2021-05-21 19:51:07 +00:00
|
|
|
/**
|
|
|
|
* Adds the pointer-events matcher
|
|
|
|
*
|
|
|
|
* @param StylePropertySanitizer $propertySanitizer
|
|
|
|
*/
|
|
|
|
public function addPointerEvents( StylePropertySanitizer $propertySanitizer ): void {
|
|
|
|
try {
|
|
|
|
$propertySanitizer->addKnownProperties( [
|
|
|
|
'pointer-events' => new KeywordMatcher( [
|
|
|
|
'auto',
|
|
|
|
'none',
|
|
|
|
'visiblePainted',
|
|
|
|
'visibleFill',
|
|
|
|
'visibleStroke',
|
|
|
|
'visible',
|
|
|
|
'painted',
|
|
|
|
'fill',
|
|
|
|
'stroke',
|
|
|
|
'all',
|
|
|
|
] )
|
|
|
|
] );
|
|
|
|
} catch ( InvalidArgumentException $e ) {
|
|
|
|
// Fail silently
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-19 21:48:47 +00:00
|
|
|
/**
|
|
|
|
* Loads a config value for a given key from the main config
|
|
|
|
* Returns null on if an ConfigException was thrown
|
|
|
|
*
|
|
|
|
* @param string $key The config key
|
|
|
|
* @param null $default
|
|
|
|
* @return mixed|null
|
|
|
|
*/
|
|
|
|
public static function getConfigValue( string $key, $default = null ) {
|
|
|
|
try {
|
|
|
|
$value = MediaWikiServices::getInstance()->getMainConfig()->get( $key );
|
|
|
|
} catch ( ConfigException $e ) {
|
|
|
|
wfLogWarning(
|
|
|
|
sprintf(
|
|
|
|
'Could not get config for "$wg%s". %s', $key,
|
|
|
|
$e->getMessage()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
return $default;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $value;
|
|
|
|
}
|
2021-03-19 20:00:11 +00:00
|
|
|
}
|