WordPress Single Post Templates

May 13th, 2008 Nathan Rice

UPDATED:
Austin recommends using a filter in your functions.php file as an alternative to the method below. IMO, his suggestion is much simpler and quite elegant. Here’s the code to add to your theme’s functions.php file. (be sure you paste this code between <?php ?> tags):

add_filter('single_template', create_function('$t', 'foreach( (array) get_the_category() as $cat ) { if ( file_exists(TEMPLATEPATH . "/single-{$cat->term_id}.php") ) return TEMPLATEPATH . "/single-{$cat->term_id}.php"; } return $t;' ));

It helps solve the multiple categories issue since it cycles through all the categories in the array and checks to see which one of them has an associated post template. When it finds one, it uses the post template file, but if it doesn’t, then it falls back on the default single.php template.

So if you are used to placing posts in multiple categories, be sure to only create post templates for the categories you know won’t ever conflict.

Again, I highly recommend using this technique. It’s much simpler and works much better. If you care to, however, the original article is still below.
END UPDATE

A few months ago, I wrote a post over at the Blog Herald explaining how to set up and use WordPress Page Templates to control the way indivudual WordPress “Pages” appeared on your blog.  The process was pretty simple … create the page template, insert the necessary code at the top of the file, and upload it to your themes folder.

What what about blog posts?  As far as I could tell, there’s no way to do a similar thing with single posts without some manual code (see the “Unique Single Template” section).

Is it possible to have post templates like page templates???

Cory sent me this simple question a few days ago, and it got me thinking. So I started investigating.

The Setup

You’re going to need 3 things:

  1. A basic knowledge of WordPress Themes
  2. A single.php file
  3. Multiple single templates, named according to category ID

The Code

What we’re going to do is create single post templates according to the ID of the category the post is filed under.  In order to determine what category a post is filed under, we’re going to use the get_the_category template tag.  Then we’re going to pull out the first category the post is filed under. Insert the following code at the TOP of your single.php file in your theme folder:

< ?php
$category = get_the_category();
$currentcat = $category[0]->cat_ID;
?>

Be aware, if you file posts under multiple categories, chances are this method will not work as you intended.

Then, we’ll add a little PHP magic to pull in the template file you created for a particular category:

< ?php
$category = get_the_category();
$currentcat = $category[0]->cat_ID;
if (file_exists(TEMPLATEPATH."/single-$currentcat".".php")) {
include(TEMPLATEPATH."/single-$currentcat".".php");
} else {
?>

This code defines the first category the post is filed under, checks to see if a single post template exists for that category (template files look like this: single-1.php, single-2.php, single-3.php, etc.), and uses it if it does.

Finally, we need to put the following code at the very bottom of the single.php file, normally just below the wp_footer function:

< ?php } ?>

The reason we have to do this is to be able to use the default single.php code as fallback in case you haven’t created a template for a certain category.

After you’ve got the code in place, just find the ID of the category you want to create a template for, then create a new file called single-ID.php (be sure to replace ID with the category number). Just insert the code you want to use for any posts filed in that category and you’re good to go!

Limitations

Like I mentioned earlier, if you are in the habit of filing posts under multiple categories, this code probably won’t work as intended. Since we’re working with templates for use on single posts in single categories, posts in multiple categories may end up falling back to the default single.php template. You have been warned!

Sample Code Download

As always, I like to provide a sample file for you to look at yourself. In this instance, I’ve edited the single.php file for the WordPress Default theme to pull in single post templates, if they exist. Take a look at the code if you have any questions about how it should look in your theme’s single.php file. I’ve also commented the code quite thoroughly.

Download single.php

The Tags: , , , , , , ,

14 Responses to “WordPress Single Post Templates”

Austin Said:

That’s an interesting approach, and it got me thinking that you could do the same thing with just the following one line added to your theme’s functions.php file:

add_filter('single_template', create_function('$t', 'foreach( (array) get_the_category() as $cat ) { if ( file_exists(TEMPLATEPATH . "/single-{$cat->term_id}.php") ) return TEMPLATEPATH . "/single-{$cat->term_id}.php"; } return $t;' ));

This should avoid the problem you mention when a post has multiple categories.

