A block template is defined as a list of block items. Such blocks can have predefined attributes, placeholder content, and be static or dynamic. Block templates allow specifying a default initial state for an editor session.
The scope of templates include:
- Setting a default state dynamically on the client. (like
defaultBlock
) - Registered as a default for a given post type.
Planned additions:
- Saved and assigned to pages as "page templates".
- Defined in a
template.php
file or pulled from a custom post type (wp_templates
) that is site specific. - As the equivalent of the theme hierarchy.
Templates can be declared in JS or in PHP as an array of blockTypes (block name and optional attributes).
The first example in PHP creates a template for posts that includes an image block to start, you can add as many or as few blocks to your template as needed.
PHP example:
<?php
function myplugin_register_template() {
$post_type_object = get_post_type_object( 'post' );
$post_type_object->template = array(
array( 'core/image' ),
);
}
add_action( 'init', 'myplugin_register_template' );
The following example in JavaScript creates a new block using InnerBlocks and templates, when inserted creates a set of blocks based off the template.
const el = React.createElement;
const { registerBlockType } = wp.blocks;
const { InnerBlocks } = wp.blockEditor;
const BLOCKS_TEMPLATE = [
[ 'core/image', {} ],
[ 'core/paragraph', { placeholder: 'Image Details' } ],
];
registerBlockType( 'myplugin/template', {
title: 'My Template Block',
category: 'widgets',
edit: ( props ) => {
return el( InnerBlocks, {
template: BLOCKS_TEMPLATE,
templateLock: false,
} );
},
save: ( props ) => {
return el( InnerBlocks.Content, {} );
},
} );
See the Meta Block Tutorial for a full example of a template in use.
To find a comprehensive list of all block attributes that you can define in a template, consult the block's block.json
file, and look at the attributes
and supports
values.
For example, packages/block-library/src/heading/block.json shows that the block has a level
attribute, and supports the anchor
parameter.
If you don't have the Gutenberg plugin installed, you can find block.json
files inside wp-includes/blocks/heading/block.json
.
A custom post type can register its own template during registration:
function myplugin_register_book_post_type() {
$args = array(
'public' => true,
'label' => 'Books',
'show_in_rest' => true,
'template' => array(
array( 'core/image', array(
'align' => 'left',
) ),
array( 'core/heading', array(
'placeholder' => 'Add Author...',
) ),
array( 'core/paragraph', array(
'placeholder' => 'Add Description...',
) ),
),
);
register_post_type( 'book', $args );
}
add_action( 'init', 'myplugin_register_book_post_type' );
Sometimes the intention might be to lock the template on the UI so that the blocks presented cannot be manipulated. This is achieved with a template_lock
property.
function myplugin_register_template() {
$post_type_object = get_post_type_object( 'post' );
$post_type_object->template = array(
array( 'core/paragraph', array(
'placeholder' => 'Add Description...',
) ),
);
$post_type_object->template_lock = 'all';
}
add_action( 'init', 'myplugin_register_template' );
Options:
contentOnly
— prevents all operations. Additionally, the block types that don't have content are hidden from the list view and can't gain focus within the block list. Unlike the other lock types, this is not overridable by children.all
— prevents all operations. It is not possible to insert new blocks, move existing blocks, or delete blocks.insert
— prevents inserting or removing blocks, but allows moving existing blocks.
Lock settings can be inherited by InnerBlocks. If templateLock
is not set in an InnerBlocks area, the locking of the parent InnerBlocks area is used. If the block is a top level block, the locking configuration of the current post type is used.
Alongside template level locking, you can lock individual blocks; you can do this using a lock
attribute on the attributes level. Block-level lock takes priority over the templateLock
feature. Currently, you can lock moving and removing blocks.
attributes: {
// Prevent a block from being moved or removed.
lock: {
remove: true,
move: true,
}
}
Options:
remove
— Locks the ability of a block from being removed.move
— Locks the ability of a block from being moved.
You can use this with templateLock
to lock all blocks except a single block by using false
in remove
or move
.
$template = array(
array( 'core/image', array(
'align' => 'left',
) ),
array( 'core/heading', array(
'placeholder' => 'Add Author...',
) ),
// Allow a Paragraph block to be moved or removed.
array( 'core/paragraph', array(
'placeholder' => 'Add Description...',
'lock' => array(
'move' => false,
'remove' => false,
),
) ),
);
Container blocks like the columns blocks also support templates. This is achieved by assigning a nested template to the block.
$template = array(
array( 'core/paragraph', array(
'placeholder' => 'Add a root-level paragraph',
) ),
array( 'core/columns', array(), array(
array( 'core/column', array(), array(
array( 'core/image', array() ),
) ),
array( 'core/column', array(), array(
array( 'core/paragraph', array(
'placeholder' => 'Add a inner paragraph'
) ),
) ),
) )
);