It may be news to you, but there is a nifty resource called Quotes on Design that serves up interesting quotes about design, curated by our very own Chris Coyier.
Up to this point, Quotes on Design (QoD) used a bit of custom code to query the WordPress database and serve up quotes. This was used for the site itself, and for its API to allow use on external sites. With the excitement surrounding the upcoming WordPress JSON REST API, we thought it would be fun to rebuild the site to use the WP API instead of our own custom code.

Here’s what we’ll cover:
- Setting up the API
- Querying the API to grab a random quote
- Doing something interesting with that quote
- Customizing the API to remove unused information
- Using the new QoD JSON REST API to grab nifty quotes for your site
Warning! The WP JSON REST API is still in development, and the query interface could change before the final version. We’ll do our best to update this tutorial if anything changes. We wouldn’t recommend using the WP API on any super-important sites just yet, unless you’re a ultra-professional WordPress developer, in which case you probably can skip this article, right?
Installing the WP API
The WP API is packaged as a plugin, available on the WordPress.org plugin repository. That means you can
- Open your WordPress dashboard
- Go to Plugins → Add New
- Search for “WP API”
- Install, activate, and you’re done
Let’s start digging in!
Grabbing a random post
For QoD, we wanted to add some simple functionality to the home page: Click a button, and have a random quote reloaded from the API.
The WP API includes several endpoints you can use to read information from the database, including meta, attachments, users – really, anything that is stored in the standard WP database is accessible to some degree.
For our use case, we just needed to grab a random post. We started with the /wp-json/posts
endpoint, which is essentially like running a WP_Query
, if you’re familiar with WordPress development. Here’s the base endpoint URL:
http://quotesondesign.com/wp-json/posts
Visiting that URL without any query parameters returns the default list of posts in JSON format — essentially, the same list of posts you’d see on the blog page:
[
{
"ID": 2328,
"content": "<p>Everything we do communicates.</p>\n",
/* ...snip most of the fields... */
"title": "Pete Episcopo",
"type": "post"
},
{
"ID": 2326,
"content": "<p>The only “intuitive” interface is the nipple. After that it’s all learned.</p>\n",
/* ...snip most of the fields... */
"title": "Bruce Ediger",
"type": "post"
},
{
"ID": 2323,
"content": "<p>Only show work you like, or you’ll end up being hired to do things you don’t like.</p>\n",
/* ...snip most of the fields... */
"title": "Victoria Pater",
"type": "post"
}
]
Note that the IDs of the quotes are sequential: they represent the three most recent quotes on the site.
For our design, we only need one post and we want it to be randomly selected. We need to filter the results, and we can do this using the same arguments we would use when calling WP_Query
or get_posts
.
In this case, we need two arguments:
- The
orderby
argument needs to berand
, to fetch a random post - The
posts_per_page
argument needs to be1
, to limit the results to a single post
See a list of all available parameters on the WP_Query codex entry.
We pass those arguments through the URL, using the filter
query variable as an array, like so:
http://quotesondesign.com/wp-json/posts?filter[orderby]=rand&filter[posts_per_page]=1
Here’s what we get back from the WP API using our filtered URL:
[
{
"ID": 2153,
"author": {
"ID": 2,
"URL": "https://css-tricks.com",
"avatar": "http://0.gravatar.com/avatar/8081b26e05bb4354f7d65ffc34cbbd67?s=96",
"description": "",
"first_name": "Chris",
"last_name": "Coyier",
/* ...snip some fields... */
},
"comment_status": "closed",
"content": "<p>Wonderfully designed > beautifully designed. Any day.</p>\n",
"date": "2013-06-27T08:34:04",
"date_gmt": "2013-06-27T16:34:04",
"date_tz": "Etc/GMT-8",
"excerpt": "<p>Wonderfully designed > beautifully designed. Any day.</p>\n",
"featured_image": null,
"format": "standard",
"guid": "http://quotesondesign.com/?p=2153",
"link": "http://quotesondesign.com/daniel-burka-2/",
"menu_order": 0,
"meta": {
/* ...snip some fields... */
},
"modified": "2013-06-27T08:34:04",
"modified_gmt": "2013-06-27T16:34:04",
"modified_tz": "Etc/GMT-8",
"parent": null,
"ping_status": "closed",
"slug": "daniel-burka-2",
"status": "publish",
"sticky": false,
"terms": {
/* ...snip some fields... */
},
"title": "Daniel Burka",
"type": "post"
}
]
Every time we refresh our filtered URL, we’ll get a different quote. Sweet!
You’ll notice that I’ve included more of the WP API JSON response in the last snippet. There’s a lot of information there — even when I hide the meta, author and terms.
For our purposes, we don’t need most of that information. Plus it’s more bandwidth-efficient to not include it all – so let’s chop it out!
Removing fields from the JSON response
A reader wrote it to say you don’t have to do this all anymore as there is a _filter
argument in the API now.
We need to filter the data being returned by the API, and fortunately the WP API provides us with exactly the filter we need, called json_prepare_post
. To use it, we create a new plugin with the following snippet of code:
function qod_remove_extra_data( $data, $post, $context ) {
// We only want to modify the 'view' context, for reading posts
if ( $context !== 'view' || is_wp_error( $data ) ) {
return $data;
}
// Here, we unset any data we don't want to see on the front end:
unset( $data['author'] );
unset( $data['status'] );
// continue unsetting whatever other fields you want
return $data;
}
add_filter( 'json_prepare_post', 'qod_remove_extra_data', 12, 3 );
Activating our plugin leaves us with a much smaller JSON response, which only contains the info we need for our use case:
[
{
"ID": 686,
"content": "<p>My secret is being not terrible at a lot of things. </p>\n",
"link": "http://quotesondesign.com/moby/",
"title": "Moby"
}
]
Adding fields to the JSON response
Now that we’re done removing unnecessary information from the JSON data, it’s only fitting that we add some of our own custom fields to stay balanced. Wax on, wax off.
Some of the quotes have a custom meta field called “Source”, which Chris uses to link to the original source of the quote, if it is available online. Let’s add that meta data to the JSON response using the same json_prepare_post
filter we used to remove data previously. In the same plugin, we add the following function:
function qod_add_custom_meta_to_posts( $data, $post, $context ) {
// We only want to modify the 'view' context, for reading posts
if ( $context !== 'view' || is_wp_error( $data ) ) {
return $data;
}
$source = get_post_meta( $post['ID'], 'Source', true );
if ( ! empty( $source ) ) {
$data['custom_meta'] = array( 'Source' => $source );
}
return $data;
}
add_filter( 'json_prepare_post', 'qod_add_custom_meta_to_posts', 10, 3 );
With that extra field appended, here’s what a JSON response looks like:
[
{
"ID": 2039,
"content": "<p>Communication that doesn’t take a chance doesn’t stand a chance.</p>\n",
"custom_meta": {
"Source": "<a href="http://altpick.com/spot/segura/segura.php">article</a>"
},
"link": "http://quotesondesign.com/carlos-segura/",
"title": "Carlos Segura"
}
]
Now that we have exactly the information we need, let’s use the JSON API to load a random quote via AJAX on the home page.
Fetching and using data from the JSON API
On the homepage of QoD, we have a button to grab another quote:

