From a19f5e4e6df003a6aab50a5b0e72e4ae3265c364 Mon Sep 17 00:00:00 2001
From: Aaron Robertshaw <60436221+aaronrobertshaw@users.noreply.github.com>
Date: Fri, 20 Dec 2024 12:30:40 +1000
Subject: [PATCH 1/5] Revert "Global Styles: Fix handling of booleans when
stabilizing block supports (#67552)"
This reverts commit 4335c452609a747faffd74f25b4846d717679e4c.
---
backport-changelog/6.8/7069.md | 1 -
lib/compat/wordpress-6.8/blocks.php | 25 +++----
phpunit/block-supports/border-test.php | 100 -------------------------
3 files changed, 10 insertions(+), 116 deletions(-)
diff --git a/backport-changelog/6.8/7069.md b/backport-changelog/6.8/7069.md
index 3e734637ddbb2f..69edbde58902d1 100644
--- a/backport-changelog/6.8/7069.md
+++ b/backport-changelog/6.8/7069.md
@@ -3,4 +3,3 @@ https://github.com/WordPress/wordpress-develop/pull/7069
* https://github.com/WordPress/gutenberg/pull/63401
* https://github.com/WordPress/gutenberg/pull/66918
* https://github.com/WordPress/gutenberg/pull/67018
-* https://github.com/WordPress/gutenberg/pull/67552
diff --git a/lib/compat/wordpress-6.8/blocks.php b/lib/compat/wordpress-6.8/blocks.php
index dc8747c04aec2f..ba321b7ac987a7 100644
--- a/lib/compat/wordpress-6.8/blocks.php
+++ b/lib/compat/wordpress-6.8/blocks.php
@@ -71,10 +71,6 @@ function gutenberg_stabilize_experimental_block_supports( $args ) {
}
$stabilize_config = function ( $unstable_config, $stable_support_key ) use ( $experimental_support_properties, $common_experimental_properties ) {
- if ( ! is_array( $unstable_config ) ) {
- return $unstable_config;
- }
-
$stable_config = array();
foreach ( $unstable_config as $key => $value ) {
// Get stable key from support-specific map, common properties map, or keep original.
@@ -120,19 +116,18 @@ function gutenberg_stabilize_experimental_block_supports( $args ) {
( $key_positions[ $support ] ?? PHP_INT_MAX ) <
( $key_positions[ $stable_support_key ] ?? PHP_INT_MAX );
- /*
- * To merge the alternative support config effectively, it also needs to be
- * stabilized before merging to keep stabilized and experimental flags in
- * sync.
- */
- $args['supports'][ $stable_support_key ] = $stabilize_config( $args['supports'][ $stable_support_key ], $stable_support_key );
- // Prevents reprocessing this support as it was stabilized above.
- $done[ $stable_support_key ] = true;
-
- if ( is_array( $stable_config ) && is_array( $args['supports'][ $stable_support_key ] ) ) {
- $stable_config = $experimental_first
+ if ( is_array( $args['supports'][ $stable_support_key ] ) ) {
+ /*
+ * To merge the alternative support config effectively, it also needs to be
+ * stabilized before merging to keep stabilized and experimental flags in
+ * sync.
+ */
+ $args['supports'][ $stable_support_key ] = $stabilize_config( $args['supports'][ $stable_support_key ], $stable_support_key );
+ $stable_config = $experimental_first
? array_merge( $stable_config, $args['supports'][ $stable_support_key ] )
: array_merge( $args['supports'][ $stable_support_key ], $stable_config );
+ // Prevents reprocessing this support as it was merged above.
+ $done[ $stable_support_key ] = true;
} else {
$stable_config = $experimental_first
? $args['supports'][ $stable_support_key ]
diff --git a/phpunit/block-supports/border-test.php b/phpunit/block-supports/border-test.php
index 510633b48aab59..6ec43b369d9a2a 100644
--- a/phpunit/block-supports/border-test.php
+++ b/phpunit/block-supports/border-test.php
@@ -730,104 +730,4 @@ public function test_should_stabilize_border_supports_using_order_based_merge()
);
$this->assertSame( $expected, $actual, 'Merged stabilized border block support config does not match when stable keys are first.' );
}
-
- /**
- * Tests that boolean border support configurations are handled correctly.
- *
- * @dataProvider data_boolean_border_supports
- *
- * @param array $supports The supports configuration to test.
- * @param boolean|array $expected_value The expected final border support value.
- */
- public function test_should_handle_boolean_border_supports( $supports, $expected_value ) {
- $args = array(
- 'supports' => $supports,
- );
-
- $actual = gutenberg_stabilize_experimental_block_supports( $args );
-
- $this->assertSame( $expected_value, $actual['supports']['border'] );
- }
-
- /**
- * Data provider for boolean border support tests.
- *
- * @return array Test parameters.
- */
- public function data_boolean_border_supports() {
- return array(
- 'experimental true only' => array(
- array(
- '__experimentalBorder' => true,
- ),
- true,
- ),
- 'experimental false only' => array(
- array(
- '__experimentalBorder' => false,
- ),
- false,
- ),
- 'experimental true before stable false' => array(
- array(
- '__experimentalBorder' => true,
- 'border' => false,
- ),
- false,
- ),
- 'stable true before experimental false' => array(
- array(
- 'border' => true,
- '__experimentalBorder' => false,
- ),
- false,
- ),
- 'experimental array before stable boolean' => array(
- array(
- '__experimentalBorder' => array(
- 'color' => true,
- 'width' => true,
- ),
- 'border' => false,
- ),
- false,
- ),
- 'stable array before experimental boolean' => array(
- array(
- 'border' => array(
- 'color' => true,
- 'width' => true,
- ),
- '__experimentalBorder' => true,
- ),
- true,
- ),
- 'experimental boolean before stable array' => array(
- array(
- '__experimentalBorder' => true,
- 'border' => array(
- 'color' => true,
- 'width' => true,
- ),
- ),
- array(
- 'color' => true,
- 'width' => true,
- ),
- ),
- 'stable boolean before experimental array' => array(
- array(
- 'border' => false,
- '__experimentalBorder' => array(
- 'color' => true,
- 'width' => true,
- ),
- ),
- array(
- 'color' => true,
- 'width' => true,
- ),
- ),
- );
- }
}
From b1915d978d49bcc087e884cb300b544bc26e319b Mon Sep 17 00:00:00 2001
From: Aaron Robertshaw <60436221+aaronrobertshaw@users.noreply.github.com>
Date: Fri, 20 Dec 2024 12:31:05 +1000
Subject: [PATCH 2/5] Revert "Block Supports: Extend stabilization to common
experimental block support flags (#67018)"
This reverts commit 5513d7aecf085a7858cae0492acf213d65a436f6.
---
backport-changelog/6.8/7069.md | 1 -
docs/explanations/architecture/styles.md | 16 +-
lib/compat/wordpress-6.8/blocks.php | 145 +++----
packages/block-editor/src/hooks/border.js | 10 +-
packages/block-editor/src/hooks/color.js | 2 +-
packages/block-editor/src/hooks/dimensions.js | 4 +-
packages/block-editor/src/hooks/style.js | 18 +-
packages/block-editor/src/hooks/test/style.js | 3 +-
packages/block-editor/src/hooks/typography.js | 2 +-
packages/block-editor/src/hooks/utils.js | 2 +-
packages/blocks/src/api/constants.js | 12 +-
.../blocks/src/store/process-block-type.js | 192 +++------
.../src/store/test/process-block-type.js | 395 ++++++++----------
.../components/global-styles/screen-block.js | 5 +-
packages/server-side-render/README.md | 2 +-
phpunit/block-supports/border-test.php | 173 +-------
16 files changed, 341 insertions(+), 641 deletions(-)
diff --git a/backport-changelog/6.8/7069.md b/backport-changelog/6.8/7069.md
index 69edbde58902d1..8b4ad69db02e75 100644
--- a/backport-changelog/6.8/7069.md
+++ b/backport-changelog/6.8/7069.md
@@ -2,4 +2,3 @@ https://github.com/WordPress/wordpress-develop/pull/7069
* https://github.com/WordPress/gutenberg/pull/63401
* https://github.com/WordPress/gutenberg/pull/66918
-* https://github.com/WordPress/gutenberg/pull/67018
diff --git a/docs/explanations/architecture/styles.md b/docs/explanations/architecture/styles.md
index 5f5e73d1372f7b..68f09f04d21d32 100644
--- a/docs/explanations/architecture/styles.md
+++ b/docs/explanations/architecture/styles.md
@@ -37,10 +37,8 @@ The user may change the state of this block by applying different styles: a text
After some user modifications to the block, the initial markup may become something like this:
```html
-
+
```
This is what we refer to as "user-provided block styles", also know as "local styles" or "serialized styles". Essentially, each tool (font size, color, etc) ends up adding some classes and/or inline styles to the block markup. The CSS styling for these classes is part of the block, global, or theme stylesheets.
@@ -125,7 +123,7 @@ The block supports API only serializes the font size value to the wrapper, resul
This is an active area of work you can follow [in the tracking issue](https://github.com/WordPress/gutenberg/issues/38167). The linked proposal is exploring a different way to serialize the user changes: instead of each block support serializing its own data (for example, classes such as `has-small-font-size`, `has-green-color`) the idea is the block would get a single class instead (for example, `wp-style-UUID`) and the CSS styling for that class will be generated in the server by WordPress.
-While work continues in that proposal, there's an escape hatch, an experimental option block authors can use. Any block support can skip the serialization to HTML markup by using `skipSerialization`. For example:
+While work continues in that proposal, there's an escape hatch, an experimental option block authors can use. Any block support can skip the serialization to HTML markup by using `__experimentalSkipSerialization`. For example:
```json
{
@@ -134,7 +132,7 @@ While work continues in that proposal, there's an escape hatch, an experimental
"supports": {
"typography": {
"fontSize": true,
- "skipSerialization": true
+ "__experimentalSkipSerialization": true
}
}
}
@@ -142,7 +140,7 @@ While work continues in that proposal, there's an escape hatch, an experimental
This means that the typography block support will do all of the things (create a UI control, bind the block attribute to the control, etc) except serializing the user values into the HTML markup. The classes and inline styles will not be automatically applied to the wrapper and it is the block author's responsibility to implement this in the `edit`, `save`, and `render_callback` functions. See [this issue](https://github.com/WordPress/gutenberg/issues/28913) for examples of how it was done for some blocks provided by WordPress.
-Note that, if `skipSerialization` is enabled for a group (typography, color, spacing) it affects _all_ block supports within this group. In the example above _all_ the properties within the `typography` group will be affected (e.g. `fontSize`, `lineHeight`, `fontFamily` .etc).
+Note that, if `__experimentalSkipSerialization` is enabled for a group (typography, color, spacing) it affects _all_ block supports within this group. In the example above _all_ the properties within the `typography` group will be affected (e.g. `fontSize`, `lineHeight`, `fontFamily` .etc).
To enable for a _single_ property only, you may use an array to declare which properties are to be skipped. In the example below, only `fontSize` will skip serialization, leaving other items within the `typography` group (e.g. `lineHeight`, `fontFamily` .etc) unaffected.
@@ -154,7 +152,7 @@ To enable for a _single_ property only, you may use an array to declare which pr
"typography": {
"fontSize": true,
"lineHeight": true,
- "skipSerialization": [ "fontSize" ]
+ "__experimentalSkipSerialization": [ "fontSize" ]
}
}
}
@@ -475,7 +473,7 @@ If blocks do this, they need to be registered in the server using the `block.jso
Every chunk of styles can only use a single selector.
-This is particularly relevant if the block is using `skipSerialization` to serialize the different style properties to different nodes other than the wrapper. See "Current limitations of blocks supports" for more.
+This is particularly relevant if the block is using `__experimentalSkipSerialization` to serialize the different style properties to different nodes other than the wrapper. See "Current limitations of blocks supports" for more.
#### 3. **Only a single property per block**
diff --git a/lib/compat/wordpress-6.8/blocks.php b/lib/compat/wordpress-6.8/blocks.php
index ba321b7ac987a7..6c026036610fe4 100644
--- a/lib/compat/wordpress-6.8/blocks.php
+++ b/lib/compat/wordpress-6.8/blocks.php
@@ -20,13 +20,8 @@ function gutenberg_stabilize_experimental_block_supports( $args ) {
return $args;
}
- $experimental_supports_map = array( '__experimentalBorder' => 'border' );
- $common_experimental_properties = array(
- '__experimentalDefaultControls' => 'defaultControls',
- '__experimentalSkipSerialization' => 'skipSerialization',
- );
- $experimental_support_properties = array(
- 'typography' => array(
+ $experimental_to_stable_keys = array(
+ 'typography' => array(
'__experimentalFontFamily' => 'fontFamily',
'__experimentalFontStyle' => 'fontStyle',
'__experimentalFontWeight' => 'fontWeight',
@@ -34,108 +29,68 @@ function gutenberg_stabilize_experimental_block_supports( $args ) {
'__experimentalTextDecoration' => 'textDecoration',
'__experimentalTextTransform' => 'textTransform',
),
+ '__experimentalBorder' => 'border',
);
- $done = array();
$updated_supports = array();
foreach ( $args['supports'] as $support => $config ) {
- /*
- * If this support config has already been stabilized, skip it.
- * A stable support key occurring after an experimental key, gets
- * stabilized then so that the two configs can be merged effectively.
- */
- if ( isset( $done[ $support ] ) ) {
- continue;
- }
-
- $stable_support_key = $experimental_supports_map[ $support ] ?? $support;
-
- /*
- * Use the support's config as is when it's not in need of stabilization.
- *
- * A support does not need stabilization if:
- * - The support key doesn't need stabilization AND
- * - Either:
- * - The config isn't an array, so can't have experimental properties OR
- * - The config is an array but has no experimental properties to stabilize.
- */
- if ( $support === $stable_support_key &&
- ( ! is_array( $config ) ||
- ( ! isset( $experimental_support_properties[ $stable_support_key ] ) &&
- empty( array_intersect_key( $common_experimental_properties, $config ) )
- )
- )
- ) {
+ // Add the support's config as is when it's not in need of stabilization.
+ if ( empty( $experimental_to_stable_keys[ $support ] ) ) {
$updated_supports[ $support ] = $config;
continue;
}
- $stabilize_config = function ( $unstable_config, $stable_support_key ) use ( $experimental_support_properties, $common_experimental_properties ) {
- $stable_config = array();
- foreach ( $unstable_config as $key => $value ) {
- // Get stable key from support-specific map, common properties map, or keep original.
- $stable_key = $experimental_support_properties[ $stable_support_key ][ $key ] ??
- $common_experimental_properties[ $key ] ??
- $key;
-
- $stable_config[ $stable_key ] = $value;
-
- /*
- * The `__experimentalSkipSerialization` key needs to be kept until
- * WP 6.8 becomes the minimum supported version. This is due to the
- * core `wp_should_skip_block_supports_serialization` function only
- * checking for `__experimentalSkipSerialization` in earlier versions.
- */
- if ( '__experimentalSkipSerialization' === $key || 'skipSerialization' === $key ) {
- $stable_config['__experimentalSkipSerialization'] = $value;
- }
+ // Stabilize the support's key if needed e.g. __experimentalBorder => border.
+ if ( is_string( $experimental_to_stable_keys[ $support ] ) ) {
+ $stabilized_key = $experimental_to_stable_keys[ $support ];
+
+ // If there is no stabilized key present, use the experimental config as is.
+ if ( ! array_key_exists( $stabilized_key, $args['supports'] ) ) {
+ $updated_supports[ $stabilized_key ] = $config;
+ continue;
}
- return $stable_config;
- };
-
- // Stabilize the config value.
- $stable_config = is_array( $config ) ? $stabilize_config( $config, $stable_support_key ) : $config;
-
- /*
- * If a plugin overrides the support config with the `register_block_type_args`
- * filter, both experimental and stable configs may be present. In that case,
- * use the order keys are defined in to determine the final value.
- * - If config is an array, merge the arrays in their order of definition.
- * - If config is not an array, use the value defined last.
- *
- * The reason for preferring the last defined key is that after filters
- * are applied, the last inserted key is likely the most up-to-date value.
- * We cannot determine with certainty which value was "last modified" so
- * the insertion order is the best guess. The extreme edge case of multiple
- * filters tweaking the same support property will become less over time as
- * extenders migrate existing blocks and plugins to stable keys.
- */
- if ( $support !== $stable_support_key && isset( $args['supports'][ $stable_support_key ] ) ) {
+
+ /*
+ * Determine the order of keys, so the last defined can be preferred.
+ *
+ * The reason for preferring the last defined key is that after filters
+ * are applied, the last inserted key is likely the most up-to-date value.
+ * We cannot determine with certainty which value was "last modified" so
+ * the insertion order is the best guess. The extreme edge case of multiple
+ * filters tweaking the same support property will become less over time as
+ * extenders migrate existing blocks and plugins to stable keys.
+ */
$key_positions = array_flip( array_keys( $args['supports'] ) );
- $experimental_first =
- ( $key_positions[ $support ] ?? PHP_INT_MAX ) <
- ( $key_positions[ $stable_support_key ] ?? PHP_INT_MAX );
-
- if ( is_array( $args['supports'][ $stable_support_key ] ) ) {
- /*
- * To merge the alternative support config effectively, it also needs to be
- * stabilized before merging to keep stabilized and experimental flags in
- * sync.
- */
- $args['supports'][ $stable_support_key ] = $stabilize_config( $args['supports'][ $stable_support_key ], $stable_support_key );
- $stable_config = $experimental_first
- ? array_merge( $stable_config, $args['supports'][ $stable_support_key ] )
- : array_merge( $args['supports'][ $stable_support_key ], $stable_config );
- // Prevents reprocessing this support as it was merged above.
- $done[ $stable_support_key ] = true;
+ $experimental_index = $key_positions[ $support ] ?? -1;
+ $stabilized_index = $key_positions[ $stabilized_key ] ?? -1;
+ $experimental_first = $experimental_index < $stabilized_index;
+
+ // Update support config, prefer the last defined value.
+ if ( is_array( $config ) ) {
+ $updated_supports[ $stabilized_key ] = $experimental_first
+ ? array_merge( $config, $args['supports'][ $stabilized_key ] )
+ : array_merge( $args['supports'][ $stabilized_key ], $config );
} else {
- $stable_config = $experimental_first
- ? $args['supports'][ $stable_support_key ]
- : $stable_config;
+ $updated_supports[ $stabilized_key ] = $experimental_first
+ ? $args['supports'][ $stabilized_key ]
+ : $config;
}
+
+ continue;
}
- $updated_supports[ $stable_support_key ] = $stable_config;
+ // Stabilize individual support feature keys e.g. __experimentalFontFamily => fontFamily.
+ if ( is_array( $experimental_to_stable_keys[ $support ] ) ) {
+ $stable_support_config = array();
+ foreach ( $config as $key => $value ) {
+ if ( array_key_exists( $key, $experimental_to_stable_keys[ $support ] ) ) {
+ $stable_support_config[ $experimental_to_stable_keys[ $support ][ $key ] ] = $value;
+ } else {
+ $stable_support_config[ $key ] = $value;
+ }
+ }
+ $updated_supports[ $support ] = $stable_support_config;
+ }
}
$args['supports'] = $updated_supports;
diff --git a/packages/block-editor/src/hooks/border.js b/packages/block-editor/src/hooks/border.js
index 14b3dbf7669b3a..4500444685befa 100644
--- a/packages/block-editor/src/hooks/border.js
+++ b/packages/block-editor/src/hooks/border.js
@@ -161,8 +161,14 @@ export function BorderPanel( { clientId, name, setAttributes, settings } ) {
}
const defaultControls = {
- ...getBlockSupport( name, [ BORDER_SUPPORT_KEY, 'defaultControls' ] ),
- ...getBlockSupport( name, [ SHADOW_SUPPORT_KEY, 'defaultControls' ] ),
+ ...getBlockSupport( name, [
+ BORDER_SUPPORT_KEY,
+ '__experimentalDefaultControls',
+ ] ),
+ ...getBlockSupport( name, [
+ SHADOW_SUPPORT_KEY,
+ '__experimentalDefaultControls',
+ ] ),
};
return (
diff --git a/packages/block-editor/src/hooks/color.js b/packages/block-editor/src/hooks/color.js
index 2fecc10a311984..ef8984c9367853 100644
--- a/packages/block-editor/src/hooks/color.js
+++ b/packages/block-editor/src/hooks/color.js
@@ -290,7 +290,7 @@ export function ColorEdit( { clientId, name, setAttributes, settings } ) {
const defaultControls = getBlockSupport( name, [
COLOR_SUPPORT_KEY,
- 'defaultControls',
+ '__experimentalDefaultControls',
] );
const enableContrastChecking =
diff --git a/packages/block-editor/src/hooks/dimensions.js b/packages/block-editor/src/hooks/dimensions.js
index c98cc34e4272c8..ffa4048b7740e3 100644
--- a/packages/block-editor/src/hooks/dimensions.js
+++ b/packages/block-editor/src/hooks/dimensions.js
@@ -88,11 +88,11 @@ export function DimensionsPanel( { clientId, name, setAttributes, settings } ) {
const defaultDimensionsControls = getBlockSupport( name, [
DIMENSIONS_SUPPORT_KEY,
- 'defaultControls',
+ '__experimentalDefaultControls',
] );
const defaultSpacingControls = getBlockSupport( name, [
SPACING_SUPPORT_KEY,
- 'defaultControls',
+ '__experimentalDefaultControls',
] );
const defaultControls = {
...defaultDimensionsControls,
diff --git a/packages/block-editor/src/hooks/style.js b/packages/block-editor/src/hooks/style.js
index db2acd01665b60..998d13cfd22247 100644
--- a/packages/block-editor/src/hooks/style.js
+++ b/packages/block-editor/src/hooks/style.js
@@ -98,16 +98,22 @@ function addAttribute( settings ) {
* @type {Record}
*/
const skipSerializationPathsEdit = {
- [ `${ BORDER_SUPPORT_KEY }.skipSerialization` ]: [ 'border' ],
- [ `${ COLOR_SUPPORT_KEY }.skipSerialization` ]: [ COLOR_SUPPORT_KEY ],
- [ `${ TYPOGRAPHY_SUPPORT_KEY }.skipSerialization` ]: [
+ [ `${ BORDER_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [ 'border' ],
+ [ `${ COLOR_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [
+ COLOR_SUPPORT_KEY,
+ ],
+ [ `${ TYPOGRAPHY_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [
TYPOGRAPHY_SUPPORT_KEY,
],
- [ `${ DIMENSIONS_SUPPORT_KEY }.skipSerialization` ]: [
+ [ `${ DIMENSIONS_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [
DIMENSIONS_SUPPORT_KEY,
],
- [ `${ SPACING_SUPPORT_KEY }.skipSerialization` ]: [ SPACING_SUPPORT_KEY ],
- [ `${ SHADOW_SUPPORT_KEY }.skipSerialization` ]: [ SHADOW_SUPPORT_KEY ],
+ [ `${ SPACING_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [
+ SPACING_SUPPORT_KEY,
+ ],
+ [ `${ SHADOW_SUPPORT_KEY }.__experimentalSkipSerialization` ]: [
+ SHADOW_SUPPORT_KEY,
+ ],
};
/**
diff --git a/packages/block-editor/src/hooks/test/style.js b/packages/block-editor/src/hooks/test/style.js
index 40e7169194b82e..2cfe299b8c8d91 100644
--- a/packages/block-editor/src/hooks/test/style.js
+++ b/packages/block-editor/src/hooks/test/style.js
@@ -133,7 +133,8 @@ describe( 'addSaveProps', () => {
const applySkipSerialization = ( features ) => {
const updatedSettings = { ...blockSettings };
Object.keys( features ).forEach( ( key ) => {
- updatedSettings.supports[ key ].skipSerialization = features[ key ];
+ updatedSettings.supports[ key ].__experimentalSkipSerialization =
+ features[ key ];
} );
return updatedSettings;
};
diff --git a/packages/block-editor/src/hooks/typography.js b/packages/block-editor/src/hooks/typography.js
index 160894eac4e610..ab9a464fe5efbc 100644
--- a/packages/block-editor/src/hooks/typography.js
+++ b/packages/block-editor/src/hooks/typography.js
@@ -133,7 +133,7 @@ export function TypographyPanel( { clientId, name, setAttributes, settings } ) {
const defaultControls = getBlockSupport( name, [
TYPOGRAPHY_SUPPORT_KEY,
- 'defaultControls',
+ '__experimentalDefaultControls',
] );
return (
diff --git a/packages/block-editor/src/hooks/utils.js b/packages/block-editor/src/hooks/utils.js
index ac6e55efe4d3bd..4334f70b9d13bf 100644
--- a/packages/block-editor/src/hooks/utils.js
+++ b/packages/block-editor/src/hooks/utils.js
@@ -124,7 +124,7 @@ export function shouldSkipSerialization(
feature
) {
const support = getBlockSupport( blockNameOrType, featureSet );
- const skipSerialization = support?.skipSerialization;
+ const skipSerialization = support?.__experimentalSkipSerialization;
if ( Array.isArray( skipSerialization ) ) {
return skipSerialization.includes( feature );
diff --git a/packages/blocks/src/api/constants.js b/packages/blocks/src/api/constants.js
index aaf6558c47bada..8ea32d591adb32 100644
--- a/packages/blocks/src/api/constants.js
+++ b/packages/blocks/src/api/constants.js
@@ -298,16 +298,7 @@ export const __EXPERIMENTAL_PATHS_WITH_OVERRIDE = {
'spacing.spacingSizes': true,
};
-export const EXPERIMENTAL_SUPPORTS_MAP = {
- __experimentalBorder: 'border',
-};
-
-export const COMMON_EXPERIMENTAL_PROPERTIES = {
- __experimentalDefaultControls: 'defaultControls',
- __experimentalSkipSerialization: 'skipSerialization',
-};
-
-export const EXPERIMENTAL_SUPPORT_PROPERTIES = {
+export const EXPERIMENTAL_TO_STABLE_KEYS = {
typography: {
__experimentalFontFamily: 'fontFamily',
__experimentalFontStyle: 'fontStyle',
@@ -316,4 +307,5 @@ export const EXPERIMENTAL_SUPPORT_PROPERTIES = {
__experimentalTextDecoration: 'textDecoration',
__experimentalTextTransform: 'textTransform',
},
+ __experimentalBorder: 'border',
};
diff --git a/packages/blocks/src/store/process-block-type.js b/packages/blocks/src/store/process-block-type.js
index 0ca28a3c3e2070..bd7e5ab43800f0 100644
--- a/packages/blocks/src/store/process-block-type.js
+++ b/packages/blocks/src/store/process-block-type.js
@@ -18,9 +18,7 @@ import { isValidIcon, normalizeIconObject, omit } from '../api/utils';
import {
BLOCK_ICON_DEFAULT,
DEPRECATED_ENTRY_KEYS,
- EXPERIMENTAL_SUPPORTS_MAP,
- COMMON_EXPERIMENTAL_PROPERTIES,
- EXPERIMENTAL_SUPPORT_PROPERTIES,
+ EXPERIMENTAL_TO_STABLE_KEYS,
} from '../api/constants';
/** @typedef {import('../api/registration').WPBlockType} WPBlockType */
@@ -68,149 +66,81 @@ function mergeBlockVariations(
return result;
}
-/**
- * Stabilizes a block support configuration by converting experimental properties
- * to their stable equivalents.
- *
- * @param {Object} unstableConfig The support configuration to stabilize.
- * @param {string} stableSupportKey The stable support key for looking up properties.
- * @return {Object} The stabilized support configuration.
- */
-function stabilizeSupportConfig( unstableConfig, stableSupportKey ) {
- const stableConfig = {};
- for ( const [ key, value ] of Object.entries( unstableConfig ) ) {
- // Get stable key from support-specific map, common properties map, or keep original.
- const stableKey =
- EXPERIMENTAL_SUPPORT_PROPERTIES[ stableSupportKey ]?.[ key ] ??
- COMMON_EXPERIMENTAL_PROPERTIES[ key ] ??
- key;
-
- stableConfig[ stableKey ] = value;
-
- /*
- * The `__experimentalSkipSerialization` key needs to be kept until
- * WP 6.8 becomes the minimum supported version. This is due to the
- * core `wp_should_skip_block_supports_serialization` function only
- * checking for `__experimentalSkipSerialization` in earlier versions.
- */
- if (
- key === '__experimentalSkipSerialization' ||
- key === 'skipSerialization'
- ) {
- stableConfig.__experimentalSkipSerialization = value;
- }
- }
- return stableConfig;
-}
-
-/**
- * Stabilizes experimental block supports by converting experimental keys and properties
- * to their stable equivalents.
- *
- * @param {Object|undefined} rawSupports The block supports configuration to stabilize.
- * @return {Object|undefined} The stabilized block supports configuration.
- */
function stabilizeSupports( rawSupports ) {
if ( ! rawSupports ) {
return rawSupports;
}
- /*
- * Create a new object to avoid mutating the original. This ensures that
- * custom block plugins that rely on immutable supports are not affected.
- * See: https://github.com/WordPress/gutenberg/pull/66849#issuecomment-2463614281
- */
+ // Create a new object to avoid mutating the original. This ensures that
+ // custom block plugins that rely on immutable supports are not affected.
+ // See: https://github.com/WordPress/gutenberg/pull/66849#issuecomment-2463614281
const newSupports = {};
- const done = {};
for ( const [ support, config ] of Object.entries( rawSupports ) ) {
- /*
- * If this support config has already been stabilized, skip it.
- * A stable support key occurring after an experimental key, gets
- * stabilized then so that the two configs can be merged effectively.
- */
- if ( done[ support ] ) {
- continue;
- }
-
- const stableSupportKey =
- EXPERIMENTAL_SUPPORTS_MAP[ support ] ?? support;
-
- /*
- * Use the support's config as is when it's not in need of stabilization.
- * A support does not need stabilization if:
- * - The support key doesn't need stabilization AND
- * - Either:
- * - The config isn't an object, so can't have experimental properties OR
- * - The config is an object but has no experimental properties to stabilize.
- */
- if (
- support === stableSupportKey &&
- ( ! isPlainObject( config ) ||
- ( ! EXPERIMENTAL_SUPPORT_PROPERTIES[ stableSupportKey ] &&
- Object.keys( config ).every(
- ( key ) => ! COMMON_EXPERIMENTAL_PROPERTIES[ key ]
- ) ) )
- ) {
+ // Add the support's config as is when it's not in need of stabilization.
+ if ( ! EXPERIMENTAL_TO_STABLE_KEYS[ support ] ) {
newSupports[ support ] = config;
continue;
}
- // Stabilize the config value.
- const stableConfig = isPlainObject( config )
- ? stabilizeSupportConfig( config, stableSupportKey )
- : config;
-
- /*
- * If a plugin overrides the support config with the `blocks.registerBlockType`
- * filter, both experimental and stable configs may be present. In that case,
- * use the order keys are defined in to determine the final value.
- * - If config is an array, merge the arrays in their order of definition.
- * - If config is not an array, use the value defined last.
- *
- * The reason for preferring the last defined key is that after filters
- * are applied, the last inserted key is likely the most up-to-date value.
- * We cannot determine with certainty which value was "last modified" so
- * the insertion order is the best guess. The extreme edge case of multiple
- * filters tweaking the same support property will become less over time as
- * extenders migrate existing blocks and plugins to stable keys.
- */
- if (
- support !== stableSupportKey &&
- Object.hasOwn( rawSupports, stableSupportKey )
- ) {
- const keyPositions = Object.keys( rawSupports ).reduce(
- ( acc, key, index ) => {
- acc[ key ] = index;
- return acc;
- },
- {}
+ // Stabilize the support's key if needed e.g. __experimentalBorder => border.
+ if ( typeof EXPERIMENTAL_TO_STABLE_KEYS[ support ] === 'string' ) {
+ const stabilizedKey = EXPERIMENTAL_TO_STABLE_KEYS[ support ];
+
+ // If there is no stabilized key present, use the experimental config as is.
+ if ( ! Object.hasOwn( rawSupports, stabilizedKey ) ) {
+ newSupports[ stabilizedKey ] = config;
+ continue;
+ }
+
+ /*
+ * Determine the order of keys, so the last defined can be preferred.
+ *
+ * The reason for preferring the last defined key is that after filters
+ * are applied, the last inserted key is likely the most up-to-date value.
+ * We cannot determine with certainty which value was "last modified" so
+ * the insertion order is the best guess. The extreme edge case of multiple
+ * filters tweaking the same support property will become less over time as
+ * extenders migrate existing blocks and plugins to stable keys.
+ */
+ const entries = Object.entries( rawSupports );
+ const experimentalIndex = entries.findIndex(
+ ( [ key ] ) => key === support
);
- const experimentalFirst =
- ( keyPositions[ support ] ?? Number.MAX_VALUE ) <
- ( keyPositions[ stableSupportKey ] ?? Number.MAX_VALUE );
-
- if ( isPlainObject( rawSupports[ stableSupportKey ] ) ) {
- /*
- * To merge the alternative support config effectively, it also needs to be
- * stabilized before merging to keep stabilized and experimental flags in sync.
- */
- rawSupports[ stableSupportKey ] = stabilizeSupportConfig(
- rawSupports[ stableSupportKey ],
- stableSupportKey
- );
- newSupports[ stableSupportKey ] = experimentalFirst
- ? { ...stableConfig, ...rawSupports[ stableSupportKey ] }
- : { ...rawSupports[ stableSupportKey ], ...stableConfig };
- // Prevents reprocessing this support as it was merged above.
- done[ stableSupportKey ] = true;
+ const stabilizedIndex = entries.findIndex(
+ ( [ key ] ) => key === stabilizedKey
+ );
+
+ // Update support config, prefer the last defined value.
+ if ( typeof config === 'object' && config !== null ) {
+ newSupports[ stabilizedKey ] =
+ experimentalIndex < stabilizedIndex
+ ? { ...config, ...rawSupports[ stabilizedKey ] }
+ : { ...rawSupports[ stabilizedKey ], ...config };
} else {
- newSupports[ stableSupportKey ] = experimentalFirst
- ? rawSupports[ stableSupportKey ]
- : stableConfig;
+ newSupports[ stabilizedKey ] =
+ experimentalIndex < stabilizedIndex
+ ? rawSupports[ stabilizedKey ]
+ : config;
}
- } else {
- newSupports[ stableSupportKey ] = stableConfig;
+ continue;
+ }
+
+ // Stabilize individual support feature keys
+ // e.g. __experimentalFontFamily => fontFamily.
+ const featureStabilizationRequired =
+ typeof EXPERIMENTAL_TO_STABLE_KEYS[ support ] === 'object' &&
+ EXPERIMENTAL_TO_STABLE_KEYS[ support ] !== null;
+ const hasConfig = typeof config === 'object' && config !== null;
+
+ if ( featureStabilizationRequired && hasConfig ) {
+ const stableConfig = {};
+ for ( const [ key, value ] of Object.entries( config ) ) {
+ const stableKey =
+ EXPERIMENTAL_TO_STABLE_KEYS[ support ][ key ] || key;
+ stableConfig[ stableKey ] = value;
+ }
+ newSupports[ support ] = stableConfig;
}
}
diff --git a/packages/blocks/src/store/test/process-block-type.js b/packages/blocks/src/store/test/process-block-type.js
index 82b2c1ad3080d7..c7f487a95301d0 100644
--- a/packages/blocks/src/store/test/process-block-type.js
+++ b/packages/blocks/src/store/test/process-block-type.js
@@ -26,7 +26,7 @@ describe( 'processBlockType', () => {
removeFilter( 'blocks.registerBlockType', 'test/filterSupports' );
} );
- it( 'should stabilize experimental block supports', () => {
+ it( 'should return the block type with stabilized supports', () => {
const blockSettings = {
...baseBlockSettings,
supports: {
@@ -66,7 +66,7 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toMatchObject( {
+ expect( processedBlockType.supports ).toEqual( {
typography: {
fontSize: true,
lineHeight: true,
@@ -77,7 +77,7 @@ describe( 'processBlockType', () => {
textTransform: true,
textDecoration: true,
__experimentalWritingMode: true,
- defaultControls: {
+ __experimentalDefaultControls: {
fontSize: true,
fontAppearance: true,
textTransform: true,
@@ -88,7 +88,79 @@ describe( 'processBlockType', () => {
radius: true,
style: true,
width: true,
- defaultControls: {
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ },
+ },
+ } );
+ } );
+
+ it( 'should return the block type with stable supports', () => {
+ const blockSettings = {
+ ...baseBlockSettings,
+ supports: {
+ typography: {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
+ fontSize: true,
+ fontAppearance: true,
+ textTransform: true,
+ },
+ },
+ __experimentalBorder: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ },
+ },
+ },
+ };
+
+ const processedBlockType = processBlockType(
+ 'test/block',
+ blockSettings
+ )( { select } );
+
+ expect( processedBlockType.supports ).toEqual( {
+ typography: {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
+ fontSize: true,
+ fontAppearance: true,
+ textTransform: true,
+ },
+ },
+ border: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ __experimentalDefaultControls: {
color: true,
radius: true,
style: true,
@@ -155,7 +227,7 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toMatchObject( {
+ expect( processedBlockType.supports ).toEqual( {
typography: {
fontSize: true,
lineHeight: true,
@@ -166,7 +238,7 @@ describe( 'processBlockType', () => {
textTransform: true,
textDecoration: true,
__experimentalWritingMode: true,
- defaultControls: {
+ __experimentalDefaultControls: {
fontSize: true,
fontAppearance: true,
textTransform: true,
@@ -177,7 +249,7 @@ describe( 'processBlockType', () => {
radius: false,
style: true,
width: true,
- defaultControls: {
+ __experimentalDefaultControls: {
color: true,
radius: true,
style: true,
@@ -187,8 +259,8 @@ describe( 'processBlockType', () => {
} );
} );
- describe( 'block deprecations', () => {
- const deprecatedBlockSettings = {
+ it( 'should stabilize experimental supports within block deprecations', () => {
+ const blockSettings = {
...baseBlockSettings,
supports: {
typography: {
@@ -249,242 +321,145 @@ describe( 'processBlockType', () => {
],
};
- beforeEach( () => {
- // Freeze the deprecated block object and its supports so that the original is not mutated.
- Object.freeze( deprecatedBlockSettings.deprecated[ 0 ] );
- Object.freeze( deprecatedBlockSettings.deprecated[ 0 ].supports );
- } );
-
- it( 'should stabilize experimental supports', () => {
- const processedBlockType = processBlockType(
- 'test/block',
- deprecatedBlockSettings
- )( { select } );
-
- expect( processedBlockType.deprecated[ 0 ].supports ).toMatchObject(
- {
- typography: {
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- },
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- defaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
- }
- );
- } );
-
- it( 'should reapply transformations after supports are filtered', () => {
- addFilter(
- 'blocks.registerBlockType',
- 'test/filterSupports',
- ( settings, name ) => {
- if (
- name === 'test/block' &&
- settings.supports.typography
- ) {
- settings.supports.typography.__experimentalFontFamily = false;
- settings.supports.typography.__experimentalFontStyle = false;
- settings.supports.typography.__experimentalFontWeight = false;
- settings.supports.__experimentalBorder = {
- radius: false,
- };
- }
- return settings;
- }
- );
-
- const processedBlockType = processBlockType(
- 'test/block',
- deprecatedBlockSettings
- )( { select } );
-
- expect( processedBlockType.deprecated[ 0 ].supports ).toMatchObject(
- {
- typography: {
- fontFamily: false,
- fontStyle: false,
- fontWeight: false,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- },
- border: {
- color: true,
- radius: false,
- style: true,
- width: true,
- defaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
- }
- );
- } );
- } );
-
- it( 'should stabilize common experimental properties across all supports', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- __experimentalDefaultControls: {
- fontSize: true,
- },
- __experimentalSkipSerialization: true,
- },
- spacing: {
- padding: true,
- __experimentalDefaultControls: {
- padding: true,
- },
- __experimentalSkipSerialization: true,
- },
- },
- };
+ // Freeze the deprecated block object and its supports so that the original is not mutated.
+ // This ensures the test covers a regression where the original object was mutated.
+ // See: https://github.com/WordPress/gutenberg/pull/63401#discussion_r1832394335.
+ Object.freeze( blockSettings.deprecated[ 0 ] );
+ Object.freeze( blockSettings.deprecated[ 0 ].supports );
const processedBlockType = processBlockType(
'test/block',
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toMatchObject( {
+ expect( processedBlockType.deprecated[ 0 ].supports ).toEqual( {
typography: {
- fontSize: true,
- defaultControls: {
- fontSize: true,
- },
- skipSerialization: true,
- __experimentalSkipSerialization: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
},
- spacing: {
- padding: true,
- defaultControls: {
- padding: true,
+ border: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
},
- skipSerialization: true,
- __experimentalSkipSerialization: true,
},
} );
} );
- it( 'should merge experimental and stable keys in order of definition', () => {
+ it( 'should reapply transformations after supports are filtered within block deprecations', () => {
const blockSettings = {
...baseBlockSettings,
supports: {
- __experimentalBorder: {
- color: true,
- radius: false,
+ typography: {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
+ fontSize: true,
+ fontAppearance: true,
+ textTransform: true,
+ },
},
border: {
- color: false,
+ color: true,
+ radius: true,
style: true,
+ width: true,
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ },
},
},
+ deprecated: [
+ {
+ supports: {
+ typography: {
+ __experimentalFontFamily: true,
+ __experimentalFontStyle: true,
+ __experimentalFontWeight: true,
+ __experimentalLetterSpacing: true,
+ __experimentalTextTransform: true,
+ __experimentalTextDecoration: true,
+ __experimentalWritingMode: true,
+ },
+ __experimentalBorder: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
+ },
+ },
+ },
+ },
+ ],
};
+ addFilter(
+ 'blocks.registerBlockType',
+ 'test/filterSupports',
+ ( settings, name ) => {
+ if ( name === 'test/block' && settings.supports.typography ) {
+ settings.supports.typography.__experimentalFontFamily = false;
+ settings.supports.typography.__experimentalFontStyle = false;
+ settings.supports.typography.__experimentalFontWeight = false;
+ settings.supports.__experimentalBorder = { radius: false };
+ }
+ return settings;
+ }
+ );
+
const processedBlockType = processBlockType(
'test/block',
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toMatchObject( {
- border: {
- color: false,
- radius: false,
- style: true,
- },
- } );
-
- const reversedSettings = {
- ...baseBlockSettings,
- supports: {
- border: {
- color: false,
- style: true,
- },
- __experimentalBorder: {
- color: true,
- radius: false,
- },
+ expect( processedBlockType.deprecated[ 0 ].supports ).toEqual( {
+ typography: {
+ fontFamily: false,
+ fontStyle: false,
+ fontWeight: false,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
},
- };
-
- const reversedProcessedType = processBlockType(
- 'test/block',
- reversedSettings
- )( { select } );
-
- expect( reversedProcessedType.supports ).toMatchObject( {
border: {
color: true,
radius: false,
style: true,
- },
- } );
- } );
-
- it( 'should handle non-object config values', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- __experimentalBorder: true,
- border: false,
- },
- };
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect( processedBlockType.supports ).toMatchObject( {
- border: false,
- } );
- } );
-
- it( 'should not modify supports that do not need stabilization', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- align: true,
- spacing: {
- padding: true,
- margin: true,
+ width: true,
+ __experimentalDefaultControls: {
+ color: true,
+ radius: true,
+ style: true,
+ width: true,
},
},
- };
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect( processedBlockType.supports ).toMatchObject( {
- align: true,
- spacing: {
- padding: true,
- margin: true,
- },
} );
} );
} );
diff --git a/packages/edit-site/src/components/global-styles/screen-block.js b/packages/edit-site/src/components/global-styles/screen-block.js
index 347d3cd1bc0a73..64f49574b6b03b 100644
--- a/packages/edit-site/src/components/global-styles/screen-block.js
+++ b/packages/edit-site/src/components/global-styles/screen-block.js
@@ -113,8 +113,9 @@ function ScreenBlock( { name, variation } ) {
if (
settingsForBlockElement?.spacing?.blockGap &&
blockType?.supports?.spacing?.blockGap &&
- ( blockType?.supports?.spacing?.skipSerialization === true ||
- blockType?.supports?.spacing?.skipSerialization?.some?.(
+ ( blockType?.supports?.spacing?.__experimentalSkipSerialization ===
+ true ||
+ blockType?.supports?.spacing?.__experimentalSkipSerialization?.some?.(
( spacingType ) => spacingType === 'blockGap'
) )
) {
diff --git a/packages/server-side-render/README.md b/packages/server-side-render/README.md
index ef7cd9bf0189c1..ba6fae302ca0a8 100644
--- a/packages/server-side-render/README.md
+++ b/packages/server-side-render/README.md
@@ -79,7 +79,7 @@ add_filter( 'rest_endpoints', 'add_rest_method');
### skipBlockSupportAttributes
-Remove attributes and style properties applied by the block supports. This prevents duplication of styles in the block wrapper and the `ServerSideRender` components. Even if certain features skip serialization to HTML markup by `skipSerialization`, all attributes and style properties are removed.
+Remove attributes and style properties applied by the block supports. This prevents duplication of styles in the block wrapper and the `ServerSideRender` components. Even if certain features skip serialization to HTML markup by `__experimentalSkipSerialization`, all attributes and style properties are removed.
- Type: `Boolean`
- Required: No
diff --git a/phpunit/block-supports/border-test.php b/phpunit/block-supports/border-test.php
index 6ec43b369d9a2a..0c320f24ebe4f2 100644
--- a/phpunit/block-supports/border-test.php
+++ b/phpunit/block-supports/border-test.php
@@ -128,11 +128,11 @@ public function test_flat_border_with_skipped_serialization() {
'test/flat-border-with-skipped-serialization',
array(
'__experimentalBorder' => array(
- 'color' => true,
- 'radius' => true,
- 'width' => true,
- 'style' => true,
- 'skipSerialization' => true,
+ 'color' => true,
+ 'radius' => true,
+ 'width' => true,
+ 'style' => true,
+ '__experimentalSkipSerialization' => true,
),
)
);
@@ -567,167 +567,4 @@ public function test_should_apply_stabilized_border_supports() {
$this->assertSame( $expected, $actual );
}
-
- /**
- * Tests that experimental border support configuration gets stabilized correctly.
- */
- public function test_should_stabilize_border_supports() {
- $block_type_args = array(
- 'supports' => array(
- '__experimentalBorder' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- '__experimentalSkipSerialization' => true,
- '__experimentalDefaultControls' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- ),
- ),
- ),
- );
-
- $actual = gutenberg_stabilize_experimental_block_supports( $block_type_args );
- $expected = array(
- 'supports' => array(
- 'border' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- 'skipSerialization' => true,
- // Has to be kept due to core's `wp_should_skip_block_supports_serialization` only checking the experimental flag until 6.8.
- '__experimentalSkipSerialization' => true,
- 'defaultControls' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- ),
- ),
- ),
- );
-
- $this->assertSame( $expected, $actual, 'Stabilized border block support config does not match.' );
- }
-
- /**
- * Tests the merging of border support configuration when stabilizing
- * experimental config. Due to the ability to filter block type args, plugins
- * or themes could filter using outdated experimental keys. While not every
- * permutation of filtering can be covered, the majority of use cases are
- * served best by merging configs based on the order they were defined if possible.
- */
- public function test_should_stabilize_border_supports_using_order_based_merge() {
- $experimental_border_config = array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- '__experimentalSkipSerialization' => true,
- '__experimentalDefaultControls' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- ),
-
- /*
- * The following simulates theme/plugin filtering using `__experimentalBorder`
- * key but stable serialization and default control keys.
- */
- 'skipSerialization' => false,
- 'defaultControls' => array(
- 'color' => true,
- 'radius' => false,
- 'style' => true,
- 'width' => true,
- ),
- );
- $stable_border_config = array(
- 'color' => true,
- 'radius' => true,
- 'style' => false,
- 'width' => true,
- 'skipSerialization' => false,
- 'defaultControls' => array(
- 'color' => true,
- 'radius' => false,
- 'style' => false,
- 'width' => true,
- ),
-
- /*
- * The following simulates theme/plugin filtering using stable `border` key
- * but experimental serialization and default control keys.
- */
- '__experimentalSkipSerialization' => true,
- '__experimentalDefaultControls' => array(
- 'color' => false,
- 'radius' => false,
- 'style' => false,
- 'width' => false,
- ),
- );
-
- $experimental_first_args = array(
- 'supports' => array(
- '__experimentalBorder' => $experimental_border_config,
- 'border' => $stable_border_config,
- ),
- );
-
- $actual = gutenberg_stabilize_experimental_block_supports( $experimental_first_args );
- $expected = array(
- 'supports' => array(
- 'border' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => false,
- 'width' => true,
- 'skipSerialization' => true,
- '__experimentalSkipSerialization' => true,
- 'defaultControls' => array(
- 'color' => false,
- 'radius' => false,
- 'style' => false,
- 'width' => false,
- ),
-
- ),
- ),
- );
- $this->assertSame( $expected, $actual, 'Merged stabilized border block support config does not match when experimental keys are first.' );
-
- $stable_first_args = array(
- 'supports' => array(
- 'border' => $stable_border_config,
- '__experimentalBorder' => $experimental_border_config,
- ),
- );
-
- $actual = gutenberg_stabilize_experimental_block_supports( $stable_first_args );
- $expected = array(
- 'supports' => array(
- 'border' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- 'skipSerialization' => false,
- '__experimentalSkipSerialization' => false,
- 'defaultControls' => array(
- 'color' => true,
- 'radius' => false,
- 'style' => true,
- 'width' => true,
- ),
- ),
- ),
- );
- $this->assertSame( $expected, $actual, 'Merged stabilized border block support config does not match when stable keys are first.' );
- }
}
From b92566f0d83c8e1877c2b4e7a9a07149e5e82c44 Mon Sep 17 00:00:00 2001
From: Aaron Robertshaw <60436221+aaronrobertshaw@users.noreply.github.com>
Date: Fri, 20 Dec 2024 12:31:39 +1000
Subject: [PATCH 3/5] Revert "Borders: Stabilize border block supports within
block processing (#66918)"
This reverts commit b5ee9da93a85290630cca0e29895a8e3e209eed0.
---
backport-changelog/6.8/7069.md | 1 -
lib/block-supports/border.php | 28 +-
lib/class-wp-theme-json-gutenberg.php | 8 +-
lib/compat/wordpress-6.8/blocks.php | 92 ++----
.../test/use-global-styles-output.js | 2 +-
.../global-styles/use-global-styles-output.js | 2 +-
packages/block-editor/src/hooks/border.js | 2 +-
packages/block-editor/src/hooks/supports.js | 2 +-
packages/blocks/src/api/constants.js | 49 ++--
.../blocks/src/store/process-block-type.js | 84 ++----
.../src/store/test/process-block-type.js | 273 ++++--------------
phpunit/block-supports/border-test.php | 108 -------
12 files changed, 141 insertions(+), 510 deletions(-)
diff --git a/backport-changelog/6.8/7069.md b/backport-changelog/6.8/7069.md
index 8b4ad69db02e75..ea3c717ec3c93a 100644
--- a/backport-changelog/6.8/7069.md
+++ b/backport-changelog/6.8/7069.md
@@ -1,4 +1,3 @@
https://github.com/WordPress/wordpress-develop/pull/7069
* https://github.com/WordPress/gutenberg/pull/63401
-* https://github.com/WordPress/gutenberg/pull/66918
diff --git a/lib/block-supports/border.php b/lib/block-supports/border.php
index f890ed84566b7f..bd4c772675a5ed 100644
--- a/lib/block-supports/border.php
+++ b/lib/block-supports/border.php
@@ -17,7 +17,7 @@ function gutenberg_register_border_support( $block_type ) {
$block_type->attributes = array();
}
- if ( block_has_support( $block_type, array( 'border' ) ) && ! array_key_exists( 'style', $block_type->attributes ) ) {
+ if ( block_has_support( $block_type, array( '__experimentalBorder' ) ) && ! array_key_exists( 'style', $block_type->attributes ) ) {
$block_type->attributes['style'] = array(
'type' => 'object',
);
@@ -52,7 +52,7 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
if (
gutenberg_has_border_feature_support( $block_type, 'radius' ) &&
isset( $block_attributes['style']['border']['radius'] ) &&
- ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'radius' )
+ ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'radius' )
) {
$border_radius = $block_attributes['style']['border']['radius'];
@@ -67,7 +67,7 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
if (
gutenberg_has_border_feature_support( $block_type, 'style' ) &&
isset( $block_attributes['style']['border']['style'] ) &&
- ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'style' )
+ ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'style' )
) {
$border_block_styles['style'] = $block_attributes['style']['border']['style'];
}
@@ -76,7 +76,7 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
if (
$has_border_width_support &&
isset( $block_attributes['style']['border']['width'] ) &&
- ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'width' )
+ ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'width' )
) {
$border_width = $block_attributes['style']['border']['width'];
@@ -91,7 +91,7 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
// Border color.
if (
$has_border_color_support &&
- ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'color' )
+ ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'color' )
) {
$preset_border_color = array_key_exists( 'borderColor', $block_attributes ) ? "var:preset|color|{$block_attributes['borderColor']}" : null;
$custom_border_color = $block_attributes['style']['border']['color'] ?? null;
@@ -103,9 +103,9 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
foreach ( array( 'top', 'right', 'bottom', 'left' ) as $side ) {
$border = $block_attributes['style']['border'][ $side ] ?? null;
$border_side_values = array(
- 'width' => isset( $border['width'] ) && ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'width' ) ? $border['width'] : null,
- 'color' => isset( $border['color'] ) && ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'color' ) ? $border['color'] : null,
- 'style' => isset( $border['style'] ) && ! wp_should_skip_block_supports_serialization( $block_type, 'border', 'style' ) ? $border['style'] : null,
+ 'width' => isset( $border['width'] ) && ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'width' ) ? $border['width'] : null,
+ 'color' => isset( $border['color'] ) && ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'color' ) ? $border['color'] : null,
+ 'style' => isset( $border['style'] ) && ! wp_should_skip_block_supports_serialization( $block_type, '__experimentalBorder', 'style' ) ? $border['style'] : null,
);
$border_block_styles[ $side ] = $border_side_values;
}
@@ -129,9 +129,9 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
/**
* Checks whether the current block type supports the border feature requested.
*
- * If the `border` support flag is a boolean `true` all border
+ * If the `__experimentalBorder` support flag is a boolean `true` all border
* support features are available. Otherwise, the specific feature's support
- * flag nested under `border` must be enabled for the feature
+ * flag nested under `experimentalBorder` must be enabled for the feature
* to be opted into.
*
* @param WP_Block_Type $block_type Block type to check for support.
@@ -141,17 +141,17 @@ function gutenberg_apply_border_support( $block_type, $block_attributes ) {
* @return boolean Whether or not the feature is supported.
*/
function gutenberg_has_border_feature_support( $block_type, $feature, $default_value = false ) {
- // Check if all border support features have been opted into via `"border": true`.
+ // Check if all border support features have been opted into via `"__experimentalBorder": true`.
if ( $block_type instanceof WP_Block_Type ) {
- $block_type_supports_border = $block_type->supports['border'] ?? $default_value;
+ $block_type_supports_border = $block_type->supports['__experimentalBorder'] ?? $default_value;
if ( true === $block_type_supports_border ) {
return true;
}
}
// Check if the specific feature has been opted into individually
- // via nested flag under `border`.
- return block_has_support( $block_type, array( 'border', $feature ), $default_value );
+ // via nested flag under `__experimentalBorder`.
+ return block_has_support( $block_type, array( '__experimentalBorder', $feature ), $default_value );
}
// Register the block support.
diff --git a/lib/class-wp-theme-json-gutenberg.php b/lib/class-wp-theme-json-gutenberg.php
index 778dcdbec78d96..afef8a5c9e72e2 100644
--- a/lib/class-wp-theme-json-gutenberg.php
+++ b/lib/class-wp-theme-json-gutenberg.php
@@ -615,10 +615,10 @@ class WP_Theme_JSON_Gutenberg {
* @var string[]
*/
const BLOCK_SUPPORT_FEATURE_LEVEL_SELECTORS = array(
- 'border' => 'border',
- 'color' => 'color',
- 'spacing' => 'spacing',
- 'typography' => 'typography',
+ '__experimentalBorder' => 'border',
+ 'color' => 'color',
+ 'spacing' => 'spacing',
+ 'typography' => 'typography',
);
/**
diff --git a/lib/compat/wordpress-6.8/blocks.php b/lib/compat/wordpress-6.8/blocks.php
index 6c026036610fe4..eb5d32afb20a59 100644
--- a/lib/compat/wordpress-6.8/blocks.php
+++ b/lib/compat/wordpress-6.8/blocks.php
@@ -6,94 +6,40 @@
*/
/**
- * Filters the block type arguments during registration to stabilize
- * experimental block supports.
+ * Filters the block type arguments during registration to stabilize experimental block supports.
*
- * This is a temporary compatibility shim as the approach in core is for this
- * to be handled within the WP_Block_Type class rather than requiring a filter.
+ * This is a temporary compatibility shim as the approach in core is for this to be handled
+ * within the WP_Block_Type class rather than requiring a filter.
*
* @param array $args Array of arguments for registering a block type.
* @return array Array of arguments for registering a block type.
*/
function gutenberg_stabilize_experimental_block_supports( $args ) {
- if ( empty( $args['supports'] ) ) {
+ if ( empty( $args['supports']['typography'] ) ) {
return $args;
}
- $experimental_to_stable_keys = array(
- 'typography' => array(
- '__experimentalFontFamily' => 'fontFamily',
- '__experimentalFontStyle' => 'fontStyle',
- '__experimentalFontWeight' => 'fontWeight',
- '__experimentalLetterSpacing' => 'letterSpacing',
- '__experimentalTextDecoration' => 'textDecoration',
- '__experimentalTextTransform' => 'textTransform',
- ),
- '__experimentalBorder' => 'border',
+ $experimental_typography_supports_to_stable = array(
+ '__experimentalFontFamily' => 'fontFamily',
+ '__experimentalFontStyle' => 'fontStyle',
+ '__experimentalFontWeight' => 'fontWeight',
+ '__experimentalLetterSpacing' => 'letterSpacing',
+ '__experimentalTextDecoration' => 'textDecoration',
+ '__experimentalTextTransform' => 'textTransform',
);
- $updated_supports = array();
- foreach ( $args['supports'] as $support => $config ) {
- // Add the support's config as is when it's not in need of stabilization.
- if ( empty( $experimental_to_stable_keys[ $support ] ) ) {
- $updated_supports[ $support ] = $config;
- continue;
- }
-
- // Stabilize the support's key if needed e.g. __experimentalBorder => border.
- if ( is_string( $experimental_to_stable_keys[ $support ] ) ) {
- $stabilized_key = $experimental_to_stable_keys[ $support ];
-
- // If there is no stabilized key present, use the experimental config as is.
- if ( ! array_key_exists( $stabilized_key, $args['supports'] ) ) {
- $updated_supports[ $stabilized_key ] = $config;
- continue;
- }
-
- /*
- * Determine the order of keys, so the last defined can be preferred.
- *
- * The reason for preferring the last defined key is that after filters
- * are applied, the last inserted key is likely the most up-to-date value.
- * We cannot determine with certainty which value was "last modified" so
- * the insertion order is the best guess. The extreme edge case of multiple
- * filters tweaking the same support property will become less over time as
- * extenders migrate existing blocks and plugins to stable keys.
- */
- $key_positions = array_flip( array_keys( $args['supports'] ) );
- $experimental_index = $key_positions[ $support ] ?? -1;
- $stabilized_index = $key_positions[ $stabilized_key ] ?? -1;
- $experimental_first = $experimental_index < $stabilized_index;
-
- // Update support config, prefer the last defined value.
- if ( is_array( $config ) ) {
- $updated_supports[ $stabilized_key ] = $experimental_first
- ? array_merge( $config, $args['supports'][ $stabilized_key ] )
- : array_merge( $args['supports'][ $stabilized_key ], $config );
- } else {
- $updated_supports[ $stabilized_key ] = $experimental_first
- ? $args['supports'][ $stabilized_key ]
- : $config;
- }
-
- continue;
- }
+ $current_typography_supports = $args['supports']['typography'];
+ $stable_typography_supports = array();
- // Stabilize individual support feature keys e.g. __experimentalFontFamily => fontFamily.
- if ( is_array( $experimental_to_stable_keys[ $support ] ) ) {
- $stable_support_config = array();
- foreach ( $config as $key => $value ) {
- if ( array_key_exists( $key, $experimental_to_stable_keys[ $support ] ) ) {
- $stable_support_config[ $experimental_to_stable_keys[ $support ][ $key ] ] = $value;
- } else {
- $stable_support_config[ $key ] = $value;
- }
- }
- $updated_supports[ $support ] = $stable_support_config;
+ foreach ( $current_typography_supports as $key => $value ) {
+ if ( array_key_exists( $key, $experimental_typography_supports_to_stable ) ) {
+ $stable_typography_supports[ $experimental_typography_supports_to_stable[ $key ] ] = $value;
+ } else {
+ $stable_typography_supports[ $key ] = $value;
}
}
- $args['supports'] = $updated_supports;
+ $args['supports']['typography'] = $stable_typography_supports;
return $args;
}
diff --git a/packages/block-editor/src/components/global-styles/test/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/test/use-global-styles-output.js
index 93e5cc9afdbb3c..5022e8ba591dbb 100644
--- a/packages/block-editor/src/components/global-styles/test/use-global-styles-output.js
+++ b/packages/block-editor/src/components/global-styles/test/use-global-styles-output.js
@@ -855,7 +855,7 @@ describe( 'global styles renderer', () => {
it( 'should return block selectors data with old experimental selectors', () => {
const imageSupports = {
- border: {
+ __experimentalBorder: {
radius: true,
__experimentalSelector: 'img, .crop-area',
},
diff --git a/packages/block-editor/src/components/global-styles/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/use-global-styles-output.js
index 7bdc95d222142d..cd4ad0cea50e0d 100644
--- a/packages/block-editor/src/components/global-styles/use-global-styles-output.js
+++ b/packages/block-editor/src/components/global-styles/use-global-styles-output.js
@@ -47,7 +47,7 @@ const ELEMENT_CLASS_NAMES = {
// List of block support features that can have their related styles
// generated under their own feature level selector rather than the block's.
const BLOCK_SUPPORT_FEATURE_LEVEL_SELECTORS = {
- border: 'border',
+ __experimentalBorder: 'border',
color: 'color',
spacing: 'spacing',
typography: 'typography',
diff --git a/packages/block-editor/src/hooks/border.js b/packages/block-editor/src/hooks/border.js
index 4500444685befa..4ab4c69a41f311 100644
--- a/packages/block-editor/src/hooks/border.js
+++ b/packages/block-editor/src/hooks/border.js
@@ -31,7 +31,7 @@ import {
import { store as blockEditorStore } from '../store';
import { __ } from '@wordpress/i18n';
-export const BORDER_SUPPORT_KEY = 'border';
+export const BORDER_SUPPORT_KEY = '__experimentalBorder';
export const SHADOW_SUPPORT_KEY = 'shadow';
const getColorByProperty = ( colors, property, value ) => {
diff --git a/packages/block-editor/src/hooks/supports.js b/packages/block-editor/src/hooks/supports.js
index 102b78bbb96e68..c0b6bb2cc8b271 100644
--- a/packages/block-editor/src/hooks/supports.js
+++ b/packages/block-editor/src/hooks/supports.js
@@ -6,7 +6,7 @@ import { Platform } from '@wordpress/element';
const ALIGN_SUPPORT_KEY = 'align';
const ALIGN_WIDE_SUPPORT_KEY = 'alignWide';
-const BORDER_SUPPORT_KEY = 'border';
+const BORDER_SUPPORT_KEY = '__experimentalBorder';
const COLOR_SUPPORT_KEY = 'color';
const CUSTOM_CLASS_NAME_SUPPORT_KEY = 'customClassName';
const FONT_FAMILY_SUPPORT_KEY = 'typography.fontFamily';
diff --git a/packages/blocks/src/api/constants.js b/packages/blocks/src/api/constants.js
index 8ea32d591adb32..85a359af754db8 100644
--- a/packages/blocks/src/api/constants.js
+++ b/packages/blocks/src/api/constants.js
@@ -58,12 +58,12 @@ export const __EXPERIMENTAL_STYLE_PROPERTY = {
},
borderColor: {
value: [ 'border', 'color' ],
- support: [ 'border', 'color' ],
+ support: [ '__experimentalBorder', 'color' ],
useEngine: true,
},
borderRadius: {
value: [ 'border', 'radius' ],
- support: [ 'border', 'radius' ],
+ support: [ '__experimentalBorder', 'radius' ],
properties: {
borderTopLeftRadius: 'topLeft',
borderTopRightRadius: 'topRight',
@@ -74,72 +74,72 @@ export const __EXPERIMENTAL_STYLE_PROPERTY = {
},
borderStyle: {
value: [ 'border', 'style' ],
- support: [ 'border', 'style' ],
+ support: [ '__experimentalBorder', 'style' ],
useEngine: true,
},
borderWidth: {
value: [ 'border', 'width' ],
- support: [ 'border', 'width' ],
+ support: [ '__experimentalBorder', 'width' ],
useEngine: true,
},
borderTopColor: {
value: [ 'border', 'top', 'color' ],
- support: [ 'border', 'color' ],
+ support: [ '__experimentalBorder', 'color' ],
useEngine: true,
},
borderTopStyle: {
value: [ 'border', 'top', 'style' ],
- support: [ 'border', 'style' ],
+ support: [ '__experimentalBorder', 'style' ],
useEngine: true,
},
borderTopWidth: {
value: [ 'border', 'top', 'width' ],
- support: [ 'border', 'width' ],
+ support: [ '__experimentalBorder', 'width' ],
useEngine: true,
},
borderRightColor: {
value: [ 'border', 'right', 'color' ],
- support: [ 'border', 'color' ],
+ support: [ '__experimentalBorder', 'color' ],
useEngine: true,
},
borderRightStyle: {
value: [ 'border', 'right', 'style' ],
- support: [ 'border', 'style' ],
+ support: [ '__experimentalBorder', 'style' ],
useEngine: true,
},
borderRightWidth: {
value: [ 'border', 'right', 'width' ],
- support: [ 'border', 'width' ],
+ support: [ '__experimentalBorder', 'width' ],
useEngine: true,
},
borderBottomColor: {
value: [ 'border', 'bottom', 'color' ],
- support: [ 'border', 'color' ],
+ support: [ '__experimentalBorder', 'color' ],
useEngine: true,
},
borderBottomStyle: {
value: [ 'border', 'bottom', 'style' ],
- support: [ 'border', 'style' ],
+ support: [ '__experimentalBorder', 'style' ],
useEngine: true,
},
borderBottomWidth: {
value: [ 'border', 'bottom', 'width' ],
- support: [ 'border', 'width' ],
+ support: [ '__experimentalBorder', 'width' ],
useEngine: true,
},
borderLeftColor: {
value: [ 'border', 'left', 'color' ],
- support: [ 'border', 'color' ],
+ support: [ '__experimentalBorder', 'color' ],
useEngine: true,
},
borderLeftStyle: {
value: [ 'border', 'left', 'style' ],
- support: [ 'border', 'style' ],
+ support: [ '__experimentalBorder', 'style' ],
useEngine: true,
},
borderLeftWidth: {
value: [ 'border', 'left', 'width' ],
- support: [ 'border', 'width' ],
+ support: [ '__experimentalBorder', 'width' ],
useEngine: true,
},
color: {
@@ -298,14 +298,11 @@ export const __EXPERIMENTAL_PATHS_WITH_OVERRIDE = {
'spacing.spacingSizes': true,
};
-export const EXPERIMENTAL_TO_STABLE_KEYS = {
- typography: {
- __experimentalFontFamily: 'fontFamily',
- __experimentalFontStyle: 'fontStyle',
- __experimentalFontWeight: 'fontWeight',
- __experimentalLetterSpacing: 'letterSpacing',
- __experimentalTextDecoration: 'textDecoration',
- __experimentalTextTransform: 'textTransform',
- },
- __experimentalBorder: 'border',
+export const TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE = {
+ __experimentalFontFamily: 'fontFamily',
+ __experimentalFontStyle: 'fontStyle',
+ __experimentalFontWeight: 'fontWeight',
+ __experimentalLetterSpacing: 'letterSpacing',
+ __experimentalTextDecoration: 'textDecoration',
+ __experimentalTextTransform: 'textTransform',
};
diff --git a/packages/blocks/src/store/process-block-type.js b/packages/blocks/src/store/process-block-type.js
index bd7e5ab43800f0..98fb044f8faadf 100644
--- a/packages/blocks/src/store/process-block-type.js
+++ b/packages/blocks/src/store/process-block-type.js
@@ -18,7 +18,7 @@ import { isValidIcon, normalizeIconObject, omit } from '../api/utils';
import {
BLOCK_ICON_DEFAULT,
DEPRECATED_ENTRY_KEYS,
- EXPERIMENTAL_TO_STABLE_KEYS,
+ TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE,
} from '../api/constants';
/** @typedef {import('../api/registration').WPBlockType} WPBlockType */
@@ -75,72 +75,24 @@ function stabilizeSupports( rawSupports ) {
// custom block plugins that rely on immutable supports are not affected.
// See: https://github.com/WordPress/gutenberg/pull/66849#issuecomment-2463614281
const newSupports = {};
-
- for ( const [ support, config ] of Object.entries( rawSupports ) ) {
- // Add the support's config as is when it's not in need of stabilization.
- if ( ! EXPERIMENTAL_TO_STABLE_KEYS[ support ] ) {
- newSupports[ support ] = config;
- continue;
- }
-
- // Stabilize the support's key if needed e.g. __experimentalBorder => border.
- if ( typeof EXPERIMENTAL_TO_STABLE_KEYS[ support ] === 'string' ) {
- const stabilizedKey = EXPERIMENTAL_TO_STABLE_KEYS[ support ];
-
- // If there is no stabilized key present, use the experimental config as is.
- if ( ! Object.hasOwn( rawSupports, stabilizedKey ) ) {
- newSupports[ stabilizedKey ] = config;
- continue;
- }
-
- /*
- * Determine the order of keys, so the last defined can be preferred.
- *
- * The reason for preferring the last defined key is that after filters
- * are applied, the last inserted key is likely the most up-to-date value.
- * We cannot determine with certainty which value was "last modified" so
- * the insertion order is the best guess. The extreme edge case of multiple
- * filters tweaking the same support property will become less over time as
- * extenders migrate existing blocks and plugins to stable keys.
- */
- const entries = Object.entries( rawSupports );
- const experimentalIndex = entries.findIndex(
- ( [ key ] ) => key === support
- );
- const stabilizedIndex = entries.findIndex(
- ( [ key ] ) => key === stabilizedKey
+ for ( const [ key, value ] of Object.entries( rawSupports ) ) {
+ if (
+ key === 'typography' &&
+ typeof value === 'object' &&
+ value !== null
+ ) {
+ newSupports.typography = Object.fromEntries(
+ Object.entries( value ).map(
+ ( [ typographyKey, typographyValue ] ) => [
+ TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE[
+ typographyKey
+ ] || typographyKey,
+ typographyValue,
+ ]
+ )
);
-
- // Update support config, prefer the last defined value.
- if ( typeof config === 'object' && config !== null ) {
- newSupports[ stabilizedKey ] =
- experimentalIndex < stabilizedIndex
- ? { ...config, ...rawSupports[ stabilizedKey ] }
- : { ...rawSupports[ stabilizedKey ], ...config };
- } else {
- newSupports[ stabilizedKey ] =
- experimentalIndex < stabilizedIndex
- ? rawSupports[ stabilizedKey ]
- : config;
- }
- continue;
- }
-
- // Stabilize individual support feature keys
- // e.g. __experimentalFontFamily => fontFamily.
- const featureStabilizationRequired =
- typeof EXPERIMENTAL_TO_STABLE_KEYS[ support ] === 'object' &&
- EXPERIMENTAL_TO_STABLE_KEYS[ support ] !== null;
- const hasConfig = typeof config === 'object' && config !== null;
-
- if ( featureStabilizationRequired && hasConfig ) {
- const stableConfig = {};
- for ( const [ key, value ] of Object.entries( config ) ) {
- const stableKey =
- EXPERIMENTAL_TO_STABLE_KEYS[ support ][ key ] || key;
- stableConfig[ stableKey ] = value;
- }
- newSupports[ support ] = stableConfig;
+ } else {
+ newSupports[ key ] = value;
}
}
diff --git a/packages/blocks/src/store/test/process-block-type.js b/packages/blocks/src/store/test/process-block-type.js
index c7f487a95301d0..c82d08cc45d651 100644
--- a/packages/blocks/src/store/test/process-block-type.js
+++ b/packages/blocks/src/store/test/process-block-type.js
@@ -26,7 +26,7 @@ describe( 'processBlockType', () => {
removeFilter( 'blocks.registerBlockType', 'test/filterSupports' );
} );
- it( 'should return the block type with stabilized supports', () => {
+ it( 'should return the block type with stabilized typography supports', () => {
const blockSettings = {
...baseBlockSettings,
supports: {
@@ -46,18 +46,6 @@ describe( 'processBlockType', () => {
textTransform: true,
},
},
- __experimentalBorder: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
};
@@ -66,39 +54,25 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toEqual( {
- typography: {
+ expect( processedBlockType.supports.typography ).toEqual( {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
+ fontAppearance: true,
textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
},
} );
} );
- it( 'should return the block type with stable supports', () => {
+ it( 'should return the block type with stable typography supports', () => {
const blockSettings = {
...baseBlockSettings,
supports: {
@@ -118,18 +92,6 @@ describe( 'processBlockType', () => {
textTransform: true,
},
},
- __experimentalBorder: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
};
@@ -138,34 +100,20 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toEqual( {
- typography: {
+ expect( processedBlockType.supports.typography ).toEqual( {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
+ fontAppearance: true,
textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
},
} );
} );
@@ -190,18 +138,6 @@ describe( 'processBlockType', () => {
textTransform: true,
},
},
- __experimentalBorder: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
};
@@ -213,10 +149,6 @@ describe( 'processBlockType', () => {
settings.supports.typography.__experimentalFontFamily = false;
settings.supports.typography.__experimentalFontStyle = false;
settings.supports.typography.__experimentalFontWeight = false;
- if ( ! settings.supports.__experimentalBorder ) {
- settings.supports.__experimentalBorder = {};
- }
- settings.supports.__experimentalBorder.radius = false;
}
return settings;
}
@@ -227,39 +159,25 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.supports ).toEqual( {
- typography: {
+ expect( processedBlockType.supports.typography ).toEqual( {
+ fontSize: true,
+ lineHeight: true,
+ fontFamily: false,
+ fontStyle: false,
+ fontWeight: false,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
+ __experimentalDefaultControls: {
fontSize: true,
- lineHeight: true,
- fontFamily: false,
- fontStyle: false,
- fontWeight: false,
- letterSpacing: true,
+ fontAppearance: true,
textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- border: {
- color: true,
- radius: false,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
},
} );
} );
- it( 'should stabilize experimental supports within block deprecations', () => {
+ it( 'should stabilize experimental typography supports within block deprecations', () => {
const blockSettings = {
...baseBlockSettings,
supports: {
@@ -279,18 +197,6 @@ describe( 'processBlockType', () => {
textTransform: true,
},
},
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
deprecated: [
{
@@ -304,18 +210,6 @@ describe( 'processBlockType', () => {
__experimentalTextDecoration: true,
__experimentalWritingMode: true,
},
- __experimentalBorder: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
},
],
@@ -332,28 +226,16 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.deprecated[ 0 ].supports ).toEqual( {
- typography: {
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- },
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
+ expect(
+ processedBlockType.deprecated[ 0 ].supports.typography
+ ).toEqual( {
+ fontFamily: true,
+ fontStyle: true,
+ fontWeight: true,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
} );
} );
@@ -377,18 +259,6 @@ describe( 'processBlockType', () => {
textTransform: true,
},
},
- border: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
deprecated: [
{
@@ -402,18 +272,6 @@ describe( 'processBlockType', () => {
__experimentalTextDecoration: true,
__experimentalWritingMode: true,
},
- __experimentalBorder: {
- color: true,
- radius: true,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
},
},
],
@@ -427,7 +285,6 @@ describe( 'processBlockType', () => {
settings.supports.typography.__experimentalFontFamily = false;
settings.supports.typography.__experimentalFontStyle = false;
settings.supports.typography.__experimentalFontWeight = false;
- settings.supports.__experimentalBorder = { radius: false };
}
return settings;
}
@@ -438,28 +295,16 @@ describe( 'processBlockType', () => {
blockSettings
)( { select } );
- expect( processedBlockType.deprecated[ 0 ].supports ).toEqual( {
- typography: {
- fontFamily: false,
- fontStyle: false,
- fontWeight: false,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- },
- border: {
- color: true,
- radius: false,
- style: true,
- width: true,
- __experimentalDefaultControls: {
- color: true,
- radius: true,
- style: true,
- width: true,
- },
- },
+ expect(
+ processedBlockType.deprecated[ 0 ].supports.typography
+ ).toEqual( {
+ fontFamily: false,
+ fontStyle: false,
+ fontWeight: false,
+ letterSpacing: true,
+ textTransform: true,
+ textDecoration: true,
+ __experimentalWritingMode: true,
} );
} );
} );
diff --git a/phpunit/block-supports/border-test.php b/phpunit/block-supports/border-test.php
index 0c320f24ebe4f2..858e4e92cc1740 100644
--- a/phpunit/block-supports/border-test.php
+++ b/phpunit/block-supports/border-test.php
@@ -457,114 +457,6 @@ public function test_split_borders_with_named_colors() {
'style' => 'border-top-width:2px;border-top-color:var(--wp--preset--color--red);border-top-style:dashed;border-right-width:0.25rem;border-right-color:var(--wp--preset--color--green);border-right-style:dotted;border-bottom-width:0.5em;border-bottom-color:var(--wp--preset--color--blue);border-bottom-style:solid;border-left-width:1px;border-left-color:var(--wp--preset--color--yellow);border-left-style:solid;',
);
- $this->assertSame( $expected, $actual );
- }
- /**
- * Tests that stabilized border supports will also apply to blocks using
- * the experimental syntax, for backwards compatibility with existing blocks.
- *
- * @covers ::gutenberg_apply_border_support
- */
- public function test_should_apply_experimental_border_supports() {
- $this->test_block_name = 'test/experimental-border-supports';
- register_block_type(
- $this->test_block_name,
- array(
- 'api_version' => 3,
- 'attributes' => array(
- 'style' => array(
- 'type' => 'object',
- ),
- ),
- 'supports' => array(
- '__experimentalBorder' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- '__experimentalDefaultControls' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- ),
- ),
- ),
- )
- );
- $registry = WP_Block_Type_Registry::get_instance();
- $block_type = $registry->get_registered( $this->test_block_name );
- $block_atts = array(
- 'style' => array(
- 'border' => array(
- 'color' => '#72aee6',
- 'radius' => '10px',
- 'style' => 'dashed',
- 'width' => '2px',
- ),
- ),
- );
-
- $actual = gutenberg_apply_border_support( $block_type, $block_atts );
- $expected = array(
- 'class' => 'has-border-color',
- 'style' => 'border-color:#72aee6;border-radius:10px;border-style:dashed;border-width:2px;',
- );
-
- $this->assertSame( $expected, $actual );
- }
-
- /**
- * Tests that stabilized border supports are applied correctly.
- *
- * @covers ::gutenberg_apply_border_support
- */
- public function test_should_apply_stabilized_border_supports() {
- $this->test_block_name = 'test/stabilized-border-supports';
- register_block_type(
- $this->test_block_name,
- array(
- 'api_version' => 3,
- 'attributes' => array(
- 'style' => array(
- 'type' => 'object',
- ),
- ),
- 'supports' => array(
- 'border' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- '__experimentalDefaultControls' => array(
- 'color' => true,
- 'radius' => true,
- 'style' => true,
- 'width' => true,
- ),
- ),
- ),
- )
- );
- $registry = WP_Block_Type_Registry::get_instance();
- $block_type = $registry->get_registered( $this->test_block_name );
- $block_atts = array(
- 'style' => array(
- 'border' => array(
- 'color' => '#72aee6',
- 'radius' => '10px',
- 'style' => 'dashed',
- 'width' => '2px',
- ),
- ),
- );
-
- $actual = gutenberg_apply_border_support( $block_type, $block_atts );
- $expected = array(
- 'class' => 'has-border-color',
- 'style' => 'border-color:#72aee6;border-radius:10px;border-style:dashed;border-width:2px;',
- );
-
$this->assertSame( $expected, $actual );
}
}
From 866968826ae4c649d26c006f0356fb301695098c Mon Sep 17 00:00:00 2001
From: Aaron Robertshaw <60436221+aaronrobertshaw@users.noreply.github.com>
Date: Fri, 20 Dec 2024 12:32:15 +1000
Subject: [PATCH 4/5] Revert "Process Block Type: Copy deprecation to a new
object instead of mutating when stabilizing supports (#66849)"
This reverts commit 457fcf8d331a92ae0c2e83db22c77f18b194da54.
---
.../blocks/src/store/process-block-type.js | 47 ++++++-------------
.../src/store/test/process-block-type.js | 6 ---
2 files changed, 15 insertions(+), 38 deletions(-)
diff --git a/packages/blocks/src/store/process-block-type.js b/packages/blocks/src/store/process-block-type.js
index 98fb044f8faadf..8e5b7962adfe17 100644
--- a/packages/blocks/src/store/process-block-type.js
+++ b/packages/blocks/src/store/process-block-type.js
@@ -71,32 +71,17 @@ function stabilizeSupports( rawSupports ) {
return rawSupports;
}
- // Create a new object to avoid mutating the original. This ensures that
- // custom block plugins that rely on immutable supports are not affected.
- // See: https://github.com/WordPress/gutenberg/pull/66849#issuecomment-2463614281
- const newSupports = {};
- for ( const [ key, value ] of Object.entries( rawSupports ) ) {
- if (
- key === 'typography' &&
- typeof value === 'object' &&
- value !== null
- ) {
- newSupports.typography = Object.fromEntries(
- Object.entries( value ).map(
- ( [ typographyKey, typographyValue ] ) => [
- TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE[
- typographyKey
- ] || typographyKey,
- typographyValue,
- ]
- )
- );
- } else {
- newSupports[ key ] = value;
- }
+ const supports = { ...rawSupports };
+ if ( supports?.typography && typeof supports.typography === 'object' ) {
+ supports.typography = Object.fromEntries(
+ Object.entries( supports.typography ).map( ( [ key, value ] ) => [
+ TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE[ key ] || key,
+ value,
+ ] )
+ );
}
- return newSupports;
+ return supports;
}
/**
@@ -165,12 +150,10 @@ export const processBlockType =
if ( settings.deprecated ) {
settings.deprecated = settings.deprecated.map( ( deprecation ) => {
// Stabilize any experimental supports before applying filters.
- let filteredDeprecation = {
- ...deprecation,
- supports: stabilizeSupports( deprecation.supports ),
- };
-
- filteredDeprecation = // Only keep valid deprecation keys.
+ deprecation.supports = stabilizeSupports(
+ deprecation.supports
+ );
+ const filteredDeprecation = // Only keep valid deprecation keys.
applyFilters(
'blocks.registerBlockType',
// Merge deprecation keys with pre-filter settings
@@ -180,10 +163,10 @@ export const processBlockType =
// Omit deprecation keys here so that deprecations
// can opt out of specific keys like "supports".
...omit( blockType, DEPRECATED_ENTRY_KEYS ),
- ...filteredDeprecation,
+ ...deprecation,
},
blockType.name,
- filteredDeprecation
+ deprecation
);
// Re-stabilize any experimental supports after applying filters.
// This ensures that any supports updated by filters are also stabilized.
diff --git a/packages/blocks/src/store/test/process-block-type.js b/packages/blocks/src/store/test/process-block-type.js
index c82d08cc45d651..3c6838e311f778 100644
--- a/packages/blocks/src/store/test/process-block-type.js
+++ b/packages/blocks/src/store/test/process-block-type.js
@@ -215,12 +215,6 @@ describe( 'processBlockType', () => {
],
};
- // Freeze the deprecated block object and its supports so that the original is not mutated.
- // This ensures the test covers a regression where the original object was mutated.
- // See: https://github.com/WordPress/gutenberg/pull/63401#discussion_r1832394335.
- Object.freeze( blockSettings.deprecated[ 0 ] );
- Object.freeze( blockSettings.deprecated[ 0 ].supports );
-
const processedBlockType = processBlockType(
'test/block',
blockSettings
From 781129d655a63489cae1dbef577c96cdf8b26c44 Mon Sep 17 00:00:00 2001
From: Aaron Robertshaw <60436221+aaronrobertshaw@users.noreply.github.com>
Date: Fri, 20 Dec 2024 12:38:01 +1000
Subject: [PATCH 5/5] Revert "Typography: Stabilize typography block supports
within block processing (#63401)"
This reverts commit 48341a13f7ca383830fecc328b3e7e3131ba831a.
---
backport-changelog/6.8/7069.md | 3 -
lib/block-supports/typography.php | 24 +-
lib/compat/wordpress-6.8/blocks.php | 41 ---
.../block-editor/src/hooks/font-family.js | 2 +-
packages/block-editor/src/hooks/supports.js | 12 +-
packages/block-editor/src/hooks/typography.js | 10 +-
packages/blocks/src/api/constants.js | 21 +-
.../blocks/src/store/process-block-type.js | 82 ++---
.../src/store/test/private-selectors.js | 12 +-
.../src/store/test/process-block-type.js | 304 ------------------
phpunit/block-supports/typography-test.php | 105 ------
11 files changed, 58 insertions(+), 558 deletions(-)
delete mode 100644 backport-changelog/6.8/7069.md
delete mode 100644 packages/blocks/src/store/test/process-block-type.js
diff --git a/backport-changelog/6.8/7069.md b/backport-changelog/6.8/7069.md
deleted file mode 100644
index ea3c717ec3c93a..00000000000000
--- a/backport-changelog/6.8/7069.md
+++ /dev/null
@@ -1,3 +0,0 @@
-https://github.com/WordPress/wordpress-develop/pull/7069
-
-* https://github.com/WordPress/gutenberg/pull/63401
diff --git a/lib/block-supports/typography.php b/lib/block-supports/typography.php
index 21086b94f15c1a..a4719b7bdd4099 100644
--- a/lib/block-supports/typography.php
+++ b/lib/block-supports/typography.php
@@ -20,16 +20,16 @@ function gutenberg_register_typography_support( $block_type ) {
return;
}
- $has_font_family_support = $typography_supports['fontFamily'] ?? false;
+ $has_font_family_support = $typography_supports['__experimentalFontFamily'] ?? false;
$has_font_size_support = $typography_supports['fontSize'] ?? false;
- $has_font_style_support = $typography_supports['fontStyle'] ?? false;
- $has_font_weight_support = $typography_supports['fontWeight'] ?? false;
- $has_letter_spacing_support = $typography_supports['letterSpacing'] ?? false;
+ $has_font_style_support = $typography_supports['__experimentalFontStyle'] ?? false;
+ $has_font_weight_support = $typography_supports['__experimentalFontWeight'] ?? false;
+ $has_letter_spacing_support = $typography_supports['__experimentalLetterSpacing'] ?? false;
$has_line_height_support = $typography_supports['lineHeight'] ?? false;
$has_text_align_support = $typography_supports['textAlign'] ?? false;
$has_text_columns_support = $typography_supports['textColumns'] ?? false;
- $has_text_decoration_support = $typography_supports['textDecoration'] ?? false;
- $has_text_transform_support = $typography_supports['textTransform'] ?? false;
+ $has_text_decoration_support = $typography_supports['__experimentalTextDecoration'] ?? false;
+ $has_text_transform_support = $typography_supports['__experimentalTextTransform'] ?? false;
$has_writing_mode_support = $typography_supports['__experimentalWritingMode'] ?? false;
$has_typography_support = $has_font_family_support
@@ -91,16 +91,16 @@ function gutenberg_apply_typography_support( $block_type, $block_attributes ) {
return array();
}
- $has_font_family_support = $typography_supports['fontFamily'] ?? false;
+ $has_font_family_support = $typography_supports['__experimentalFontFamily'] ?? false;
$has_font_size_support = $typography_supports['fontSize'] ?? false;
- $has_font_style_support = $typography_supports['fontStyle'] ?? false;
- $has_font_weight_support = $typography_supports['fontWeight'] ?? false;
- $has_letter_spacing_support = $typography_supports['letterSpacing'] ?? false;
+ $has_font_style_support = $typography_supports['__experimentalFontStyle'] ?? false;
+ $has_font_weight_support = $typography_supports['__experimentalFontWeight'] ?? false;
+ $has_letter_spacing_support = $typography_supports['__experimentalLetterSpacing'] ?? false;
$has_line_height_support = $typography_supports['lineHeight'] ?? false;
$has_text_align_support = $typography_supports['textAlign'] ?? false;
$has_text_columns_support = $typography_supports['textColumns'] ?? false;
- $has_text_decoration_support = $typography_supports['textDecoration'] ?? false;
- $has_text_transform_support = $typography_supports['textTransform'] ?? false;
+ $has_text_decoration_support = $typography_supports['__experimentalTextDecoration'] ?? false;
+ $has_text_transform_support = $typography_supports['__experimentalTextTransform'] ?? false;
$has_writing_mode_support = $typography_supports['__experimentalWritingMode'] ?? false;
// Whether to skip individual block support features.
diff --git a/lib/compat/wordpress-6.8/blocks.php b/lib/compat/wordpress-6.8/blocks.php
index eb5d32afb20a59..6cfa98691020ef 100644
--- a/lib/compat/wordpress-6.8/blocks.php
+++ b/lib/compat/wordpress-6.8/blocks.php
@@ -5,47 +5,6 @@
* @package gutenberg
*/
-/**
- * Filters the block type arguments during registration to stabilize experimental block supports.
- *
- * This is a temporary compatibility shim as the approach in core is for this to be handled
- * within the WP_Block_Type class rather than requiring a filter.
- *
- * @param array $args Array of arguments for registering a block type.
- * @return array Array of arguments for registering a block type.
- */
-function gutenberg_stabilize_experimental_block_supports( $args ) {
- if ( empty( $args['supports']['typography'] ) ) {
- return $args;
- }
-
- $experimental_typography_supports_to_stable = array(
- '__experimentalFontFamily' => 'fontFamily',
- '__experimentalFontStyle' => 'fontStyle',
- '__experimentalFontWeight' => 'fontWeight',
- '__experimentalLetterSpacing' => 'letterSpacing',
- '__experimentalTextDecoration' => 'textDecoration',
- '__experimentalTextTransform' => 'textTransform',
- );
-
- $current_typography_supports = $args['supports']['typography'];
- $stable_typography_supports = array();
-
- foreach ( $current_typography_supports as $key => $value ) {
- if ( array_key_exists( $key, $experimental_typography_supports_to_stable ) ) {
- $stable_typography_supports[ $experimental_typography_supports_to_stable[ $key ] ] = $value;
- } else {
- $stable_typography_supports[ $key ] = $value;
- }
- }
-
- $args['supports']['typography'] = $stable_typography_supports;
-
- return $args;
-}
-
-add_filter( 'register_block_type_args', 'gutenberg_stabilize_experimental_block_supports', PHP_INT_MAX, 1 );
-
function gutenberg_apply_block_hooks_to_post_content( $content ) {
// The `the_content` filter does not provide the post that the content is coming from.
// However, we can infer it by calling `get_post()`, which will return the current post
diff --git a/packages/block-editor/src/hooks/font-family.js b/packages/block-editor/src/hooks/font-family.js
index e5d8e02ab8ec02..ba9a66a8bcf04f 100644
--- a/packages/block-editor/src/hooks/font-family.js
+++ b/packages/block-editor/src/hooks/font-family.js
@@ -13,7 +13,7 @@ import { shouldSkipSerialization } from './utils';
import { TYPOGRAPHY_SUPPORT_KEY } from './typography';
import { unlock } from '../lock-unlock';
-export const FONT_FAMILY_SUPPORT_KEY = 'typography.fontFamily';
+export const FONT_FAMILY_SUPPORT_KEY = 'typography.__experimentalFontFamily';
const { kebabCase } = unlock( componentsPrivateApis );
/**
diff --git a/packages/block-editor/src/hooks/supports.js b/packages/block-editor/src/hooks/supports.js
index c0b6bb2cc8b271..75f2bdf2dc219e 100644
--- a/packages/block-editor/src/hooks/supports.js
+++ b/packages/block-editor/src/hooks/supports.js
@@ -9,17 +9,17 @@ const ALIGN_WIDE_SUPPORT_KEY = 'alignWide';
const BORDER_SUPPORT_KEY = '__experimentalBorder';
const COLOR_SUPPORT_KEY = 'color';
const CUSTOM_CLASS_NAME_SUPPORT_KEY = 'customClassName';
-const FONT_FAMILY_SUPPORT_KEY = 'typography.fontFamily';
+const FONT_FAMILY_SUPPORT_KEY = 'typography.__experimentalFontFamily';
const FONT_SIZE_SUPPORT_KEY = 'typography.fontSize';
const LINE_HEIGHT_SUPPORT_KEY = 'typography.lineHeight';
/**
* Key within block settings' support array indicating support for font style.
*/
-const FONT_STYLE_SUPPORT_KEY = 'typography.fontStyle';
+const FONT_STYLE_SUPPORT_KEY = 'typography.__experimentalFontStyle';
/**
* Key within block settings' support array indicating support for font weight.
*/
-const FONT_WEIGHT_SUPPORT_KEY = 'typography.fontWeight';
+const FONT_WEIGHT_SUPPORT_KEY = 'typography.__experimentalFontWeight';
/**
* Key within block settings' supports array indicating support for text
* align e.g. settings found in `block.json`.
@@ -34,7 +34,7 @@ const TEXT_COLUMNS_SUPPORT_KEY = 'typography.textColumns';
* Key within block settings' supports array indicating support for text
* decorations e.g. settings found in `block.json`.
*/
-const TEXT_DECORATION_SUPPORT_KEY = 'typography.textDecoration';
+const TEXT_DECORATION_SUPPORT_KEY = 'typography.__experimentalTextDecoration';
/**
* Key within block settings' supports array indicating support for writing mode
* e.g. settings found in `block.json`.
@@ -44,13 +44,13 @@ const WRITING_MODE_SUPPORT_KEY = 'typography.__experimentalWritingMode';
* Key within block settings' supports array indicating support for text
* transforms e.g. settings found in `block.json`.
*/
-const TEXT_TRANSFORM_SUPPORT_KEY = 'typography.textTransform';
+const TEXT_TRANSFORM_SUPPORT_KEY = 'typography.__experimentalTextTransform';
/**
* Key within block settings' supports array indicating support for letter-spacing
* e.g. settings found in `block.json`.
*/
-const LETTER_SPACING_SUPPORT_KEY = 'typography.letterSpacing';
+const LETTER_SPACING_SUPPORT_KEY = 'typography.__experimentalLetterSpacing';
const LAYOUT_SUPPORT_KEY = 'layout';
const TYPOGRAPHY_SUPPORT_KEYS = [
LINE_HEIGHT_SUPPORT_KEY,
diff --git a/packages/block-editor/src/hooks/typography.js b/packages/block-editor/src/hooks/typography.js
index ab9a464fe5efbc..cf3f4327c8f034 100644
--- a/packages/block-editor/src/hooks/typography.js
+++ b/packages/block-editor/src/hooks/typography.js
@@ -27,12 +27,12 @@ function omit( object, keys ) {
);
}
-const LETTER_SPACING_SUPPORT_KEY = 'typography.letterSpacing';
-const TEXT_TRANSFORM_SUPPORT_KEY = 'typography.textTransform';
-const TEXT_DECORATION_SUPPORT_KEY = 'typography.textDecoration';
+const LETTER_SPACING_SUPPORT_KEY = 'typography.__experimentalLetterSpacing';
+const TEXT_TRANSFORM_SUPPORT_KEY = 'typography.__experimentalTextTransform';
+const TEXT_DECORATION_SUPPORT_KEY = 'typography.__experimentalTextDecoration';
const TEXT_COLUMNS_SUPPORT_KEY = 'typography.textColumns';
-const FONT_STYLE_SUPPORT_KEY = 'typography.fontStyle';
-const FONT_WEIGHT_SUPPORT_KEY = 'typography.fontWeight';
+const FONT_STYLE_SUPPORT_KEY = 'typography.__experimentalFontStyle';
+const FONT_WEIGHT_SUPPORT_KEY = 'typography.__experimentalFontWeight';
const WRITING_MODE_SUPPORT_KEY = 'typography.__experimentalWritingMode';
export const TYPOGRAPHY_SUPPORT_KEY = 'typography';
export const TYPOGRAPHY_SUPPORT_KEYS = [
diff --git a/packages/blocks/src/api/constants.js b/packages/blocks/src/api/constants.js
index 85a359af754db8..620dfcbb8599c0 100644
--- a/packages/blocks/src/api/constants.js
+++ b/packages/blocks/src/api/constants.js
@@ -183,7 +183,7 @@ export const __EXPERIMENTAL_STYLE_PROPERTY = {
},
fontFamily: {
value: [ 'typography', 'fontFamily' ],
- support: [ 'typography', 'fontFamily' ],
+ support: [ 'typography', '__experimentalFontFamily' ],
useEngine: true,
},
fontSize: {
@@ -193,12 +193,12 @@ export const __EXPERIMENTAL_STYLE_PROPERTY = {
},
fontStyle: {
value: [ 'typography', 'fontStyle' ],
- support: [ 'typography', 'fontStyle' ],
+ support: [ 'typography', '__experimentalFontStyle' ],
useEngine: true,
},
fontWeight: {
value: [ 'typography', 'fontWeight' ],
- support: [ 'typography', 'fontWeight' ],
+ support: [ 'typography', '__experimentalFontWeight' ],
useEngine: true,
},
lineHeight: {
@@ -240,17 +240,17 @@ export const __EXPERIMENTAL_STYLE_PROPERTY = {
},
textDecoration: {
value: [ 'typography', 'textDecoration' ],
- support: [ 'typography', 'textDecoration' ],
+ support: [ 'typography', '__experimentalTextDecoration' ],
useEngine: true,
},
textTransform: {
value: [ 'typography', 'textTransform' ],
- support: [ 'typography', 'textTransform' ],
+ support: [ 'typography', '__experimentalTextTransform' ],
useEngine: true,
},
letterSpacing: {
value: [ 'typography', 'letterSpacing' ],
- support: [ 'typography', 'letterSpacing' ],
+ support: [ 'typography', '__experimentalLetterSpacing' ],
useEngine: true,
},
writingMode: {
@@ -297,12 +297,3 @@ export const __EXPERIMENTAL_PATHS_WITH_OVERRIDE = {
'typography.fontSizes': true,
'spacing.spacingSizes': true,
};
-
-export const TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE = {
- __experimentalFontFamily: 'fontFamily',
- __experimentalFontStyle: 'fontStyle',
- __experimentalFontWeight: 'fontWeight',
- __experimentalLetterSpacing: 'letterSpacing',
- __experimentalTextDecoration: 'textDecoration',
- __experimentalTextTransform: 'textTransform',
-};
diff --git a/packages/blocks/src/store/process-block-type.js b/packages/blocks/src/store/process-block-type.js
index 8e5b7962adfe17..bc7b1a0e10e774 100644
--- a/packages/blocks/src/store/process-block-type.js
+++ b/packages/blocks/src/store/process-block-type.js
@@ -15,11 +15,7 @@ import warning from '@wordpress/warning';
* Internal dependencies
*/
import { isValidIcon, normalizeIconObject, omit } from '../api/utils';
-import {
- BLOCK_ICON_DEFAULT,
- DEPRECATED_ENTRY_KEYS,
- TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE,
-} from '../api/constants';
+import { BLOCK_ICON_DEFAULT, DEPRECATED_ENTRY_KEYS } from '../api/constants';
/** @typedef {import('../api/registration').WPBlockType} WPBlockType */
@@ -66,24 +62,6 @@ function mergeBlockVariations(
return result;
}
-function stabilizeSupports( rawSupports ) {
- if ( ! rawSupports ) {
- return rawSupports;
- }
-
- const supports = { ...rawSupports };
- if ( supports?.typography && typeof supports.typography === 'object' ) {
- supports.typography = Object.fromEntries(
- Object.entries( supports.typography ).map( ( [ key, value ] ) => [
- TYPOGRAPHY_SUPPORTS_EXPERIMENTAL_TO_STABLE[ key ] || key,
- value,
- ] )
- );
- }
-
- return supports;
-}
-
/**
* Takes the unprocessed block type settings, merges them with block type metadata
* and applies all the existing filters for the registered block type.
@@ -124,9 +102,6 @@ export const processBlockType =
),
};
- // Stabilize any experimental supports before applying filters.
- blockType.supports = stabilizeSupports( blockType.supports );
-
const settings = applyFilters(
'blocks.registerBlockType',
blockType,
@@ -134,10 +109,6 @@ export const processBlockType =
null
);
- // Re-stabilize any experimental supports after applying filters.
- // This ensures that any supports updated by filters are also stabilized.
- blockType.supports = stabilizeSupports( blockType.supports );
-
if (
settings.description &&
typeof settings.description !== 'string'
@@ -148,38 +119,29 @@ export const processBlockType =
}
if ( settings.deprecated ) {
- settings.deprecated = settings.deprecated.map( ( deprecation ) => {
- // Stabilize any experimental supports before applying filters.
- deprecation.supports = stabilizeSupports(
- deprecation.supports
- );
- const filteredDeprecation = // Only keep valid deprecation keys.
- applyFilters(
- 'blocks.registerBlockType',
- // Merge deprecation keys with pre-filter settings
- // so that filters that depend on specific keys being
- // present don't fail.
- {
- // Omit deprecation keys here so that deprecations
- // can opt out of specific keys like "supports".
- ...omit( blockType, DEPRECATED_ENTRY_KEYS ),
- ...deprecation,
- },
- blockType.name,
- deprecation
- );
- // Re-stabilize any experimental supports after applying filters.
- // This ensures that any supports updated by filters are also stabilized.
- filteredDeprecation.supports = stabilizeSupports(
- filteredDeprecation.supports
- );
-
- return Object.fromEntries(
- Object.entries( filteredDeprecation ).filter( ( [ key ] ) =>
+ settings.deprecated = settings.deprecated.map( ( deprecation ) =>
+ Object.fromEntries(
+ Object.entries(
+ // Only keep valid deprecation keys.
+ applyFilters(
+ 'blocks.registerBlockType',
+ // Merge deprecation keys with pre-filter settings
+ // so that filters that depend on specific keys being
+ // present don't fail.
+ {
+ // Omit deprecation keys here so that deprecations
+ // can opt out of specific keys like "supports".
+ ...omit( blockType, DEPRECATED_ENTRY_KEYS ),
+ ...deprecation,
+ },
+ blockType.name,
+ deprecation
+ )
+ ).filter( ( [ key ] ) =>
DEPRECATED_ENTRY_KEYS.includes( key )
)
- );
- } );
+ )
+ );
}
if ( ! isPlainObject( settings ) ) {
diff --git a/packages/blocks/src/store/test/private-selectors.js b/packages/blocks/src/store/test/private-selectors.js
index 2c173b96b0bcb1..ada2bd7c8cbcfe 100644
--- a/packages/blocks/src/store/test/private-selectors.js
+++ b/packages/blocks/src/store/test/private-selectors.js
@@ -127,12 +127,12 @@ describe( 'private selectors', () => {
name: 'core/example-block',
supports: {
typography: {
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- textDecoration: true,
- textTransform: true,
- letterSpacing: true,
+ __experimentalFontFamily: true,
+ __experimentalFontStyle: true,
+ __experimentalFontWeight: true,
+ __experimentalTextDecoration: true,
+ __experimentalTextTransform: true,
+ __experimentalLetterSpacing: true,
fontSize: true,
lineHeight: true,
},
diff --git a/packages/blocks/src/store/test/process-block-type.js b/packages/blocks/src/store/test/process-block-type.js
deleted file mode 100644
index 3c6838e311f778..00000000000000
--- a/packages/blocks/src/store/test/process-block-type.js
+++ /dev/null
@@ -1,304 +0,0 @@
-/**
- * WordPress dependencies
- */
-import { addFilter, removeFilter } from '@wordpress/hooks';
-
-/**
- * Internal dependencies
- */
-import { processBlockType } from '../process-block-type';
-
-describe( 'processBlockType', () => {
- const baseBlockSettings = {
- apiVersion: 3,
- attributes: {},
- edit: () => null,
- name: 'test/block',
- save: () => null,
- title: 'Test Block',
- };
-
- const select = {
- getBootstrappedBlockType: () => null,
- };
-
- afterEach( () => {
- removeFilter( 'blocks.registerBlockType', 'test/filterSupports' );
- } );
-
- it( 'should return the block type with stabilized typography supports', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- lineHeight: true,
- __experimentalFontFamily: true,
- __experimentalFontStyle: true,
- __experimentalFontWeight: true,
- __experimentalLetterSpacing: true,
- __experimentalTextTransform: true,
- __experimentalTextDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- },
- };
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect( processedBlockType.supports.typography ).toEqual( {
- fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- } );
- } );
-
- it( 'should return the block type with stable typography supports', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- },
- };
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect( processedBlockType.supports.typography ).toEqual( {
- fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- } );
- } );
-
- it( 'should reapply transformations after supports are filtered', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- lineHeight: true,
- __experimentalFontFamily: true,
- __experimentalFontStyle: true,
- __experimentalFontWeight: true,
- __experimentalLetterSpacing: true,
- __experimentalTextTransform: true,
- __experimentalTextDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- },
- };
-
- addFilter(
- 'blocks.registerBlockType',
- 'test/filterSupports',
- ( settings, name ) => {
- if ( name === 'test/block' && settings.supports.typography ) {
- settings.supports.typography.__experimentalFontFamily = false;
- settings.supports.typography.__experimentalFontStyle = false;
- settings.supports.typography.__experimentalFontWeight = false;
- }
- return settings;
- }
- );
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect( processedBlockType.supports.typography ).toEqual( {
- fontSize: true,
- lineHeight: true,
- fontFamily: false,
- fontStyle: false,
- fontWeight: false,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- } );
- } );
-
- it( 'should stabilize experimental typography supports within block deprecations', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- },
- deprecated: [
- {
- supports: {
- typography: {
- __experimentalFontFamily: true,
- __experimentalFontStyle: true,
- __experimentalFontWeight: true,
- __experimentalLetterSpacing: true,
- __experimentalTextTransform: true,
- __experimentalTextDecoration: true,
- __experimentalWritingMode: true,
- },
- },
- },
- ],
- };
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect(
- processedBlockType.deprecated[ 0 ].supports.typography
- ).toEqual( {
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- } );
- } );
-
- it( 'should reapply transformations after supports are filtered within block deprecations', () => {
- const blockSettings = {
- ...baseBlockSettings,
- supports: {
- typography: {
- fontSize: true,
- lineHeight: true,
- fontFamily: true,
- fontStyle: true,
- fontWeight: true,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- __experimentalDefaultControls: {
- fontSize: true,
- fontAppearance: true,
- textTransform: true,
- },
- },
- },
- deprecated: [
- {
- supports: {
- typography: {
- __experimentalFontFamily: true,
- __experimentalFontStyle: true,
- __experimentalFontWeight: true,
- __experimentalLetterSpacing: true,
- __experimentalTextTransform: true,
- __experimentalTextDecoration: true,
- __experimentalWritingMode: true,
- },
- },
- },
- ],
- };
-
- addFilter(
- 'blocks.registerBlockType',
- 'test/filterSupports',
- ( settings, name ) => {
- if ( name === 'test/block' && settings.supports.typography ) {
- settings.supports.typography.__experimentalFontFamily = false;
- settings.supports.typography.__experimentalFontStyle = false;
- settings.supports.typography.__experimentalFontWeight = false;
- }
- return settings;
- }
- );
-
- const processedBlockType = processBlockType(
- 'test/block',
- blockSettings
- )( { select } );
-
- expect(
- processedBlockType.deprecated[ 0 ].supports.typography
- ).toEqual( {
- fontFamily: false,
- fontStyle: false,
- fontWeight: false,
- letterSpacing: true,
- textTransform: true,
- textDecoration: true,
- __experimentalWritingMode: true,
- } );
- } );
-} );
diff --git a/phpunit/block-supports/typography-test.php b/phpunit/block-supports/typography-test.php
index 1804659c11af3c..eafd505db6ec65 100644
--- a/phpunit/block-supports/typography-test.php
+++ b/phpunit/block-supports/typography-test.php
@@ -283,111 +283,6 @@ public function test_should_generate_classname_for_font_family() {
$this->assertSame( $expected, $actual );
}
- /**
- * Tests that stabilized typography supports will also apply to blocks using
- * the experimental syntax, for backwards compatibility with existing blocks.
- *
- * @covers ::gutenberg_apply_typography_support
- */
- public function test_should_apply_experimental_typography_supports() {
- $this->test_block_name = 'test/experimental-typography-supports';
- register_block_type(
- $this->test_block_name,
- array(
- 'api_version' => 3,
- 'attributes' => array(
- 'style' => array(
- 'type' => 'object',
- ),
- ),
- 'supports' => array(
- 'typography' => array(
- '__experimentalFontFamily' => true,
- '__experimentalFontStyle' => true,
- '__experimentalFontWeight' => true,
- '__experimentalLetterSpacing' => true,
- '__experimentalTextDecoration' => true,
- '__experimentalTextTransform' => true,
- ),
- ),
- )
- );
- $registry = WP_Block_Type_Registry::get_instance();
- $block_type = $registry->get_registered( $this->test_block_name );
- $block_atts = array(
- 'fontFamily' => 'serif',
- 'style' => array(
- 'typography' => array(
- 'fontStyle' => 'italic',
- 'fontWeight' => 'bold',
- 'letterSpacing' => '1px',
- 'textDecoration' => 'underline',
- 'textTransform' => 'uppercase',
- ),
- ),
- );
-
- $actual = gutenberg_apply_typography_support( $block_type, $block_atts );
- $expected = array(
- 'class' => 'has-serif-font-family',
- 'style' => 'font-style:italic;font-weight:bold;text-decoration:underline;text-transform:uppercase;letter-spacing:1px;',
- );
-
- $this->assertSame( $expected, $actual );
- }
-
- /**
- * Tests that stabilized typography supports are applied correctly.
- *
- * @covers ::gutenberg_apply_typography_support
- */
- public function test_should_apply_stabilized_typography_supports() {
- $this->test_block_name = 'test/experimental-typography-supports';
- register_block_type(
- $this->test_block_name,
- array(
- 'api_version' => 3,
- 'attributes' => array(
- 'style' => array(
- 'type' => 'object',
- ),
- ),
- 'supports' => array(
- 'typography' => array(
- 'fontFamily' => true,
- 'fontStyle' => true,
- 'fontWeight' => true,
- 'letterSpacing' => true,
- 'textDecoration' => true,
- 'textTransform' => true,
- ),
- ),
- )
- );
- $registry = WP_Block_Type_Registry::get_instance();
- $block_type = $registry->get_registered( $this->test_block_name );
- $block_atts = array(
- 'fontFamily' => 'serif',
- 'style' => array(
- 'typography' => array(
- 'fontStyle' => 'italic',
- 'fontWeight' => 'bold',
- 'letterSpacing' => '1px',
- 'textDecoration' => 'underline',
- 'textTransform' => 'uppercase',
- ),
- ),
- );
-
- $actual = gutenberg_apply_typography_support( $block_type, $block_atts );
- $expected = array(
- 'class' => 'has-serif-font-family',
- 'style' => 'font-style:italic;font-weight:bold;text-decoration:underline;text-transform:uppercase;letter-spacing:1px;',
- );
-
- $this->assertSame( $expected, $actual );
- }
-
/**
* Tests generating font size values, including fluid formulae, from fontSizes preset.
*