<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="content-type" content="text/html; charset=utf-8" />
<title>[52049] trunk: Update theme.json classes for WordPress 5.9.</title>
</head>
<body>

<style type="text/css"><!--
#msg dl.meta { border: 1px #006 solid; background: #369; padding: 6px; color: #fff; }
#msg dl.meta dt { float: left; width: 6em; font-weight: bold; }
#msg dt:after { content:':';}
#msg dl, #msg dt, #msg ul, #msg li, #header, #footer, #logmsg { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt;  }
#msg dl a { font-weight: bold}
#msg dl a:link    { color:#fc3; }
#msg dl a:active  { color:#ff0; }
#msg dl a:visited { color:#cc6; }
h3 { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt; font-weight: bold; }
#msg pre { white-space: pre-line; overflow: auto; background: #ffc; border: 1px #fa0 solid; padding: 6px; }
#logmsg { background: #ffc; border: 1px #fa0 solid; padding: 1em 1em 0 1em; }
#logmsg p, #logmsg pre, #logmsg blockquote { margin: 0 0 1em 0; }
#logmsg p, #logmsg li, #logmsg dt, #logmsg dd { line-height: 14pt; }
#logmsg h1, #logmsg h2, #logmsg h3, #logmsg h4, #logmsg h5, #logmsg h6 { margin: .5em 0; }
#logmsg h1:first-child, #logmsg h2:first-child, #logmsg h3:first-child, #logmsg h4:first-child, #logmsg h5:first-child, #logmsg h6:first-child { margin-top: 0; }
#logmsg ul, #logmsg ol { padding: 0; list-style-position: inside; margin: 0 0 0 1em; }
#logmsg ul { text-indent: -1em; padding-left: 1em; }#logmsg ol { text-indent: -1.5em; padding-left: 1.5em; }
#logmsg > ul, #logmsg > ol { margin: 0 0 1em 0; }
#logmsg pre { background: #eee; padding: 1em; }
#logmsg blockquote { border: 1px solid #fa0; border-left-width: 10px; padding: 1em 1em 0 1em; background: white;}
#logmsg dl { margin: 0; }
#logmsg dt { font-weight: bold; }
#logmsg dd { margin: 0; padding: 0 0 0.5em 0; }
#logmsg dd:before { content:'\00bb';}
#logmsg table { border-spacing: 0px; border-collapse: collapse; border-top: 4px solid #fa0; border-bottom: 1px solid #fa0; background: #fff; }
#logmsg table th { text-align: left; font-weight: normal; padding: 0.2em 0.5em; border-top: 1px dotted #fa0; }
#logmsg table td { text-align: right; border-top: 1px dotted #fa0; padding: 0.2em 0.5em; }
#logmsg table thead th { text-align: center; border-bottom: 1px solid #fa0; }
#logmsg table th.Corner { text-align: left; }
#logmsg hr { border: none 0; border-top: 2px dashed #fa0; height: 1px; }
#header, #footer { color: #fff; background: #636; border: 1px #300 solid; padding: 6px; }
#patch { width: 100%; }
#patch h4 {font-family: verdana,arial,helvetica,sans-serif;font-size:10pt;padding:8px;background:#369;color:#fff;margin:0;}
#patch .propset h4, #patch .binary h4 {margin:0;}
#patch pre {padding:0;line-height:1.2em;margin:0;}
#patch .diff {width:100%;background:#eee;padding: 0 0 10px 0;overflow:auto;}
#patch .propset .diff, #patch .binary .diff  {padding:10px 0;}
#patch span {display:block;padding:0 10px;}
#patch .modfile, #patch .addfile, #patch .delfile, #patch .propset, #patch .binary, #patch .copfile {border:1px solid #ccc;margin:10px 0;}
#patch ins {background:#dfd;text-decoration:none;display:block;padding:0 10px;}
#patch del {background:#fdd;text-decoration:none;display:block;padding:0 10px;}
#patch .lines, .info {color:#888;background:#fff;}
--></style>
<div id="msg">
<dl class="meta" style="font-size: 105%">
<dt style="float: left; width: 6em; font-weight: bold">Revision</dt> <dd><a style="font-weight: bold" href="https://core.trac.wordpress.org/changeset/52049">52049</a><script type="application/ld+json">{"@context":"http://schema.org","@type":"EmailMessage","description":"Review this Commit","action":{"@type":"ViewAction","url":"https://core.trac.wordpress.org/changeset/52049","name":"Review Commit"}}</script></dd>
<dt style="float: left; width: 6em; font-weight: bold">Author</dt> <dd>jorgefilipecosta</dd>
<dt style="float: left; width: 6em; font-weight: bold">Date</dt> <dd>2021-11-08 19:18:39 +0000 (Mon, 08 Nov 2021)</dd>
</dl>

<pre style='padding-left: 1em; margin: 2em 0; border-left: 2px solid #ccc; line-height: 1.25; font-size: 105%; font-family: sans-serif'>Update theme.json classes for WordPress 5.9.

This commit ports to core the changes to the classes that deal with theme.json code.

See <a href="https://core.trac.wordpress.org/ticket/54336">#54336</a>.
Props oandregal, spacedmonkey, noisysocks, hellofromtonya, youknowriad.</pre>

<h3>Modified Paths</h3>
<ul>
<li><a href="#trunksrcwpincludesblockeditorphp">trunk/src/wp-includes/block-editor.php</a></li>
<li><a href="#trunksrcwpincludesblocksupportsduotonephp">trunk/src/wp-includes/block-supports/duotone.php</a></li>
<li><a href="#trunksrcwpincludesclasswpthemejsonresolverphp">trunk/src/wp-includes/class-wp-theme-json-resolver.php</a></li>
<li><a href="#trunksrcwpincludesclasswpthemejsonphp">trunk/src/wp-includes/class-wp-theme-json.php</a></li>
<li><a href="#trunksrcwpincludesdefaultfiltersphp">trunk/src/wp-includes/default-filters.php</a></li>
<li><a href="#trunksrcwpincludesksesphp">trunk/src/wp-includes/kses.php</a></li>
<li><a href="#trunksrcwpincludesscriptloaderphp">trunk/src/wp-includes/script-loader.php</a></li>
<li><a href="#trunksrcwpincludesthemei18njson">trunk/src/wp-includes/theme-i18n.json</a></li>
<li><a href="#trunksrcwpincludesthemejson">trunk/src/wp-includes/theme.json</a></li>
<li><a href="#trunksrcwpsettingsphp">trunk/src/wp-settings.php</a></li>
<li><a href="#trunktestsphpunitdatalanguagesthemesblockthemepl_PLpo">trunk/tests/phpunit/data/languages/themes/block-theme-pl_PL.po</a></li>
<li><a href="#trunktestsphpunitdatathemedir1blockthemethemejson">trunk/tests/phpunit/data/themedir1/block-theme/theme.json</a></li>
<li><a href="#trunktestsphpunitteststhemethemeDirphp">trunk/tests/phpunit/tests/theme/themeDir.php</a></li>
<li><a href="#trunktestsphpunitteststhemewpThemeJsonphp">trunk/tests/phpunit/tests/theme/wpThemeJson.php</a></li>
<li><a href="#trunktestsphpunitteststhemewpThemeJsonResolverphp">trunk/tests/phpunit/tests/theme/wpThemeJsonResolver.php</a></li>
</ul>

<h3>Added Paths</h3>
<ul>
<li><a href="#trunksrcwpincludesclasswpthemejsonschemaphp">trunk/src/wp-includes/class-wp-theme-json-schema.php</a></li>
<li>trunk/tests/phpunit/data/themedir1/block-theme-child/</li>
<li><a href="#trunktestsphpunitdatathemedir1blockthemechildstylecss">trunk/tests/phpunit/data/themedir1/block-theme-child/style.css</a></li>
<li><a href="#trunktestsphpunitdatathemedir1blockthemechildthemejson">trunk/tests/phpunit/data/themedir1/block-theme-child/theme.json</a></li>
<li><a href="#trunktestsphpunitteststhemewpThemeJsonSchemaphp">trunk/tests/phpunit/tests/theme/wpThemeJsonSchema.php</a></li>
</ul>