Using jQuery event handlers and the jQuery.ajax function, we grab a quote and update the HTML with the following snippet:
jQuery( function( $ ) {
$( '#get-another-quote-button' ).on( 'click', function ( e ) {
e.preventDefault();
$.ajax( {
url: '/wp-json/posts?filter[orderby]=rand&filter[posts_per_page]=1',
success: function ( data ) {
var post = data.shift(); // The data is an array of posts. Grab the first one.
$( '#quote-title' ).text( post.title );
$( '#quote-content' ).html( post.content );
// If the Source is available, use it. Otherwise hide it.
if ( typeof post.custom_meta !== 'undefined' && typeof post.custom_meta.Source !== 'undefined' ) {
$( '#quote-source' ).html( 'Source:' + post.custom_meta.Source );
} else {
$( '#quote-source' ).text( '' );
}
},
cache: false
} );
} );
} );
Now, when the button is pressed a new random quote is automagically loaded. Try it out.
Using the API on your site
You can query Quotes on Design using the API to show nifty quotes for your own site. We’ve updated the API page to use the new WP JSON API. As mentioned earlier, the WP API is subject to change, so don’t rely on this API for uber-important sites just yet.
More API goodness is coming
Development on the WP API is ongoing, and there is a lot of excitement to see what folks can build with it. Our example is pretty basic, but the thought of a JSON API has already gotten our gears turning about the possibilities.
Check out Quotes on Design and share your thoughts and questions in the comments. And if you have a great quote you’d like to see on the site, submit it!
Hey Andy, thanks for sharing this! It’s nice to see an example of the API in the wild.
Fwiw, from a practical standpoint, the API is actually at version 1.2.x for the stable version, and the version that should get proposed for core inclusion will be version 2.x, and 2.0 is in beta right now. The article notes it’s not yet at 1.0, which isn’t accurate.
There are lots of awesome potential use cases for the API and I’m excited to see this as another way folks are using it. I’m using it on my own site for an ad delivery system, actually. I think we’ll see all sorts of exciting use cases in the future.
Thanks Brian, you’re absolutely right: 1.0 is old news. Fixed!
This was a pretty basic example of the API, and I’m sure we’ll start being blown away by what folks do with it.
Without wishing to be disparaging – instead of the regular WordPress articles of late – if there’s going to be non-CSS-related code here in that sphere, I’d really like it to be related to static site generators. I’d really love it if more web designers/developers were familiar with them so that we can maybe get a few more people away from WordPress and that ilk, and shut down the repeated crops of botnets that result from their security vulnerabilities (and the even more frequent vulnerabilities in their poorly written plugins).
that’s a valid point. That is why I plan to use only the wordpress backoffice and recreate a fonrt office, free from WP plugins, themes, and vulnerabilities) as depicted in this other post on CSS tricks https://css-tricks.com/thoughts-on-an-api-first-wordpress/
I have a question about his method.
Each time you click the button on the website, an http request is made to the WP API. Am I right ?
I am planning on using WordPress as an API for my website(s) but I can’t imagine doing that. Because let’s say the Quotes on Design website becomes very popular. So popular that, say, 4000+ people click on the button at the same time, and repeatedly. The API calls would probably crash the server where WordPress is installed on.
What would be good solution to “cache” thoses API calls ? Should we imagein to make a CRON job that would automaticaly fetch all (and then only new quotes) – and put them on a local database or file storage ?) and make the random choice of quote directly on QoD’s server ?
Is there any system already existing that can help me manage API calls like this ?
Thanks a lot !
Yes, you would probably want to cache each unique URL, and invalidate the cache whenever posts related to that URL are updated. I don’t know if existing caching plugins handle API URLs out of the box, but I’d expect them to handle API URLs once the API gets more traction.
Thanks for the article!
I just wanted to add that when I wrote my thoughts on an API-first WordPress here on CSS-Tricks, I mentioned a couple of features that I couldn’t find on the WP-API which led me to work on my custom solution, and the main one being the ability to request multiple batches of content with a single request.
For those interested in using the WP-API while still having that feature, I’ve created a plugin that extends the WP-API implementing that functionality. It’s available on GitHub.
Hi,
I try to run your code with WP-API v 2.0 and it’s not working. I just tried to unset some keys from the $data object, but making call to the endpoint still returns them.
We’re running the latest stable version. 2.0 is in development still – but we’ll (try to) update this post when 2.0 comes out!
Fantastic resource. I see loads of documentation on the REST API, but it’s nice to find something at a level that lets API novices like me actually try a working project.
It’ll be fun to see how easily I can integrate this into an Angular project..
Hi Andy,
Very nice idea. But I am wondering how can I use my custom quotes here.
Hey Daniel,
You’d probably need to set up your own “Quotes on Design” WordPress site – unless I’m misunderstanding your goal?
This is very helpful, would it be possible to shed some light on how you achieved the pretty URLs as well on each click? as I couldn’t see the method through the code.
What is interesting there is I thought this method would not be so great for SEO as the pages might not get indexed, but it seems that I was wrong, ran a quick “site:http://quotesondesign.com/” search on google and all the quote seems to be indexed which is great news.
I am wondering if Yoast took care of that part?
I think that part was handled by Chris – each quote lives on it’s own URL, but he used some JS to replace the URL:
I like to share you this addon than we are currently working on. Is for using WordPress as a backend for Ember.
https://www.npmjs.com/package/ember-cli-to-wp-theme
Just wanted to pop in and let you know that the tweet links on QoS break if the quote has an unescaped apostrophe in it – example: http://quotesondesign.com/nate-simpson/
Correcting myself… The behavior is actually a little bit weird. If I load a quote and click the tweet link, it doesn’t show the full excerpt. However if I reload the page and click the tweet link again, it populates the tweet correctly. The tweet also works properly if I refresh the page once before ever clicking.
Thanks for this post. Very useful!
One thing i noticed is that while the url gets updated (which is cool), the back button isn’t working, which would be nice too to be able to go back to the last quote.
Hey Andy! Its a very helpful post, I was having an error in fetching data using the wp api but your post solved my problem. Thank you for sharing this post with us.
Hi Andy,
Very very helpful post thank you so much
but i have only one issue with WP API i have to use private query filter to use offset parameter but i can’t understand the Authentication if you have any solution this well be very helpful
thanks
I don’t think I’ll be much help other than linking to the WP API documentation on Authentication. You could also try posting your question on the WordPress StackExchange – just make sure to give lots of detail.
Thanks you so much
re: Yes, you would probably want to cache each unique URL,
Is it possible to use transients with the API? with “small” repetitive request I typically prefer to transient the results so I’m not going to the DB for next to nothing again and again and again.
Yes, absolutely! You could also use a caching plugin (which would avoid the DB query for a transient) assuming your unique API URLs were sending the same content on every request.
Hi Andy,
I’ve tried to use your code, but I can’t remove author data. I’m able to remove simple key:value data, but when it is an array like in author – your method is not working. Do you have any clue?
Hey Lucas! I don’t have a clue – I would figure
unset
would do the trick. Can you post a snippet here or on a Gist so we can dig through it?Hey Lucas, I commented on the gist. Seems to be working for me, but we can dig in!
Hey – here is my code: https://gist.github.com/anonymous/1375b3f28d125354e5ff
Hey Andy I am wondering if you know how to unset $data for a specific post type? The above approach works for post but if I add the post type the unset doesn’t work. i.e. /posts/?type=post_type
Are you sure your code is being executed? Try putting
var_dump( $data ); exit;
in the place you’re trying to unset and visit the page. Did the code get there? If not, where is it stopping? If so, what is the state of the$data
variable? That’s where I’d start.Sorry so I am trying to unset specific values for a curl. I did a var dump and all of the content passes through. I am using this line. curl -i localhost/wp-json/posts?type=custom_post_type. Custom Post Type is a custom post type that I created. If I just do a curl on posts the unset content is removed but if I do the version with ?type=custom_post_type the unset values still show up.
I am assuming that the var dump is not working on the /posts?type=custom_post_type. I just do not know why or if there is something specific I need to add to the function that you wrote above to make it work with specific post types.