Clifton Callender

Jekyll From Scratch - Extending Jekyll

Jekyll From Scratch - Extending Jekyll

In my previous posts I’ve covered the basics of Jekyll, and building a static website framework from the ground up. In this post, I’ll demonstrate ways in which you can enhance your otherwise-static Jekyll website or blog with interactive components like comment threads, site search, and contact forms, and I’ll share a few additional tips and tricks I’ve run into as well.

Table of Contents

Building Client-Side Interaction

Now that the framework is in place, we’ve got a fully functioning static website and blog. But what about the interactive component? What about the things that require server-side scripting and databases like comment threads or search? Well, many dynamic features of modern blogs can be incorporated through third party services nowadays. Here I’ll go over a few of them.

User Comments with Disqus

There are a few services out there offering embedded comment integration at the moment. I picked Disqus because it’s been around a while, has good adoption, and generally seems to work well. Close contenders include Livefyre and Facebook’s Comments Box plugin. Each has its pros and cons.

To implement Disqus comments, you simply sign up for an account and add some javascript to your templates. You can follow the instructions they give you, or you can just use my consolidated script below.

Include the following lines somewhere in your main template, replacing your_shortname with the shortname you were given when you signed up. Ensure this is embedded on every page where you’d want to embed disqus comments or list comment counts.

<script type="text/javascript">
	var disqus_shortname  = 'your_shortname',
	    disqus_identifier = '{{ page.path | split:'/' | last | cgi_escape }}',
	    disqus_url        = '{{ site.url }}{{ page.url | uri_escape }}'
	;
	(function() {
		var load = function(src){
			var s = document.createElement('script'); s.type = 'text/javascript'; s.async = true; s.src = src;
			var e = document.getElementsByTagName('script')[0]; e.parentNode.insertBefore(s, e);
		};
		load('//' + disqus_shortname + '.disqus.com/count.js');
		if (document.getElementById('disqus_thread')) {
			load('//' + disqus_shortname + '.disqus.com/embed.js');
		}
	})();
</script>

On any page where you’d want to display a comment thread, add the following line somewhere after page.content:

<div id="disqus_thread"></div>

Also wherever you want to include a link displaying the number of comments at the target url, include the following line:

<a href="{{ post.url }}#disqus_thread" data-disqus-identifier="{{ post.path | split:'/' | last | cgi_escape }}">View Comments</a>

The “View Comments” text will be replaced with “# Comments” when the script is loaded.

Important: if you aren’t looping over pages using a post variable like I am in this example, change post to whatever variable you’re using. If you’re linking to the comments on the current page, use page instead of post on both the href and data attributes.

In these examples I’ve utilized the disqus_url variable to ensure that comment threads are linked to the canonical url, and the disqus_identifier variable to tie the thread to the post filename. This way if you ever restructure your blog, the comment threads will still work so long as the post filenames are the same.

Social Widgets

Integrating social plugins into your website is pretty straight forward. Usually it’s just a little bit of javascript to add to a template. If you use any of these, this is where the rel="canonical" declaration that I suggested in the previous post can become important.

Facebook — Go to Facebook’s Developer Page to add “Like” buttons, “Share” widgets, and so on.

Twitter — Go to Twitter’s Widget Settings to create embedded tweet timelines, or favorites lists. Go to Twitter’s Resources Page to find “Tweet This”, “Follow Me”, or “@Mention Me” buttons.

LinkedIn — Go to LinkedIn’s Plugins Page to find “Follow” and “Share” buttons.

There’s also Reddit, Delicious and a billion others. I personally think they’re mostly clutter and fairly pointless unless you have a sizable audience, so I don’t intend to use them at the moment.

Contact Forms with jotForm

There are a lot of free contact form services out there. Wufoo, Formstack, 123ContactForm and jotForm are some popular ones just to name a few. One guy I saw even co-opted his GitHub Issue Tracker as a feedback/contact form.

I ended up using jotForm because it was fairly simple and included CAPTCHA support. To incorporate this onto my own contact page I created an account, made a simple form, and clicked the “Embed Form” button.

I opted for the “Source Code” option as opposed to “Embed” or “iFrame” because I wanted primary control over the CSS. The default stylesheet clashed with my fonts and wasn’t compatable with my responsive design, so I threw it out and used my own.

I was surprised to find that there are actually a few clever options for embedded interactive searching on static websites like Jekyll.

