Lesson 6: Inner Blocks / Nested Blocks
Learning Outcomes
- Learn what inner blocks are and how to use them
- Learn how to decide when to use inner blocks
- Learn about controlled parent/child relationships in inner blocks
What are "Inner Blocks"?
Inner Blocks in WordPress let you nest blocks within other blocks. You may have come across this when using the Core Group or Columns Block. This ability is very powerful because it allows you to compose different blocks to build your content. Composition is not the only thing nested blocks are useful for. Nesting also allows you to reduce complexity when you have to create repeatable items in your blocks.
The Group block for example is meant to be a wrapper for other blocks. It creates a container with some options like a background color for that section and allows you to insert other blocks into that wrapper. Other examples of this kind of nesting can be found in the Media & Text and the Cover Block. They all allow you to place other blocks within an inner block area.
There is another great use case for inner blocks. We can see that when looking at the core columns block. It may seem very similar at first because it also allows you to place blocks within the columns.
But this is not the case for the columns block. It only allows individual column blocks as its direct decedents. The individual column blocks then allow you to place other blocks in them. The column blocks cannot get used outside the columns block.
Instead of storing a very complicated array of the columns with all their options and building custom logic to add, move or remove columns, all of that gets managed by using inner blocks with a wrapping columns
block that only allows column
blocks as its direct children.
Another example of this is the core buttons block. It only allows button blocks as its children and uses inner blocks to allow editors to add, move and remove the individual button blocks.
Exercise Overview
In this lesson, we are going to build two different blocks ("Hero" Block and "Card Grid" blocks) to take a look at what use-cases inner blocks can solve. Both of these blocks have a starter
folder that already contains a lot of the skeleton for these blocks. But the actual InnerBlocks
need to get added to the components to make them work. If you get stuck you can always take a peek at the completed
folder for a complete version of the block.
1. "Hero" Block
First, we will take a look at how we can use inner blocks to allow an editor to place anything they want into a predefined area of your component. In this example, we have a "simple" design of a "Hero" component that has an image at the top with a content area that overlaps the image.
The client wants to be able to place other blocks into the content area of the "Hero" component to make it as flexible as possible for them.
For the "Hero" Block we need to define an area of our markup where other blocks can get inserted.
To get started there is a starter block scaffolded out located in the blocks folder of the theme with the name inner-blocks-one-starter
. If you are stuck you can take a look at the inner-blocks-one-complete
folder for a completed version of the block.
Most of the markup and block scaffolding already is in place. What is left to do is wire up the InnerBlock
area both in the editor and on the frontend. You will find comments outlining where the inner blocks area should go in both the edit.js
and the markup.php
files. We go through these steps below in our Using Inner Blocks
section.
2. "Card Grid" Block
The second block we will be building is a "Card Grid" component. This Card grid should allow editors to place as many card components into a grid that automatically reflows based on the screen size of the device they are using.
The "Card Grid" consists of two elements. The actual grid can contain the cards and then the individual cards. This is similar to how core has the Buttons block that contains individual Button blocks.
So we can archive this by creating two different blocks. The "Card Grid" block needs to define an inner block area where only the card block can get inserted. And then we need the card block that cannot get used anywhere outside of the "Card Grid".
To get started there are two starter blocks scaffolded out located in the blocks folder of the theme with the name inner-blocks-two-card-grid-starter
and inner-blocks-two-card-starter
. If you are stuck you can take a look at the inner-blocks-two-card-grid-complete
and inner-blocks-two-card-complete
folders for a completed version of the block.
As in the first example, most of the markup and block are already scaffolded out. The inner blocks area needs to get added and configured again and the "Card" block needs to get hidden from the inserter unless it is being used within the "Card Grid" block. We go through these steps below in our Using Inner Blocks
section.
Using Inner Blocks
Inner Blocks are the name of a react component that is part of the "Block Editor" package. We can import it into our block edit component and use it within the markup of our block.
In the block one starter edit.js
file, do the following:
Update the import to include InnerBlocks
:
import { useBlockProps, InnerBlocks } from '@wordpress/block-editor';
Then, use the example below to add the InnerBlocks
component where the inner blocks should be located:
const BlockEdit = () => {
const blockProps = useBlockProps();
return (
<div {...blockProps}>
<InnerBlocks />
</div>
)
}
export default BlockEdit;
You should end up with this:
<div className={`${className}__content`}>
<InnerBlocks />
</div>
Connect block to build scripts
As always, you should have npm run watch
running in your terminal. Making this change, did it do anything? Let's check the editor. Search the block inserter for Inner Blocks One - Starter
... is it found? Only see Inner Blocks One - Complete
? That is because these files are not being targeted by our build scripts. Let's fix that.
First, look at the package.json
file. You will see in that file an object 10up-toolkit
and within another object of entry
. This entry
object contains the entry points for @10up/toolkit
(used in many of our internal projects, including our default scaffolding). As you can see here, targeted directories to compile blocks. In particular, you can see inner-blocks-one-complete-block
listed. First, to get this block rigged up, we want to add "inner-blocks-one-starter-block": "./includes/blocks/inner-blocks-one-starter/index.js"
to the package json:
"entry": {
"admin": "./assets/js/admin/admin.js",
"blocks": "./includes/blocks/blocks.js",
"frontend": "./assets/js/frontend/frontend.js",
"shared": "./assets/js/shared/shared.js",
"styleguide": "./assets/js/styleguide/styleguide.js",
"admin-style": "./assets/css/admin/admin-style.css",
"editor-style": "./assets/css/frontend/editor-style.css",
"shared-style": "./assets/css/shared/shared-style.css",
"style": "./assets/css/frontend/style.css",
"styleguide-style": "./assets/css/styleguide/styleguide.css",
"core-block-overrides": "./includes/core-block-overrides.js",
"rich-text-formats": "./includes/text-formats/index.js",
"cta-complete-block": "./includes/blocks/cta-complete/index.js",
"inner-blocks-one-starter-block": "./includes/blocks/inner-blocks-one-starter/index.js",
"inner-blocks-one-complete-block": "./includes/blocks/inner-blocks-one-complete/index.js",
"inner-blocks-two-card-grid-complete-block": "./includes/blocks/inner-blocks-two-card-grid-complete/index.js",
"inner-blocks-two-card-complete-block": "./includes/blocks/inner-blocks-two-card-complete/index.js",
"cta-starter-block": "./includes/blocks/cta-starter/index.js"
}
Second, we need to register the block in blocks.php
. There are two steps to this. Step one: find the comment // Require custom blocks
and the line require_once TENUP_THEME_BLOCK_DIR . '/inner-blocks-one-complete/register.php';
. We want to duplicate this line and reference the starter register file:
// Require custom blocks.
require_once TENUP_THEME_BLOCK_DIR . '/cta-complete/register.php';
require_once TENUP_THEME_BLOCK_DIR . '/cta-starter/register.php';
require_once TENUP_THEME_BLOCK_DIR . '/inner-blocks-one-starter/register.php';
require_once TENUP_THEME_BLOCK_DIR . '/inner-blocks-one-complete/register.php';
require_once TENUP_THEME_BLOCK_DIR . '/inner-blocks-two-card-grid-complete/register.php';
require_once TENUP_THEME_BLOCK_DIR . '/inner-blocks-two-card-complete/register.php';
Then, we want to find the section in the file that begins with the comment // Call block register functions for each block.
Here you can see the InnerBlocksOneComplete\register();
call. We want to replicate this and reference our starter block as InnerBlocksOneStarter\register();
.
// Call block register functions for each block.
CTAComplete\register();
InnerBlocksOneStarter\register();
InnerBlocksOneComplete\register();
InnerBlocksTwoCardGridComplete\register();
InnerBlocksTwoCardComplete\register();
CTAStarter\register();
Let's test that and see if it works. Stop your task runner in the terminal and restart it. Now, go back to the edit.js
file and save. You should see the task runner update in the terminal.
With this done you can already go to the editor and see that there is now a Block Appender rendering inside your block that allows you to insert any blocks you want (you may have to refresh the page).
Saving the Content of our Inner Blocks Area
If you now try to save the post and view it on the frontend you will find that your changes are not actually saved. The way the editor stores inner blocks is in the actual post content. But right now our blocks just return null
in their save
method in the index.js
file.
We can fix this by replacing the null
with the <InnerBlocks.Content />
component:
/**
* WordPress dependencies
*/
import { registerBlockType } from '@wordpress/blocks';
import { InnerBlocks } from '@wordpress/block-editor';
/**
* Internal dependencies
*/
import edit from './edit';
import block from './block.json';
/**
* Register block
*/
registerBlockType(block.name, {
edit,
save: () => <InnerBlocks.Content />,
});
Now that the content is saved in the database we also need to somehow use this content in our markup in PHP. When we build blocks dynamically we provide the register_block_type_from_metadata
function with a render_callback
. This function gets called with three arguments from WordPress. The block attributes, the blocks content (inner blocks) as a markup string and the block as a WP_Block
class. We are interested in the second parameter for this.
To rig this up, let's go to our markup.php
file. If we look down at the content container, we will see the comment // The inner blocks content should get rendered here.
Replace this comment with the following:
/*
* the $args['content'] is the html generated from innerBlocks
* it is being created from the save method in JS or the render_callback
* in php and is sanitized.
*
* Re sanitizing it through `wp_kses_post` causes
* embed blocks to break and other core filters don't apply.
* therefore no additional sanitization is done and it is being output as is
*/
echo $args['content']; // phpcs:disable
From here we are most of the way there. You are now able to insert any blocks you want and the content gets saved into the database and shows on the frontend.
There are a few things to keep in mind though to improve the User Experience of our block.
Use allowedBlocks
to better control what editors can do
Right now editors are able to insert any block they want into the inner blocks area. This might not always be what we want. In the case of our "Hero" it, for example, doesn't really make sense to allow any layout-related blocks like the "Banner", "Columns" or "Group" block.
We can improve the UX by adding a list of allowed blocks to the inner blocks component.
return (
<InnerBlocks
allowedBlocks={[
'core/heading',
'core/paragraph',
'core/buttons',
'core/button',
]}
/>
)
Defining a template
to Get Editors Up and Running Quicker
Another thing that isn't great about the UX right now is that the initial state when we insert our "Hero" block is only an empty grey box.
We know that in an ideal case we want our editors to have a title and a paragraph inside the "Hero" so we can make that the default when the block is inserted.
To achieve this we can define a template
on the inner block area. The template gets defined the same way you already saw in Lesson 5: Block Variations. It is an array with individual items also being represented as an array with the block name as the first element, the attributes of the block as the second and child blocks as the third element.
return (
<InnerBlocks
template={[
['core/heading', { level: 2, placeholder: 'Insert your heading here...' }],
['core/paragraph', { placeholder: 'Write some description text here...' }],
]}
/>
)
Parent / Child Relationships of Blocks
Before continuing, be sure you have done the following for the inner-blocks-two-card-starter
and the inner-blocks-two-card-grid-starter.
(per our guidance above):
- Add the block entry file paths in
package.json
- Register the blocks in
blocks.php
(Remember, there are two steps to this). - Add
<InnerBlocks />
toedit.js
(Don't forget to importInnerBlocks
at the top of the file). - Update the
save
method in theindex.js
file (save: () => <InnerBlocks.Content />
). - Add the markup output in
markup.php
(echo $args['content'];
).
If you want to create parent/child relationships between blocks like we need to do for our "Card Grid" and "Card" blocks that consists of two things. For one, you need to define the allowedBlocks
on the InnerBlocks
in the parent block to only contain the child block you want to have show up. If the allowedBlocks
array only contains one item the inserter will no longer show the block picker popover but instead directly insert that one block. Which is a nice little UX improvement we get for free.
So, for the "Card Grid" block, we want to update the edit.js
do define allowedBlocks
for <InnerBlocks />
:
<InnerBlocks
allowedBlocks={['gutenberg-lessons/inner-blocks-two-card-starter']}
/>
BONUS: To enhance our UX and better define the editorial output, let's add orientation and a template to <InnerBlocks />
in edit.js
:
<InnerBlocks
allowedBlocks={['gutenberg-lessons/inner-blocks-two-card-complete']}
orientation="horizontal"
template={[
['gutenberg-lessons/inner-blocks-two-card-starter'],
['gutenberg-lessons/inner-blocks-two-card-starter'],
]}
/>
For the above, we have already included styles in the index.css
to assist with our layout - more on this below.
The second thing we need to do is make sure our "child" block, in this case, the "Card", doesn't appear in the regular inserter or that it cannot get moved anywhere outside of a "Card Grid". And for that, we can use the parent
option in the block.json
file. When you define a parent
the block will no longer show up in the inserter and it also cannot get used anywhere outside of the defined parent.
{
"name": "...",
"...",
"parent": [ "namespace/block-name" ]
}
So, let's go ahead and update the "Card" block.json
to include the following:
"parent": [ "gutenberg-lessons/inner-blocks-two-card-grid-starter" ]
Now, when using the block inserter, you no longer see reference to the "Card" block, unless adding content to the "Card Grid" block.
Styling Considerations in the Editor
One thing to be aware of when using InnerBlocks
is that at the moment it makes it a bit more difficult to match styling between the editor and the frontend of the site because the InnerBlocks
component in the editor wraps the markup of the child blocks in this markup:
<div class="block-editor-inner-blocks">
<div class="block-editor-block-list__layout">
<!-- Markup of the Child Blocks will get inserted here -->
<div class="block-list-appender">...</div>
</div>
</div>
This means that we will have to adjust our styling between the frontend and the editor.
In our "Card Grid" we use CSS Grid to make the main wrapper a grid container with the child blocks being direct children of the wrapper. In the editor however, this won't work because the .block-editor-inner-blocks
and .block-editor-block-list__layout
elements are inserted in between the wrapper and the children. So we need to get creating in our styles and override some of the frontend styles and add new ones to visually match the component in the editor.
There currently is a new experimental API in core called __experimentalUseInnerBlocksProps
that will allow us to remove these additional elements and get markup parity between the editor and the frontend. But that API is not fully production-ready and may change in the future.
Takeaways
The takeaway you should take from this lesson is that inner blocks are a very powerful concept in the editor that allows us to use different blocks to compose more complex content. They can also allow us to simplify the logic we have to code when it comes to managing repeating elements and have many options to control the user experience.
Next Steps
Now that you have built two blocks that use InnerBlocks
take a look at the Component Reference for the InnerBlocks
component and see whether there are any other things you can to improve the editorial user experience.