Zola borrows the concept of shortcodes from WordPress.
In our case, a shortcode corresponds to a template defined in the templates/shortcodes
directory that can be used in a Markdown file.
Broadly speaking, Zola's shortcodes cover two distinct use cases:
The latter may also be solved by writing HTML, however Zola allows the use of Markdown based shortcodes which end in .md
rather than .html
. This may be particularly useful if you want to include headings generated by the shortcode in the
table of contents.
If you want to use something similar to shortcodes in your templates, you can use Tera macros. They are functions or components that you can call to return some text.
Let's write a shortcode to embed YouTube videos as an example.
In a file called youtube.html
in the templates/shortcodes
directory, paste the
following:
<div {% if class %}class="{{class}}"{% endif %}>
<iframe
src="https://www.youtube.com/embed/{{id}}{% if autoplay %}?autoplay=1{% endif %}"
webkitallowfullscreen
mozallowfullscreen
allowfullscreen>
</iframe>
</div>
This template is very straightforward: an iframe pointing to the YouTube embed URL wrapped in a <div>
.
In terms of input, this shortcode expects at least one variable: id
(example here).
Because the other variables are in an if
statement, they are optional.
That's it. Zola will now recognise this template as a shortcode named youtube
(the filename minus the .html
extension).
The Markdown renderer will wrap an inline HTML node such as <a>
or <span>
into a paragraph.
If you want to disable this behaviour, wrap your shortcode in a <div>
.
A Markdown based shortcode in turn will be treated as if what it returned was part of the page's body. If we create
books.md
in templates/shortcodes
for example:
{% set data = load_data(path=path) -%}
{% for book in data.books %}
### {{ book.title }}
{{ book.description | safe }}
{% endfor %}
This will create a shortcode books
with the argument path
pointing to a .toml
file where it loads lists of books with
titles and descriptions. They will flow with the rest of the document in which books
is called.
Shortcodes are rendered before the page's Markdown is parsed so they don't have access to the page's table of contents.
Because of that, you also cannot use the get_page
/ get_section
/ get_taxonomy
/ get_taxonomy_term
global functions. It might work while
running zola serve
because it has been loaded but it will fail during zola build
.
There are two kinds of shortcodes:
In both cases, the arguments must be named and they will all be passed to the template. Parentheses are mandatory even if there are no arguments.
Note that while shortcodes look like normal Tera expressions, they are not Tera at all -- they can pretty much just shuttle arguments to their template. Several limitions of note are:
If the shortcode is invalid, it will not be interpreted by the markdown parser and will instead get rendered directly into the final HTML.
Lastly, a shortcode name (and thus the corresponding .html
file) as well as any argument names
can only contain numbers, letters and underscores, and must start with a letter or underscore.
In Regex terms, ^[A-Za-z_][0-9A-Za-z_]+$
.
Argument values can be of one of five types:
true
or false
Malformed values will be silently ignored.
Both types of shortcode will also get either a page
or section
variable depending on where they were used
and a config
variable. These values will overwrite any arguments passed to a shortcode so these variable names
should not be used as argument names in shortcodes.
Simply call the shortcode as if it was a Tera function in a variable block.
Here is a YouTube video:
{{ youtube(id="dQw4w9WgXcQ") }}
{{ youtube(id="dQw4w9WgXcQ", autoplay=true) }}
An inline {{ youtube(id="dQw4w9WgXcQ", autoplay=true, class="youtube") }} shortcode
Note that if you want to have some content that looks like a shortcode but not have Zola try to render it,
you will need to escape it by using {{/*
and */}}
instead of {{
and }}
.
Let's imagine that we have the following shortcode quote.html
template:
<blockquote>
{{ body }} <br>
-- {{ author}}
</blockquote>
We could use it in our Markdown file like so:
As someone said:
{% quote(author="Vincent") %}
A quote
{% end %}
The body of the shortcode will be automatically passed down to the rendering context as the body
variable and needs
to be on a new line.
Note that for both cases that the parentheses for shortcodes are necessary. A shortcode without the parentheses will render as plaintext and no warning will be emitted.
As an example, this is how an aside
shortcode-with-body with no arguments would be defined in aside.html
:
<aside>
{{ body }}
</aside>
We could use it in our Markdown file like so:
Readers can refer to the aside for more information.
{% aside() %}
An aside
{% end %}
If you want to have some content that looks like a shortcode but not have Zola try to render it,
you will need to escape it by using {%/*
and */%}
instead of {%
and %}
. You won't need to escape
anything else until the closing tag.
Every shortcode can access some variables, beyond what you explicitly passed as parameter. These variables are explained in the following subsections:
nth
)lang
), unless called from the markdown
template filter (in which case it will always be the same value as default_language
in configuration, or en
when it is unset)colocated_path
When one of these variables conflict with a variable passed as argument, the argument value will be used.
nth
: invocation countEvery shortcode context is passed in a variable named nth
that tracks how many times a particular shortcode has
been invoked in the current Markdown file. Given a shortcode true_statement.html
template:
<p id="number{{ nth }}">{{ value }} is equal to {{ nth }}.</p>
It could be used in our Markdown as follows:
{{ true_statement(value=1) }}
{{ true_statement(value=2) }}
This is useful when implementing custom markup for features such as sidenotes or end notes.
lang
: current languageNOTE: When calling a shortcode from within the markdown
template filter, the lang
variable will always be en
.
If you feel like you need that, please consider using template macros instead.
If you really need that, you can rewrite your Markdown content to pass lang
as argument to the shortcode.
Every shortcode can access the current language in the lang
variable in the context.
This is useful for presenting/filtering information in a shortcode depending in a per-language manner. For example, to display a per-language book cover for the current page in a shortcode called bookcover.md
:
![Book cover in {{ lang }}](cover.{{ lang }}.png)
page
or section
You can access a slighty stripped down version of the equivalent variables in the normal templates. The following attributes will be empty:
(Note: this is because the rendering of markdown is done before populating the sections)
A useful attribute to page
in shortcodes is colocated_path
.
This is used when you want to pass the name of some assets to shortcodes without repeating the full folders path.
Mostly useful when combined with load_data
or resize_image
.
{% set resized = resize_image(format="jpg", path=page.colocated_path ~ img_name, width=width, op="fit_width") %}
<img alt="{{ alt }}" src="{{ resized.url | safe }}" />
Here are some shortcodes for inspiration.
Embed a responsive player for a YouTube video.
The arguments are:
id
: the video id (mandatory)playlist
: the playlist id (optional)class
: a class to add to the <div>
surrounding the iframeautoplay
: when set to "true", the video autoplays on loadCode:
<div {% if class %}class="{{class}}"{% endif %}>
<iframe src="https://www.youtube-nocookie.com/embed/{{id}}{% if playlist %}?list={{playlist}}{% endif %}{% if autoplay %}?autoplay=1{% endif %}" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe>
</div>
Usage example:
{{ youtube(id="dCKeXuVHl1o") }}
{{ youtube(id="dCKeXuVHl1o", playlist="RDdQw4w9WgXcQ") }}
{{ youtube(id="dCKeXuVHl1o", autoplay=true) }}
{{ youtube(id="dCKeXuVHl1o", autoplay=true, class="youtube") }}
See content processing page for code and example.