Filtering the theme.json file values (such as default color palette and other global styles) in WordPress

The theme.json file is a central place for various WordPress theme settings. Things like color palettes, gradients, fonts, block spacing, and other global styles and default settings of the block editor are all defined in this single file.

There are a few options to adjust the values defined in that file. Within the Site Editor, things defined at the theme.json level can be customized and saved in the website options table, thus decoupled from the theme.json file itself.

There’s also a built-in tool in WordPress core that can be used to generate a new block theme based on the customizations made. That’s handy if one prefers building the theme “live” rather than from the IDE point of view.

In both cases, however, a connection with the original theme is lost, at least partially in the first and fully in the second. That’s not always a problem; sometimes, it’s expected and desired. But what if we want to keep the “foundation,” the basic theme, and be able to receive its updates as soon as they are available but still modify some values of the theme.json file to meet the project’s specific needs?

How to filter the theme.json data with PHP

Four new filters were introduced in WordPress 6.1 (released on November 2, 2022) that can be used specifically for that purpose. These filters’ scope varies depending on which data layer is to be modified:

All of these hooks are defined in the WP_Theme_JSON_Resolver class and are passing the WP_Theme_JSON_Data object as a first argument; this object can be used in a custom callback function to update the underlying data.

Filtering default theme.json file data