</div>
<div id="patch">
<h3>Diff</h3>
<a id="trunksrcwpincludesblockeditorphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/block-editor.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/block-editor.php    2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/block-editor.php      2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -303,15 +303,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                $custom_settings
</span><span class="cx" style="display: block; padding: 0 10px">        );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        $theme_json = WP_Theme_JSON_Resolver::get_merged_data( $editor_settings );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ $theme_json = WP_Theme_JSON_Resolver::get_merged_data();
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        if ( WP_Theme_JSON_Resolver::theme_has_support() ) {
</span><span class="cx" style="display: block; padding: 0 10px">                $editor_settings['styles'][] = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'css'            => $theme_json->get_stylesheet( 'block_styles' ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'css'            => $theme_json->get_stylesheet( array( 'styles', 'presets' ) ),
</ins><span class="cx" style="display: block; padding: 0 10px">                         '__unstableType' => 'globalStyles',
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px">                $editor_settings['styles'][] = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'css'                     => $theme_json->get_stylesheet( 'css_variables' ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'css'                     => $theme_json->get_stylesheet( array( 'variables' ) ),
</ins><span class="cx" style="display: block; padding: 0 10px">                         '__experimentalNoWrapper' => true,
</span><span class="cx" style="display: block; padding: 0 10px">                        '__unstableType'          => 'globalStyles',
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -358,17 +358,17 @@
</span><span class="cx" style="display: block; padding: 0 10px">                $editor_settings['disableCustomFontSizes'] = ! $editor_settings['__experimentalFeatures']['typography']['customFontSize'];
</span><span class="cx" style="display: block; padding: 0 10px">                unset( $editor_settings['__experimentalFeatures']['typography']['customFontSize'] );
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        if ( isset( $editor_settings['__experimentalFeatures']['typography']['customLineHeight'] ) ) {
-               $editor_settings['enableCustomLineHeight'] = $editor_settings['__experimentalFeatures']['typography']['customLineHeight'];
-               unset( $editor_settings['__experimentalFeatures']['typography']['customLineHeight'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ if ( isset( $editor_settings['__experimentalFeatures']['typography']['lineHeight'] ) ) {
+               $editor_settings['enableCustomLineHeight'] = $editor_settings['__experimentalFeatures']['typography']['lineHeight'];
+               unset( $editor_settings['__experimentalFeatures']['typography']['lineHeight'] );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px">        if ( isset( $editor_settings['__experimentalFeatures']['spacing']['units'] ) ) {
</span><span class="cx" style="display: block; padding: 0 10px">                $editor_settings['enableCustomUnits'] = $editor_settings['__experimentalFeatures']['spacing']['units'];
</span><span class="cx" style="display: block; padding: 0 10px">                unset( $editor_settings['__experimentalFeatures']['spacing']['units'] );
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        if ( isset( $editor_settings['__experimentalFeatures']['spacing']['customPadding'] ) ) {
-               $editor_settings['enableCustomSpacing'] = $editor_settings['__experimentalFeatures']['spacing']['customPadding'];
-               unset( $editor_settings['__experimentalFeatures']['spacing']['customPadding'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ if ( isset( $editor_settings['__experimentalFeatures']['spacing']['padding'] ) ) {
+               $editor_settings['enableCustomSpacing'] = $editor_settings['__experimentalFeatures']['spacing']['padding'];
+               unset( $editor_settings['__experimentalFeatures']['spacing']['padding'] );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span></span></pre></div>
<a id="trunksrcwpincludesblocksupportsduotonephp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/block-supports/duotone.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/block-supports/duotone.php  2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/block-supports/duotone.php    2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -70,6 +70,28 @@
</span><span class="cx" style="display: block; padding: 0 10px"> }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px"> /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ * Direct port of tinycolor's boundAlpha function to maintain consistency with
+ * how tinycolor works.
+ *
+ * @see https://github.com/bgrins/TinyColor
+ *
+ * @since 5.9.0
+ * @access private
+ *
+ * @param  mixed $n   Number of unknown type.
+ * @return float      Value in the range [0,1].
+ */
+function _wp_tinycolor_bound_alpha( $n ) {
+       if ( is_numeric( $n ) ) {
+               $n = (float) $n;
+               if ( $n >= 0 && $n <= 1 ) {
+                       return $n;
+               }
+       }
+       return 1;
+}
+
+/**
</ins><span class="cx" style="display: block; padding: 0 10px">  * Round and convert values of an RGB object.
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span><span class="cx" style="display: block; padding: 0 10px">  * Direct port of TinyColor's function, lightly simplified to maintain
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -170,8 +192,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px"> /**
</span><span class="cx" style="display: block; padding: 0 10px">  * Parses hex, hsl, and rgb CSS strings using the same regex as TinyColor v1.4.2
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">- * used in the JavaScript. Only colors output from react-color are implemented
- * and the alpha value is ignored as it is not used in duotone.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ * used in the JavaScript. Only colors output from react-color are implemented.
</ins><span class="cx" style="display: block; padding: 0 10px">  *
</span><span class="cx" style="display: block; padding: 0 10px">  * Direct port of TinyColor's function, lightly simplified to maintain
</span><span class="cx" style="display: block; padding: 0 10px">  * consistency with TinyColor.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -180,6 +201,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">  * @see https://github.com/casesandberg/react-color/
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span><span class="cx" style="display: block; padding: 0 10px">  * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ * @since 5.9.0 Added alpha processing.
</ins><span class="cx" style="display: block; padding: 0 10px">  * @access private
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span><span class="cx" style="display: block; padding: 0 10px">  * @param string $color_str CSS color string.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -199,7 +221,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $rgb_regexp = '/^rgb' . $permissive_match3 . '$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $rgb_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => $match[1],
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => $match[2],
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -206,11 +228,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => $match[3],
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = 1;
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $rgba_regexp = '/^rgba' . $permissive_match4 . '$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $rgba_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => $match[1],
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => $match[2],
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -217,11 +243,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => $match[3],
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = _wp_tinycolor_bound_alpha( $match[4] );
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hsl_regexp = '/^hsl' . $permissive_match3 . '$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $hsl_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_hsl_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_hsl_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'h' => $match[1],
</span><span class="cx" style="display: block; padding: 0 10px">                                's' => $match[2],
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -228,6 +258,10 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'l' => $match[3],
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = 1;
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hsla_regexp = '/^hsla' . $permissive_match4 . '$/';
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -239,11 +273,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'l' => $match[3],
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = _wp_tinycolor_bound_alpha( $match[4] );
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hex8_regexp = '/^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $hex8_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => base_convert( $match[1], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => base_convert( $match[2], 16, 10 ),
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -250,11 +288,17 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => base_convert( $match[3], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = _wp_tinycolor_bound_alpha(
+                       base_convert( $match[4], 16, 10 ) / 255
+               );
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hex6_regexp = '/^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $hex6_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => base_convert( $match[1], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => base_convert( $match[2], 16, 10 ),
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -261,11 +305,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => base_convert( $match[3], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = 1;
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hex4_regexp = '/^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $hex4_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => base_convert( $match[1] . $match[1], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => base_convert( $match[2] . $match[2], 16, 10 ),
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -272,11 +320,17 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => base_convert( $match[3] . $match[3], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = _wp_tinycolor_bound_alpha(
+                       base_convert( $match[4] . $match[4], 16, 10 ) / 255
+               );
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        $hex3_regexp = '/^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/';
</span><span class="cx" style="display: block; padding: 0 10px">        if ( preg_match( $hex3_regexp, $color_str, $match ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return wp_tinycolor_rgb_to_rgb(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $rgb = wp_tinycolor_rgb_to_rgb(
</ins><span class="cx" style="display: block; padding: 0 10px">                         array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'r' => base_convert( $match[1] . $match[1], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'g' => base_convert( $match[2] . $match[2], 16, 10 ),
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -283,7 +337,24 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'b' => base_convert( $match[3] . $match[3], 16, 10 ),
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $rgb['a'] = 1;
+
+               return $rgb;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+       /*
+        * The JS color picker considers the string "transparent" to be a hex value,
+        * so we need to handle it here as a special case.
+        */
+       if ( 'transparent' === $color_str ) {
+               return array(
+                       'r' => 0,
+                       'g' => 0,
+                       'b' => 0,
+                       'a' => 0,
+               );
+       }
</ins><span class="cx" style="display: block; padding: 0 10px"> }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -313,7 +384,96 @@
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+/**
+ * Renders the duotone filter SVG and returns the CSS filter property to
+ * reference the rendered SVG.
+ *
+ * @since 5.9.0
+ *
+ * @param array $preset Duotone preset value as seen in theme.json.
+ * @return string Duotone CSS filter property.
+ */
+function wp_render_duotone_filter_preset( $preset ) {
+       $duotone_id     = $preset['slug'];
+       $duotone_colors = $preset['colors'];
+       $filter_id      = 'wp-duotone-' . $duotone_id;
+       $duotone_values = array(
+               'r' => array(),
+               'g' => array(),
+               'b' => array(),
+               'a' => array(),
+       );
+       foreach ( $duotone_colors as $color_str ) {
+               $color = wp_tinycolor_string_to_rgb( $color_str );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                $duotone_values['r'][] = $color['r'] / 255;
+               $duotone_values['g'][] = $color['g'] / 255;
+               $duotone_values['b'][] = $color['b'] / 255;
+               $duotone_values['a'][] = $color['a'];
+       }
+
+       ob_start();
+
+       ?>
+
+       <svg
+               xmlns="http://www.w3.org/2000/svg"
+               viewBox="0 0 0 0"
+               width="0"
+               height="0"
+               focusable="false"
+               role="none"
+               style="visibility: hidden; position: absolute; left: -9999px; overflow: hidden;"
+       >
+               <defs>
+                       <filter id="<?php echo esc_attr( $filter_id ); ?>">
+                               <feColorMatrix
+                                       color-interpolation-filters="sRGB"
+                                       type="matrix"
+                                       values="
+                                               .299 .587 .114 0 0
+                                               .299 .587 .114 0 0
+                                               .299 .587 .114 0 0
+                                               .299 .587 .114 0 0
+                                       "
+                               />
+                               <feComponentTransfer color-interpolation-filters="sRGB" >
+                                       <feFuncR type="table" tableValues="<?php echo esc_attr( implode( ' ', $duotone_values['r'] ) ); ?>" />
+                                       <feFuncG type="table" tableValues="<?php echo esc_attr( implode( ' ', $duotone_values['g'] ) ); ?>" />
+                                       <feFuncB type="table" tableValues="<?php echo esc_attr( implode( ' ', $duotone_values['b'] ) ); ?>" />
+                                       <feFuncA type="table" tableValues="<?php echo esc_attr( implode( ' ', $duotone_values['a'] ) ); ?>" />
+                               </feComponentTransfer>
+                               <feComposite in2="SourceGraphic" operator="in" />
+                       </filter>
+               </defs>
+       </svg>
+
+       <?php
+
+       $svg = ob_get_clean();
+
+       if ( ! defined( 'SCRIPT_DEBUG' ) || ! SCRIPT_DEBUG ) {
+               // Clean up the whitespace.
+               $svg = preg_replace( "/[\r\n\t ]+/", ' ', $svg );
+               $svg = preg_replace( '/> </', '><', $svg );
+               $svg = trim( $svg );
+       }
+
+       add_action(
+               /*
+                * Safari doesn't render SVG filters defined in data URIs,
+                * and SVG filters won't render in the head of a document,
+                * so the next best place to put the SVG is in the footer.
+                */
+               is_admin() ? 'admin_footer' : 'wp_footer',
+               static function () use ( $svg ) {
+                       echo $svg;
+               }
+       );
+
+       return "url('#" . $filter_id . "')";
+}
+
</ins><span class="cx" style="display: block; padding: 0 10px"> /**
</span><span class="cx" style="display: block; padding: 0 10px">  * Render out the duotone stylesheet and SVG.
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span></span></pre></div>
<a id="trunksrcwpincludesclasswpthemejsonresolverphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/class-wp-theme-json-resolver.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/class-wp-theme-json-resolver.php    2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/class-wp-theme-json-resolver.php      2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -11,6 +11,10 @@
</span><span class="cx" style="display: block; padding: 0 10px">  * Class that abstracts the processing of the different data sources
</span><span class="cx" style="display: block; padding: 0 10px">  * for site-level config and offers an API to work with them.
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ * This class is for internal core usage and is not supposed to be used by extenders (plugins and/or themes).
+ * This is a low-level API that may need to do breaking changes. Please,
+ * use get_global_settings, get_global_styles, and get_global_stylesheet instead.
+ *
</ins><span class="cx" style="display: block; padding: 0 10px">  * @access private
</span><span class="cx" style="display: block; padding: 0 10px">  */
</span><span class="cx" style="display: block; padding: 0 10px"> class WP_Theme_JSON_Resolver {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -40,9 +44,27 @@
</span><span class="cx" style="display: block; padding: 0 10px">        private static $theme_has_support = null;
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Container for data coming from the user.
+        *
+        * @since 5.9.0
+        * @var WP_Theme_JSON
+        */
+       private static $user = null;
+
+       /**
+        * Stores the ID of the custom post type
+        * that holds the user data.
+        *
+        * @since 5.9.0
+        * @var integer
+        */
+       private static $user_custom_post_type_id = null;
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * Container to keep loaded i18n schema for `theme.json`.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * @since 5.9.0
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * @since 5.8.0
+        * @since 5.9.0 Renamed from $theme_json_i18n
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var array
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="cx" style="display: block; padding: 0 10px">        private static $i18n_schema = null;
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -122,34 +144,45 @@
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * Returns the theme's data.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Data from theme.json can be augmented via the $theme_support_data variable.
-        * This is useful, for example, to backfill the gaps in theme.json that a theme
-        * has declared via add_theme_supports.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * Data from theme.json will be backfilled from existing
+        * theme supports, if any. Note that if the same data
+        * is present in theme.json and in theme supports,
+        * the theme.json takes precendence.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Note that if the same data is present in theme.json and in $theme_support_data,
-        * the theme.json's is not overwritten.
-        *
</del><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Theme supports have been inlined and the argument removed.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * @param array $theme_support_data Optional. Theme support data in theme.json format.
-        *                                  Default empty array.
</del><span class="cx" style="display: block; padding: 0 10px">          * @return WP_Theme_JSON Entity that holds theme data.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        public static function get_theme_data( $theme_support_data = array() ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ public static function get_theme_data( $deprecated = array() ) {
+               if ( ! empty( $deprecated ) ) {
+                       _deprecated_argument( __METHOD__, '5.9' );
+               }
</ins><span class="cx" style="display: block; padding: 0 10px">                 if ( null === self::$theme ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        $theme_json_data = self::read_json_file( self::get_file_path_from_theme( 'theme.json' ) );
</span><span class="cx" style="display: block; padding: 0 10px">                        $theme_json_data = self::translate( $theme_json_data, wp_get_theme()->get( 'TextDomain' ) );
</span><span class="cx" style="display: block; padding: 0 10px">                        self::$theme     = new WP_Theme_JSON( $theme_json_data );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                }
</del><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                if ( empty( $theme_support_data ) ) {
-                       return self::$theme;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 if ( wp_get_theme()->parent() ) {
+                               // Get parent theme.json.
+                               $parent_theme_json_data = self::read_json_file( self::get_file_path_from_theme( 'theme.json', true ) );
+                               $parent_theme_json_data = self::translate( $parent_theme_json_data, wp_get_theme()->parent()->get( 'TextDomain' ) );
+                               $parent_theme           = new WP_Theme_JSON( $parent_theme_json_data );
+
+                               // Merge the child theme.json into the parent theme.json.
+                               // The child theme takes precedence over the parent.
+                               $parent_theme->merge( self::$theme );
+                               self::$theme = $parent_theme;
+                       }
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                /*
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                 * We want the presets and settings declared in theme.json
-                * to override the ones declared via add_theme_support.
-                */
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * We want the presets and settings declared in theme.json
+               * to override the ones declared via theme supports.
+               * So we take theme supports, transform it to theme.json shape
+               * and merge the self::$theme upon that.
+               */
+               $theme_support_data  = WP_Theme_JSON::get_from_editor_settings( get_default_block_editor_settings() );
</ins><span class="cx" style="display: block; padding: 0 10px">                 $with_theme_supports = new WP_Theme_JSON( $theme_support_data );
</span><span class="cx" style="display: block; padding: 0 10px">                $with_theme_supports->merge( self::$theme );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -157,40 +190,180 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * There are different sources of data for a site: core and theme.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * Returns the CPT that contains the user's origin config
+        * for the current theme or a void array if none found.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * While the getters {@link get_core_data}, {@link get_theme_data} return the raw data
-        * from the respective origins, this method merges them all together.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * It can also create and return a new draft CPT.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * If the same piece of data is declared in different origins (core and theme),
-        * the last origin overrides the previous. For example, if core disables custom colors
-        * but a theme enables them, the theme config wins.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * @since 5.9.0
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param bool  $should_create_cpt  Optional. Whether a new CPT should be created if no one was found.
+        *                                  False by default.
+        * @param array $post_status_filter Filter Optional. CPT by post status.
+        *                                   ['publish'] by default, so it only fetches published posts.
+        *
+        * @return array Custom Post Type for the user's origin config.
+        */
+       private static function get_user_data_from_custom_post_type( $should_create_cpt = false, $post_status_filter = array( 'publish' ) ) {
+               $user_cpt         = array();
+               $post_type_filter = 'wp_global_styles';
+               $query            = new WP_Query(
+                       array(
+                               'posts_per_page' => 1,
+                               'orderby'        => 'date',
+                               'order'          => 'desc',
+                               'post_type'      => $post_type_filter,
+                               'post_status'    => $post_status_filter,
+                               'tax_query'      => array(
+                                       array(
+                                               'taxonomy' => 'wp_theme',
+                                               'field'    => 'name',
+                                               'terms'    => wp_get_theme()->get_stylesheet(),
+                                       ),
+                               ),
+                       )
+               );
+
+               if ( is_array( $query->posts ) && ( 1 === $query->post_count ) ) {
+                       $user_cpt = $query->posts[0]->to_array();
+               } elseif ( $should_create_cpt ) {
+                       $cpt_post_id = wp_insert_post(
+                               array(
+                                       'post_content' => '{"version": ' . WP_Theme_JSON::LATEST_SCHEMA . ', "isGlobalStylesUserThemeJSON": true }',
+                                       'post_status'  => 'publish',
+                                       'post_title'   => __( 'Custom Styles', 'default' ),
+                                       'post_type'    => $post_type_filter,
+                                       'post_name'    => 'wp-global-styles-' . urlencode( wp_get_theme()->get_stylesheet() ),
+                                       'tax_input'    => array(
+                                               'wp_theme' => array( wp_get_theme()->get_stylesheet() ),
+                                       ),
+                               ),
+                               true
+                       );
+
+                       if ( is_wp_error( $cpt_post_id ) ) {
+                               $user_cpt = array();
+                       } else {
+                               $user_cpt = get_post( $cpt_post_id, ARRAY_A );
+                       }
+               }
+
+               return $user_cpt;
+       }
+
+       /**
+        * Returns the user's origin config.
+        *
+        * @since 5.9.0
+        *
+        * @return WP_Theme_JSON Entity that holds user data.
+        */
+       public static function get_user_data() {
+               if ( null !== self::$user ) {
+                       return self::$user;
+               }
+
+               $config   = array();
+               $user_cpt = self::get_user_data_from_custom_post_type();
+
+               if ( array_key_exists( 'post_content', $user_cpt ) ) {
+                       $decoded_data = json_decode( $user_cpt['post_content'], true );
+
+                       $json_decoding_error = json_last_error();
+                       if ( JSON_ERROR_NONE !== $json_decoding_error ) {
+                               trigger_error( 'Error when decoding a theme.json schema for user data. ' . json_last_error_msg() );
+                               return new WP_Theme_JSON( $config, 'user' );
+                       }
+
+                       // Very important to verify if the flag isGlobalStylesUserThemeJSON is true.
+                       // If is not true the content was not escaped and is not safe.
+                       if (
+                               is_array( $decoded_data ) &&
+                               isset( $decoded_data['isGlobalStylesUserThemeJSON'] ) &&
+                               $decoded_data['isGlobalStylesUserThemeJSON']
+                       ) {
+                               unset( $decoded_data['isGlobalStylesUserThemeJSON'] );
+                               $config = $decoded_data;
+                       }
+               }
+               self::$user = new WP_Theme_JSON( $config, 'user' );
+
+               return self::$user;
+       }
+
+       /**
+        * There are three sources of data (origins) for a site:
+        * core, theme, and user. The user's has higher priority
+        * than the theme's, and the theme's higher than core's.
+        *
+        * Unlike the getters {@link get_core_data},
+        * {@link get_theme_data}, and {@link get_user_data},
+        * this method returns data after it has been merged
+        * with the previous origins. This means that if the same piece of data
+        * is declared in different origins (user, theme, and core),
+        * the last origin overrides the previous.
+        *
+        * For example, if the user has set a background color
+        * for the paragraph block, and the theme has done it as well,
+        * the user preference wins.
+        *
</ins><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Add user data and change the arguments.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * @param array $settings Optional. Existing block editor settings. Default empty array.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * @param string $origin Optional. To what level should we merge data.
+        *                       Valid values are 'theme' or 'user'.
+        *                       Default is 'user'.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return WP_Theme_JSON
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        public static function get_merged_data( $settings = array() ) {
-               $theme_support_data = WP_Theme_JSON::get_from_editor_settings( $settings );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ public static function get_merged_data( $origin = 'user' ) {
+               if ( is_array( $origin ) ) {
+                       _deprecated_argument( __FUNCTION__, '5.9' );
+               }
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                $result = new WP_Theme_JSON();
</span><span class="cx" style="display: block; padding: 0 10px">                $result->merge( self::get_core_data() );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $result->merge( self::get_theme_data( $theme_support_data ) );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $result->merge( self::get_theme_data() );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                if ( 'user' === $origin ) {
+                       $result->merge( self::get_user_data() );
+               }
+
</ins><span class="cx" style="display: block; padding: 0 10px">                 return $result;
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Returns the ID of the custom post type
+        * that stores user data.
+        *
+        * @since 5.9.0
+        *
+        * @return integer|null
+        */
+       public static function get_user_custom_post_type_id() {
+               if ( null !== self::$user_custom_post_type_id ) {
+                       return self::$user_custom_post_type_id;
+               }
+
+               $user_cpt = self::get_user_data_from_custom_post_type( true );
+
+               if ( array_key_exists( 'ID', $user_cpt ) ) {
+                       self::$user_custom_post_type_id = $user_cpt['ID'];
+               }
+
+               return self::$user_custom_post_type_id;
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * Whether the current theme has a theme.json file.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Also check in the parent theme.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @return bool
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="cx" style="display: block; padding: 0 10px">        public static function theme_has_support() {
</span><span class="cx" style="display: block; padding: 0 10px">                if ( ! isset( self::$theme_has_support ) ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        self::$theme_has_support = (bool) self::get_file_path_from_theme( 'theme.json' );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 self::$theme_has_support = is_readable( get_theme_file_path( 'theme.json' ) );
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                return self::$theme_has_support;
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -202,24 +375,17 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * If it isn't, returns an empty string, otherwise returns the whole file path.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Adapt to work with child themes.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param string $file_name Name of the file.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param bool   $template  Optional. Use template theme directory. Default false.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return string The whole file path or empty if the file doesn't exist.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private static function get_file_path_from_theme( $file_name ) {
-               /*
-                * This used to be a locate_template call. However, that method proved problematic
-                * due to its use of constants (STYLESHEETPATH) that threw errors in some scenarios.
-                *
-                * When the theme.json merge algorithm properly supports child themes,
-                * this should also fall back to the template path, as locate_template did.
-                */
-               $located   = '';
-               $candidate = get_stylesheet_directory() . '/' . $file_name;
-               if ( is_readable( $candidate ) ) {
-                       $located = $candidate;
-               }
-               return $located;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private static function get_file_path_from_theme( $file_name, $template = false ) {
+               $path      = $template ? get_template_directory() : get_stylesheet_directory();
+               $candidate = $path . '/' . $file_name;
+
+               return is_readable( $candidate ) ? $candidate : '';
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -226,11 +392,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * Cleans the cached data so it can be recalculated.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added new variables to reset.
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public static function clean_cached_data() {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                self::$core              = null;
-               self::$theme             = null;
-               self::$theme_has_support = null;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         self::$core                     = null;
+               self::$theme                    = null;
+               self::$user                     = null;
+               self::$user_custom_post_type_id = null;
+               self::$theme_has_support        = null;
+               self::$i18n_schema              = null;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px"> }
</span></span></pre></div>
<a id="trunksrcwpincludesclasswpthemejsonschemaphp"></a>
<div class="addfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Added: trunk/src/wp-includes/class-wp-theme-json-schema.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/class-wp-theme-json-schema.php                              (rev 0)
+++ trunk/src/wp-includes/class-wp-theme-json-schema.php        2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -0,0 +1,141 @@
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+<?php
+/**
+ * WP_Theme_JSON_Schema class
+ *
+ * @package WordPress
+ * @subpackage Theme
+ * @since 5.9.0
+ */
+
+/**
+ * Class that migrates a given theme.json structure to the latest schema.
+ *
+ * This class is for internal core usage and is not supposed to be used by extenders (plugins and/or themes).
+ * This is a low-level API that may need to do breaking changes. Please,
+ * use get_global_settings, get_global_styles, and get_global_stylesheet instead.
+ *
+ * @access private
+ */
+class WP_Theme_JSON_Schema {
+
+       /**
+        * Maps old properties to their new location within the schema's settings.
+        * This will be applied at both the defaults and individual block levels.
+        */
+       const V1_TO_V2_RENAMED_PATHS = array(
+               'border.customRadius'         => 'border.radius',
+               'spacing.customMargin'        => 'spacing.margin',
+               'spacing.customPadding'       => 'spacing.padding',
+               'typography.customLineHeight' => 'typography.lineHeight',
+       );
+
+       /**
+        * Function that migrates a given theme.json structure to the last version.
+        *
+        * @param array $theme_json The structure to migrate.
+        *
+        * @return array The structure in the last version.
+        */
+       public static function migrate( $theme_json ) {
+               if ( ! isset( $theme_json['version'] ) ) {
+                       $theme_json = array(
+                               'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                       );
+               }
+
+               if ( 1 === $theme_json['version'] ) {
+                       $theme_json = self::migrate_v1_to_v2( $theme_json );
+               }
+
+               return $theme_json;
+       }
+
+       /**
+        * Removes the custom prefixes for a few properties
+        * that were part of v1:
+        *
+        * 'border.customRadius'         => 'border.radius',
+        * 'spacing.customMargin'        => 'spacing.margin',
+        * 'spacing.customPadding'       => 'spacing.padding',
+        * 'typography.customLineHeight' => 'typography.lineHeight',
+        *
+        * @param array $old Data to migrate.
+        *
+        * @return array Data without the custom prefixes.
+        */
+       private static function migrate_v1_to_v2( $old ) {
+               // Copy everything.
+               $new = $old;
+
+               // Overwrite the things that changed.
+               if ( isset( $old['settings'] ) ) {
+                       $new['settings'] = self::rename_paths( $old['settings'], self::V1_TO_V2_RENAMED_PATHS );
+               }
+
+               // Set the new version.
+               $new['version'] = 2;
+
+               return $new;
+       }
+
+       /**
+        * Processes the settings subtree.
+        *
+        * @param array $settings        Array to process.
+        * @param array $paths_to_rename Paths to rename.
+        *
+        * @return array The settings in the new format.
+        */
+       private static function rename_paths( $settings, $paths_to_rename ) {
+               $new_settings = $settings;
+
+               // Process any renamed/moved paths within default settings.
+               self::rename_settings( $new_settings, $paths_to_rename );
+
+               // Process individual block settings.
+               if ( isset( $new_settings['blocks'] ) && is_array( $new_settings['blocks'] ) ) {
+                       foreach ( $new_settings['blocks'] as &$block_settings ) {
+                               self::rename_settings( $block_settings, $paths_to_rename );
+                       }
+               }
+
+               return $new_settings;
+       }
+
+       /**
+        * Processes a settings array, renaming or moving properties.
+        *
+        * @param array $settings        Reference to settings either defaults or an individual block's.
+        * @param arary $paths_to_rename Paths to rename.
+        */
+       private static function rename_settings( &$settings, $paths_to_rename ) {
+               foreach ( $paths_to_rename as $original => $renamed ) {
+                       $original_path = explode( '.', $original );
+                       $renamed_path  = explode( '.', $renamed );
+                       $current_value = _wp_array_get( $settings, $original_path, null );
+
+                       if ( null !== $current_value ) {
+                               _wp_array_set( $settings, $renamed_path, $current_value );
+                               self::unset_setting_by_path( $settings, $original_path );
+                       }
+               }
+       }
+
+       /**
+        * Removes a property from within the provided settings by its path.
+        *
+        * @param array $settings Reference to the current settings array.
+        * @param array $path Path to the property to be removed.
+        *
+        * @return void
+        */
+       private static function unset_setting_by_path( &$settings, $path ) {
+               $tmp_settings = &$settings; // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable
+               $last_key     = array_pop( $path );
+               foreach ( $path as $key ) {
+                       $tmp_settings = &$tmp_settings[ $key ];
+               }
+
+               unset( $tmp_settings[ $last_key ] );
+       }
+}
</ins></span></pre></div>
<a id="trunksrcwpincludesclasswpthemejsonphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/class-wp-theme-json.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/class-wp-theme-json.php     2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/class-wp-theme-json.php       2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -10,6 +10,10 @@
</span><span class="cx" style="display: block; padding: 0 10px"> /**
</span><span class="cx" style="display: block; padding: 0 10px">  * Class that encapsulates the processing of structures that adhere to the theme.json spec.
</span><span class="cx" style="display: block; padding: 0 10px">  *
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ * This class is for internal core usage and is not supposed to be used by extenders (plugins and/or themes).
+ * This is a low-level API that may need to do breaking changes. Please,
+ * use get_global_settings, get_global_styles, and get_global_stylesheet instead.
+ *
</ins><span class="cx" style="display: block; padding: 0 10px">  * @access private
</span><span class="cx" style="display: block; padding: 0 10px">  */
</span><span class="cx" style="display: block; padding: 0 10px"> class WP_Theme_JSON {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -70,119 +74,153 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * This contains the necessary metadata to process them:
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * - path          => where to find the preset within the settings section
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * - path       => where to find the preset within the settings section
+        * - value_key  => the key that represents the value
+        * - value_func => the callback to render the value (either value_key or value_func should be present)
+        * - css_vars   => template string to use in generating the CSS Custom Property.
+        *                 Example output: "--wp--preset--duotone--blue: <value>" will generate as many CSS Custom Properties as presets defined
+        *                 substituting the $slug for the slug's value for each preset value.
+        * - classes    => array containing a structure with the classes to generate for the presets.
+        *                 Each key is a template string to resolve similarly to the css_vars and each value is the CSS property to use for that class.
+        *                 Example output: ".has-blue-color { color: <value> }"
+        * - properties  => a list of CSS properties to be used by kses to check the preset value is safe.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * - value_key     => the key that represents the value
-        *
-        * - css_var_infix => infix to use in generating the CSS Custom Property. Example:
-        *                   --wp--preset--<preset_infix>--<slug>: <preset_value>
-        *
-        * - classes      => array containing a structure with the classes to
-        *                   generate for the presets. Each class should have
-        *                   the class suffix and the property name. Example:
-        *
-        *                   .has-<slug>-<class_suffix> {
-        *                       <property_name>: <preset_value>
-        *                   }
-        *
</del><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added new presets and simplified the metadata structure.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var array
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="cx" style="display: block; padding: 0 10px">        const PRESETS_METADATA = array(
</span><span class="cx" style="display: block; padding: 0 10px">                array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'path'          => array( 'color', 'palette' ),
-                       'value_key'     => 'color',
-                       'css_var_infix' => 'color',
-                       'classes'       => array(
-                               array(
-                                       'class_suffix'  => 'color',
-                                       'property_name' => 'color',
-                               ),
-                               array(
-                                       'class_suffix'  => 'background-color',
-                                       'property_name' => 'background-color',
-                               ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'path'       => array( 'color', 'palette' ),
+                       'value_key'  => 'color',
+                       'css_vars'   => '--wp--preset--color--$slug',
+                       'classes'    => array(
+                               '.has-$slug-color'            => 'color',
+                               '.has-$slug-background-color' => 'background-color',
+                               '.has-$slug-border-color'     => 'border-color',
</ins><span class="cx" style="display: block; padding: 0 10px">                         ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'properties' => array( 'color', 'background-color', 'border-color' ),
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'path'          => array( 'color', 'gradients' ),
-                       'value_key'     => 'gradient',
-                       'css_var_infix' => 'gradient',
-                       'classes'       => array(
-                               array(
-                                       'class_suffix'  => 'gradient-background',
-                                       'property_name' => 'background',
-                               ),
-                       ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'path'       => array( 'color', 'gradients' ),
+                       'value_key'  => 'gradient',
+                       'css_vars'   => '--wp--preset--gradient--$slug',
+                       'classes'    => array( '.has-$slug-gradient-background' => 'background' ),
+                       'properties' => array( 'background' ),
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'path'          => array( 'typography', 'fontSizes' ),
-                       'value_key'     => 'size',
-                       'css_var_infix' => 'font-size',
-                       'classes'       => array(
-                               array(
-                                       'class_suffix'  => 'font-size',
-                                       'property_name' => 'font-size',
-                               ),
-                       ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'path'       => array( 'color', 'duotone' ),
+                       'value_func' => 'wp_render_duotone_filter_preset',
+                       'css_vars'   => '--wp--preset--duotone--$slug',
+                       'classes'    => array(),
+                       'properties' => array( 'filter' ),
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                array(
+                       'path'       => array( 'typography', 'fontSizes' ),
+                       'value_key'  => 'size',
+                       'css_vars'   => '--wp--preset--font-size--$slug',
+                       'classes'    => array( '.has-$slug-font-size' => 'font-size' ),
+                       'properties' => array( 'font-size' ),
+               ),
+               array(
+                       'path'       => array( 'typography', 'fontFamilies' ),
+                       'value_key'  => 'fontFamily',
+                       'css_vars'   => '--wp--preset--font-family--$slug',
+                       'classes'    => array( '.has-$slug-font-family' => 'font-family' ),
+                       'properties' => array( 'font-family' ),
+               ),
</ins><span class="cx" style="display: block; padding: 0 10px">         );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * Metadata for style properties.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Each property declares:
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * Each element is a direct mapping from the CSS property name to the
+        * path to the value in theme.json & block attributes.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * - 'value': path to the value in theme.json and block attributes.
-        *
</del><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added new properties and simplified the metadata structure.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var array
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="cx" style="display: block; padding: 0 10px">        const PROPERTIES_METADATA = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                'background'       => array(
-                       'value' => array( 'color', 'gradient' ),
-               ),
-               'background-color' => array(
-                       'value' => array( 'color', 'background' ),
-               ),
-               'color'            => array(
-                       'value' => array( 'color', 'text' ),
-               ),
-               'font-size'        => array(
-                       'value' => array( 'typography', 'fontSize' ),
-               ),
-               'line-height'      => array(
-                       'value' => array( 'typography', 'lineHeight' ),
-               ),
-               'margin'           => array(
-                       'value'      => array( 'spacing', 'margin' ),
-                       'properties' => array( 'top', 'right', 'bottom', 'left' ),
-               ),
-               'padding'          => array(
-                       'value'      => array( 'spacing', 'padding' ),
-                       'properties' => array( 'top', 'right', 'bottom', 'left' ),
-               ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         'background'                 => array( 'color', 'gradient' ),
+               'background-color'           => array( 'color', 'background' ),
+               'border-radius'              => array( 'border', 'radius' ),
+               'border-top-left-radius'     => array( 'border', 'radius', 'topLeft' ),
+               'border-top-right-radius'    => array( 'border', 'radius', 'topRight' ),
+               'border-bottom-left-radius'  => array( 'border', 'radius', 'bottomLeft' ),
+               'border-bottom-right-radius' => array( 'border', 'radius', 'bottomRight' ),
+               'border-color'               => array( 'border', 'color' ),
+               'border-width'               => array( 'border', 'width' ),
+               'border-style'               => array( 'border', 'style' ),
+               'color'                      => array( 'color', 'text' ),
+               'font-family'                => array( 'typography', 'fontFamily' ),
+               'font-size'                  => array( 'typography', 'fontSize' ),
+               'font-style'                 => array( 'typography', 'fontStyle' ),
+               'font-weight'                => array( 'typography', 'fontWeight' ),
+               'letter-spacing'             => array( 'typography', 'letterSpacing' ),
+               'line-height'                => array( 'typography', 'lineHeight' ),
+               'margin'                     => array( 'spacing', 'margin' ),
+               'margin-top'                 => array( 'spacing', 'margin', 'top' ),
+               'margin-right'               => array( 'spacing', 'margin', 'right' ),
+               'margin-bottom'              => array( 'spacing', 'margin', 'bottom' ),
+               'margin-left'                => array( 'spacing', 'margin', 'left' ),
+               'padding'                    => array( 'spacing', 'padding' ),
+               'padding-top'                => array( 'spacing', 'padding', 'top' ),
+               'padding-right'              => array( 'spacing', 'padding', 'right' ),
+               'padding-bottom'             => array( 'spacing', 'padding', 'bottom' ),
+               'padding-left'               => array( 'spacing', 'padding', 'left' ),
+               '--wp--style--block-gap'     => array( 'spacing', 'blockGap' ),
+               'text-decoration'            => array( 'typography', 'textDecoration' ),
+               'text-transform'             => array( 'typography', 'textTransform' ),
+               'filter'                     => array( 'filter', 'duotone' ),
</ins><span class="cx" style="display: block; padding: 0 10px">         );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Protected style properties.
+        *
+        * These style properties are only rendered if a setting enables it
+        * via a value other than `null`.
+        *
+        * Each element maps the style property to the corresponding theme.json
+        * setting key.
+        *
+        * @since 5.9.0
+        */
+       const PROTECTED_PROPERTIES = array(
+               'spacing.blockGap' => array( 'spacing', 'blockGap' ),
+       );
+
+       /**
+        * The top-level keys a theme.json can have.
+        *
</ins><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Renamed from ALLOWED_TOP_LEVEL_KEYS and added new values.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var string[]
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        const ALLOWED_TOP_LEVEL_KEYS = array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ const VALID_TOP_LEVEL_KEYS = array(
+               'customTemplates',
</ins><span class="cx" style="display: block; padding: 0 10px">                 'settings',
</span><span class="cx" style="display: block; padding: 0 10px">                'styles',
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                'templateParts',
</ins><span class="cx" style="display: block; padding: 0 10px">                 'version',
</span><span class="cx" style="display: block; padding: 0 10px">        );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * The valid properties under the settings key.
+        *
</ins><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Renamed from ALLOWED_SETTINGS, gained new properties, and renamed others according to the new schema.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var array
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        const ALLOWED_SETTINGS = array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ const VALID_SETTINGS = array(
</ins><span class="cx" style="display: block; padding: 0 10px">                 'border'     => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'customRadius' => null,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'color'  => null,
+                       'radius' => null,
+                       'style'  => null,
+                       'width'  => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                'color'      => array(
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'background'     => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                         'custom'         => null,
</span><span class="cx" style="display: block; padding: 0 10px">                        'customDuotone'  => null,
</span><span class="cx" style="display: block; padding: 0 10px">                        'customGradient' => null,
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -190,6 +228,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'gradients'      => null,
</span><span class="cx" style="display: block; padding: 0 10px">                        'link'           => null,
</span><span class="cx" style="display: block; padding: 0 10px">                        'palette'        => null,
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'text'           => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                'custom'     => null,
</span><span class="cx" style="display: block; padding: 0 10px">                'layout'     => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -197,25 +236,38 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'wideSize'    => null,
</span><span class="cx" style="display: block; padding: 0 10px">                ),
</span><span class="cx" style="display: block; padding: 0 10px">                'spacing'    => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'customMargin'  => null,
-                       'customPadding' => null,
-                       'units'         => null,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'blockGap' => null,
+                       'margin'   => null,
+                       'padding'  => null,
+                       'units'    => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'customFontSize'   => null,
-                       'customLineHeight' => null,
-                       'dropCap'          => null,
-                       'fontSizes'        => null,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'customFontSize' => null,
+                       'dropCap'        => null,
+                       'fontFamilies'   => null,
+                       'fontSizes'      => null,
+                       'fontStyle'      => null,
+                       'fontWeight'     => null,
+                       'letterSpacing'  => null,
+                       'lineHeight'     => null,
+                       'textDecoration' => null,
+                       'textTransform'  => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">        );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * The valid properties under the styles key.
+        *
</ins><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Renamed from ALLOWED_SETTINGS, gained new properties.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var array
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        const ALLOWED_STYLES = array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ const VALID_STYLES = array(
</ins><span class="cx" style="display: block; padding: 0 10px">                 'border'     => array(
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'color'  => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                         'radius' => null,
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'style'  => null,
+                       'width'  => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                'color'      => array(
</span><span class="cx" style="display: block; padding: 0 10px">                        'background' => null,
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -222,23 +274,23 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'gradient'   => null,
</span><span class="cx" style="display: block; padding: 0 10px">                        'text'       => null,
</span><span class="cx" style="display: block; padding: 0 10px">                ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                'filter'     => array(
+                       'duotone' => null,
+               ),
</ins><span class="cx" style="display: block; padding: 0 10px">                 'spacing'    => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'margin'  => array(
-                               'top'    => null,
-                               'right'  => null,
-                               'bottom' => null,
-                               'left'   => null,
-                       ),
-                       'padding' => array(
-                               'bottom' => null,
-                               'left'   => null,
-                               'right'  => null,
-                               'top'    => null,
-                       ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'margin'   => null,
+                       'padding'  => null,
+                       'blockGap' => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'fontSize'   => null,
-                       'lineHeight' => null,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'fontFamily'     => null,
+                       'fontSize'       => null,
+                       'fontStyle'      => null,
+                       'fontWeight'     => null,
+                       'letterSpacing'  => null,
+                       'lineHeight'     => null,
+                       'textDecoration' => null,
+                       'textTransform'  => null,
</ins><span class="cx" style="display: block; padding: 0 10px">                 ),
</span><span class="cx" style="display: block; padding: 0 10px">        );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -257,10 +309,13 @@
</span><span class="cx" style="display: block; padding: 0 10px">        );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * The latest version of the schema in use.
+        *
</ins><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Changed value.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @var int
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        const LATEST_SCHEMA = 1;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ const LATEST_SCHEMA = 2;
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * Constructor.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -276,18 +331,16 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        $origin = 'theme';
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                if ( ! isset( $theme_json['version'] ) || self::LATEST_SCHEMA !== $theme_json['version'] ) {
-                       $this->theme_json = array();
-                       return;
-               }
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->theme_json    = WP_Theme_JSON_Schema::migrate( $theme_json );
+               $valid_block_names   = array_keys( self::get_blocks_metadata() );
+               $valid_element_names = array_keys( self::ELEMENTS );
+               $this->theme_json    = self::sanitize( $this->theme_json, $valid_block_names, $valid_element_names );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->theme_json = self::sanitize( $theme_json );
-
</del><span class="cx" style="display: block; padding: 0 10px">                 // Internally, presets are keyed by origin.
</span><span class="cx" style="display: block; padding: 0 10px">                $nodes = self::get_setting_nodes( $this->theme_json );
</span><span class="cx" style="display: block; padding: 0 10px">                foreach ( $nodes as $node ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        foreach ( self::PRESETS_METADATA as $preset ) {
-                               $path   = array_merge( $node['path'], $preset['path'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 foreach ( self::PRESETS_METADATA as $preset_metadata ) {
+                               $path   = array_merge( $node['path'], $preset_metadata['path'] );
</ins><span class="cx" style="display: block; padding: 0 10px">                                 $preset = _wp_array_get( $this->theme_json, $path, null );
</span><span class="cx" style="display: block; padding: 0 10px">                                if ( null !== $preset ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                        _wp_array_set( $this->theme_json, $path, array( $origin => $preset ) );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -300,11 +353,14 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * Sanitizes the input according to the schemas.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Has new parameters.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $input Structure to sanitize.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param array $valid_block_names List of valid block names.
+        * @param array $valid_element_names List of valid element names.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return array The sanitized output.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private static function sanitize( $input ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private static function sanitize( $input, $valid_block_names, $valid_element_names ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $output = array();
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                if ( ! is_array( $input ) ) {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -311,31 +367,25 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        return $output;
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $allowed_top_level_keys = self::ALLOWED_TOP_LEVEL_KEYS;
-               $allowed_settings       = self::ALLOWED_SETTINGS;
-               $allowed_styles         = self::ALLOWED_STYLES;
-               $allowed_blocks         = array_keys( self::get_blocks_metadata() );
-               $allowed_elements       = array_keys( self::ELEMENTS );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $output = array_intersect_key( $input, array_flip( self::VALID_TOP_LEVEL_KEYS ) );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $output = array_intersect_key( $input, array_flip( $allowed_top_level_keys ) );
-
-               // Build the schema.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         // Build the schema based on valid block & element names.
</ins><span class="cx" style="display: block; padding: 0 10px">                 $schema                 = array();
</span><span class="cx" style="display: block; padding: 0 10px">                $schema_styles_elements = array();
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                foreach ( $allowed_elements as $element ) {
-                       $schema_styles_elements[ $element ] = $allowed_styles;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         foreach ( $valid_element_names as $element ) {
+                       $schema_styles_elements[ $element ] = self::VALID_STYLES;
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px">                $schema_styles_blocks   = array();
</span><span class="cx" style="display: block; padding: 0 10px">                $schema_settings_blocks = array();
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                foreach ( $allowed_blocks as $block ) {
-                       $schema_settings_blocks[ $block ]           = $allowed_settings;
-                       $schema_styles_blocks[ $block ]             = $allowed_styles;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         foreach ( $valid_block_names as $block ) {
+                       $schema_settings_blocks[ $block ]           = self::VALID_SETTINGS;
+                       $schema_styles_blocks[ $block ]             = self::VALID_STYLES;
</ins><span class="cx" style="display: block; padding: 0 10px">                         $schema_styles_blocks[ $block ]['elements'] = $schema_styles_elements;
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $schema['styles']             = $allowed_styles;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $schema['styles']             = self::VALID_STYLES;
</ins><span class="cx" style="display: block; padding: 0 10px">                 $schema['styles']['blocks']   = $schema_styles_blocks;
</span><span class="cx" style="display: block; padding: 0 10px">                $schema['styles']['elements'] = $schema_styles_elements;
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $schema['settings']           = $allowed_settings;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $schema['settings']           = self::VALID_SETTINGS;
</ins><span class="cx" style="display: block; padding: 0 10px">                 $schema['settings']['blocks'] = $schema_settings_blocks;
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                // Remove anything that's not present in the schema.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -360,7 +410,6 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                return $output;
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-
</del><span class="cx" style="display: block; padding: 0 10px">         /**
</span><span class="cx" style="display: block; padding: 0 10px">         * Returns the metadata for each block.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -377,14 +426,16 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *       'core/heading': {
</span><span class="cx" style="display: block; padding: 0 10px">         *         'selector': 'h1',
</span><span class="cx" style="display: block; padding: 0 10px">         *         'elements': {}
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         *       }
-        *       'core/group': {
-        *         'selector': '.wp-block-group',
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  *       },
+        *       'core/image': {
+        *         'selector': '.wp-block-image',
+        *         'duotone': 'img',
</ins><span class="cx" style="display: block; padding: 0 10px">          *         'elements': {}
</span><span class="cx" style="display: block; padding: 0 10px">         *       }
</span><span class="cx" style="display: block; padding: 0 10px">         *     }
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added duotone key with CSS selector.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @return array Block metadata.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -407,11 +458,16 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                self::$blocks_metadata[ $block_name ]['selector'] = '.wp-block-' . str_replace( '/', '-', str_replace( 'core/', '', $block_name ) );
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        /*
-                        * Assign defaults, then overwrite those that the block sets by itself.
-                        * If the block selector is compounded, will append the element to each
-                        * individual block selector.
-                        */
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 if (
+                               isset( $block_type->supports['color']['__experimentalDuotone'] ) &&
+                               is_string( $block_type->supports['color']['__experimentalDuotone'] )
+                       ) {
+                               self::$blocks_metadata[ $block_name ]['duotone'] = $block_type->supports['color']['__experimentalDuotone'];
+                       }
+
+                       // Assign defaults, then overwrite those that the block sets by itself.
+                       // If the block selector is compounded, will append the element to each
+                       // individual block selector.
</ins><span class="cx" style="display: block; padding: 0 10px">                         $block_selectors = explode( ',', self::$blocks_metadata[ $block_name ]['selector'] );
</span><span class="cx" style="display: block; padding: 0 10px">                        foreach ( self::ELEMENTS as $el_name => $el_selector ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                $element_selector = array();
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -493,28 +549,98 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * the theme.json structure this object represents.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Changed the arguments passed to the function.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * @param string $type Optional. Type of stylesheet we want. Accepts 'all',
-        *                     'block_styles', and 'css_variables'. Default 'all'.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * @param array $types    Types of styles to load. Will load all by default. It accepts:
+        *                         'variables': only the CSS Custom Properties for presets & custom ones.
+        *                         'styles': only the styles section in theme.json.
+        *                         'presets': only the classes for the presets.
+        * @param array $origins A list of origins to include. By default it includes self::VALID_ORIGINS.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return string Stylesheet.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        public function get_stylesheet( $type = 'all' ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ public function get_stylesheet( $types = array( 'variables', 'styles', 'presets' ), $origins = self::VALID_ORIGINS ) {
+               if ( is_string( $types ) ) {
+                       // Dispatch error and map old arguments to new ones.
+                       _deprecated_argument( __FUNCTION__, '5.9' );
+                       if ( 'block_styles' === $types ) {
+                               $types = array( 'styles', 'presets' );
+                       } elseif ( 'css_variables' === $types ) {
+                               $types = array( 'variables' );
+                       } else {
+                               $types = array( 'variables', 'styles', 'presets' );
+                       }
+               }
+
</ins><span class="cx" style="display: block; padding: 0 10px">                 $blocks_metadata = self::get_blocks_metadata();
</span><span class="cx" style="display: block; padding: 0 10px">                $style_nodes     = self::get_style_nodes( $this->theme_json, $blocks_metadata );
</span><span class="cx" style="display: block; padding: 0 10px">                $setting_nodes   = self::get_setting_nodes( $this->theme_json, $blocks_metadata );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                switch ( $type ) {
-                       case 'block_styles':
-                               return $this->get_block_styles( $style_nodes, $setting_nodes );
-                       case 'css_variables':
-                               return $this->get_css_variables( $setting_nodes );
-                       default:
-                               return $this->get_css_variables( $setting_nodes ) . $this->get_block_styles( $style_nodes, $setting_nodes );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $stylesheet = '';
+
+               if ( in_array( 'variables', $types, true ) ) {
+                       $stylesheet .= $this->get_css_variables( $setting_nodes, $origins );
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                if ( in_array( 'styles', $types, true ) ) {
+                       $stylesheet .= $this->get_block_classes( $style_nodes );
+               }
+
+               if ( in_array( 'presets', $types, true ) ) {
+                       $stylesheet .= $this->get_preset_classes( $setting_nodes, $origins );
+               }
+
+               return $stylesheet;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Returns the page templates of the current theme.
+        *
+        * @since 5.9.0
+        *
+        * @return array
+        */
+       public function get_custom_templates() {
+               $custom_templates = array();
+               if ( ! isset( $this->theme_json['customTemplates'] ) ) {
+                       return $custom_templates;
+               }
+
+               foreach ( $this->theme_json['customTemplates'] as $item ) {
+                       if ( isset( $item['name'] ) ) {
+                               $custom_templates[ $item['name'] ] = array(
+                                       'title'     => isset( $item['title'] ) ? $item['title'] : '',
+                                       'postTypes' => isset( $item['postTypes'] ) ? $item['postTypes'] : array( 'page' ),
+                               );
+                       }
+               }
+               return $custom_templates;
+       }
+
+       /**
+        * Returns the template part data of current theme.
+        *
+        * @since 5.9.0
+        *
+        * @return array
+        */
+       public function get_template_parts() {
+               $template_parts = array();
+               if ( ! isset( $this->theme_json['templateParts'] ) ) {
+                       return $template_parts;
+               }
+
+               foreach ( $this->theme_json['templateParts'] as $item ) {
+                       if ( isset( $item['name'] ) ) {
+                               $template_parts[ $item['name'] ] = array(
+                                       'title' => isset( $item['title'] ) ? $item['title'] : '',
+                                       'area'  => isset( $item['area'] ) ? $item['area'] : '',
+                               );
+                       }
+               }
+               return $template_parts;
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * Converts each style section into a list of rulesets
</span><span class="cx" style="display: block; padding: 0 10px">         * containing the block styles to be appended to the stylesheet.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -526,37 +652,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *     style-property-one: value;
</span><span class="cx" style="display: block; padding: 0 10px">         *   }
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Additionally, it'll also create new rulesets
-        * as classes for each preset value such as:
-        *
-        *     .has-value-color {
-        *       color: value;
-        *     }
-        *
-        *     .has-value-background-color {
-        *       background-color: value;
-        *     }
-        *
-        *     .has-value-font-size {
-        *       font-size: value;
-        *     }
-        *
-        *     .has-value-gradient-background {
-        *       background: value;
-        *     }
-        *
-        *     p.has-value-gradient-background {
-        *       background: value;
-        *     }
-        *
</del><span class="cx" style="display: block; padding: 0 10px">          * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Renamed to get_block_classes and no longer returns preset classes.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * @param array $style_nodes   Nodes with styles.
-        * @param array $setting_nodes Nodes with settings.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  * @param array $style_nodes Nodes with styles.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return string The new stylesheet.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private function get_block_styles( $style_nodes, $setting_nodes ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private function get_block_classes( $style_nodes ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $block_rules = '';
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
</ins><span class="cx" style="display: block; padding: 0 10px">                 foreach ( $style_nodes as $metadata ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( null === $metadata['selector'] ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                continue;
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -564,11 +668,77 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        $node         = _wp_array_get( $this->theme_json, $metadata['path'], array() );
</span><span class="cx" style="display: block; padding: 0 10px">                        $selector     = $metadata['selector'];
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        $declarations = self::compute_style_properties( $node );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 $settings     = _wp_array_get( $this->theme_json, array( 'settings' ) );
+                       $declarations = self::compute_style_properties( $node, $settings );
+
+                       // 1. Separate the ones who use the general selector
+                       // and the ones who use the duotone selector.
+                       $declarations_duotone = array();
+                       foreach ( $declarations as $index => $declaration ) {
+                               if ( 'filter' === $declaration['name'] ) {
+                                       unset( $declarations[ $index ] );
+                                       $declarations_duotone[] = $declaration;
+                               }
+                       }
+
+                       // 2. Generate the rules that use the general selector.
</ins><span class="cx" style="display: block; padding: 0 10px">                         $block_rules .= self::to_ruleset( $selector, $declarations );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+                       // 3. Generate the rules that use the duotone selector.
+                       if ( isset( $metadata['duotone'] ) && ! empty( $declarations_duotone ) ) {
+                               $selector_duotone = self::scope_selector( $metadata['selector'], $metadata['duotone'] );
+                               $block_rules     .= self::to_ruleset( $selector_duotone, $declarations_duotone );
+                       }
+
+                       if ( self::ROOT_BLOCK_SELECTOR === $selector ) {
+                               $block_rules .= 'body { margin: 0; }';
+                               $block_rules .= '.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }';
+                               $block_rules .= '.wp-site-blocks > .alignright { float: right; margin-left: 2em; }';
+                               $block_rules .= '.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }';
+
+                               $has_block_gap_support = _wp_array_get( $this->theme_json, array( 'settings', 'spacing', 'blockGap' ) ) !== null;
+                               if ( $has_block_gap_support ) {
+                                       $block_rules .= '.wp-site-blocks > * { margin-top: 0; margin-bottom: 0; }';
+                                       $block_rules .= '.wp-site-blocks > * + * { margin-top: var( --wp--style--block-gap ); }';
+                               }
+                       }
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                return $block_rules;
+       }
+
+       /**
+        * Creates new rulesets as classes for each preset value such as:
+        *
+        *   .has-value-color {
+        *     color: value;
+        *   }
+        *
+        *   .has-value-background-color {
+        *     background-color: value;
+        *   }
+        *
+        *   .has-value-font-size {
+        *     font-size: value;
+        *   }
+        *
+        *   .has-value-gradient-background {
+        *     background: value;
+        *   }
+        *
+        *   p.has-value-gradient-background {
+        *     background: value;
+        *   }
+        *
+        * @since 5.9.0
+        *
+        * @param array $setting_nodes Nodes with settings.
+        * @param array $origins       List of origins to process presets from.
+        * @return string The new stylesheet.
+        */
+       private function get_preset_classes( $setting_nodes, $origins ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $preset_rules = '';
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
</ins><span class="cx" style="display: block; padding: 0 10px">                 foreach ( $setting_nodes as $metadata ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( null === $metadata['selector'] ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                continue;
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -576,10 +746,10 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        $selector      = $metadata['selector'];
</span><span class="cx" style="display: block; padding: 0 10px">                        $node          = _wp_array_get( $this->theme_json, $metadata['path'], array() );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        $preset_rules .= self::compute_preset_classes( $node, $selector );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 $preset_rules .= self::compute_preset_classes( $node, $selector, $origins );
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                return $block_rules . $preset_rules;
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         return $preset_rules;
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -597,11 +767,13 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *     }
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added origins parameter.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $nodes Nodes with settings.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param array $origins List of origins to process.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return string The new stylesheet.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private function get_css_variables( $nodes ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private function get_css_variables( $nodes, $origins ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $stylesheet = '';
</span><span class="cx" style="display: block; padding: 0 10px">                foreach ( $nodes as $metadata ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( null === $metadata['selector'] ) {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -611,7 +783,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        $selector = $metadata['selector'];
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        $node         = _wp_array_get( $this->theme_json, $metadata['path'], array() );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        $declarations = array_merge( self::compute_preset_vars( $node ), self::compute_theme_vars( $node ) );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 $declarations = array_merge( self::compute_preset_vars( $node, $origins ), self::compute_theme_vars( $node ) );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        $stylesheet .= self::to_ruleset( $selector, $declarations );
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -668,45 +840,18 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Given an array of presets keyed by origin and the value key of the preset,
-        * it returns an array where each key is the preset slug and each value the preset value.
-        *
-        * @since 5.8.0
-        *
-        * @param array  $preset_per_origin Array of presets keyed by origin.
-        * @param string $value_key         The property of the preset that contains its value.
-        * @return array Array of presets where each key is a slug and each value is the preset value.
-        */
-       private static function get_merged_preset_by_slug( $preset_per_origin, $value_key ) {
-               $result = array();
-               foreach ( self::VALID_ORIGINS as $origin ) {
-                       if ( ! isset( $preset_per_origin[ $origin ] ) ) {
-                               continue;
-                       }
-                       foreach ( $preset_per_origin[ $origin ] as $preset ) {
-                               /*
-                                * We don't want to use kebabCase here,
-                                * see https://github.com/WordPress/gutenberg/issues/32347
-                                * However, we need to make sure the generated class or CSS variable
-                                * doesn't contain spaces.
-                                */
-                               $result[ preg_replace( '/\s+/', '-', $preset['slug'] ) ] = $preset[ $value_key ];
-                       }
-               }
-               return $result;
-       }
-
-       /**
</del><span class="cx" style="display: block; padding: 0 10px">          * Given a settings array, it returns the generated rulesets
</span><span class="cx" style="display: block; padding: 0 10px">         * for the preset classes.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added origins parameter.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array  $settings Settings to process.
</span><span class="cx" style="display: block; padding: 0 10px">         * @param string $selector Selector wrapping the classes.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param array  $origins  List of origins to process.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return string The result of processing the presets.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private static function compute_preset_classes( $settings, $selector ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private static function compute_preset_classes( $settings, $selector, $origins ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 if ( self::ROOT_BLOCK_SELECTOR === $selector ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        // Classes at the global level do not need any CSS prefixed,
</span><span class="cx" style="display: block; padding: 0 10px">                        // and we don't want to increase its specificity.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -714,17 +859,18 @@
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                $stylesheet = '';
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                foreach ( self::PRESETS_METADATA as $preset ) {
-                       $preset_per_origin = _wp_array_get( $settings, $preset['path'], array() );
-                       $preset_by_slug    = self::get_merged_preset_by_slug( $preset_per_origin, $preset['value_key'] );
-                       foreach ( $preset['classes'] as $class ) {
-                               foreach ( $preset_by_slug as $slug => $value ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         foreach ( self::PRESETS_METADATA as $preset_metadata ) {
+                       $slugs = self::get_settings_slugs( $settings, $preset_metadata, $origins );
+                       foreach ( $preset_metadata['classes'] as $class => $property ) {
+                               foreach ( $slugs as $slug ) {
+                                       $css_var     = self::replace_slug_in_string( $preset_metadata['css_vars'], $slug );
+                                       $class_name  = self::replace_slug_in_string( $class, $slug );
</ins><span class="cx" style="display: block; padding: 0 10px">                                         $stylesheet .= self::to_ruleset(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                self::append_to_selector( $selector, '.has-' . _wp_to_kebab_case( $slug ) . '-' . $class['class_suffix'] ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                         self::append_to_selector( $selector, $class_name ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                                'name'  => $class['property_name'],
-                                                               'value' => 'var(--wp--preset--' . $preset['css_var_infix'] . '--' . _wp_to_kebab_case( $slug ) . ') !important',
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                         'name'  => $property,
+                                                               'value' => 'var(' . $css_var . ') !important',
</ins><span class="cx" style="display: block; padding: 0 10px">                                                         ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                )
</span><span class="cx" style="display: block; padding: 0 10px">                                        );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -736,6 +882,147 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Function that scopes a selector with another one. This works a bit like
+        * SCSS nesting except the `&` operator isn't supported.
+        *
+        * <code>
+        * $scope = '.a, .b .c';
+        * $selector = '> .x, .y';
+        * $merged = scope_selector( $scope, $selector );
+        * // $merged is '.a > .x, .a .y, .b .c > .x, .b .c .y'
+        * </code>
+        *
+        * @since 5.9.0
+        *
+        * @param string $scope    Selector to scope to.
+        * @param string $selector Original selector.
+        *
+        * @return string Scoped selector.
+        */
+       private static function scope_selector( $scope, $selector ) {
+               $scopes    = explode( ',', $scope );
+               $selectors = explode( ',', $selector );
+
+               $selectors_scoped = array();
+               foreach ( $scopes as $outer ) {
+                       foreach ( $selectors as $inner ) {
+                               $selectors_scoped[] = trim( $outer ) . ' ' . trim( $inner );
+                       }
+               }
+
+               return implode( ', ', $selectors_scoped );
+       }
+
+       /**
+        * Gets preset values keyed by slugs based on settings and metadata.
+        *
+        * <code>
+        * $settings = array(
+        *     'typography' => array(
+        *         'fontFamilies' => array(
+        *             array(
+        *                 'slug'       => 'sansSerif',
+        *                 'fontFamily' => '"Helvetica Neue", sans-serif',
+        *             ),
+        *             array(
+        *                 'slug'   => 'serif',
+        *                 'colors' => 'Georgia, serif',
+        *             )
+        *         ),
+        *     ),
+        * );
+        * $meta = array(
+        *    'path'      => array( 'typography', 'fontFamilies' ),
+        *    'value_key' => 'fontFamily',
+        * );
+        * $values_by_slug = get_settings_values_by_slug();
+        * // $values_by_slug === array(
+        * //   'sans-serif' => '"Helvetica Neue", sans-serif',
+        * //   'serif'      => 'Georgia, serif',
+        * // );
+        * </code>
+        *
+        * @since 5.9.0
+        *
+        * @param array $settings Settings to process.
+        * @param array $preset_metadata One of the PRESETS_METADATA values.
+        * @param array $origins List of origins to process.
+        * @return array Array of presets where each key is a slug and each value is the preset value.
+        */
+       private static function get_settings_values_by_slug( $settings, $preset_metadata, $origins ) {
+               $preset_per_origin = _wp_array_get( $settings, $preset_metadata['path'], array() );
+
+               $result = array();
+               foreach ( $origins as $origin ) {
+                       if ( ! isset( $preset_per_origin[ $origin ] ) ) {
+                               continue;
+                       }
+                       foreach ( $preset_per_origin[ $origin ] as $preset ) {
+                               $slug = _wp_to_kebab_case( $preset['slug'] );
+
+                               $value = '';
+                               if ( isset( $preset_metadata['value_key'] ) ) {
+                                       $value_key = $preset_metadata['value_key'];
+                                       $value     = $preset[ $value_key ];
+                               } elseif (
+                                       isset( $preset_metadata['value_func'] ) &&
+                                       is_callable( $preset_metadata['value_func'] )
+                               ) {
+                                       $value_func = $preset_metadata['value_func'];
+                                       $value      = call_user_func( $value_func, $preset );
+                               } else {
+                                       // If we don't have a value, then don't add it to the result.
+                                       continue;
+                               }
+
+                               $result[ $slug ] = $value;
+                       }
+               }
+               return $result;
+       }
+
+       /**
+        * Similar to get_settings_values_by_slug, but doesn't compute the value.
+        *
+        * @since 5.9.0
+        *
+        * @param array $settings Settings to process.
+        * @param array $preset_metadata One of the PRESETS_METADATA values.
+        * @param array $origins List of origins to process.
+        * @return array Array of presets where the key and value are both the slug.
+        */
+       private static function get_settings_slugs( $settings, $preset_metadata, $origins = self::VALID_ORIGINS ) {
+               $preset_per_origin = _wp_array_get( $settings, $preset_metadata['path'], array() );
+
+               $result = array();
+               foreach ( $origins as $origin ) {
+                       if ( ! isset( $preset_per_origin[ $origin ] ) ) {
+                               continue;
+                       }
+                       foreach ( $preset_per_origin[ $origin ] as $preset ) {
+                               $slug = _wp_to_kebab_case( $preset['slug'] );
+
+                               // Use the array as a set so we don't get duplicates.
+                               $result[ $slug ] = $slug;
+                       }
+               }
+               return $result;
+       }
+
+       /**
+        * Transform a slug into a CSS Custom Property.
+        *
+        * @since 5.9.0
+        *
+        * @param string $input String to replace.
+        * @param string $slug The slug value to use to generate the custom property.
+        * @return string The CSS Custom Property. Something along the lines of --wp--preset--color--black.
+        */
+       private static function replace_slug_in_string( $input, $slug ) {
+               return strtr( $input, array( '$slug' => $slug ) );
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * Given the block settings, it extracts the CSS Custom Properties
</span><span class="cx" style="display: block; padding: 0 10px">         * for the presets and adds them to the $declarations array
</span><span class="cx" style="display: block; padding: 0 10px">         * following the format:
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -748,16 +1035,16 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $settings Settings to process.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param array $origins  List of origins to process.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return array Returns the modified $declarations.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private static function compute_preset_vars( $settings ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private static function compute_preset_vars( $settings, $origins ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $declarations = array();
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                foreach ( self::PRESETS_METADATA as $preset ) {
-                       $preset_per_origin = _wp_array_get( $settings, $preset['path'], array() );
-                       $preset_by_slug    = self::get_merged_preset_by_slug( $preset_per_origin, $preset['value_key'] );
-                       foreach ( $preset_by_slug as $slug => $value ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         foreach ( self::PRESETS_METADATA as $preset_metadata ) {
+                       $values_by_slug = self::get_settings_values_by_slug( $settings, $preset_metadata, $origins );
+                       foreach ( $values_by_slug as $slug => $value ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                                 $declarations[] = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'name'  => '--wp--preset--' . $preset['css_var_infix'] . '--' . _wp_to_kebab_case( $slug ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'name'  => self::replace_slug_in_string( $preset_metadata['css_vars'], $slug ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                         'value' => $value,
</span><span class="cx" style="display: block; padding: 0 10px">                                );
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -864,46 +1151,42 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *     )
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Added theme setting and properties parameters.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $styles Styles to process.
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @param array $settings Theme settings.
+        * @param array $properties Properties metadata.
</ins><span class="cx" style="display: block; padding: 0 10px">          * @return array Returns the modified $declarations.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        private static function compute_style_properties( $styles ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ private static function compute_style_properties( $styles, $settings = array(), $properties = self::PROPERTIES_METADATA ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                 $declarations = array();
</span><span class="cx" style="display: block; padding: 0 10px">                if ( empty( $styles ) ) {
</span><span class="cx" style="display: block; padding: 0 10px">                        return $declarations;
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $properties = array();
-               foreach ( self::PROPERTIES_METADATA as $name => $metadata ) {
-                       /*
-                        * Some properties can be shorthand properties, meaning that
-                        * they contain multiple values instead of a single one.
-                        * An example of this is the padding property.
-                        */
-                       if ( self::has_properties( $metadata ) ) {
-                               foreach ( $metadata['properties'] as $property ) {
-                                       $properties[] = array(
-                                               'name'  => $name . '-' . $property,
-                                               'value' => array_merge( $metadata['value'], array( $property ) ),
-                                       );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         foreach ( $properties as $css_property => $value_path ) {
+                       $value = self::get_property_value( $styles, $value_path );
+
+                       // Look up protected properties, keyed by value path.
+                       // Skip protected properties that are explicitly set to `null`.
+                       if ( is_array( $value_path ) ) {
+                               $path_string = implode( '.', $value_path );
+                               if (
+                                       array_key_exists( $path_string, self::PROTECTED_PROPERTIES ) &&
+                                       _wp_array_get( $settings, self::PROTECTED_PROPERTIES[ $path_string ], null ) === null
+                               ) {
+                                       continue;
</ins><span class="cx" style="display: block; padding: 0 10px">                                 }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        } else {
-                               $properties[] = array(
-                                       'name'  => $name,
-                                       'value' => $metadata['value'],
-                               );
</del><span class="cx" style="display: block; padding: 0 10px">                         }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                }
</del><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                foreach ( $properties as $prop ) {
-                       $value = self::get_property_value( $styles, $prop['value'] );
-                       if ( empty( $value ) ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 // Skip if empty and not "0" or value represents array of longhand values.
+                       $has_missing_value = empty( $value ) && ! is_numeric( $value );
+                       if ( $has_missing_value || is_array( $value ) ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                                 continue;
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        $declarations[] = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                'name'  => $prop['name'],
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                         'name'  => $css_property,
</ins><span class="cx" style="display: block; padding: 0 10px">                                 'value' => $value,
</span><span class="cx" style="display: block; padding: 0 10px">                        );
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -912,22 +1195,6 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         * Whether the metadata contains a key named properties.
-        *
-        * @since 5.8.0
-        *
-        * @param array $metadata Description of the style property.
-        * @return bool True if properties exists, false otherwise.
-        */
-       private static function has_properties( $metadata ) {
-               if ( array_key_exists( 'properties', $metadata ) ) {
-                       return true;
-               }
-
-               return false;
-       }
-
-       /**
</del><span class="cx" style="display: block; padding: 0 10px">          * Returns the style property for the given path.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * It also converts CSS Custom Property stored as
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -935,6 +1202,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * "--wp--preset--color--secondary".
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Consider $value that are arrays as well.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $styles Styles subtree.
</span><span class="cx" style="display: block; padding: 0 10px">         * @param array $path   Which property to process.
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -943,7 +1211,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">        private static function get_property_value( $styles, $path ) {
</span><span class="cx" style="display: block; padding: 0 10px">                $value = _wp_array_get( $styles, $path, '' );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                if ( '' === $value ) {
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         if ( '' === $value || is_array( $value ) ) {
</ins><span class="cx" style="display: block; padding: 0 10px">                         return $value;
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1015,7 +1283,6 @@
</span><span class="cx" style="display: block; padding: 0 10px">                return $nodes;
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-
</del><span class="cx" style="display: block; padding: 0 10px">         /**
</span><span class="cx" style="display: block; padding: 0 10px">         * Builds metadata for the style nodes, which returns in the form of:
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1022,11 +1289,13 @@
</span><span class="cx" style="display: block; padding: 0 10px">         *     [
</span><span class="cx" style="display: block; padding: 0 10px">         *       [
</span><span class="cx" style="display: block; padding: 0 10px">         *         'path'     => [ 'path', 'to', 'some', 'node' ],
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         *         'selector' => 'CSS selector for some node'
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  *         'selector' => 'CSS selector for some node',
+        *         'duotone'  => 'CSS selector for duotone for some node'
</ins><span class="cx" style="display: block; padding: 0 10px">          *       ],
</span><span class="cx" style="display: block; padding: 0 10px">         *       [
</span><span class="cx" style="display: block; padding: 0 10px">         *         'path'     => ['path', 'to', 'other', 'node' ],
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-         *         'selector' => 'CSS selector for other node'
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+  *         'selector' => 'CSS selector for other node',
+        *         'duotone'  => null
</ins><span class="cx" style="display: block; padding: 0 10px">          *       ],
</span><span class="cx" style="display: block; padding: 0 10px">         *     ]
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1068,9 +1337,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                $selector = $selectors[ $name ]['selector'];
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        $duotone_selector = null;
+                       if ( isset( $selectors[ $name ]['duotone'] ) ) {
+                               $duotone_selector = $selectors[ $name ]['duotone'];
+                       }
+
</ins><span class="cx" style="display: block; padding: 0 10px">                         $nodes[] = array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'path'     => array( 'styles', 'blocks', $name ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'selector' => $selector,
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                'duotone'  => $duotone_selector,
</ins><span class="cx" style="display: block; padding: 0 10px">                         );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( isset( $theme_json['styles']['blocks'][ $name ]['elements'] ) ) {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1090,6 +1365,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">         * Merge new incoming data.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @since 5.9.0 Duotone preset also has origins.
</ins><span class="cx" style="display: block; padding: 0 10px">          *
</span><span class="cx" style="display: block; padding: 0 10px">         * @param WP_Theme_JSON $incoming Data to merge.
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1098,14 +1374,14 @@
</span><span class="cx" style="display: block; padding: 0 10px">                $this->theme_json = array_replace_recursive( $this->theme_json, $incoming_data );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                /*
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                 * The array_replace_recursive() algorithm merges at the leaf level.
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+          * The array_replace_recursive algorithm merges at the leaf level.
</ins><span class="cx" style="display: block; padding: 0 10px">                  * For leaf values that are arrays it will use the numeric indexes for replacement.
</span><span class="cx" style="display: block; padding: 0 10px">                 * In those cases, we want to replace the existing with the incoming value, if it exists.
</span><span class="cx" style="display: block; padding: 0 10px">                 */
</span><span class="cx" style="display: block; padding: 0 10px">                $to_replace   = array();
</span><span class="cx" style="display: block; padding: 0 10px">                $to_replace[] = array( 'spacing', 'units' );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $to_replace[] = array( 'color', 'duotone' );
</del><span class="cx" style="display: block; padding: 0 10px">                 foreach ( self::VALID_ORIGINS as $origin ) {
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        $to_replace[] = array( 'color', 'duotone', $origin );
</ins><span class="cx" style="display: block; padding: 0 10px">                         $to_replace[] = array( 'color', 'palette', $origin );
</span><span class="cx" style="display: block; padding: 0 10px">                        $to_replace[] = array( 'color', 'gradients', $origin );
</span><span class="cx" style="display: block; padding: 0 10px">                        $to_replace[] = array( 'typography', 'fontSizes', $origin );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1122,9 +1398,167 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                }
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * Removes insecure data from theme.json.
+        *
+        * @since 5.9.0
+        *
+        * @param array $theme_json Structure to sanitize.
+        * @return array Sanitized structure.
+        */
+       public static function remove_insecure_properties( $theme_json ) {
+               $sanitized = array();
+
+               $theme_json = WP_Theme_JSON_Schema::migrate( $theme_json );
+
+               $valid_block_names   = array_keys( self::get_blocks_metadata() );
+               $valid_element_names = array_keys( self::ELEMENTS );
+               $theme_json          = self::sanitize( $theme_json, $valid_block_names, $valid_element_names );
+
+               $blocks_metadata = self::get_blocks_metadata();
+               $style_nodes     = self::get_style_nodes( $theme_json, $blocks_metadata );
+               foreach ( $style_nodes as $metadata ) {
+                       $input = _wp_array_get( $theme_json, $metadata['path'], array() );
+                       if ( empty( $input ) ) {
+                               continue;
+                       }
+
+                       $output = self::remove_insecure_styles( $input );
+                       if ( ! empty( $output ) ) {
+                               _wp_array_set( $sanitized, $metadata['path'], $output );
+                       }
+               }
+
+               $setting_nodes = self::get_setting_nodes( $theme_json );
+               foreach ( $setting_nodes as $metadata ) {
+                       $input = _wp_array_get( $theme_json, $metadata['path'], array() );
+                       if ( empty( $input ) ) {
+                               continue;
+                       }
+
+                       $output = self::remove_insecure_settings( $input );
+                       if ( ! empty( $output ) ) {
+                               _wp_array_set( $sanitized, $metadata['path'], $output );
+                       }
+               }
+
+               if ( empty( $sanitized['styles'] ) ) {
+                       unset( $theme_json['styles'] );
+               } else {
+                       $theme_json['styles'] = $sanitized['styles'];
+               }
+
+               if ( empty( $sanitized['settings'] ) ) {
+                       unset( $theme_json['settings'] );
+               } else {
+                       $theme_json['settings'] = $sanitized['settings'];
+               }
+
+               return $theme_json;
+       }
+
+       /**
+        * Processes a setting node and returns the same node
+        * without the insecure settings.
+        *
+        * @since 5.9.0
+        *
+        * @param array $input Node to process.
+        * @return array
+        */
+       private static function remove_insecure_settings( $input ) {
+               $output = array();
+               foreach ( self::PRESETS_METADATA as $preset_metadata ) {
+                       $presets = _wp_array_get( $input, $preset_metadata['path'], null );
+                       if ( null === $presets ) {
+                               continue;
+                       }
+
+                       $escaped_preset = array();
+                       foreach ( $presets as $preset ) {
+                               if (
+                                       esc_attr( esc_html( $preset['name'] ) ) === $preset['name'] &&
+                                       sanitize_html_class( $preset['slug'] ) === $preset['slug']
+                               ) {
+                                       $value = null;
+                                       if ( isset( $preset_metadata['value_key'] ) ) {
+                                               $value = $preset[ $preset_metadata['value_key'] ];
+                                       } elseif (
+                                               isset( $preset_metadata['value_func'] ) &&
+                                               is_callable( $preset_metadata['value_func'] )
+                                       ) {
+                                               $value = call_user_func( $preset_metadata['value_func'], $preset );
+                                       }
+
+                                       $preset_is_valid = true;
+                                       foreach ( $preset_metadata['properties'] as $property ) {
+                                               if ( ! self::is_safe_css_declaration( $property, $value ) ) {
+                                                       $preset_is_valid = false;
+                                                       break;
+                                               }
+                                       }
+
+                                       if ( $preset_is_valid ) {
+                                               $escaped_preset[] = $preset;
+                                       }
+                               }
+                       }
+
+                       if ( ! empty( $escaped_preset ) ) {
+                               _wp_array_set( $output, $preset_metadata['path'], $escaped_preset );
+                       }
+               }
+
+               return $output;
+       }
+
+       /**
+        * Processes a style node and returns the same node
+        * without the insecure styles.
+        *
+        * @since 5.9.0
+        *
+        * @param array $input Node to process.
+        * @return array
+        */
+       private static function remove_insecure_styles( $input ) {
+               $output       = array();
+               $declarations = self::compute_style_properties( $input );
+
+               foreach ( $declarations as $declaration ) {
+                       if ( self::is_safe_css_declaration( $declaration['name'], $declaration['value'] ) ) {
+                               $path = self::PROPERTIES_METADATA[ $declaration['name'] ];
+
+                               // Check the value isn't an array before adding so as to not
+                               // double up shorthand and longhand styles.
+                               $value = _wp_array_get( $input, $path, array() );
+                               if ( ! is_array( $value ) ) {
+                                       _wp_array_set( $output, $path, $value );
+                               }
+                       }
+               }
+               return $output;
+       }
+
+       /**
+        * Checks that a declaration provided by the user is safe.
+        *
+        * @since 5.9.0
+        *
+        * @param string $property_name Property name in a CSS declaration, i.e. the `color` in `color: red`.
+        * @param string $property_value Value in a CSS declaration, i.e. the `red` in `color: red`.
+        * @return boolean
+        */
+       private static function is_safe_css_declaration( $property_name, $property_value ) {
+               $style_to_validate = $property_name . ': ' . $property_value;
+               $filtered          = esc_html( safecss_filter_attr( $style_to_validate ) );
+               return ! empty( trim( $filtered ) );
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * Returns the raw data.
</span><span class="cx" style="display: block; padding: 0 10px">         *
</span><span class="cx" style="display: block; padding: 0 10px">         * @since 5.8.0
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1176,7 +1610,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( ! isset( $theme_settings['settings']['typography'] ) ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                $theme_settings['settings']['typography'] = array();
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        $theme_settings['settings']['typography']['customLineHeight'] = $settings['enableCustomLineHeight'];
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 $theme_settings['settings']['typography']['lineHeight'] = $settings['enableCustomLineHeight'];
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                if ( isset( $settings['enableCustomUnits'] ) ) {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1220,7 +1654,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        if ( ! isset( $theme_settings['settings']['spacing'] ) ) {
</span><span class="cx" style="display: block; padding: 0 10px">                                $theme_settings['settings']['spacing'] = array();
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        $theme_settings['settings']['spacing']['customPadding'] = $settings['enableCustomSpacing'];
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 $theme_settings['settings']['spacing']['padding'] = $settings['enableCustomSpacing'];
</ins><span class="cx" style="display: block; padding: 0 10px">                 }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                return $theme_settings;
</span></span></pre></div>
<a id="trunksrcwpincludesdefaultfiltersphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/default-filters.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/default-filters.php 2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/default-filters.php   2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -336,6 +336,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> add_action( 'init', 'check_theme_switched', 99 );
</span><span class="cx" style="display: block; padding: 0 10px"> add_action( 'init', array( 'WP_Block_Supports', 'init' ), 22 );
</span><span class="cx" style="display: block; padding: 0 10px"> add_action( 'switch_theme', array( 'WP_Theme_JSON_Resolver', 'clean_cached_data' ) );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+add_action( 'start_previewing_theme', array( 'WP_Theme_JSON_Resolver', 'clean_cached_data' ) );
</ins><span class="cx" style="display: block; padding: 0 10px"> add_action( 'after_switch_theme', '_wp_menus_changed' );
</span><span class="cx" style="display: block; padding: 0 10px"> add_action( 'after_switch_theme', '_wp_sidebars_changed' );
</span><span class="cx" style="display: block; padding: 0 10px"> add_action( 'wp_print_styles', 'print_emoji_styles' );
</span></span></pre></div>
<a id="trunksrcwpincludesksesphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/kses.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/kses.php    2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/kses.php      2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -2260,6 +2260,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-bottom-color',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-bottom-style',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-bottom-width',
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'border-bottom-right-radius',
+                       'border-bottom-left-radius',
</ins><span class="cx" style="display: block; padding: 0 10px">                         'border-left',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-left-color',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-left-style',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -2268,6 +2270,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-top-color',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-top-style',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-top-width',
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'border-top-left-radius',
+                       'border-top-right-radius',
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-spacing',
</span><span class="cx" style="display: block; padding: 0 10px">                        'border-collapse',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -2282,6 +2286,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'column-width',
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                        'color',
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'filter',
</ins><span class="cx" style="display: block; padding: 0 10px">                         'font',
</span><span class="cx" style="display: block; padding: 0 10px">                        'font-family',
</span><span class="cx" style="display: block; padding: 0 10px">                        'font-size',
</span></span></pre></div>
<a id="trunksrcwpincludesscriptloaderphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/script-loader.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/script-loader.php   2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/script-loader.php     2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -2321,8 +2321,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        if ( null === $stylesheet ) {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $settings   = get_default_block_editor_settings();
-               $theme_json = WP_Theme_JSON_Resolver::get_merged_data( $settings );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $theme_json = WP_Theme_JSON_Resolver::get_merged_data();
</ins><span class="cx" style="display: block; padding: 0 10px">                 $stylesheet = $theme_json->get_stylesheet();
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                if ( $can_use_cache ) {
</span></span></pre></div>
<a id="trunksrcwpincludesthemei18njson"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/theme-i18n.json</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/theme-i18n.json     2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/theme-i18n.json       2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -5,6 +5,11 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        {
</span><span class="cx" style="display: block; padding: 0 10px">                                                "name": "Font size name"
</span><span class="cx" style="display: block; padding: 0 10px">                                        }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                ],
+                               "fontFamilies": [
+                                       {
+                                               "name": "Font family name"
+                                       }
</ins><span class="cx" style="display: block; padding: 0 10px">                                 ]
</span><span class="cx" style="display: block; padding: 0 10px">                },
</span><span class="cx" style="display: block; padding: 0 10px">                "color": {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -31,6 +36,11 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                {
</span><span class="cx" style="display: block; padding: 0 10px">                                                        "name": "Font size name"
</span><span class="cx" style="display: block; padding: 0 10px">                                                }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                        ],
+                                       "fontFamilies": [
+                                               {
+                                                       "name": "Font family name"
+                                               }
</ins><span class="cx" style="display: block; padding: 0 10px">                                         ]
</span><span class="cx" style="display: block; padding: 0 10px">                                },
</span><span class="cx" style="display: block; padding: 0 10px">                                "color": {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -47,5 +57,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                }
</span><span class="cx" style="display: block; padding: 0 10px">                        }
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        }
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ },
+       "customTemplates": [
+               {
+                       "title": "Custom template name"
+               }
+       ],
+       "templateParts": [
+               {
+                       "title": "Template part name"
+               }
+       ]
</ins><span class="cx" style="display: block; padding: 0 10px"> }
</span></span></pre></div>
<a id="trunksrcwpincludesthemejson"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-includes/theme.json</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-includes/theme.json  2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-includes/theme.json    2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1,8 +1,11 @@
</span><span class="cx" style="display: block; padding: 0 10px"> {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-        "version": 1,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+ "version": 2,
</ins><span class="cx" style="display: block; padding: 0 10px">         "settings": {
</span><span class="cx" style="display: block; padding: 0 10px">                "border": {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        "customRadius": false
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 "color": false,
+                       "radius": false,
+                       "style": false,
+                       "width": false
</ins><span class="cx" style="display: block; padding: 0 10px">                 },
</span><span class="cx" style="display: block; padding: 0 10px">                "color": {
</span><span class="cx" style="display: block; padding: 0 10px">                        "custom": true,
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -9,6 +12,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        "customDuotone": true,
</span><span class="cx" style="display: block; padding: 0 10px">                        "customGradient": true,
</span><span class="cx" style="display: block; padding: 0 10px">                        "link": false,
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        "background": true,
+                       "text": true,
</ins><span class="cx" style="display: block; padding: 0 10px">                         "duotone": [
</span><span class="cx" style="display: block; padding: 0 10px">                                {
</span><span class="cx" style="display: block; padding: 0 10px">                                        "name":  "Dark grayscale" ,
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -177,14 +182,20 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        ]
</span><span class="cx" style="display: block; padding: 0 10px">                },
</span><span class="cx" style="display: block; padding: 0 10px">                "spacing": {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        "customMargin": false,
-                       "customPadding": false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 "blockGap": null,
+                       "margin": false,
+                       "padding": false,
</ins><span class="cx" style="display: block; padding: 0 10px">                         "units": [ "px", "em", "rem", "vh", "vw", "%" ]
</span><span class="cx" style="display: block; padding: 0 10px">                },
</span><span class="cx" style="display: block; padding: 0 10px">                "typography": {
</span><span class="cx" style="display: block; padding: 0 10px">                        "customFontSize": true,
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        "customLineHeight": false,
</del><span class="cx" style="display: block; padding: 0 10px">                         "dropCap": true,
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        "fontStyle": true,
+                       "fontWeight": true,
+                       "letterSpacing": true,
+                       "lineHeight": false,
+                       "textDecoration": true,
+                       "textTransform": true,
</ins><span class="cx" style="display: block; padding: 0 10px">                         "fontSizes": [
</span><span class="cx" style="display: block; padding: 0 10px">                                {
</span><span class="cx" style="display: block; padding: 0 10px">                                        "name": "Small",
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -216,9 +227,20 @@
</span><span class="cx" style="display: block; padding: 0 10px">                "blocks": {
</span><span class="cx" style="display: block; padding: 0 10px">                        "core/button": {
</span><span class="cx" style="display: block; padding: 0 10px">                                "border": {
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        "customRadius": true
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 "radius": true
</ins><span class="cx" style="display: block; padding: 0 10px">                                 }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        },
+                       "core/pullquote": {
+                               "border": {
+                                       "color": true,
+                                       "radius": true,
+                                       "style": true,
+                                       "width": true
+                               }
</ins><span class="cx" style="display: block; padding: 0 10px">                         }
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+        },
+       "styles": {
+               "spacing": { "blockGap": "24px" }
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> }
</span></span></pre></div>
<a id="trunksrcwpsettingsphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/src/wp-settings.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/src/wp-settings.php 2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/src/wp-settings.php   2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -170,6 +170,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/class-wp-date-query.php';
</span><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/theme.php';
</span><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/class-wp-theme.php';
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+require ABSPATH . WPINC . '/class-wp-theme-json-schema.php';
</ins><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/class-wp-theme-json.php';
</span><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/class-wp-theme-json-resolver.php';
</span><span class="cx" style="display: block; padding: 0 10px"> require ABSPATH . WPINC . '/class-wp-block-template.php';
</span></span></pre></div>
<a id="trunktestsphpunitdatalanguagesthemesblockthemepl_PLpo"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/tests/phpunit/data/languages/themes/block-theme-pl_PL.po</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/data/languages/themes/block-theme-pl_PL.po    2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/tests/phpunit/data/languages/themes/block-theme-pl_PL.po      2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -22,6 +22,10 @@
</span><span class="cx" style="display: block; padding: 0 10px"> msgid "Homepage template"
</span><span class="cx" style="display: block; padding: 0 10px"> msgstr "Szablon strony głównej"
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+msgctxt "Template part name"
+msgid "Small Header"
+msgstr "MaÅ‚y nagłówek"
+
</ins><span class="cx" style="display: block; padding: 0 10px"> msgctxt "Color name"
</span><span class="cx" style="display: block; padding: 0 10px"> msgid "Light"
</span><span class="cx" style="display: block; padding: 0 10px"> msgstr "Jasny"
</span></span></pre></div>
<a id="trunktestsphpunitdatathemedir1blockthemethemejson"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/tests/phpunit/data/themedir1/block-theme/theme.json</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/data/themedir1/block-theme/theme.json 2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/tests/phpunit/data/themedir1/block-theme/theme.json   2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -64,6 +64,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">        "templateParts": [
</span><span class="cx" style="display: block; padding: 0 10px">                {
</span><span class="cx" style="display: block; padding: 0 10px">                        "name": "small-header",
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        "title": "Small Header",
</ins><span class="cx" style="display: block; padding: 0 10px">                         "area": "header"
</span><span class="cx" style="display: block; padding: 0 10px">                }
</span><span class="cx" style="display: block; padding: 0 10px">        ]
</span></span></pre></div>
<a id="trunktestsphpunitdatathemedir1blockthemechildstylecss"></a>
<div class="addfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Added: trunk/tests/phpunit/data/themedir1/block-theme-child/style.css</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/data/themedir1/block-theme-child/style.css                            (rev 0)
+++ trunk/tests/phpunit/data/themedir1/block-theme-child/style.css      2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -0,0 +1,8 @@
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+/*
+Theme Name: Block Theme Child Theme
+Theme URI: https://wordpress.org/
+Description: For testing purposes only.
+Template: block-theme
+Version: 1.0.0
+Text Domain: block-theme-child
+*/
</ins></span></pre></div>
<a id="trunktestsphpunitdatathemedir1blockthemechildthemejson"></a>
<div class="addfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Added: trunk/tests/phpunit/data/themedir1/block-theme-child/theme.json</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/data/themedir1/block-theme-child/theme.json                           (rev 0)
+++ trunk/tests/phpunit/data/themedir1/block-theme-child/theme.json     2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -0,0 +1,51 @@
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+{
+       "version": 1,
+       "settings": {
+               "color": {
+                       "palette": [
+                               {
+                                       "slug": "light",
+                                       "name": "Light",
+                                       "color": "#f3f4f6"
+                               },
+                               {
+                                       "slug": "primary",
+                                       "name": "Primary",
+                                       "color": "#3858e9"
+                               },
+                               {
+                                       "slug": "dark",
+                                       "name": "Dark",
+                                       "color": "#111827"
+                               }
+                       ],
+                       "link": true
+               },
+               "blocks": {
+                       "core/post-title": {
+                               "color": {
+                                       "palette": [
+                                               {
+                                                       "slug": "light",
+                                                       "name": "Light",
+                                                       "color": "#f3f4f6"
+                                               }
+                                       ]
+                               }
+                       }
+               }
+       },
+       "customTemplates": [
+               {
+                       "name": "page-home",
+                       "title": "Homepage"
+               }
+       ],
+       "templateParts": [
+               {
+                       "name": "small-header",
+                       "title": "Small Header",
+                       "area": "header"
+               }
+       ]
+}
</ins></span></pre></div>
<a id="trunktestsphpunitteststhemethemeDirphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/tests/phpunit/tests/theme/themeDir.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/tests/theme/themeDir.php      2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/tests/phpunit/tests/theme/themeDir.php        2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -162,6 +162,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'camelCase',
</span><span class="cx" style="display: block; padding: 0 10px">                        'REST Theme',
</span><span class="cx" style="display: block; padding: 0 10px">                        'Block Theme',
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'Block Theme Child Theme',
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                sort( $theme_names );
</span></span></pre></div>
<a id="trunktestsphpunitteststhemewpThemeJsonphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/tests/phpunit/tests/theme/wpThemeJson.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/tests/theme/wpThemeJson.php   2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/tests/phpunit/tests/theme/wpThemeJson.php     2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -15,6 +15,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_settings() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -24,6 +25,10 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        'color'       => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'custom' => false,
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                        'layout'      => array(
+                                               'contentSize' => 'value',
+                                               'invalid/key' => 'value',
+                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                         'invalid/key' => 'value',
</span><span class="cx" style="display: block; padding: 0 10px">                                        'blocks'      => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'core/group' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -52,6 +57,9 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'color'  => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'custom' => false,
</span><span class="cx" style="display: block; padding: 0 10px">                        ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                        'layout' => array(
+                               'contentSize' => 'value',
+                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                         'blocks' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'core/group' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                        'color' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -61,7 +69,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSameSetsWithIndex( $expected, $actual );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEqualSetsWithIndex( $expected, $actual );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -193,7 +201,116 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
+        */
+       public function test_get_stylesheet_support_for_shorthand_and_longhand_values() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                               'styles'  => array(
+                                       'blocks' => array(
+                                               'core/group' => array(
+                                                       'border'  => array(
+                                                               'radius' => '10px',
+                                                       ),
+                                                       'spacing' => array(
+                                                               'padding' => '24px',
+                                                               'margin'  => '1em',
+                                                       ),
+                                               ),
+                                               'core/image' => array(
+                                                       'border'  => array(
+                                                               'radius' => array(
+                                                                       'topLeft'     => '10px',
+                                                                       'bottomRight' => '1em',
+                                                               ),
+                                                       ),
+                                                       'spacing' => array(
+                                                               'padding' => array(
+                                                                       'top' => '15px',
+                                                               ),
+                                                               'margin'  => array(
+                                                                       'bottom' => '30px',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $styles = 'body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }.wp-block-group{border-radius: 10px;margin: 1em;padding: 24px;}.wp-block-image{border-top-left-radius: 10px;border-bottom-right-radius: 1em;margin-bottom: 30px;padding-top: 15px;}';
+               $this->assertEquals( $styles, $theme_json->get_stylesheet() );
+               $this->assertEquals( $styles, $theme_json->get_stylesheet( array( 'styles' ) ) );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_get_stylesheet_skips_disabled_protected_properties() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                               'settings' => array(
+                                       'spacing' => array(
+                                               'blockGap' => null,
+                                       ),
+                               ),
+                               'styles'   => array(
+                                       'spacing' => array(
+                                               'blockGap' => '1em',
+                                       ),
+                                       'blocks'  => array(
+                                               'core/columns' => array(
+                                                       'spacing' => array(
+                                                               'blockGap' => '24px',
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $expected = 'body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }';
+               $this->assertEquals( $expected, $theme_json->get_stylesheet() );
+               $this->assertEquals( $expected, $theme_json->get_stylesheet( array( 'styles' ) ) );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_get_stylesheet_renders_enabled_protected_properties() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                               'settings' => array(
+                                       'spacing' => array(
+                                               'blockGap' => true,
+                                       ),
+                               ),
+                               'styles'   => array(
+                                       'spacing' => array(
+                                               'blockGap' => '1em',
+                                       ),
+                                       'blocks'  => array(
+                                               'core/columns' => array(
+                                                       'spacing' => array(
+                                                               'blockGap' => '24px',
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $expected = 'body{--wp--style--block-gap: 1em;}body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }.wp-site-blocks > * { margin-top: 0; margin-bottom: 0; }.wp-site-blocks > * + * { margin-top: var( --wp--style--block-gap ); }.wp-block-columns{--wp--style--block-gap: 24px;}';
+               $this->assertEquals( $expected, $theme_json->get_stylesheet() );
+               $this->assertEquals( $expected, $theme_json->get_stylesheet( array( 'styles' ) ) );
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * @ticket 53175
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_stylesheet() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -221,6 +338,9 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                        'spacing'    => array(
+                                               'blockGap' => false,
+                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                         'misc'       => 'value',
</span><span class="cx" style="display: block; padding: 0 10px">                                        'blocks'     => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'core/group' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -250,6 +370,9 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        'blocks'   => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'core/group'     => array(
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                        'border'   => array(
+                                                               'radius' => '10px',
+                                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                                         'elements' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                                'link' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                                        'color' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -258,10 +381,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                        'spacing'  => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                                'padding' => array(
-                                                                       'top'    => '12px',
-                                                                       'bottom' => '24px',
-                                                               ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                         'padding' => '24px',
</ins><span class="cx" style="display: block; padding: 0 10px">                                                         ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                'core/heading'   => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -293,6 +413,19 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                'core/image'     => array(
+                                                       'border'  => array(
+                                                               'radius' => array(
+                                                                       'topLeft'     => '10px',
+                                                                       'bottomRight' => '1em',
+                                                               ),
+                                                       ),
+                                                       'spacing' => array(
+                                                               'margin' => array(
+                                                                       'bottom' => '30px',
+                                                               ),
+                                                       ),
+                                               ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                         ),
</span><span class="cx" style="display: block; padding: 0 10px">                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'misc'     => 'value',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -299,22 +432,19 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSame(
-                       'body{--wp--preset--color--grey: grey;}.wp-block-group{--wp--custom--base-font: 16;--wp--custom--line-height--small: 1.2;--wp--custom--line-height--medium: 1.4;--wp--custom--line-height--large: 1.8;}body{color: var(--wp--preset--color--grey);}a{background-color: #333;color: #111;}.wp-block-group{padding-top: 12px;padding-bottom: 24px;}.wp-block-group a{color: #111;}h1,h2,h3,h4,h5,h6{color: #123456;}h1 a,h2 a,h3 a,h4 a,h5 a,h6 a{background-color: #333;color: #111;font-size: 60px;}.wp-block-post-date{color: #123456;}.wp-block-post-date a{background-color: #777;color: #555;}.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}',
-                       $theme_json->get_stylesheet()
-               );
-               $this->assertSame(
-                       'body{color: var(--wp--preset--color--grey);}a{background-color: #333;color: #111;}.wp-block-group{padding-top: 12px;padding-bottom: 24px;}.wp-block-group a{color: #111;}h1,h2,h3,h4,h5,h6{color: #123456;}h1 a,h2 a,h3 a,h4 a,h5 a,h6 a{background-color: #333;color: #111;font-size: 60px;}.wp-block-post-date{color: #123456;}.wp-block-post-date a{background-color: #777;color: #555;}.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}',
-                       $theme_json->get_stylesheet( 'block_styles' )
-               );
-               $this->assertSame(
-                       'body{--wp--preset--color--grey: grey;}.wp-block-group{--wp--custom--base-font: 16;--wp--custom--line-height--small: 1.2;--wp--custom--line-height--medium: 1.4;--wp--custom--line-height--large: 1.8;}',
-                       $theme_json->get_stylesheet( 'css_variables' )
-               );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $variables = 'body{--wp--preset--color--grey: grey;--wp--preset--font-family--small: 14px;--wp--preset--font-family--big: 41px;}.wp-block-group{--wp--custom--base-font: 16;--wp--custom--line-height--small: 1.2;--wp--custom--line-height--medium: 1.4;--wp--custom--line-height--large: 1.8;}';
+               $styles    = 'body{color: var(--wp--preset--color--grey);}body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }.wp-site-blocks > * { margin-top: 0; margin-bottom: 0; }.wp-site-blocks > * + * { margin-top: var( --wp--style--block-gap ); }a{background-color: #333;color: #111;}.wp-block-group{border-radius: 10px;padding: 24px;}.wp-block-group a{color: #111;}h1,h2,h3,h4,h5,h6{color: #123456;}h1 a,h2 a,h3 a,h4 a,h5 a,h6 a{background-color: #333;color: #111;font-size: 60px;}.wp-block-post-date{color: #123456;}.wp-block-post-date a{background-color: #777;color: #555;}.wp-block-image{border-top-left-radius: 10px;border-bottom-right-radius: 1em;margin-bottom: 30px;}';
+               $presets   = '.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}.has-grey-border-color{border-color: var(--wp--preset--color--grey) !important;}.has-small-font-family{font-family: var(--wp--preset--font-family--small) !important;}.has-big-font-family{font-family: var(--wp--preset--font-family--big) !important;}';
+               $all       = $variables . $styles . $presets;
+               $this->assertEquals( $all, $theme_json->get_stylesheet() );
+               $this->assertEquals( $styles, $theme_json->get_stylesheet( array( 'styles' ) ) );
+               $this->assertEquals( $presets, $theme_json->get_stylesheet( array( 'presets' ) ) );
+               $this->assertEquals( $variables, $theme_json->get_stylesheet( array( 'variables' ) ) );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_stylesheet_preset_classes_work_with_compounded_selectors() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -337,14 +467,15 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSame(
-                       'h1.has-white-color,h2.has-white-color,h3.has-white-color,h4.has-white-color,h5.has-white-color,h6.has-white-color{color: var(--wp--preset--color--white) !important;}h1.has-white-background-color,h2.has-white-background-color,h3.has-white-background-color,h4.has-white-background-color,h5.has-white-background-color,h6.has-white-background-color{background-color: var(--wp--preset--color--white) !important;}',
-                       $theme_json->get_stylesheet( 'block_styles' )
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEquals(
+                       'h1.has-white-color,h2.has-white-color,h3.has-white-color,h4.has-white-color,h5.has-white-color,h6.has-white-color{color: var(--wp--preset--color--white) !important;}h1.has-white-background-color,h2.has-white-background-color,h3.has-white-background-color,h4.has-white-background-color,h5.has-white-background-color,h6.has-white-background-color{background-color: var(--wp--preset--color--white) !important;}h1.has-white-border-color,h2.has-white-border-color,h3.has-white-border-color,h4.has-white-border-color,h5.has-white-border-color,h6.has-white-border-color{border-color: var(--wp--preset--color--white) !important;}',
+                       $theme_json->get_stylesheet( array( 'presets' ) )
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 53175
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_stylesheet_preset_rules_come_after_block_rules() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -376,18 +507,62 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        )
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSame(
-                       '.wp-block-group{--wp--preset--color--grey: grey;}.wp-block-group{color: red;}.wp-block-group.has-grey-color{color: var(--wp--preset--color--grey) !important;}.wp-block-group.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}',
-                       $theme_json->get_stylesheet()
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $styles    = 'body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }.wp-block-group{color: red;}';
+               $presets   = '.wp-block-group.has-grey-color{color: var(--wp--preset--color--grey) !important;}.wp-block-group.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}.wp-block-group.has-grey-border-color{border-color: var(--wp--preset--color--grey) !important;}';
+               $variables = '.wp-block-group{--wp--preset--color--grey: grey;}';
+               $all       = $variables . $styles . $presets;
+               $this->assertEquals( $all, $theme_json->get_stylesheet() );
+               $this->assertEquals( $styles, $theme_json->get_stylesheet( array( 'styles' ) ) );
+               $this->assertEquals( $presets, $theme_json->get_stylesheet( array( 'presets' ) ) );
+               $this->assertEquals( $variables, $theme_json->get_stylesheet( array( 'variables' ) ) );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_get_stylesheet_generates_proper_classes_from_slugs() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                               'settings' => array(
+                                       'color' => array(
+                                               'palette' => array(
+                                                       array(
+                                                               'slug'  => 'grey',
+                                                               'color' => 'grey',
+                                                       ),
+                                                       array(
+                                                               'slug'  => 'dark grey',
+                                                               'color' => 'grey',
+                                                       ),
+                                                       array(
+                                                               'slug'  => 'light-grey',
+                                                               'color' => 'grey',
+                                                       ),
+                                                       array(
+                                                               'slug'  => 'white2black',
+                                                               'color' => 'grey',
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSame(
-                       '.wp-block-group{color: red;}.wp-block-group.has-grey-color{color: var(--wp--preset--color--grey) !important;}.wp-block-group.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}',
-                       $theme_json->get_stylesheet( 'block_styles' )
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+
+               $this->assertEquals(
+                       '.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-dark-grey-color{color: var(--wp--preset--color--dark-grey) !important;}.has-light-grey-color{color: var(--wp--preset--color--light-grey) !important;}.has-white-2-black-color{color: var(--wp--preset--color--white-2-black) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}.has-dark-grey-background-color{background-color: var(--wp--preset--color--dark-grey) !important;}.has-light-grey-background-color{background-color: var(--wp--preset--color--light-grey) !important;}.has-white-2-black-background-color{background-color: var(--wp--preset--color--white-2-black) !important;}.has-grey-border-color{border-color: var(--wp--preset--color--grey) !important;}.has-dark-grey-border-color{border-color: var(--wp--preset--color--dark-grey) !important;}.has-light-grey-border-color{border-color: var(--wp--preset--color--light-grey) !important;}.has-white-2-black-border-color{bor
 der-color: var(--wp--preset--color--white-2-black) !important;}',
+                       $theme_json->get_stylesheet( array( 'presets' ) )
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                $this->assertEquals(
+                       'body{--wp--preset--color--grey: grey;--wp--preset--color--dark-grey: grey;--wp--preset--color--light-grey: grey;--wp--preset--color--white-2-black: grey;}',
+                       $theme_json->get_stylesheet( array( 'variables' ) )
+               );
+
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 53175
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_stylesheet_preset_values_are_marked_as_important() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -421,8 +596,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'core'
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSame(
-                       'body{--wp--preset--color--grey: grey;}p{background-color: blue;color: red;font-size: 12px;line-height: 1.3;}.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}',
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEquals(
+                       'body{--wp--preset--color--grey: grey;}body { margin: 0; }.wp-site-blocks > .alignleft { float: left; margin-right: 2em; }.wp-site-blocks > .alignright { float: right; margin-left: 2em; }.wp-site-blocks > .aligncenter { justify-content: center; margin-left: auto; margin-right: auto; }p{background-color: blue;color: red;font-size: 12px;line-height: 1.3;}.has-grey-color{color: var(--wp--preset--color--grey) !important;}.has-grey-background-color{background-color: var(--wp--preset--color--grey) !important;}.has-grey-border-color{border-color: var(--wp--preset--color--grey) !important;}',
</ins><span class="cx" style="display: block; padding: 0 10px">                         $theme_json->get_stylesheet()
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -429,6 +604,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_merge_incoming_data() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -608,7 +784,7 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'fontSizes' => array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'fontSizes'    => array(
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 'theme' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                                'slug' => 'fontSize',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -616,6 +792,14 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                        'fontFamilies' => array(
+                                               'theme' => array(
+                                                       array(
+                                                               'slug'       => 'fontFamily',
+                                                               'fontFamily' => 'fontFamily',
+                                                       ),
+                                               ),
+                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'blocks'     => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                        'core/paragraph' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -669,6 +853,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 53175
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_merge_incoming_data_empty_presets() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -736,7 +921,9 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        'version'  => WP_Theme_JSON::LATEST_SCHEMA,
</span><span class="cx" style="display: block; padding: 0 10px">                        'settings' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'color'      => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'duotone'   => array(),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'duotone'   => array(
+                                               'theme' => array(),
+                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                         'gradients' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'theme' => array(),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -760,6 +947,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 53175
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_merge_incoming_data_null_presets() {
</span><span class="cx" style="display: block; padding: 0 10px">                $theme_json = new WP_Theme_JSON(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -810,10 +998,10 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                        'custom' => false,
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                'spacing'    => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                        'customMargin' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                 'margin' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                        'customLineHeight' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                                 'lineHeight' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                )
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -827,9 +1015,11 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                'color'      => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                        'custom'    => false,
</span><span class="cx" style="display: block; padding: 0 10px">                                        'duotone'   => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                                array(
-                                                       'slug'   => 'value',
-                                                       'colors' => array( 'red', 'green' ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                         'theme' => array(
+                                                       array(
+                                                               'slug'   => 'value',
+                                                               'colors' => array( 'red', 'green' ),
+                                                       ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        'gradients' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -850,12 +1040,12 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'spacing'    => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'customMargin' => false,
-                                       'units'        => array( 'px', 'em' ),
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'margin' => false,
+                                       'units'  => array( 'px', 'em' ),
</ins><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'customLineHeight' => false,
-                                       'fontSizes'        => array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'lineHeight' => false,
+                                       'fontSizes'  => array(
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 'theme' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                                'slug'  => 'size',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -871,6 +1061,558 @@
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
+        */
+       public function test_remove_insecure_properties_removes_unsafe_styles() {
+               $actual = WP_Theme_JSON::remove_insecure_properties(
+                       array(
+                               'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                               'styles'  => array(
+                                       'color'    => array(
+                                               'gradient' => 'url(\'\')',
+                                               'text'     => 'var:preset|color|dark-red',
+                                       ),
+                                       'elements' => array(
+                                               'link' => array(
+                                                       'color' => array(
+                                                               'gradient'   => 'url(\'\')',
+                                                               'text'       => 'var:preset|color|dark-pink',
+                                                               'background' => 'var:preset|color|dark-red',
+                                                       ),
+                                               ),
+                                       ),
+                                       'blocks'   => array(
+                                               'core/image'  => array(
+                                                       'filter' => array(
+                                                               'duotone' => 'var:preset|duotone|blue-red',
+                                                       ),
+                                               ),
+                                               'core/cover'  => array(
+                                                       'filter' => array(
+                                                               'duotone' => 'var(--wp--preset--duotone--blue-red, var(--fallback-unsafe))',
+                                                       ),
+                                               ),
+                                               'core/group'  => array(
+                                                       'color'    => array(
+                                                               'gradient' => 'url(\'\')',
+                                                               'text'     => 'var:preset|color|dark-gray',
+                                                       ),
+                                                       'elements' => array(
+                                                               'link' => array(
+                                                                       'color' => array(
+                                                                               'gradient' => 'url(\'\')',
+                                                                               'text'     => 'var:preset|color|dark-pink',
+                                                                       ),
+                                                               ),
+                                                       ),
+                                               ),
+                                               'invalid/key' => array(
+                                                       'background' => 'green',
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $expected = array(
+                       'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                       'styles'  => array(
+                               'color'    => array(
+                                       'text' => 'var:preset|color|dark-red',
+                               ),
+                               'elements' => array(
+                                       'link' => array(
+                                               'color' => array(
+                                                       'text'       => 'var:preset|color|dark-pink',
+                                                       'background' => 'var:preset|color|dark-red',
+                                               ),
+                                       ),
+                               ),
+                               'blocks'   => array(
+                                       'core/image' => array(
+                                               'filter' => array(
+                                                       'duotone' => 'var:preset|duotone|blue-red',
+                                               ),
+                                       ),
+                                       'core/group' => array(
+                                               'color'    => array(
+                                                       'text' => 'var:preset|color|dark-gray',
+                                               ),
+                                               'elements' => array(
+                                                       'link' => array(
+                                                               'color' => array(
+                                                                       'text' => 'var:preset|color|dark-pink',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_remove_insecure_properties_removes_unsafe_styles_sub_properties() {
+               $actual = WP_Theme_JSON::remove_insecure_properties(
+                       array(
+                               'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                               'styles'  => array(
+                                       'border'   => array(
+                                               'radius' => array(
+                                                       'topLeft'     => '6px',
+                                                       'topRight'    => 'var(--top-right, var(--unsafe-fallback))',
+                                                       'bottomRight' => '6px',
+                                                       'bottomLeft'  => '6px',
+                                               ),
+                                       ),
+                                       'spacing'  => array(
+                                               'padding' => array(
+                                                       'top'    => '1px',
+                                                       'right'  => '1px',
+                                                       'bottom' => 'var(--bottom, var(--unsafe-fallback))',
+                                                       'left'   => '1px',
+                                               ),
+                                       ),
+                                       'elements' => array(
+                                               'link' => array(
+                                                       'spacing' => array(
+                                                               'padding' => array(
+                                                                       'top'    => '2px',
+                                                                       'right'  => '2px',
+                                                                       'bottom' => 'var(--bottom, var(--unsafe-fallback))',
+                                                                       'left'   => '2px',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                                       'blocks'   => array(
+                                               'core/group' => array(
+                                                       'border'   => array(
+                                                               'radius' => array(
+                                                                       'topLeft'     => '5px',
+                                                                       'topRight'    => 'var(--top-right, var(--unsafe-fallback))',
+                                                                       'bottomRight' => '5px',
+                                                                       'bottomLeft'  => '5px',
+                                                               ),
+                                                       ),
+                                                       'spacing'  => array(
+                                                               'padding' => array(
+                                                                       'top'    => '3px',
+                                                                       'right'  => '3px',
+                                                                       'bottom' => 'var(bottom, var(--unsafe-fallback))',
+                                                                       'left'   => '3px',
+                                                               ),
+                                                       ),
+                                                       'elements' => array(
+                                                               'link' => array(
+                                                                       'spacing' => array(
+                                                                               'padding' => array(
+                                                                                       'top'    => '4px',
+                                                                                       'right'  => '4px',
+                                                                                       'bottom' => 'var(--bottom, var(--unsafe-fallback))',
+                                                                                       'left'   => '4px',
+                                                                               ),
+                                                                       ),
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+                       true
+               );
+
+               $expected = array(
+                       'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                       'styles'  => array(
+                               'border'   => array(
+                                       'radius' => array(
+                                               'topLeft'     => '6px',
+                                               'bottomRight' => '6px',
+                                               'bottomLeft'  => '6px',
+                                       ),
+                               ),
+                               'spacing'  => array(
+                                       'padding' => array(
+                                               'top'   => '1px',
+                                               'right' => '1px',
+                                               'left'  => '1px',
+                                       ),
+                               ),
+                               'elements' => array(
+                                       'link' => array(
+                                               'spacing' => array(
+                                                       'padding' => array(
+                                                               'top'   => '2px',
+                                                               'right' => '2px',
+                                                               'left'  => '2px',
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                               'blocks'   => array(
+                                       'core/group' => array(
+                                               'border'   => array(
+                                                       'radius' => array(
+                                                               'topLeft'     => '5px',
+                                                               'bottomRight' => '5px',
+                                                               'bottomLeft'  => '5px',
+                                                       ),
+                                               ),
+                                               'spacing'  => array(
+                                                       'padding' => array(
+                                                               'top'   => '3px',
+                                                               'right' => '3px',
+                                                               'left'  => '3px',
+                                                       ),
+                                               ),
+                                               'elements' => array(
+                                                       'link' => array(
+                                                               'spacing' => array(
+                                                                       'padding' => array(
+                                                                               'top'   => '4px',
+                                                                               'right' => '4px',
+                                                                               'left'  => '4px',
+                                                                       ),
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_remove_insecure_properties_removes_non_preset_settings() {
+               $actual = WP_Theme_JSON::remove_insecure_properties(
+                       array(
+                               'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                               'settings' => array(
+                                       'color'   => array(
+                                               'custom'  => true,
+                                               'palette' => array(
+                                                       array(
+                                                               'name'  => 'Red',
+                                                               'slug'  => 'red',
+                                                               'color' => '#ff0000',
+                                                       ),
+                                                       array(
+                                                               'name'  => 'Green',
+                                                               'slug'  => 'green',
+                                                               'color' => '#00ff00',
+                                                       ),
+                                                       array(
+                                                               'name'  => 'Blue',
+                                                               'slug'  => 'blue',
+                                                               'color' => '#0000ff',
+                                                       ),
+                                               ),
+                                       ),
+                                       'spacing' => array(
+                                               'padding' => false,
+                                       ),
+                                       'blocks'  => array(
+                                               'core/group' => array(
+                                                       'color'   => array(
+                                                               'custom'  => true,
+                                                               'palette' => array(
+                                                                       array(
+                                                                               'name'  => 'Yellow',
+                                                                               'slug'  => 'yellow',
+                                                                               'color' => '#ff0000',
+                                                                       ),
+                                                                       array(
+                                                                               'name'  => 'Pink',
+                                                                               'slug'  => 'pink',
+                                                                               'color' => '#00ff00',
+                                                                       ),
+                                                                       array(
+                                                                               'name'  => 'Orange',
+                                                                               'slug'  => 'orange',
+                                                                               'color' => '#0000ff',
+                                                                       ),
+                                                               ),
+                                                       ),
+                                                       'spacing' => array(
+                                                               'padding' => false,
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $expected = array(
+                       'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                       'settings' => array(
+                               'color'  => array(
+                                       'palette' => array(
+                                               array(
+                                                       'name'  => 'Red',
+                                                       'slug'  => 'red',
+                                                       'color' => '#ff0000',
+                                               ),
+                                               array(
+                                                       'name'  => 'Green',
+                                                       'slug'  => 'green',
+                                                       'color' => '#00ff00',
+                                               ),
+                                               array(
+                                                       'name'  => 'Blue',
+                                                       'slug'  => 'blue',
+                                                       'color' => '#0000ff',
+                                               ),
+                                       ),
+                               ),
+                               'blocks' => array(
+                                       'core/group' => array(
+                                               'color' => array(
+                                                       'palette' => array(
+                                                               array(
+                                                                       'name'  => 'Yellow',
+                                                                       'slug'  => 'yellow',
+                                                                       'color' => '#ff0000',
+                                                               ),
+                                                               array(
+                                                                       'name'  => 'Pink',
+                                                                       'slug'  => 'pink',
+                                                                       'color' => '#00ff00',
+                                                               ),
+                                                               array(
+                                                                       'name'  => 'Orange',
+                                                                       'slug'  => 'orange',
+                                                                       'color' => '#0000ff',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_remove_insecure_properties_removes_unsafe_preset_settings() {
+               $actual = WP_Theme_JSON::remove_insecure_properties(
+                       array(
+                               'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                               'settings' => array(
+                                       'color'      => array(
+                                               'palette' => array(
+                                                       array(
+                                                               'name'  => 'Red/><b>ok</ok>',
+                                                               'slug'  => 'red',
+                                                               'color' => '#ff0000',
+                                                       ),
+                                                       array(
+                                                               'name'  => 'Green',
+                                                               'slug'  => 'a" attr',
+                                                               'color' => '#00ff00',
+                                                       ),
+                                                       array(
+                                                               'name'  => 'Blue',
+                                                               'slug'  => 'blue',
+                                                               'color' => 'var(--color, var(--unsafe-fallback))',
+                                                       ),
+                                                       array(
+                                                               'name'  => 'Pink',
+                                                               'slug'  => 'pink',
+                                                               'color' => '#FFC0CB',
+                                                       ),
+                                               ),
+                                       ),
+                                       'typography' => array(
+                                               'fontFamilies' => array(
+                                                       array(
+                                                               'name'       => 'Helvetica Arial/><b>test</b>',
+                                                               'slug'       => 'helvetica-arial',
+                                                               'fontFamily' => 'Helvetica Neue, Helvetica, Arial, sans-serif',
+                                                       ),
+                                                       array(
+                                                               'name'       => 'Geneva',
+                                                               'slug'       => 'geneva#asa',
+                                                               'fontFamily' => 'Geneva, Tahoma, Verdana, sans-serif',
+                                                       ),
+                                                       array(
+                                                               'name'       => 'Cambria',
+                                                               'slug'       => 'cambria',
+                                                               'fontFamily' => 'Cambria, Georgia, serif',
+                                                       ),
+                                                       array(
+                                                               'name'       => 'Helvetica Arial',
+                                                               'slug'       => 'helvetica-arial',
+                                                               'fontFamily' => 'var(--fontFamily, var(--unsafe-fallback))',
+                                                       ),
+                                               ),
+                                       ),
+                                       'blocks'     => array(
+                                               'core/group' => array(
+                                                       'color' => array(
+                                                               'palette' => array(
+                                                                       array(
+                                                                               'name'  => 'Red/><b>ok</ok>',
+                                                                               'slug'  => 'red',
+                                                                               'color' => '#ff0000',
+                                                                       ),
+                                                                       array(
+                                                                               'name'  => 'Green',
+                                                                               'slug'  => 'a" attr',
+                                                                               'color' => '#00ff00',
+                                                                       ),
+                                                                       array(
+                                                                               'name'  => 'Blue',
+                                                                               'slug'  => 'blue',
+                                                                               'color' => 'var(--color, var(--unsafe--fallback))',
+                                                                       ),
+                                                                       array(
+                                                                               'name'  => 'Pink',
+                                                                               'slug'  => 'pink',
+                                                                               'color' => '#FFC0CB',
+                                                                       ),
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       )
+               );
+
+               $expected = array(
+                       'version'  => WP_Theme_JSON::LATEST_SCHEMA,
+                       'settings' => array(
+                               'color'      => array(
+                                       'palette' => array(
+                                               array(
+                                                       'name'  => 'Pink',
+                                                       'slug'  => 'pink',
+                                                       'color' => '#FFC0CB',
+                                               ),
+                                       ),
+                               ),
+                               'typography' => array(
+                                       'fontFamilies' => array(
+                                               array(
+                                                       'name'       => 'Cambria',
+                                                       'slug'       => 'cambria',
+                                                       'fontFamily' => 'Cambria, Georgia, serif',
+                                               ),
+                                       ),
+                               ),
+                               'blocks'     => array(
+                                       'core/group' => array(
+                                               'color' => array(
+                                                       'palette' => array(
+                                                               array(
+                                                                       'name'  => 'Pink',
+                                                                       'slug'  => 'pink',
+                                                                       'color' => '#FFC0CB',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_remove_insecure_properties_applies_safe_styles() {
+               $actual = WP_Theme_JSON::remove_insecure_properties(
+                       array(
+                               'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                               'styles'  => array(
+                                       'color' => array(
+                                               'text' => '#abcabc ', // Trailing space.
+                                       ),
+                               ),
+                       ),
+                       true
+               );
+
+               $expected = array(
+                       'version' => WP_Theme_JSON::LATEST_SCHEMA,
+                       'styles'  => array(
+                               'color' => array(
+                                       'text' => '#abcabc ',
+                               ),
+                       ),
+               );
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_get_custom_templates() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version'         => 1,
+                               'customTemplates' => array(
+                                       array(
+                                               'name'  => 'page-home',
+                                               'title' => 'Homepage template',
+                                       ),
+                               ),
+                       )
+               );
+
+               $page_templates = $theme_json->get_custom_templates();
+
+               $this->assertEqualSetsWithIndex(
+                       $page_templates,
+                       array(
+                               'page-home' => array(
+                                       'title'     => 'Homepage template',
+                                       'postTypes' => array( 'page' ),
+                               ),
+                       )
+               );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       public function test_get_template_parts() {
+               $theme_json = new WP_Theme_JSON(
+                       array(
+                               'version'       => 1,
+                               'templateParts' => array(
+                                       array(
+                                               'name'  => 'small-header',
+                                               'title' => 'Small Header',
+                                               'area'  => 'header',
+                                       ),
+                               ),
+                       )
+               );
+
+               $template_parts = $theme_json->get_template_parts();
+
+               $this->assertEqualSetsWithIndex(
+                       $template_parts,
+                       array(
+                               'small-header' => array(
+                                       'title' => 'Small Header',
+                                       'area'  => 'header',
+                               ),
+                       )
+               );
+       }
+
+       /**
</ins><span class="cx" style="display: block; padding: 0 10px">          * @ticket 52991
</span><span class="cx" style="display: block; padding: 0 10px">         */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_from_editor_settings() {
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -928,9 +1670,9 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        'units' => array( 'px', 'em', 'rem', 'vh', 'vw', '%' ),
</span><span class="cx" style="display: block; padding: 0 10px">                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'customFontSize'   => false,
-                                       'customLineHeight' => true,
-                                       'fontSizes'        => array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'customFontSize' => false,
+                                       'lineHeight'     => true,
+                                       'fontSizes'      => array(
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        'slug' => 'size-slug',
</span><span class="cx" style="display: block; padding: 0 10px">                                                        'name' => 'Size Name',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -948,6 +1690,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_editor_settings_no_theme_support() {
</span><span class="cx" style="display: block; padding: 0 10px">                $input = array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -990,8 +1733,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                        'units' => false,
</span><span class="cx" style="display: block; padding: 0 10px">                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'customFontSize'   => true,
-                                       'customLineHeight' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'customFontSize' => true,
+                                       'lineHeight'     => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1003,6 +1746,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_editor_settings_blank() {
</span><span class="cx" style="display: block; padding: 0 10px">                $expected = array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -1011,58 +1755,57 @@
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><span class="cx" style="display: block; padding: 0 10px">                $actual   = WP_Theme_JSON::get_from_editor_settings( array() );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $this->assertSameSetsWithIndex( $expected, $actual );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEqualSetsWithIndex( $expected, $actual );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_editor_settings_custom_units_can_be_disabled() {
</span><span class="cx" style="display: block; padding: 0 10px">                add_theme_support( 'custom-units', array() );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $input = get_default_block_editor_settings();
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $actual = WP_Theme_JSON::get_from_editor_settings( get_default_block_editor_settings() );
+               remove_theme_support( 'custom-units' );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                $expected = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'units'         => array( array() ),
-                       'customPadding' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'units'   => array( array() ),
+                       'padding' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $actual = WP_Theme_JSON::get_from_editor_settings( $input );
-
-               $this->assertSameSetsWithIndex( $expected, $actual['settings']['spacing'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEqualSetsWithIndex( $expected, $actual['settings']['spacing'] );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_editor_settings_custom_units_can_be_enabled() {
</span><span class="cx" style="display: block; padding: 0 10px">                add_theme_support( 'custom-units' );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $input = get_default_block_editor_settings();
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $actual = WP_Theme_JSON::get_from_editor_settings( get_default_block_editor_settings() );
+               remove_theme_support( 'custom-units' );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                $expected = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'units'         => array( 'px', 'em', 'rem', 'vh', 'vw', '%' ),
-                       'customPadding' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'units'   => array( 'px', 'em', 'rem', 'vh', 'vw', '%' ),
+                       'padding' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $actual = WP_Theme_JSON::get_from_editor_settings( $input );
-
-               $this->assertSameSetsWithIndex( $expected, $actual['settings']['spacing'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEqualSetsWithIndex( $expected, $actual['settings']['spacing'] );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_get_editor_settings_custom_units_can_be_filtered() {
</span><span class="cx" style="display: block; padding: 0 10px">                add_theme_support( 'custom-units', 'rem', 'em' );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                $input = get_default_block_editor_settings();
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $actual = WP_Theme_JSON::get_from_editor_settings( get_default_block_editor_settings() );
+               remove_theme_support( 'custom-units' );
</ins><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                $expected = array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                        'units'         => array( 'rem', 'em' ),
-                       'customPadding' => false,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                 'units'   => array( 'rem', 'em' ),
+                       'padding' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                 );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-
-               $actual = WP_Theme_JSON::get_from_editor_settings( $input );
-
-               $this->assertSameSetsWithIndex( $expected, $actual['settings']['spacing'] );
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         $this->assertEqualSetsWithIndex( $expected, $actual['settings']['spacing'] );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px"> }
</span></span></pre></div>
<a id="trunktestsphpunitteststhemewpThemeJsonResolverphp"></a>
<div class="modfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Modified: trunk/tests/phpunit/tests/theme/wpThemeJsonResolver.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/tests/theme/wpThemeJsonResolver.php   2021-11-08 17:07:49 UTC (rev 52048)
+++ trunk/tests/phpunit/tests/theme/wpThemeJsonResolver.php     2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -46,6 +46,7 @@
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="cx" style="display: block; padding: 0 10px">         * @ticket 52991
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+         * @ticket 54336
</ins><span class="cx" style="display: block; padding: 0 10px">          */
</span><span class="cx" style="display: block; padding: 0 10px">        public function test_translations_are_applied() {
</span><span class="cx" style="display: block; padding: 0 10px">                add_filter( 'locale', array( $this, 'filter_set_locale_to_polish' ) );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -52,7 +53,6 @@
</span><span class="cx" style="display: block; padding: 0 10px">                load_textdomain( 'block-theme', realpath( DIR_TESTDATA . '/languages/themes/block-theme-pl_PL.mo' ) );
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                switch_theme( 'block-theme' );
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-
</del><span class="cx" style="display: block; padding: 0 10px">                 $actual = WP_Theme_JSON_Resolver::get_theme_data();
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">                unload_textdomain( 'block-theme' );
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -62,6 +62,8 @@
</span><span class="cx" style="display: block; padding: 0 10px">                $this->assertSame(
</span><span class="cx" style="display: block; padding: 0 10px">                        array(
</span><span class="cx" style="display: block; padding: 0 10px">                                'color'      => array(
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                        'custom'         => false,
+                                       'customGradient' => false,
</ins><span class="cx" style="display: block; padding: 0 10px">                                         'palette'        => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                'theme' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -85,11 +87,11 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'custom'         => false,
-                                       'customGradient' => false,
</del><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'typography' => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'fontSizes'        => array(
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'customFontSize' => false,
+                                       'lineHeight'     => true,
+                                       'fontSizes'      => array(
</ins><span class="cx" style="display: block; padding: 0 10px">                                                 'theme' => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                        array(
</span><span class="cx" style="display: block; padding: 0 10px">                                                                'name' => 'Custom',
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -98,14 +100,10 @@
</span><span class="cx" style="display: block; padding: 0 10px">                                                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                                                ),
</span><span class="cx" style="display: block; padding: 0 10px">                                        ),
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'customFontSize'   => false,
-                                       'customLineHeight' => true,
</del><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'spacing'    => array(
</span><del style="background-color: #fdd; text-decoration:none; display:block; padding: 0 10px">-                                        'units'         => array(
-                                               'rem',
-                                       ),
-                                       'customPadding' => true,
</del><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                                 'units'   => array( 'rem' ),
+                                       'padding' => true,
</ins><span class="cx" style="display: block; padding: 0 10px">                                 ),
</span><span class="cx" style="display: block; padding: 0 10px">                                'blocks'     => array(
</span><span class="cx" style="display: block; padding: 0 10px">                                        'core/paragraph' => array(
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -125,6 +123,24 @@
</span><span class="cx" style="display: block; padding: 0 10px">                        ),
</span><span class="cx" style="display: block; padding: 0 10px">                        $actual->get_settings()
</span><span class="cx" style="display: block; padding: 0 10px">                );
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+                $this->assertSame(
+                       $actual->get_custom_templates(),
+                       array(
+                               'page-home' => array(
+                                       'title'     => 'Szablon strony głównej',
+                                       'postTypes' => array( 'page' ),
+                               ),
+                       )
+               );
+               $this->assertSame(
+                       $actual->get_template_parts(),
+                       array(
+                               'small-header' => array(
+                                       'title' => 'MaÅ‚y nagłówek',
+                                       'area'  => 'header',
+                               ),
+                       )
+               );
</ins><span class="cx" style="display: block; padding: 0 10px">         }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><span class="cx" style="display: block; padding: 0 10px">        /**
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -143,4 +159,157 @@
</span><span class="cx" style="display: block; padding: 0 10px">                $this->assertTrue( $has_theme_json_support );
</span><span class="cx" style="display: block; padding: 0 10px">        }
</span><span class="cx" style="display: block; padding: 0 10px"> 
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+        /**
+        * @ticket 54336
+        */
+       function test_add_theme_supports_are_loaded_for_themes_without_theme_json() {
+               switch_theme( 'default' );
+               $color_palette = array(
+                       array(
+                               'name'  => 'Primary',
+                               'slug'  => 'primary',
+                               'color' => '#F00',
+                       ),
+                       array(
+                               'name'  => 'Secondary',
+                               'slug'  => 'secondary',
+                               'color' => '#0F0',
+                       ),
+                       array(
+                               'name'  => 'Tertiary',
+                               'slug'  => 'tertiary',
+                               'color' => '#00F',
+                       ),
+               );
+               add_theme_support( 'editor-color-palette', $color_palette );
+               add_theme_support( 'custom-line-height' );
+
+               $settings = WP_Theme_JSON_Resolver::get_theme_data()->get_settings();
+
+               remove_theme_support( 'custom-line-height' );
+               remove_theme_support( 'editor-color-palette' );
+
+               $this->assertFalse( WP_Theme_JSON_Resolver::theme_has_support() );
+               $this->assertTrue( $settings['typography']['lineHeight'] );
+               $this->assertSame( $color_palette, $settings['color']['palette']['theme'] );
+       }
+
+       /**
+        * Recursively applies ksort to an array.
+        */
+       private static function recursive_ksort( &$array ) {
+               foreach ( $array as &$value ) {
+                       if ( is_array( $value ) ) {
+                               self::recursive_ksort( $value );
+                       }
+               }
+               ksort( $array );
+       }
+
+       /**
+        * @ticket 54336
+        */
+       function test_merges_child_theme_json_into_parent_theme_json() {
+               switch_theme( 'block-theme-child' );
+
+               $actual_settings   = WP_Theme_JSON_Resolver::get_theme_data()->get_settings();
+               $expected_settings = array(
+                       'color'      => array(
+                               'custom'         => false,
+                               'customGradient' => false,
+                               'gradients'      => array(
+                                       'theme' => array(
+                                               array(
+                                                       'name'     => 'Custom gradient',
+                                                       'gradient' => 'linear-gradient(135deg,rgba(0,0,0) 0%,rgb(0,0,0) 100%)',
+                                                       'slug'     => 'custom-gradient',
+                                               ),
+                                       ),
+                               ),
+                               'palette'        => array(
+                                       'theme' => array(
+                                               array(
+                                                       'slug'  => 'light',
+                                                       'name'  => 'Light',
+                                                       'color' => '#f3f4f6',
+                                               ),
+                                               array(
+                                                       'slug'  => 'primary',
+                                                       'name'  => 'Primary',
+                                                       'color' => '#3858e9',
+                                               ),
+                                               array(
+                                                       'slug'  => 'dark',
+                                                       'name'  => 'Dark',
+                                                       'color' => '#111827',
+                                               ),
+                                       ),
+                               ),
+                               'link'           => true,
+                       ),
+                       'typography' => array(
+                               'customFontSize' => false,
+                               'lineHeight'     => true,
+                               'fontSizes'      => array(
+                                       'theme' => array(
+                                               array(
+                                                       'name' => 'Custom',
+                                                       'slug' => 'custom',
+                                                       'size' => '100px',
+                                               ),
+                                       ),
+                               ),
+                       ),
+                       'spacing'    => array(
+                               'units'   => array( 'rem' ),
+                               'padding' => true,
+                       ),
+                       'blocks'     => array(
+                               'core/paragraph'  => array(
+                                       'color' => array(
+                                               'palette' => array(
+                                                       'theme' => array(
+                                                               array(
+                                                                       'slug'  => 'light',
+                                                                       'name'  => 'Light',
+                                                                       'color' => '#f5f7f9',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                               'core/post-title' => array(
+                                       'color' => array(
+                                               'palette' => array(
+                                                       'theme' => array(
+                                                               array(
+                                                                       'slug'  => 'light',
+                                                                       'name'  => 'Light',
+                                                                       'color' => '#f3f4f6',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+               self::recursive_ksort( $actual_settings );
+               self::recursive_ksort( $expected_settings );
+
+               // Should merge settings.
+               $this->assertSame(
+                       $expected_settings,
+                       $actual_settings
+               );
+
+               $this->assertSame(
+                       WP_Theme_JSON_Resolver::get_theme_data()->get_custom_templates(),
+                       array(
+                               'page-home' => array(
+                                       'title'     => 'Homepage',
+                                       'postTypes' => array( 'page' ),
+                               ),
+                       )
+               );
+       }
</ins><span class="cx" style="display: block; padding: 0 10px"> }
</span></span></pre></div>
<a id="trunktestsphpunitteststhemewpThemeJsonSchemaphp"></a>
<div class="addfile"><h4 style="background-color: #eee; color: inherit; margin: 1em 0; padding: 1.3em; font-size: 115%">Added: trunk/tests/phpunit/tests/theme/wpThemeJsonSchema.php</h4>
<pre class="diff"><span>
<span class="info" style="display: block; padding: 0 10px; color: #888">--- trunk/tests/phpunit/tests/theme/wpThemeJsonSchema.php                             (rev 0)
+++ trunk/tests/phpunit/tests/theme/wpThemeJsonSchema.php       2021-11-08 19:18:39 UTC (rev 52049)
</span><span class="lines" style="display: block; padding: 0 10px; color: #888">@@ -0,0 +1,193 @@
</span><ins style="background-color: #dfd; text-decoration:none; display:block; padding: 0 10px">+<?php
+
+/**
+ * Test WP_Theme_JSON_Schema class.
+ *
+ * @package WordPress
+ * @subpackage Theme
+ *
+ * @since 5.9.0
+ *
+ * @group themes
+ */
+class Tests_Theme_wpThemeJsonSchema extends WP_UnitTestCase {
+       /**
+        * The current theme.json schema version.
+        */
+       const LATEST_SCHEMA_VERSION = WP_Theme_JSON::LATEST_SCHEMA;
+
+       /**
+        * @ticket 54336
+        */
+       function test_migrate_v1_to_v2() {
+               $theme_json_v1 = array(
+                       'version'  => 1,
+                       'settings' => array(
+                               'color'      => array(
+                                       'palette' => array(
+                                               array(
+                                                       'name'  => 'Pale Pink',
+                                                       'slug'  => 'pale-pink',
+                                                       'color' => '#f78da7',
+                                               ),
+                                               array(
+                                                       'name'  => 'Vivid Red',
+                                                       'slug'  => 'vivid-red',
+                                                       'color' => '#cf2e2e',
+                                               ),
+                                       ),
+                                       'custom'  => false,
+                                       'link'    => true,
+                               ),
+                               'border'     => array(
+                                       'color'        => false,
+                                       'customRadius' => false,
+                                       'style'        => false,
+                                       'width'        => false,
+                               ),
+                               'typography' => array(
+                                       'fontStyle'      => false,
+                                       'fontWeight'     => false,
+                                       'letterSpacing'  => false,
+                                       'textDecoration' => false,
+                                       'textTransform'  => false,
+                               ),
+                               'blocks'     => array(
+                                       'core/group' => array(
+                                               'border'     => array(
+                                                       'color'        => true,
+                                                       'customRadius' => true,
+                                                       'style'        => true,
+                                                       'width'        => true,
+                                               ),
+                                               'typography' => array(
+                                                       'fontStyle'      => true,
+                                                       'fontWeight'     => true,
+                                                       'letterSpacing'  => true,
+                                                       'textDecoration' => true,
+                                                       'textTransform'  => true,
+                                               ),
+                                       ),
+                               ),
+                       ),
+                       'styles'   => array(
+                               'color'    => array(
+                                       'background' => 'purple',
+                               ),
+                               'blocks'   => array(
+                                       'core/group' => array(
+                                               'color'    => array(
+                                                       'background' => 'red',
+                                               ),
+                                               'spacing'  => array(
+                                                       'padding' => array(
+                                                               'top' => '10px',
+                                                       ),
+                                               ),
+                                               'elements' => array(
+                                                       'link' => array(
+                                                               'color' => array(
+                                                                       'text' => 'yellow',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                               'elements' => array(
+                                       'link' => array(
+                                               'color' => array(
+                                                       'text' => 'red',
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+
+               $actual = WP_Theme_JSON_Schema::migrate( $theme_json_v1 );
+
+               $expected = array(
+                       'version'  => self::LATEST_SCHEMA_VERSION,
+                       'settings' => array(
+                               'color'      => array(
+                                       'palette' => array(
+                                               array(
+                                                       'name'  => 'Pale Pink',
+                                                       'slug'  => 'pale-pink',
+                                                       'color' => '#f78da7',
+                                               ),
+                                               array(
+                                                       'name'  => 'Vivid Red',
+                                                       'slug'  => 'vivid-red',
+                                                       'color' => '#cf2e2e',
+                                               ),
+                                       ),
+                                       'custom'  => false,
+                                       'link'    => true,
+                               ),
+                               'border'     => array(
+                                       'color'  => false,
+                                       'radius' => false,
+                                       'style'  => false,
+                                       'width'  => false,
+                               ),
+                               'typography' => array(
+                                       'fontStyle'      => false,
+                                       'fontWeight'     => false,
+                                       'letterSpacing'  => false,
+                                       'textDecoration' => false,
+                                       'textTransform'  => false,
+                               ),
+                               'blocks'     => array(
+                                       'core/group' => array(
+                                               'border'     => array(
+                                                       'color'  => true,
+                                                       'radius' => true,
+                                                       'style'  => true,
+                                                       'width'  => true,
+                                               ),
+                                               'typography' => array(
+                                                       'fontStyle'      => true,
+                                                       'fontWeight'     => true,
+                                                       'letterSpacing'  => true,
+                                                       'textDecoration' => true,
+                                                       'textTransform'  => true,
+                                               ),
+                                       ),
+                               ),
+                       ),
+                       'styles'   => array(
+                               'color'    => array(
+                                       'background' => 'purple',
+                               ),
+                               'blocks'   => array(
+                                       'core/group' => array(
+                                               'color'    => array(
+                                                       'background' => 'red',
+                                               ),
+                                               'spacing'  => array(
+                                                       'padding' => array(
+                                                               'top' => '10px',
+                                                       ),
+                                               ),
+                                               'elements' => array(
+                                                       'link' => array(
+                                                               'color' => array(
+                                                                       'text' => 'yellow',
+                                                               ),
+                                                       ),
+                                               ),
+                                       ),
+                               ),
+                               'elements' => array(
+                                       'link' => array(
+                                               'color' => array(
+                                                       'text' => 'red',
+                                               ),
+                                       ),
+                               ),
+                       ),
+               );
+
+               $this->assertEqualSetsWithIndex( $expected, $actual );
+       }
+}
</ins></span></pre>
</div>
</div>

</body>
</html>