WordPress Block Development: Complete Guide to Multiple Blocks & Interactivity API

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!