Skip to content

Customization

Zensical makes it easy to get started with a modern documentation site. However, sometimes you may want to adjust the look and feel of your documentation to better match your project's branding or to add custom functionality. This guide explains how you can customize Zensical.

Adding assets

Zensical offers several options for theme customization. To change the appearance of your pages, you can add your own CSS files located within your documentation directory. For more advanced modifications, you can include custom JavaScript code.

Additional CSS

To customize the appearance of your site, you can add a custom style sheet that overrides or extends Zensical’s default styles. Whether you want to adjust the design or apply specific branding, simply place your style sheet file within the docs directory:

.
├─ docs/
│  └─ stylesheets/
│     └─ extra.css
└─ zensical.toml

Then, add the following lines to zensical.toml:

[project]
extra_css = ["stylesheets/extra.css"]
.
├─ docs/
│  └─ stylesheets/
│     └─ extra.css
└─ mkdocs.yml

Then, add the following lines to mkdocs.yml:

extra_css:
  - stylesheets/extra.css

Additional JavaScript

To enhance your documentation with custom functionality or interactive features, you can add JavaScript files to your project. Place your custom scripts within the docs directory:

.
├─ docs/
│  └─ javascripts/
│     └─ extra.js
└─ zensical.toml

Then, add the following lines to zensical.toml:

[project]
extra_javascript = ["javascripts/extra.js"]
.
├─ docs/
│  └─ javascripts/
│     └─ extra.js
└─ mkdocs.yml

Then, add the following lines to mkdocs.yml:

extra_javascript:
  - javascripts/extra.js
Making sure your JavaScript code runs at the right time

It is likely that you will want to run some initialization code in your JavaScript once the page has been fully loaded by the browser. This means installing a callback function subscribing to events on the document$ observable exported by Zensical.

Using the document$ observable is particularly important if you are using instant navigation since it will not result in a page refresh in the browser, but subscribers on the observable will be notified.

document$.subscribe(function() {
  console.log("Initialize third-party libraries here")
})

Modules, async, defer

If you want to import code as a JavaScript module, you can simply make sure that the file has the .mjs extension or you can explicitly specify that it is to be loaded as a module:

[[project.extra_javascript]]
path = "javascripts/extra.js"
type = "module"
extra_javascript:
  - path: javascripts/extra.js
    type: module

This configuration will result in a type="module" attribute being added to the <script> tag for the extra JavaScript.

Likewise, you can add defer and async attributes to the script tag to further influence how the JavaScript is loaded. For example, for the async case:

[[project.extra_javascript]]
path = "javascripts/extra.js"
async = true
extra_javascript:
  - path: javascripts/extra.js
    async: true

Note that Zensical will auto-detect modules by looking at the file extension only when the extra_javascript element is plain text. That means that if you want to load a module using async, you also need to specify the type attribute.

Extending the theme

Zensical uses MiniJinja, a Rust-based template engine inspired by Python’s popular Jinja system, to render the HTML structure of your site – including the header, footer, and navigation sidebars. It rendered the entire scaffold of your documentation site, while the main content is generated from your Markdown files.

Templates are HTML files enhanced with MiniJinja instructions for dynamic rendering. The default page template is main.html, which inherits from base.html and includes additional templates found in the partials directory. The template can be customized per page.

Configuring overrides

To add new templates or to override parts of existing one, you first need to configure the custom_dir setting to point to a directory in which you store your template overrides:

[project.theme]
custom_dir = "overrides"
theme:
  custom_dir: overrides

The custom_dir path is resolved relative to your configuration file.

Theme structure

In order to override theme templates, it's important to understand the file system structure, since any files you store in custom_dir override the templates and partials of the same name as they are provided by Zensical. Likewise, it's important to know what files exist if you want to add your new templates or partials:

.
├─ .icons/                             # Icon sets
├─ assets/
  ├─ images/                          # Images and icons
  ├─ javascripts/                     # JavaScript files
  └─ stylesheets/                     # Style sheets