Comment made on May 13th, 2008 at 8:26 pm
Nathan Rice Said:

@Austin,
Forgive my ignorance … I’m not a plugin writer, so bear with me :-)

What is the “term_id”? It’s not in the codex doc for the “get_the_category” function.

Also, does this account for a default fallback? In the event that a single post template doesn’t exist for that particular category, does the theme just parse the single.php file as usual?

I’m not sure if this helps the “multiple categories” problem though. Won’t this just return the first (or perhaps last) category in the array?

Thanks!

Comment made on May 13th, 2008 at 10:02 pm
Austin Said:

What is the “term_id”?

That’s the category id. Since WP version 2.3 categories and tags are both “terms.”

Also, does this account for a default fallback?

Yes, if there is no such file, it just goes with the default single template.

I’m not sure if this helps the “multiple categories” problem though. Won’t this just return the first (or perhaps last) category in the array?

It will return the first category for which there is a corresponding single-[category id].php template.

Comment made on May 13th, 2008 at 10:16 pm
Nathan Rice Said:

That’s the category id. Since WP version 2.3 categories and tags are both “terms.”

Yes, if there is no such file, it just goes with the default single template.

It will return the first category for which there is a corresponding single-[category id].php template.

So, essentially it does everything my code does … just way better :-) I’m gonna add it to the top of this story as an alternate, better way of doing it. Thanks!!!

Comment made on May 13th, 2008 at 11:14 pm
Nathan Rice Said:

@Austin
Do you think it would be possible for WordPress to parse post templates like they do page templates and let you choose (on a per post basis) which post template they would like to use (again, just like the page templates work)?

That would be an interesting addition to the core, IMO.

Comment made on May 13th, 2008 at 11:31 pm
Austin Said:

That’s sort of what’s going on in the new Monotone theme: each post has a differently-styled template, based on the colors of its photo.

That’s automatically done, but I think you’re talking about something that would allow someone to manually select a template for individual posts. I don’t see that becoming part of core, because I think it wouldn’t be commonly used, and the core WordPress devs don’t like there to be too many options. :)

Comment made on May 14th, 2008 at 11:55 am
Nathan Rice Said:

@Austin:
I guess it would have to be a plugin. I can see it being useful, but definitely not useful enough to warrant inclusion in the core. Good point.

Comment made on May 14th, 2008 at 4:16 pm
James Beechinor Said:

This is excellent - top work both Nathan and Austin.

I was wondering though - would it be possible to do the same thing using tags, rather than categories?

Comment made on May 27th, 2008 at 3:58 pm
Kathy Said:

I like this approach. I’m using Brain Gardners vertigo theme, and there isn’t a sigle.php file in it. So I’m confused, how would I implement this?

Comment made on June 15th, 2008 at 3:24 pm
Maya Said:

Thank you! This was exactly what I was looking for. Worked out great.

Comment made on June 20th, 2008 at 11:20 am
Nathan Rice Said:

@Kathy:
I’ve always used index.php as my front page template, and single.php as my single post template, but Brian does it a little different …

He uses home.php as his front page, and index.php as his single post template. So just copy the index.php file, rename the copy as single.php and use then start the instructions.

Nathan

Comment made on July 5th, 2008 at 7:44 pm
Adam Kayce Said:

This is great - thanks for the code, and hats off to Austin for the upgrade.

My question is this: I’m trying to create a site that has four different categories (I’m controlling styles through the body tag), but I can’t find a way to link directly to the latest post in each category (from the home page or the nav bar, basically).

If I link to the category, I get a category/archive look (i.e. no comments). Is there a way to link directly to the single view of the latest post in a given category?

Thanks; if anyone has a solution, this’ll totally save my bacon.

Comment made on July 18th, 2008 at 9:12 am

Trackbacks

Ulanism » How-To create templates for individual posts using Wordpress Said:

[...] simply adore stuff like this. While it took me 15 or 20 minutes to figure out what they were talking about, once I did it was [...]

Comment made on July 2nd, 2008 at 6:46 am
Changes to w(t)f | wind(the)frog(dot)net Said:

[...] made the single post pages look correct as well. Full info on how to do that can be found here. Handy handy [...]

Comment made on July 18th, 2008 at 10:53 pm
 

Leave a Comment