One option is a service called Tapir. This service will take an RSS or Atom feed for your site’s content, index it, and provide a keyword search api which you can interact with through a jQuery widget. I’m not exactly sure how they make money on this, but it’s a great service.

To integrate it, you’d simply take the Atom feed we created earlier, remove the post limit, and add the non-blogpost pages to create a full site feed for Tapir to digest. I don’t yet have enough content on my site to warrant a full keyword search, but I’ll definitely come back to this if I decide I need it in the future.

Another option is a pretty ingenious idea from Alex Pearce and Development Seed which involves creating a search.json document with Jekyll, populated with your site’s content including tags and other metadata which you can pull in through a custom javascript plugin.

User Analytics

No website is complete without the ability to track page hits and user engagement. Since GitHub Pages doesn’t offer access to server logs, using a third party tracker is your only option. I went with Google Analytics because it’s familiar, but any of the alternatives would do just fine as well. Enabling any of them is as simple as copying a few lines of javascript to the bottom of your site template.

Other Jekyll Tips and Tricks

Reading Time

This is a neat idea that I saw in a couple of places. Since Jekyll provides a handy number_of_words Liquid filter, you can estimate the reading time of your posts based on a rate of 180 words-per-minute with {{ page.content | number_of_words | divided_by:180 }}.

Since there’s no way to round a number using Liquid, the filter ends up spitting out the “floor” of the number (essentially it ignores everything after the decimal point). The original author added append: '.0' to get a long floating point number and then used JavaScript to round it up to a whole number, but that’s a really hacky solution and it’s actually unnecessary. You could just use plus:91 | divided_by:180 to trick it into rounding upward. I ended up using the following:

{% capture readtime %}{{ post.content | number_of_words | plus:91 | divided_by:180.0 | append:'.' | split:'.' | first }}{% endcapture %}
{% if readtime == '0' %} &lt; 1{% else %}{{ readtime }}{% endif %} min. read

Update — Aug 21st, 2014: A recent update to Liquid broke my original read time trick. It used to truncate everything after the decimal place, but now it will display uneven divisions as a fraction or a long floating point number. You can get around this by adding .0 to the divisor and then adding | append:'.' | split:'.' | first. I have updated the example code above. It’s ugly but it still works.

Similar Posts

Jekyll provides a built-in mechanism to identify and display “related” posts with the site.related_posts variable. This may be useful to display a handful of links for further reading below your current post. Alternatively, you could loop through similarly-tagged posts and list those using site.tags[tagname].

Site Last Updated Notice

You can use {{ site.time | date_to_string }} in your page footer to display a site-wide “last updated” time. This will be set to the most recent time Jekyll has compiled the blog (presumably on your last git push).

Table of Contents /w Maruku Kramdown

Update — May 8th, 2014: Earlier in 2014 Jekyll and GitHub Pages both announced that they are deprecating maruku as Jekyll’s default markdown compiler. Luckily, “kramdown” now has pretty solid feature parity including aspects like table of contents support.

If you're using "Maruku" as your Markdown interpreter (which at the moment is Jekyll's default), you can take advantage of a few extra features which it defines as a "superset" of the Markdown syntax. One feature which I've found handy for this post in particular is the ability to generate a linked table of contents using all of the headers in your document.

Adding a table of contents to your post is pretty simple; just include the following in your post somewhere:

* Table of Contents Placeholder
{:toc}

Strangely for this to work, you'll need an `h1` tag in your document somewhere. I don't like this requirement because I prefer to have my post title be `h1` and all post content using `h2` or lower. I got around it with a somewhat hacky solution. I just made sure to include `# Table of Contents` right before the lines above, and replaced it manually in my post template. I also removed the inline styling on the list it generates because I prefer to use my own CSS.

Static Asset Combination

As demonstrated by Development Seed and others, you can utilize liquid {% include file.ext %} tags to consolidate static assets for your site. Say you have a handful of javascript or css files which are included on each page. Simply move them to the _includes folder and create one consolidated file like so:

---
---
{% include file1.css %}
{% include file2.css %}
...

This will reduce the number of requests the browser has to make and speed up load times.

Conclusion

With that, I think I’ll close out my series of posts on Jekyll. I do intend to keep updating these posts as I run across new ideas and examples from other blogs. Please let me know if you run across any other handy tips or third-party services to add to this list in the comment thread.