by Andrey Pokrovskiy
UPDATE 06/16/2017:
I updated the project to use ReactRouter 4 and Webpack 2. Some parts were refactored and simplified. Included links to the frontend React demo and Wordpress backend demo.
WordPress is a powerful content management tool. But when it comes to developing on top of it, can be quite frustrating. WordPress’s crazy mixture of HTML and PHP loops can often prove unintuitive to grasp and hard to maintain.
There’s a light at the end of the tunnel, though! Starting with version 4.7, WordPress comes with a built-in REST API, and plugins are no longer required. This makes it easier to use WordPress strictly as a backend data storage or CMS, while allowing for a totally custom front end solution of your choice.
You no longer need to have a local WordPress installation and deal with setting up vhosts. Your local development process can be limited to building the front end that is connected with a WordPress installation hosted on a remote server.
In this article I’m going to use ReactJS to build the front end part of the application, React Router for routing, and Webpack for bundling it all together. I’ll also show you how to integrate Advanced Custom Fields, for those of us who have come to rely on it as a tool to create an intuitive solution for our clients.
The stack looks like this:
- ReactJs
- React Router v4
- Alt (for Flux implementation)
- Webpack v2
GitHub repo: https://github.com/DreySkee/wp-api-react
React frontend demo url: http://wp-api-react.surge.sh/
Wordpress backend demo url: http://andreypokrovskiy.com/projects/wp-api-react/wp-admin
User: demo
Pass: wp-react-demo
Project setup
Let’s name the project “wp-api-react”. To follow along, first thing you need to do is include this in your package.json and run npm install
:
Install webpack and webpack-dev-server globally as well if you have not done this already:
npm i webpack webpack-dev-server -g
Now in the project folder create wepack.dev.js
for development configuration and webpack.production.js
with configuration for building the project for production.
Paste this in webpack.dev.config.js
:
And this in webpack.production.config.js
:
Create “src” folder in the project root and create index.html
inside of it. The index.html
file will include this chunk of code:
Now let’s add a few more folders to the project. Inside of the “src” folder create “scripts” folder and inside of “scripts” create “components”, “flux” and index.js
file. This structure will help to keep files organized.
By now the folder structure should look like this:
wp-api-react/
— node_modules/
— src/
— — scripts/
— — — components/
— — — flux/
— — — index.js
— — index.html
— package.json
— webpack.config.js
index.js
is the entry point for Webpack and it will hold all routes for the project. Let’s include React, React Router and basic routing structure in the file:
index.js
is referencing Home component in imports. We need to create it in the “components” folder. Home.js
will be the template component for the homepage. Include this in the file:
If you run npm start
in the terminal inside of the project folder and open http://localhost:8080/ in the browser you should see a “Hello world!” message. If you start changing files Webpack will hot-reload the page for you.
Flux with Alt
Now it’s time to implement flux using Alt. You will need to create three new folders inside of the “flux” folder: “alt”, “stores” and “actions”:
wp-api/
— node_modules/
— src/
— — scripts/
— — — flux/
— — — — alt/
— — — — actions/
— — — — stores/
— — — components/
— — — — Home.js
— — — index.js
— — index.html
— package.json
— webpack.config.js
Create Alt.js
inside of the “alt” folder and paste this in the file:
All this file is doing is exporting the Alt instance that we will use in stores and actions.
Create DataActions.js
in the “actions” folder. This file will have all the logic for getting the data from the WordPress REST API endpoints. For talking to the API we’ll use axios. Include this in DataActions.js
:
Don’t forget to replace the WordPress installation example URL with yours.
Create DataStore.js
in “stores” folder. This file will be listening to DataActions.js’ getSuccess()
method that returns data from the WordPress API. It will then store and manipulate the data. Paste this in DataStore.js
:
To get data from the WordPress API and make it available for the app you need to include DataActions.js
in index.js
and wrap the render function in DataActions.getPages()
. The returned response will later be used to dynamically create routes:
Now every time the app gets initiated DataActions.getPages()
calls the WordPress API endpoint and stores returned data in DataStore.js
.
To access it simply include the DataStore.js
in any component and call the appropriate method. For example let’s get all the data inside the Home.js
file and console.log
it:
After Webpack refreshes the page you should see the returned data object in the console:
Object {pages: Array[1], posts: Array[1]}
Dynamic Routes
Right now there are no routes set in the app other than the index route. If you have a few pages created in WordPress backend, chances are you want them to be available for the front end. To dynamically add routes to React Router we need to add another method in index.js
, let’s call it buildRoutes()
:
Include {this.buildRoutes(response)}
inside of React Router right after <Route path=”/” component={ Home } exact
/>. The method simply loops through all pages returned by the WordPress API and returns new routes. Notice how for each route it adds the “Home” component. This means that the “Home” component will be used for every route.
Let’s say in WordPress you have a page with a slug “about”. If you go to the route for that page “/about” it will load but you will still see the same “Hello World” message. In the case that you only need one template for every page, you can leave it as is and get page specific data by calling DataStore.getPageBySlug(slug)
and providing the current page slug.
In most cases, though, you would need to have multiple templates for different pages.
Page Templates
In order to use page templates we need to let React know what template to use with any given page. We can use the page slug that gets returned by the API to map templates to different routes.
Let’s assume we have two pages with slugs “home” and “about”. We need to create an object that maps page slugs to React component paths. Let’s name the object templates and include it in index.js
:
We also made updates to the buildRoutes()
method to require the right component. Don’t forget to create the About.js
component to map the “about” slug.
In order to get page-specific data, all you need to do is call the DataStore.getPageBySlug(slug)
method and provide the current page slug:
render() { let page = DataStore.getPageBySlug(‘about’);
return ( <div> <h1>{page.title.rendered}</h1> </div> );}
Dynamic Navigation
Now we’re going to add a global navigation that will reflect all WordPress backend page links. First create a Header.js
component in the “components” folder:
We get all pages from WordPress using DataStore.getAllPages()
, then we’re sorting them by “menu_order” with lodash and looping through them to spit out the React Router’s <Link
/>. Note that the homepage route is being excluded from the allPages array and included as a separate link.
Include the Header.js
component into index.js
and you’ll see the dynamic nav included on every page:
Advanced Custom Fields
Most WordPress developers are familiar with Advanced Custom Fields plugin. It makes the WordPress CMS fully customizable and user friendly. Fortunately, it’s very easy to access ACF data by utilizing WordPress API.
To get ACF data from API endpoints we need to install another plugin called ACF to REST API. This will include an acf property in the object returned by the WordPress API. You can access the acf fields like so:
render() { let page = DataStore.getPageBySlug(‘about’); let acf = page.acf; // Advanced Custom Fields data
return ( <div> <h1>{acf.yourCustomFieldName}</h1> </div> );}
Next Steps
All right, we’ve covered the most common use case for leveraging the WordPress CMS admin, along with a React front-end.
Some next steps might be to add styling for the project in Less or Sass. Or maybe extending the DataAction.js
file by adding additional API endpoint calls to pull more data like comments, categories, and taxonomies.
I strongly recommend checking out the official WordPress REST API handbook, where the API functionality is well documented. There you’ll find information about CRUD, pagination, authentication, querying, creating custom endpoints, and more. These resources will help extend what we’ve built here.