├─ partials/
  ├─ integrations/                    # Third-party integrations
    ├─ analytics/                    # Analytics integrations
    └─ analytics.html                # Analytics setup
  ├─ languages/                       # Translation languages
  ├─ actions.html                     # Actions
  ├─ alternate.html                   # Site language selector
  ├─ comments.html                    # Comment system (empty by default)
  ├─ consent.html                     # Consent
  ├─ content.html                     # Page content
  ├─ copyright.html                   # Copyright and theme information
  ├─ feedback.html                    # Was this page helpful?
  ├─ footer.html                      # Footer bar
  ├─ header.html                      # Header bar
  ├─ icons.html                       # Custom icons
  ├─ language.html                    # Translation setup
  ├─ logo.html                        # Logo in header and sidebar
  ├─ nav.html                         # Main navigation
  ├─ nav-item.html                    # Main navigation item
  ├─ palette.html                     # Color palette toggle
  ├─ progress.html                    # Progress indicator
  ├─ search.html                      # Search interface
  ├─ social.html                      # Social links
  ├─ source.html                      # Repository information
  ├─ source-file.html                 # Source file information
  ├─ tabs.html                        # Tabs navigation
  ├─ tabs-item.html                   # Tabs navigation item
  ├─ tags.html                        # Tags
  ├─ toc.html                         # Table of contents
  ├─ toc-item.html                    # Table of contents item
  └─ top.html                         # Back-to-top button
├─ 404.html                            # 404 error page
├─ base.html                           # Base template
└─ main.html                           # Default page

Custom templates

Some of your pages may need a different structure. Say, you want to have a custom layout for the homepage of your documentation site, for reference pages or your blog posts. You can use the template language provided by MiniJinja to produce any HTML structure you like. Use the templates provided by Zensical as a starter or start from a clean slate.

Add a new template to the overrides directory you configured. Make sure its name is not the same as one of the files provided by Zensical (such as main.html and base.html). Now you can specify which template should be used for a page by adding the template option to the page header:

---
template: "my_template.html"
---

# Page title

...

Custom error pages

The overrides directory is also the place for adding a 404 error page, which can be configured as a fallback in your web server. If you set custom_dir to overrides, use the following layout:

├─ overrides/
│  └─ 404.html
└─ zensical.toml

Note that Zensical provides a default 404.html template.

Template overrides

There are two methods to override templates: You can either override entire templates or partials, or you can override specific blocks within templates, which is the recommended approach.

For more information, refer to the Jinja Template Designer Documentation.

Overriding blocks recommended

Zensical templates contain blocks that wrap specific features and that can be re-defined or extended by an override. Zensical defines a main.html template that inherits all its functionality from base.html. The most robust way of making changes via overrides is therefore to override main.html instead of base.html, which is more likely to change in future versions of Zensical.

In order to set up block overrides, create a main.html file inside the overrides directory:

.
└─ overrides/
   └─ main.html

Then, e.g. to override the site title, add the following lines to main.html:

{% extends "base.html" %}

{% block htmltitle %}
  <title>Lorem ipsum dolor sit amet</title>
{% endblock %}

If you intend to add something to a block rather than to replace it altogether with new content, use {{ super() }} inside the block to include the original block content. This is particularly useful when adding third-party scripts to your docs, e.g.

{% extends "base.html" %}

{% block scripts %}
  <!-- Add scripts that need to run before here -->
  {{ super() }}
  <!-- Add scripts that need to run afterwards here -->
{% endblock %}

The following template blocks are provided by the theme:

Block name Purpose
analytics Wraps the Google Analytics integration
announce Wraps the announcement bar
config Wraps the JavaScript application config
container Wraps the main content container
content Wraps the main content
extrahead Empty block to add custom meta tags
fonts Wraps the font definitions
footer Wraps the footer with navigation and copyright
header Wraps the fixed header bar
hero Wraps the hero teaser (if available)
htmltitle Wraps the <title> tag
libs Wraps the JavaScript libraries (header)
outdated Wraps the version warning
scripts Wraps the JavaScript application (footer)
site_meta Wraps the meta tags in the document head
site_nav Wraps the site navigation and table of contents
styles Wraps the style sheets (also extra sources)
tabs Wraps the tabs navigation (if available)

Overriding partials

In order to override a partial, you can replace it with a file of the same name and location in the overrides directory. For example, to replace the original footer.html partial, create a new footer.html partial in the overrides directory:

.
└─ overrides/
   └─ partials/
      └─ footer.html

Zensical will now use the new partial when rendering the theme.

Packaging themes

If you want to share your theme extension with others, you can package it as a Python distribution. This allows users to install it with pip and reference it by name in their configuration.

Future improvements

The current packaging process closely mirrors MkDocs, using the mkdocs.themes entry point, which allows existing Material for MkDocs theme derivations to run on Zensical with minimal changes. With the upcoming component system, this process will become more flexible, enabling reuse at the component level.

Package layout

The following layout is recommended for a packaged theme extension. The package directory contains an optional mkdocs_theme.yml configuration file alongside your template and media files, and a pyproject.toml at the top level declares the package metadata and entry point:

.
├─ pyproject.toml
└─ my_theme/
   ├─ __init__.py #(1)!
   ├─ ... #(2)!
   ├─ main.html
   └─ mkdocs_theme.yml
  1. The __init__.py file is required to make the theme directory a Python package that can be imported. It can be empty.

  2. You can add any files you like here. As a recommendation for a sensible structure, you can follow the patterns laid out in the theme structure section.

Package configuration

The pyproject.toml file should contain the following content, with the placeholders – marked as highlighted lines – replaced with your actual values:

[build-system]
requires = ["hatchling"]
build-backend = "hatchling.build"

[project]
name = "my-theme"
version = "0.1.0"
description = "My theme extension"
authors = [
  { name = "Jane Doe", email = "jane@example.com" }
]
license = "MIT"
requires-python = ">=3.9"
dependencies = ["zensical>=0.0.37"]

[tool.hatch.build.targets.wheel]
include = ["my_theme"]

[project.entry-points."mkdocs.themes"]
my_theme = "my_theme"

The entry point group mkdocs.themes is what Zensical uses to discover installed themes. The key on the left (my_theme) is the name users will set in their configuration file, and the value on the right (my_theme) is the directory inside your package that contains the theme files.

Theme configuration

A packaged theme extension may include a mkdocs_theme.yml file in the theme directory to set default configuration values and declare which theme it extends:

extends: material #(1)!

# Default configuration - here, we just set a different font to demonstrate the
# concept, but you can set any theme configuration option here
font:
  text: Roboto
  code: Roboto
  1. If you're building on top of Zensical's default theme, set extends to material. If you're building on top of another extension, set it to the name of that extension.

    For now, we deliberately keep the name of the default theme as material for compatibility with existing Material for MkDocs extensions.

The extends key tells Zensical which theme this extension builds on. Users can override any of the defaults defined here in their own mkdocs.yml or zensical.toml. If no mkdocs_theme.yml is provided, the extension is treated as a full standalone theme.

Differences from MkDocs theme packaging

Zensical makes theming more flexible than MkDocs:

  • mkdocs_theme.yml is optional, even for packaged themes. MkDocs requires it for any packaged theme; Zensical simply looks for it and uses it when present.
  • mkdocs_theme.yml is also read from custom_dir. MkDocs ignores it outside of packaged themes; Zensical reads it regardless, so you can include theme configuration in a local override directory too.

Using packaged themes

Once your package is ready, install it locally with:

pip install .

Or publish it to PyPI and let users install it with:

pip install my-theme

Users can then reference your extension by its entry point name in their configuration:

[project.theme]
name = "my_theme"
theme:
  name: my_theme