6 min read

Jekyll KaTeX Block

Table of Contents
⚠️

This post was migrated from an old version of my blog, which used Jekyll. The content is likely to be out of date and the rendering will be broken, but I have kept it here so that I don’t break any permalinks.

The newer incarnation of my blog uses Astro instead, with support for KaTeX\KaTeX support providing via remark-math and rehype-katex.

If you follow Hacker News, then you might’ve seen the latest development from Khan Academy: KaTeX.

Although it hasn’t yet implemented the full set of mathematical symbols, KaTeX\KaTeX promises to replace MathJax as the web technology for displaying mathematical expressions. This plugin utilises Jekyll’s plugin system to enable you to easily add mathematical equations to your statically generated site, by adding a Liquid block corresponding to content to be rendered by KaTeX.

It’s as simple as adding this plugin to your _plugins folder, pointing the plugin to your KaTeX\KaTeX JavaScript file in _config.yml, and including the KaTeX\KaTeX CSS in the resulting web pages.

For more information, continue reading, or if you want to just go straight to the code, head over to https://github.com/drewsberry/jekyll-katex-block.

#

Why KaTeX\KaTeX?

So why should you use KaTeX\KaTeX? Well, here’s a few reasons:

  • It’s super fast
  • It supports server side rendering
  • Pure CSS (the JS is only necessary is you’re rendering client-side)
  • It’s super fast.
#

The Jekyll KaTeX\KaTeX plugin

So how does this Jekyll plugin fit in?

Well, there are three types of Jekyll plugins: Generators Converters Tags

A subset of the Tags plugin is the Block plugin. What this does is provides access to a Liquid block, that takes the content inside the block and does with it as you specify in your plugin. Here’s an example of a Liquid block:

1
{% myblock %}
2
This is the content inside the block!
3
{% endmyblock %}

The Jekyll KaTeX block then provides the {% latex %} block, which uses KaTeX to compile the contents of the block into an equation. So this is how you’d use it in your post:

1
{% latex %}
2
E = \gamma mc^2
3
{% endlatex %}

In addition, you can pass tokens to blocks like so:

1
{% myblock mytoken %}
2
Block contents
3
{% endmyblock %}

So if you pass the centred token to the latex block like so:

1
{% latex centred %}
2
E = \gamma mc^2
3
{% endlatex %}

Then the equation will also be cented on the page, with some space above and below. This is achieved using a little bit of inline CSS injected into the equation.

#

How to install it

If you’ve already got the KaTeX\KaTeX JavaScript installed on your system, as well as the CSS files and the font files, then all you need to do is drop jekyll-katex-block.rb into your _plugins directory in your project, and you’re ready to go!

Just make sure that you’re loading the CSS in your web page (but not the JavaScript) and have the font files available.

#

Options

You can specify where to tell the plugin to look for your JavaScript file in your _config.yml with the following:

1
katex:
2
path_to_js: "./the/path/to/your/js"
#

Troubleshooting

If you’re seeing a bunch of weird symbols instead of your equations, then it’s probably because you’re not using UTF-8. This is what it’ll look like:

without utf-8, your browser will get very confused

Put the following in the head of your HTML to sort it out:

1
<meta charset="utf-8" />

If you’re seeing your equations, but they don’t have any special formatting and just look like weird, squished, ugly versions of what you want, like this:

this is what happens if you don't have the font files in the right place

Then it’s probably because your server can’t find the font files. You can test this out by doing the following:

Terminal window
1
$ jekyll build
2
$ cd _site
3
$ python -m http.server

Go to your web page in question with your web browser, and look at the output of your Python session. It should say something like:

1
Serving HTTP on 0.0.0.0 port 8000 ...
2
127.0.0.1 - - [04/Nov/2014 12:18:07] "GET / HTTP/1.1" 200 -

But if your server can’t find the font files, it’ll output the following:

1
127.0.0.1 - - [04/Nov/2014 12:23:26] code 404, message File not found
2
127.0.0.1 - - [04/Nov/2014 12:23:26] "GET /fonts/KaTeX_Math-Italic.woff HTTP/1.1" 404 -
3
127.0.0.1 - - [04/Nov/2014 12:23:26] code 404, message File not found
4
127.0.0.1 - - [04/Nov/2014 12:23:26] "GET /fonts/KaTeX_Size1-Regular.woff HTTP/1.1" 404 -
5
127.0.0.1 - - [04/Nov/2014 12:23:26] code 404, message File not found
6
127.0.0.1 - - [04/Nov/2014 12:23:26] "GET /fonts/KaTeX_Size3-Regular.woff HTTP/1

To solve this, put all your KaTeX\KaTeX font files in in the same directory as your KaTeX\KaTeX CSS file, that way it’ll automatically look for and find it. If you’re still having problems, have a look at the example Jekyll site on the GitHub repo, or comment below.

Now you should be back to how it’s supposed to look:

K0=mihtem(xx)2ih2πtK_0 = \sqrt{\frac{m}{ih t}} e^{-\frac{m(x-x')^2}{i\frac{h}{2\pi} t}} #

How it works

The plugin basically works by compiling KaTeX\KaTeX within the Ruby script using the ExecJS module. The JavaScript function renderToString is then called from this compiled JavaScript on the content inside the block. This converts the content from LaTeX to HTML.

The Ruby code to actually do this looks like this:

1
# Compile the KaTeX JavaScript
2
katexsrc = open(path_to_katex_js).read
3
@katex = ExecJS.compile(katexsrc)
4
5
# Turn a LaTeX string into HTML equation
6
equation_string = "E = \\gamma mc^2"
7
html = @katex.call("katex.renderToString", equation_string)

The @ sign indicates that the katex variable is an instance variable, which basically means that it’s available to all the methods within the current class. The block itself is a class inheriting from Liquid::Block. It’s a member of the Jekyll::Tags module, which makes it accessible to Jekyll on runtime.

As with all tags, it then needs to be registered. This is where you tell Liquid what identifier the tag should have, i.e. what string inside the and brackets should activate the plugin. This looks like the following:

1
Liquid::Template.register_tag('latex', Jekyll::Tags::KatexBlock)

Where KatexBlock is the KaTeX\KaTeX block class.

To see the full code, head over to the GitHub repository.