-
Notifications
You must be signed in to change notification settings - Fork 4.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Duotone: Limit SVG filter output to used filters (#48995)
Adds the needed duotone code to the frontend output if that duotone definition is being used. * Adds a new WP_Duotone class that handles what filters need to be added to the page based on what duotone filters, if any, are in use. * Removes all duotone output initially via removing the actions that add them * Registers all duotone filters from global styles and store in the WP_Duotone class * Registers all block-level duotone filters from global styles and stores in the WP_Duotone class * Process all blocks via `render_block` filter to determine the correct duotone output needed. This is the same as before, but we're only outputting the duotone code if needed. * On the 'wp_enqueue_scripts' action, output all the necessary CSS for duotone filters collected from WP_Duotone::$output`. * On the 'wp_footer' action, output all necessary SVG for duotone filters from WP_Duotone::$output
- Loading branch information
Showing
6 changed files
with
358 additions
and
65 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,150 @@ | ||
<?php | ||
/** | ||
* WP_Duotone class | ||
* | ||
* @package gutenberg | ||
* @since 6.3.0 | ||
*/ | ||
|
||
/** | ||
* Manages which duotone filters need to be output on the page. | ||
* | ||
* @access public | ||
*/ | ||
class WP_Duotone { | ||
/** | ||
* An array of Duotone presets from global, theme, and custom styles. | ||
* | ||
* Example: | ||
* [ | ||
* 'blue-orange' => | ||
* [ | ||
* 'slug' => 'blue-orange', | ||
* 'colors' => [ '#0000ff', '#ffcc00' ], | ||
* ] | ||
* ], | ||
* … | ||
* ] | ||
* | ||
* @since 6.3.0 | ||
* @var array | ||
*/ | ||
static $global_styles_presets = array(); | ||
|
||
/** | ||
* An array of block names from global, theme, and custom styles that have duotone presets. We'll use this to quickly | ||
* check if a block being rendered needs to have duotone applied, and which duotone preset to use. | ||
* | ||
* Example: | ||
* [ | ||
* 'core/featured-image' => 'blue-orange', | ||
* … | ||
* ] | ||
* | ||
* @since 6.3.0 | ||
* @var array | ||
*/ | ||
static $global_styles_block_names = array(); | ||
|
||
/** | ||
* An array of Duotone SVG and CSS ouput needed for the frontend duotone rendering based on what is | ||
* being ouptput on the page. Organized by a slug of the preset/color group and the information needed | ||
* to generate the SVG and CSS at render. | ||
* | ||
* Example: | ||
* [ | ||
* 'blue-orange' => [ | ||
* 'slug' => 'blue-orange', | ||
* 'colors' => [ '#0000ff', '#ffcc00' ], | ||
* ], | ||
* 'wp-duotone-000000-ffffff-2' => [ | ||
* 'slug' => 'wp-duotone-000000-ffffff-2', | ||
* 'colors' => [ '#000000', '#ffffff' ], | ||
* ], | ||
* ] | ||
* | ||
* @since 6.3.0 | ||
* @var array | ||
*/ | ||
static $output = array(); | ||
|
||
/** | ||
* Get all possible duotone presets from global and theme styles and store as slug => [ colors array ] | ||
* We only want to process this one time. On block render we'll access and output only the needed presets for that page. | ||
*/ | ||
static function set_global_styles_presets() { | ||
// Get the per block settings from the theme.json. | ||
$tree = gutenberg_get_global_settings(); | ||
$presets_by_origin = _wp_array_get( $tree, array( 'color', 'duotone' ), array() ); | ||
|
||
foreach ( $presets_by_origin as $presets ) { | ||
foreach ( $presets as $preset ) { | ||
self::$global_styles_presets[ _wp_to_kebab_case( $preset['slug'] ) ] = array( | ||
'slug' => $preset['slug'], | ||
'colors' => $preset['colors'], | ||
); | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* Scrape all block names from global styles and store in WP_Duotone::$global_styles_block_names | ||
*/ | ||
static function set_global_style_block_names() { | ||
// Get the per block settings from the theme.json. | ||
$tree = WP_Theme_JSON_Resolver::get_merged_data(); | ||
$block_nodes = $tree->get_styles_block_nodes(); | ||
$theme_json = $tree->get_raw_data(); | ||
|
||
foreach ( $block_nodes as $block_node ) { | ||
// This block definition doesn't include any duotone settings. Skip it. | ||
if ( empty( $block_node['duotone'] ) ) { | ||
continue; | ||
} | ||
|
||
// Value looks like this: 'var(--wp--preset--duotone--blue-orange)' or 'var:preset|duotone|default-filter'. | ||
$duotone_attr_path = array_merge( $block_node['path'], array( 'filter', 'duotone' ) ); | ||
$duotone_attr = _wp_array_get( $theme_json, $duotone_attr_path, array() ); | ||
|
||
if ( empty( $duotone_attr ) ) { | ||
continue; | ||
} | ||
// If it has a duotone filter preset, save the block name and the preset slug. | ||
$slug = self::gutenberg_get_slug_from_attr( $duotone_attr ); | ||
|
||
if ( $slug && $slug !== $duotone_attr ) { | ||
self::$global_styles_block_names[ $block_node['name'] ] = $slug; | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* Take the inline CSS duotone variable from a block and return the slug. Handles styles slugs like: | ||
* var:preset|duotone|default-filter | ||
* var(--wp--preset--duotone--blue-orange) | ||
* | ||
* @param string $duotone_attr The duotone attribute from a block. | ||
* @return string The slug of the duotone preset or an empty string if no slug is found. | ||
*/ | ||
static function gutenberg_get_slug_from_attr( $duotone_attr ) { | ||
// Uses Branch Reset Groups `(?|…)` to return one capture group. | ||
preg_match( '/(?|var:preset\|duotone\|(\S+)|var\(--wp--preset--duotone--(\S+)\))/', $duotone_attr, $matches ); | ||
|
||
return ! empty( $matches[1] ) ? $matches[1] : ''; | ||
} | ||
|
||
/** | ||
* Check if we have a valid duotone preset. | ||
* | ||
* @param string $duotone_attr The duotone attribute from a block. | ||
* @return bool True if the duotone preset present and valid. | ||
*/ | ||
static function is_preset( $duotone_attr ) { | ||
$slug = WP_Duotone::gutenberg_get_slug_from_attr( $duotone_attr ); | ||
|
||
return array_key_exists( $slug, WP_Duotone::$global_styles_presets ); | ||
} | ||
} | ||
|
||
add_action( 'wp_loaded', array( 'WP_Duotone', 'set_global_styles_presets' ), 10 ); | ||
add_action( 'wp_loaded', array( 'WP_Duotone', 'set_global_style_block_names' ), 10 ); |
Oops, something went wrong.