The first hook on the list allows filtering default theme.json data, as provided by WordPress. These include various Block Editor settings (like whether or not to show the “appearance tools,” the core duotone, gradients, and shadows configuration, common spacing and typography configuration, etc.

Let’s assume you want to modify the default “black” color – it is set to #000000 in WordPress core, but most modern websites rarely use “pure black” color for text or other elements. Let’s say we want to change it to #111111, but rather than defining the custom color, we still want to refer to it as core “black.”

That’s where the wp_theme_json_data_default filter comes in handy:

<?php
/**
* Change the core "black" color from "#000000" to "#111111"
*
* This will affect the "--wp--preset--color--black" global CSS variable.
*/
add_filter(
'wp_theme_json_data_default',
/**
* Filter the theme JSON data
*
* @param \WP_Theme_JSON_Data $theme_json The theme.json data class object to access and update the underlying data.
*
* @return \WP_Theme_JSON_Data Same instance as provided.
*/
function ( \WP_Theme_JSON_Data $theme_json ): \WP_Theme_JSON_Data {
// Get the theme.json data and default color palette.
$theme_json_data = $theme_json->get_data();
$color_palette = $theme_json_data['settings']['color']['palette']['default'] ?? null;
if ( null !== $color_palette ) {
// Find the color we're looking for in the palette.
$index = array_search( 'black', array_column( $color_palette, 'slug' ), true );
// Modify the color value if it's found in the palette.
if ( false !== $index ) {
$color_palette[ $index ]['color'] = '#111111';
// Update the theme.json structure.
$theme_json = $theme_json->update_with(
[
'version' => 2,
'settings' => [
'color' => [
'palette' => [
'default' => $color_palette,
],
],
],
],
);
}
}
return $theme_json;
}
);

This filter above will modify the --wp--preset--color--black CSS variable value from #000000 to #111111. Note: if you’re updating a single element of an array of elements, you must provide a whole array to the update_with method, not only the single updated value. Otherwise, only the provided value will be used, and the rest will be lost.

This behavior, however, could be used for another purpose – removing default values that are not used and needed. By default, WordPress core provides many colors and gradients, which you might not want to see on your website for many reasons. Using the same hook, you can get rid of them:

<?php
/**
* Remove default duotone and gradients
*/
add_filter(
'wp_theme_json_data_default',
/**
* Filter the theme JSON data
*
* @param \WP_Theme_JSON_Data $theme_json The theme.json data class object to access and update the underlying data.
*
* @return \WP_Theme_JSON_Data Same instance as provided.
*/
function ( \WP_Theme_JSON_Data $theme_json ): \WP_Theme_JSON_Data {
return $theme_json->update_with(
[
'version' => 2,
'settings' => [
'color' => [
'duotone' => [
'default' => [],
],
'gradients' => [
'default' => [],
],
],
],
],
);
}
);

All the remaining elements of the theme.json structure will remain unchanged, so you don’t need to worry about passing the whole theme.json data into the update_with method.

One more thing to note: the version key should reflect the theme.json file version. You can find which version your theme uses by inspecting the theme.json file, which is usually located in the theme’s root directory. That’s required to ensure that settings will be merged correctly and to maintain required backward and forward compatibility.

Filtering the theme-related data

In this following example, I’d like to filter the “Extra Extra Large” (xx-large) font size defined in the Twenty Twenty Four theme, which is currently set as fluid between 2.5rem and 3.27rem. I want to ensure it will always be set to 4rem. In this case, I’ll use the wp_theme_json_data_theme hook:

<?php
/**
* Adjust font size defined in theme
*/
add_filter(
'wp_theme_json_data_theme',
/**
* Filter the theme JSON data
*
* @param \WP_Theme_JSON_Data $theme_json The theme.json data class object to access and update the underlying data.
*
* @return \WP_Theme_JSON_Data Same instance as provided.
*/
function ( \WP_Theme_JSON_Data $theme_json ): \WP_Theme_JSON_Data {
// Get the theme.json data and typography configuration.
$theme_json_data = $theme_json->get_data();
$font_sizes = $theme_json_data['settings']['typography']['fontSizes']['theme'] ?? null;
if ( null !== $font_sizes ) {
// Find the font size we're looking for.
$index = array_search( 'xx-large', array_column( $font_sizes, 'slug' ), true );
// Modify the font size value if it's found.
if ( false !== $index ) {
$font_sizes[ $index ]['fluid'] = false;
$font_sizes[ $index ]['size'] = '4rem';
// Update the theme.json structure.
$theme_json = $theme_json->update_with(
[
'version' => 2,
'settings' => [
'typography' => [
'fontSizes' => [
'theme' => $font_sizes,
],
],
],
],
);
}
}
return $theme_json;
}
);

Depending on a project’s needs, any required business logic can be applied inside that filter. Rather than displaying the default value of the site’s background color as defined in a theme, you might want to display a “random” background color on every render (because why not?). In this case, it would be as simple as:

<?php
/**
* Randomize theme background colors
*/
add_filter(
'wp_theme_json_data_theme',
/**
* Filter the theme JSON data
*
* @param \WP_Theme_JSON_Data $theme_json The theme.json data class object to access and update the underlying data.
*
* @return \WP_Theme_JSON_Data Same instance as provided.
*/
function ( \WP_Theme_JSON_Data $theme_json ): \WP_Theme_JSON_Data {
$colors = [
'#8D6A9F',
'#C5CBD3',
'#8CBCB9',
'#DDA448',
'#BB342F',
];
return $theme_json->update_with(
[
'version' => 2,
'styles' => [
'color' => [
'background' => $colors[ wp_rand( 0, count( $colors ) - 1 ) ],
],
],
],
);
}
);

How to filter the theme.json data with JavaScript

WordPress 6.2 (released on March 29, 2023) introduced a new JS filter (blockEditor.useSetting.before), which allows engineers to modify the theme.json settings before the Block Editor is rendered. This filter allows to modify the block settings depending on the user role, other blocks used on a page, post type, and any other data available in the Block Editor context.

With this filter, you can modify the settings and block configuration in the Block Editor context; these changes will not apply on the website’s front-end – so the use cases for this approach are quite different.

If you’re interested more about this topic, please check the “further reading” section below for recommended sources and articles.

Conclusion

WordPress hooks mentioned in this article give the engineers much control over the theme.json file contents. That might be useful in some cases, like where a project specifics are to allow users some level of freedom, but still, there should be some validation and protection in place; where certain things should be removed to reduce the size of a CSS rendered on the front-end (front-end performance improvements); or, where some things of a chosen theme should be modified, without detaching the rest of the settings.

But this will not always be the best choice; sometimes, it’s better to modify the theme.json directly (if it’s a fully custom theme) or to build/export a new theme based on applied modifications. It all depends on the specific needs of a project. But it’s good to have more options!

Further reading

If you’d like to find out more about this topic, below are the resources I found helpful when researching this topic for the first time: