Creating multiple Gutenberg blocks while ensuring interactivity and customization can seem challenging, but with the right tools and approach, it becomes manageable. This guide is tailored for developers who want to streamline their WordPress build setup, making it both efficient and flexible. We’ll dive into optimizing webpack configurations, using the interactivity API, and building multiple blocks—all while keeping the codebase clean.
Here’s a representation of the folder structure we’ll use for this guide:
resources/ // Source folder
├── blocks/
├── js/
│ ├── index.js
│ ├── admin.js
├── css/
│ ├── public.css
│ ├── admin.css
build/ // Destination folder
├── index.js
├── index.js.map
├── admin.js
├── admin.js.map
├── public.css
├── public.css.map
├── admin.css
├── admin.css.map
└── blocks/
├── block-1/
└── block-2/
Note on Creating Multiple Blocks
It’s out of scope for this guide to provide a detailed step-by-step on how to create multiple blocks. If you’re looking for a comprehensive guide, I highly recommend checking out How to Build a Multi-Block Plugin. It covers everything you need to get started in detail.
Step 1: Initial Setup
The first step in building interactive WordPress blocks is getting your environment set up correctly. My journey started by adding scripts in package.json
that enabled block development through the WordPress wp-scripts
library.
Here’s what the initial setup looked like:
"scripts": {
"start:blocks": "wp-scripts start --webpack-src-dir=resources/blocks/ --output-path=build/blocks/ --webpack-copy-php --experimental-modules",
"start:others": "wp-scripts start --webpack-src-dir=resources/js/ --output-path=build/others/ --config custom-webpack.config.js"
},
These scripts allowed me to start developing blocks under resources/blocks
and other JavaScript files in resources/js
using custom configurations. However, this initial setup felt cumbersome, and it required running two commands in a specific order (start:blocks
before start:others
), as start:others
could overwrite the output of start:blocks
. The same applied when building, even though it’s not detailed here. I knew there was a more efficient way to handle it.
Step 2: Creating a Custom Webpack Configuration
I created a webpack.config.js
to handle additional entries, such as an admin-specific JavaScript file. Here’s what that configuration looked like:
const defaultConfig = require('@wordpress/scripts/config/webpack.config');
const path = require('path');
const { getWebpackEntryPoints } = require('@wordpress/scripts/utils/config');
const path = require('path');
module.exports = {
...defaultConfig,
entry: {
...defaultConfig.entry(),
admin: path.resolve(__dirname, 'resources/js/admin.js'),
},
};
This configuration extended WordPress’s default webpack configuration to include admin.js
, which was perfect for adding admin-specific functionality. However, there was one drawback—when using experimental scripts, defaultConfig
is returned as an array, which made the initial setup tricky to maintain as I added more interactivity.
Step 3: Optimizing the Configuration
To simplify the development process, I optimized the package.json
and webpack.config.js
files. This made my build configuration cleaner and easier to manage, especially when working with multiple blocks and JavaScript files.
Here’s what my package.json
script became:
"scripts": {
"start": "wp-scripts start --webpack-src-dir=resources/ --webpack-copy-php --experimental-modules"
},
To add the build command with the same parameters, simply extend the scripts
section of your package.json
to include a build
command. This command will compile and bundle the files for production use, similar to the start
command but optimized for deployment.
Here’s how you can add it:
"scripts": {
"start": "wp-scripts start --webpack-src-dir=resources/ --webpack-copy-php --experimental-modules",
"build": "wp-scripts build --webpack-src-dir=resources/ --webpack-copy-php --experimental-modules"
},
The build
command uses wp-scripts build
instead of start
, which compiles the files for production, typically resulting in minified and optimized output.
This allowed me to handle all JavaScript and PHP copying with a single command.
The updated webpack.config.js
looks like this:
const defaultConfig = require('@wordpress/scripts/config/webpack.config');
const path = require('path');
module.exports = [
...defaultConfig,
{
...defaultConfig[0],
entry: {
...defaultConfig[0].entry(),
index: path.resolve(__dirname, 'resources/js/index.js'),
admin: path.resolve(__dirname, 'resources/js/admin.js'),
}
}
];
In this optimized version, I accounted for the fact that defaultConfig
is now returned as an array when enabling experimental scripts. I made sure to properly spread the configuration and added new entries for index.js
and admin.js
without compromising the existing webpack settings.
Step 4: The Final Output Structure
After implementing these changes, the output structure looked like this:
build/
├── index.js
├── index.js.map
├── admin.js
├── admin.js.map
├── public.css
├── public.css.map
├── admin.css
├── admin.css.map
└── blocks/
├── block-1/
└── block-2/
This streamlined structure ensures that each component—whether it’s a block or admin functionality—has its own dedicated build file, improving maintainability and performance.
Summary
By optimizing scripts in package.json
and leveraging a customized webpack.config.js
, I was able to efficiently build multiple blocks, add interactive JavaScript, and create custom admin functionalities in WordPress. The experimental modules flag allowed for advanced configurations that greatly simplified the workflow.
This process provides a scalable way to manage interactive and non-interactive JavaScript, ensuring a consistent development experience when working with Gutenberg blocks and other WordPress features.
Final Thoughts
Building blocks in WordPress can be complex, especially when you want to add interactive elements or customize functionality for specific areas like the admin dashboard. However, by leveraging the tools provided by WordPress and customizing webpack configurations, you can make this process efficient and scalable.
If you’re diving into Blocks development and want to add custom interactivity, start by getting comfortable with wp-scripts
and consider customizing your webpack setup—it makes a world of difference!