Angular Framework Angular is an open source framework that allows the development of risky and robust web applications. Being a Google product, angular is well-known for end-to-end application development. All the big names like Google, Forbes, Wh. Working with the Angular.js framework is fast and rewarding, and combined with WordPress it can make a really nice SPA (Single-page Application) in a short time-span. With all the CMS controls and plugins WordPress offers, this is a interesting short-cut. Setting Up the Theme. Angular is part of the well-known MEAN stack that combines Angular with MongoDB, ExpressJS, and NodeJS. Similar to the MERN stack, it relies entirely on JavaScript for both the front-end and back-end. Common Technologies. Angular, React, and Vue can all be used to develop Progressive Web Apps, also known as PWAs.
The tutorial will show you how to setup and work with Angular.js inside of a WordPress plugin to create a stand-alone, API powered, Angular.js single page app or microsite that is available at a defined path of a WordPress website.
By the end of this tutorial you should grasp a few key things about working with Angular.js inside of WordPress. Using a self-installed WordPress install as a backend API service for serving HTTP requests to a front-end Angular.js client is a powerful combination. If you’re an eager beaver you can dive right into the sample plugin on GitHub which provides the source for the working demo. This example app should demonstrate the following concepts I commonly come across when building Angular apps in WordPress:
- How to work with HTML5 pushState routing
- Creating custom API endpoints that are consumed by Angular’s
$resource
service - Work with gulp to compile your front-end’s LESS, JS and more with a build process
- Auto-version your CSS and JS includes
- Setup browser cache rules for
/api/**
routes to reduce requests per second (when it makes sense to) - How to handle it all inside of an isolated WordPress plugin
If there’s a common scenario I didn’t mention that you would like to recommend please let me know in the blog post comments.
Table of Contents
- WordPress Plugin
- Why a plugin and not a theme?
- Server-side routing
- Auto-versioning CSS/JS
- API Requests to WordPress
- Angular Front-end
- Index template
- Front-end routing
- Concatenating JS includes
- HTML5
pushState
routing
- Gulp Build
- Watch & Build CSS/JS
- JS Compile Tasks
- LESS Compile Tasks
WordPress Plugin
To begin we we’ll start by creating a fresh WordPress plugin. I’m assuming you know how to do this already, but if not then I recommend reading the detailed plugin guidelines provided on WordPress.org. This plugin will serve as a backend to our angular app, and will handling the following logic for us:
- Add routing rules to WordPress for serving custom API endpoints
- Add routing rules to WordPress to load our base app index template at
/wordpress-angular-plugin/**
. The trailing wildcard is critical for supporting HTML5 pushState routing within our Angular app. - Process and cache HTTP API requests to third-party providers on the server-side (i.e. the US National Weather Service)
Why a plugin and not a theme?
There are a few key benefits to building both the backend and front-end inside of a single WordPress plugin:
- Simplicity: I don’t have to support two separate servers, one for a WordPress backend API and another for serving the Angular.js app’s HTML. With this approach I can easily do both from one environment.
- Access to WP: I’ve found that it’s useful to have easy server-side access to WordPress when working with it as a backend. A few scenarios include processing Gravity Forms submissions, passing values from server-side to client-side with
wp_localize_script()
when users are logged in, and various other things. - Portability By isolating everything into a WordPress plugin we can easily move our entire app from site to site, enabling and disabling on demand.
All of the logic described in this tutorial could be used within a WordPress theme as well, the same concepts apply.
Server-side routing
Our WordPress plugin defines the URL where our Angular app will load based on the path of the plugin directory. The intercept_wp_router()
method is applied to the do_parse_request
filter to handle this:
If you want to change the base URL for your app to something custom you’ll need to change the value of the public variable base_href
. This is set in the __constuct()
method of the ngApp
Class. That’s a mouthful, but basically you would find and modify this line within the plugin:
In the case of this tutorial, our main plugin file is wordpress-angular-plugin/wordpress-angular-plugin.php
so the Angular app will load at /wordpress-angular-plugin/
out of the box. You can change this whatever you like in the plugin to customize the URL.
Once you load up https://www.yoursite.com/wordpress-angular-plugin/
you should see the same Angular app demo currently available at:
Auto-versioning CSS/JS
Google Chrome and other browsers will cache our *.css and *.js for an indefinite period of time. If we make changes to our Angular app’s code, or our LESS stylesheet, browsers won’t know the file has changed and could serve the old, previously cached version of the file to repeat visitors. For this reason, it’s very important that we add version strings to static resources, or in our case the /dist/js/main.js
and /dist/css/main.css
files. This is especially important for single page apps because we are effectively loading EVERYTHING in the browser.
Luckily, I’ve included a setup in this plugin that will handle this for you automatically. This is the only thing that PHP is actually used for in the index.php
template.
Here’s the method that handles this for us:
Using PHP’s filemtime()
function we check the modified time of the CSS and JS files, and then we add the timestamp returned to the end of each file as a “version string” like this:
/dist/css/main.css?v=1497114238
/dist/js/main.js?v=1497114238
Now you’ll always have up to date assets loading for your users!
API Requests to WordPress
Now that we have the basic structure for an Angular app setup within a WordPress plugin let’s look at how to make requests from client (Angular) to server (WordPress) by defining a few custom HTTP API routes. For demo purposes I’ve wired together a backend API that will:
- Handle incoming requests to
/api/weather/{latitude:longitude}/
- Lookup a weather forecast for the provided lat/long using the National Weather Service API
- Return the response body as JSON back to the client
In a real-world scenario we could just do this inside of Angular entirely, but it serves as a good example to cover common situations where:
- You want to make requests to a secured API without exposing keys on the client-side
- You want to cache the results of remote API responses locally to server faster responses, and avoid overage costs for API’s where you pay by the request
- You want to serve cached results of remote API requests that have been performed by a worker process (bonus points for performance)
In the context of our demo app, an API request is made to the weather API endpoint defined in our plugin to retreive the weather for a users location. This provides a basic demonstration of how to write a PHP backend service that processes input from our Angular app.
Important Note
I am deliberately NOT using the official WP REST API here. I personally believe in building minimal systems that solve specific problems, I think it make a big difference in terms of maintenance, sustainability and security. This is entirely my own opinion, but I beleive it’s better to build your own microservices like this rather than load in the entire WP JSON api for many circumstances.
Angular App Front-end
The Angular.js app is currently using Angular 1. Living earth. Most of the project I work on at the moment are still using Angular 1, only a few have made the switch to 2. Because some of this code is directly pulled from those projects I found it easier to work with Angular 1. In the future I will update this to Angular 2 on another branch.
The Angular 1 app in this demo is very basic, but it does handle a few important things well:
- Concatenates, minifies and proressively enhances LESS and JS using Gulp
- Provides HTML5 pushState routing – NO hashbangs here (e.g. www.you.com/pathname/#/slug)
- Handles 404’s beneath our plugin’s top level URL path
- Route separation – Individual routes/views/controllers are broken down into separate files. When Angular projects get large and span multiple developers this is very helpful.
- Uses the
$resource
service to interface with a custom HTTP API we’ll define in WordPress - Doesn’t rely on the WordPress JSON API in anyway
- Provides access to the Lodash as an Angular service and template helper
Index template
Our Angular app is served from a single file: ./views/index.php
. This is where we define our [ng-app]
, the structure of our HTML doc, and the <ui-view />
directive provided by ui-router. This tag will specify where every view inside of our Angular app will load when the URL changes or initially loads.
Using *.php and not *.html allows us to easily pass values from WordPress into Angular by adding them to an inline JSON object before we load main.js
. This is a similar approadch to using the wp_localize_script()
function in WordPress to pass PHP values from a plugin or theme into JS.
Front-end Routing
The Angular app is served from a single file within the plugin: ./views/index.php
. This is where we define our [ng-app]
, the structure of our HTML doc, and the <ui-view />
directive provided by ui-router. This tag will specify where every view inside of our Angular app will load when the URL changes or initially loads at or below the app path defined by our plugin.
This means that anything underneath this URL is handled by Angular.js routing and the ui-router module. We’ve specifically setup the way we route to and load in our /views/index.php
file to support this structure. This will handle the WordPress angular routing in a seamless way so that:
- If you visit the URL of an Angular defined route directly it will load in the expected route/view/controller configuration
- When you browse from view-to-view within the app fully qualified URL’s will be loaded into the browser using the HTML5 pushState API
It is only possible to do this if EVERYTHING underneath our /wordpress-angular-plugin/
page is handled by Angular and UI router.
UI-Router defines routes for everything served by Angular undereath the /wordpress-angular-plugin/
directory. Here’s what a typical route definition looks like.
This is pulled right from the contents of the /views/home.html
template. The mainApp
is our main ng module defined in /js/main.js
with angular.module()
.
Concatenating JS Includes
When you look at the source of /js/main.js
you’ll see a number of lines that look like this:
These define external JS files that we want to include in our JS app. These are handled by gulp during the compile process with the help of the gulp-include plugin. For now I find that this approach is straight forward and easier to work with than Webpack or CommonJS, but if you find that your /dist/js/main.js
file is getting too large then it may be best to work with Webpack instead.
More information on the build process can be found in the compiling with gulp section.
Compiling with Gulp
The plugin uses Gulp to compile our CSS/JS and also prepares our HTML, with a few smart additions.
Watch & Build CSS/JS
I recommend that you use the watch process to build your CSS/JS on-demand as it changes. This provides a faster web development workflow. To start gulp in watch mode open a Terminal window and go to the directory where you’ve cloned this plugin. Enter the gulp
command in your prompt and Gulp will begin watching for JS/CSS changes.
Samsung kies lite download. If you’ve just cloned the repo you can build everything with the gulp build
command. This will run through every process needed to everything we need to serve our micro app (CSS/JS/HTML).
JS Compile Task
Wordpress Angular Template
The js-main
gulp task uses a few gulp plugins to compile and prepare our JavaScript files for the app. This part of the turorial is entirely opinionated, it’s a set of tasks that I commonly use so I’ve included them here.
- Sourcemaps will be generated for easier debugging
- The gulp-include plugin will provide an interface similar to CodeKit, allowing you to concatenate multiple JS files into a single compiled file
- The ng-annotate plugin adds and removes AngularJS dependency injection annotations.
- Uglify handles minification to compress our source
- Caching is added to help speed up Uglify minification
Angular Vs Wordpress
LESS Compile Task
The less
task is pretty straight forward. It compiles LESS to CSS, with the following helpful additions:
- Backwards support to automatically add browser prefixes with autoprefixer
- Concatenation and minification handled by cleanCSS
Conclusion
The combination of an Angular.js front-end and a WordPress API backend provides a powerful framework for building all kinds of find things. Hopefully this tutorial gives you a few ideas about how work with the two technologies in your projects. I do myself all the time. If you have any questions, comments or feedback please let me know in the comments below.
Meet the Author
I’ve been recently learning Angular so that I can incorporate it into a PHP application. I was surprised to find, though, that despite Angular’s popularity it is difficult to find resources on the best way to include/embed an Angular app inside an existing PHP page. After a couple hours of searching and consulting other users on StackOverflow, I figured out a workable solution.
Am I the only one doing this?
From the get-go, I’ve been quite puzzled by the lack of resource on this topic. Of course, you’ll find a handful of resources across the web claiming that they show how to integrate Angular with PHP (in some cases with Laravel, for example), but in the end you’ll realize that they’re talking about the old-fashioned AngularJS, not the latest and greatest Angular (which is, at the time of this writing, at version 7).
As a result, when I finally turned to StackOverflow, I was half-expecting to hear “you should never include Angular inside a PHP application, and here’s why…” Honestly, that would have been just as informative to me.
Instead, though, I got some useful answers. Here are two options that I think are pretty good.
Option 1: Ditch your app’s index.html
and include its contents instead
Turns out, all that your app really needs in order to work on your PHP page is (all credit to StackOverflow user ivanivan for this):
- The presence of
<app-root></app-root>
- Your app’s script files
Adding the <app-root>
tags to your PHP page is no sweat, but by default it’s difficult to include the script files because the Angular compiler creates dynamic file names. To turn those dynamic file names off, you’ll want to set 'outputHashing'
to 'none'
in angular.json
(credit to StackOverflow user user184994).
Once you’ve taken care of that, you can use a simple PHP function like this to simplify inserting those two prerequisites:
Then, you can use it in your PHP file like this:
Of course, if you’ve chosen to put other complex elements or script includes inside your app’s `index.html` file, you’ll want to update this helper function to include those files as well.
Option 2: Unwrap the body tags with PHP
In the comments of the aforementioned StackOverflow page, someone suggested just using PHP’s include()
and including the entire index.html
file inside PHP. That poses some problems because while you can remove the <html>
and <head>
tags just fine, Angular requires you to have a <body>
tag in that file.
In order to make this work, then, you’ll want to:
- Make sure that the
<body></body>
tags are the outermost tags in yourapp.component.html
file. - Strip the body tags from
index.html
at runtime via PHP
This helper function will strip the body tags:
I prefer the first option I showed, though.
Anyways, that about wraps it up!