A guide to creating and distributing custom themes.
NOTE: If you are looking for existing third party themes, they are listed in the community wiki page and the MkDocs project catalog. If you want to share a theme you create, you should list it there.
When creating a new theme, you can either follow the steps in this guide to
create one from scratch or you can download the mkdocs-basic-theme
as a
basic, yet complete, theme with all the boilerplate required. You can find
this base theme on GitHub. It contains detailed comments in
the code to describe the different features and their usage.
The bare minimum required for a custom theme is a main.html
Jinja2
template file which is placed in a directory that is not a child of the
docs_dir. Within mkdocs.yml
, set the theme.custom_dir
option to the path of the directory containing main.html
. The path should be
relative to the configuration file. For example, given this example project
layout:
mkdocs.yml
docs/
index.md
about.md
custom_theme/
main.html
...
... you would include the following settings in mkdocs.yml
to use the custom theme
directory:
theme:
name: null
custom_dir: 'custom_theme/'
NOTE: Generally, when building your own custom theme, the theme.name configuration setting would be set to
null
. However, if the theme.custom_dir configuration value is used in combination with an existing theme, the theme.custom_dir can be used to replace only specific parts of a built-in theme. For example, with the above layout and if you setname: "mkdocs"
then themain.html
file in the theme.custom_dir would replace the file of the same name in themkdocs
theme but otherwise themkdocs
theme would remain unchanged. This is useful if you want to make small adjustments to an existing theme.For more specific information, see Customizing Your Theme.
WARNING: A theme's configuration defined in a
mkdocs_theme.yml
file is not loaded fromtheme.custom_dir
. When an entire theme exists intheme.custom_dir
andtheme.name
is set tonull
, then the entire theme configuration must be defined in the theme configuration option in themkdocs.yml
file.However, when a theme is packaged up for distribution, and loaded using the
theme.name
configuration option, then amkdocs_theme.yml
file is required for the theme.
The simplest main.html
file is the following:
<!DOCTYPE html>
<html>
<head>
<title>{% if page.title %}{{ page.title }} - {% endif %}{{ config.site_name }}</title>
{%- for path in config.extra_css %}
<link href="{{ path | url }}" rel="stylesheet">
{%- endfor %}
</head>
<body>
{{ page.content }}
{%- for script in config.extra_javascript %}
{{ script | script_tag }}
{%- endfor %}
</body>
</html>
The body content from each page specified in mkdocs.yml
is inserted using the
{{ page.content }}
tag. Style-sheets and scripts can be brought into this
theme as with a normal HTML file. Navbars and tables of contents can also be
generated and included automatically, through the nav
and toc
objects,
respectively. If you wish to write your own theme, it is recommended to start
with one of the built-in themes and modify it accordingly.
NOTE:
As MkDocs uses Jinja as its template engine, you have access to all the
power of Jinja, including template inheritance. You may notice that the
themes included with MkDocs make extensive use of template inheritance and
blocks, allowing users to easily override small bits and pieces of the
templates from the theme custom_dir. Therefore, the built-in themes are
implemented in a base.html
file, which main.html
extends. Although not
required, third party template authors are encouraged to follow a similar
pattern and may want to define the same blocks as are used in the built-in
themes for consistency.
MkDocs defines the top-level extra_css and extra_javascript configs. These are lists of files.
The theme must include the HTML that links the items from these configs, otherwise the configs will be non-functional. You can see the recommended way to render both of them in the base example above.
NEW: Changed in version 1.5:
The items of the
config.extra_javascript
list used to be simple strings but now became objects that have these fields:path
,type
,async
,defer
.In that version, MkDocs also gained the
script_tag
filter.? EXAMPLE: Obsolete style:
{%- for path in extra_javascript %} <script src="{{ path }}"></script> {%- endfor %}This old-style example even uses the obsolete top-level
extra_javascript
list. Please always useconfig.extra_javascript
instead.So, a slightly more modern approach is the following, but it is still obsolete because it ignores the extra attributes of the script:
{%- for path in config.extra_javascript %} <script src="{{ path | url }}"></script> {%- endfor %}? EXAMPLE: New style:
{%- for script in config.extra_javascript %} {{ script | script_tag }} {%- endfor %}If you wish to be able to pick up the new customizations while keeping your theme compatible with older versions of MkDocs, use this snippet:
! EXAMPLE: Backwards-compatible style:
{%- for script in config.extra_javascript %} {%- if script.path %} {# Detected MkDocs 1.5+ which has `script.path` and `script_tag` #} {{ script | script_tag }} {%- else %} {# Fallback - examine the file name directly #} <script src="{{ script | url }}"{% if script.endswith(".mjs") %} type="module"{% endif %}></script> {%- endif %} {%- endfor %}
There are various files which a theme treats special in some way. Any other
files are simply copied from the theme directory to the same path in the
site_dir
when the site it built. For example image and CSS files have no
special significance and are copied as-is. Note, however, that if the user
provides a file with the same path in their docs_dir
, then the user's file
will replace the theme file.
Any files with the .html
extension are considered to be template files and are
not copied from the theme directory or any subdirectories. Also, any files
listed in static_templates are treated as templates regardless of their file
extension.
The various files required for packaging a theme are also ignored. Specifically,
the mkdocs_theme.yml
configuration file and any Python files.
Theme authors can explicitly force MkDocs to ignore files by starting a file or directory name with a dot. Any of the following files would be ignored:
.ignored.txt
.ignored/file.txt
foo/.ignored.txt
foo/.ignored/file.txt
All documentation files are ignored. Specifically, any Markdown files (using any of the file extensions supported by MKDocs). Additionally, any README files which may exist in the theme directories are ignored.
Each template in a theme is built with a template context. These are the variables that are available to themes. The context varies depending on the template that is being built. At the moment templates are either built with the global context or with a page specific context. The global context is used for HTML pages that don't represent an individual Markdown document, for example a 404.html page or search.html.
The following variables are available globally on any template.
The config
variable is an instance of MkDocs' config object generated from the
mkdocs.yml
config file. While you can use any config option, some commonly
used options include:
- config.site_name
- config.site_url
- config.site_author
- config.site_description
- config.theme.locale (See also Theme Configuration below)
- config.extra_javascript
- config.extra_css
- config.repo_url
- config.repo_name
- config.copyright
The nav
variable is used to create the navigation for the documentation. The
nav
object is an iterable of navigation objects as
defined by the nav configuration setting.
::: mkdocs.structure.nav.Navigation options: show_root_heading: false show_root_toc_entry: true members: [] heading_level: 4
In addition to the iterable of navigation objects, the
nav
object contains the following attributes:
::: mkdocs.structure.nav.Navigation.homepage options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Navigation.pages options: show_root_full_path: false heading_level: 5
This list is not necessarily a complete list of all site pages as it does not contain pages which are not included in the navigation. This list does match the list and order of pages used for all "next page" and "previous page" links. For a list of all pages, use the pages template variable.
Following is a basic usage example which outputs the first and second level navigation as a nested list.
{% if nav|length > 1 %}
<ul>
{% for nav_item in nav %}
{% if nav_item.children %}
<li>{{ nav_item.title }}
<ul>
{% for nav_item in nav_item.children %}
<li class="{% if nav_item.active %}current{% endif %}">
<a href="{{ nav_item.url|url }}">{{ nav_item.title }}</a>
</li>
{% endfor %}
</ul>
</li>
{% else %}
<li class="{% if nav_item.active %}current{% endif %}">
<a href="{{ nav_item.url|url }}">{{ nav_item.title }}</a>
</li>
{% endif %}
{% endfor %}
</ul>
{% endif %}
The base_url
provides a relative path to the root of the MkDocs project. While
this can be used directly by prepending it to a local relative URL, it is best
to use the url template filter, which is smarter about how it applies
base_url
.
Contains the current MkDocs version.
A Python datetime object that represents the date and time the documentation was built in UTC. This is useful for showing how recently the documentation was updated.
A flat list of File
objects for all pages in the project. This list can
contain pages not included in the global navigation and may not match
the order of pages within that navigation. The page object for each
File
can be accessed from file.page
.
In templates which are not rendered from a Markdown source file, the page
variable is None
. In templates which are rendered from a Markdown source file,
the page
variable contains a page
object. The same page
objects are used
as page
navigation objects in the global
navigation and in the pages template variable.
::: mkdocs.structure.pages.Page options: show_root_heading: false show_root_toc_entry: true members: [] heading_level: 4
All page
objects contain the following attributes:
::: mkdocs.structure.pages.Page.title options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.content options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.toc options: show_root_full_path: false heading_level: 5
The following example would display the top two levels of the Table of Contents for a page.
<ul>
{% for toc_item in page.toc %}
<li><a href="{{ toc_item.url }}">{{ toc_item.title }}</a></li>
{% for toc_item in toc_item.children %}
<li><a href="{{ toc_item.url }}">{{ toc_item.title }}</a></li>
{% endfor %}
{% endfor %}
</ul>
::: mkdocs.structure.pages.Page.meta options: show_root_full_path: false heading_level: 5
In this example we define a source
property above the page title:
source: generics.py
mixins.py
# Page title
Content...
A template can access this metadata for the page with the meta.source
variable. This could then be used to link to source files related to the
documentation page.
{% for filename in page.meta.source %}
<a class="github" href="https://github.com/.../{{ filename }}">
<span class="label label-info">{{ filename }}</span>
</a>
{% endfor %}
::: mkdocs.structure.pages.Page.url options: show_root_full_path: false heading_level: 5
It is expected that this be used with the url filter to ensure the URL is relative to the current page.
<a href="{{ page.url|url }}">{{ page.title }}</a>
::: mkdocs.structure.pages.Page.file options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.abs_url options: show_root_full_path: false heading_level: 5
For example, if site_url: https://example.com/
, then the value of
page.abs_url
for the page foo.md
would be /foo/
. However, if
site_url: https://example.com/bar/
, then the value of page.abs_url
for the
page foo.md
would be /bar/foo/
.
::: mkdocs.structure.pages.Page.canonical_url options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.edit_url options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.is_homepage options: show_root_full_path: false heading_level: 5
This can be used in conjunction with other attributes of the page
object to alter the behavior. For example, to display a different title
on the homepage:
{% if not page.is_homepage %}{{ page.title }} - {% endif %}{{ site_name }}
::: mkdocs.structure.pages.Page.previous_page options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.next_page options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.StructureItem.parent options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.children options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.active options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.is_section options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.is_page options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.pages.Page.is_link options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.toc.AnchorLink options: show_root_heading: false show_root_toc_entry: true heading_level: 5
Navigation objects contained in the nav template variable may be one of section objects, page objects, and link objects. While section objects may contain nested navigation objects, pages and links do not.
Page objects are the full page object as used for the current page with all of the same attributes available. Section and Link objects contain a subset of those attributes as defined below:
A section
navigation object defines a named section in the navigation and
contains a list of child navigation objects. Note that sections do not contain
URLs and are not links of any kind. However, by default, MkDocs sorts index
pages to the top and the first child might be used as the URL for a section if a
theme chooses to do so.
::: mkdocs.structure.nav.Section options: show_root_heading: false show_root_toc_entry: true members: [] heading_level: 4
The following attributes are available on section
objects:
::: mkdocs.structure.nav.Section.title options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.StructureItem.parent options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Section.children options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Section.active options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Section.is_section options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Section.is_page options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Section.is_link options: show_root_full_path: false heading_level: 5
A link
navigation object contains a link which does not point to an internal
MkDocs page.
::: mkdocs.structure.nav.Link options: show_root_heading: false show_root_toc_entry: true members: [] heading_level: 4
The following attributes are available on link
objects:
::: mkdocs.structure.nav.Link.title options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.url options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.StructureItem.parent options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.children options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.active options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.is_section options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.is_page options: show_root_full_path: false heading_level: 5
::: mkdocs.structure.nav.Link.is_link options: show_root_full_path: false heading_level: 5
Additional variables can be passed to the template with the
extra
configuration option. This is a
set of key value pairs that can make custom templates far more flexible.
For example, this could be used to include the project version of all pages
and a list of links related to the project. This can be achieved with the
following extra
configuration:
extra:
version: 0.13.0
links:
- https://github.com/mkdocs
- https://docs.readthedocs.org/en/latest/builds.html#mkdocs
- https://www.mkdocs.org/
And then displayed with this HTML in the custom theme.
{{ config.extra.version }}
{% if config.extra.links %}
<ul>
{% for link in config.extra.links %}
<li>{{ link }}</li>
{% endfor %}
</ul>
{% endif %}
In addition to Jinja's default filters, the following custom filters are available to use in MkDocs templates:
Normalizes a URL. Absolute URLs are passed through unaltered. If the URL is relative and the template context includes a page object, then the URL is returned relative to the page object. Otherwise, the URL is returned with base_url prepended.
<a href="{{ page.url|url }}">{{ page.title }}</a>
Safely convert a Python object to a value in a JavaScript script.
<script>
var mkdocs_page_name = {{ page.title|tojson|safe }};
</script>
NEW: New in version 1.5.
Convert an item from extra_javascript
to a <script>
tag that takes into account all customizations of this config and has the equivalent of |url
behavior built-in.
See how to use it in the base example above
As of MkDocs version 0.17 client side search support has been added to MkDocs
via the search
plugin. A theme needs to provide a few things for the plugin to
work with the theme.
While the search
plugin is activated by default, users can disable the plugin
and themes should account for this. It is recommended that theme templates wrap
search specific markup with a check for the plugin:
{% if 'search' in config.plugins %}
search stuff here...
{% endif %}
At its most basic functionality, the search plugin will simply provide an index file which is no more than a JSON file containing the content of all pages. The theme would need to implement its own search functionality client-side. However, with a few settings and the necessary templates, the plugin can provide a complete functioning client-side search tool based on lunr.js.
The following HTML needs to be added to the theme so that the provided JavaScript is able to properly load the search scripts and make relative links to the search results from the current page.
<script>var base_url = {{ base_url|tojson }};</script>
With properly configured settings, the following HTML in a template will add a full search implementation to your theme.
<h1 id="search">Search Results</h1>
<form action="search.html">
<input name="q" id="mkdocs-search-query" type="text" >
</form>
<div id="mkdocs-search-results">
Sorry, page not found.
</div>
The JavaScript in the plugin works by looking for the specific ID's used in the
above HTML. The form input for the user to type the search query must be
identified with id="mkdocs-search-query"
and the div where the results will be
placed must be identified with id="mkdocs-search-results"
.
The plugin supports the following options being set in the theme's
configuration file, mkdocs_theme.yml
:
Determines whether the search plugin expects the theme to provide a dedicated
search page via a template located at search/search.html
.
When include_search_page
is set to true
, the search template will be built
and available at search/search.html
. This method is used by the readthedocs
theme.
When include_search_page
is set to false
or not defined, it is expected that
the theme provide some other mechanisms for displaying search results. For
example, the mkdocs
theme displays results on any page via a modal.
Determines whether the search plugin should only generate a search index or a complete search solution.
When search_index_only
is set to false
, then the search plugin modifies the
Jinja environment by adding its own templates
directory (with a lower
precedence than the theme) and adds its scripts to the extra_javascript
config
setting.
When search_index_only
is set to true
or not defined, the search plugin
makes no modifications to the Jinja environment. A complete solution using the
provided index file is the responsibility of the theme.
The search index is written to a JSON file at search/search_index.json
in the
site_dir. The JSON object contained within the file may contain up to three
objects.
{
config: {...},
docs: [...],
index: {...}
}
If present, the config
object contains the key/value pairs of config options
defined for the plugin in the user's mkdocs.yml
config file under
plugings.search
. The config
object was new in MkDocs version 1.0.
The docs
object contains a list of document objects. Each document object is
made up of a location
(URL), a title
, and text
which can be used to create
a search index and/or display search results.
If present, the index
object contains a pre-built index which offers
performance improvements for larger sites. Note that the pre-built index is only
created if the user explicitly enables the prebuild_index config option.
Themes should expect the index to not be present, but can choose to use the
index when it is available. The index
object was new in MkDocs version 1.0.
MkDocs makes use of Python packaging to distribute themes. This comes with a few requirements.
To see an example of a package containing one theme, see the MkDocs Bootstrap theme and to see a package that contains many themes, see the MkDocs Bootswatch theme.
NOTE:
It is not strictly necessary to package a theme, as the entire theme
can be contained in the custom_dir
. If you have created a "one-off theme,"
that should be sufficient. However, if you intend to distribute your theme
for others to use, packaging the theme has some advantages. By packaging
your theme, your users can more easily install it, they can rely on a default
configuration being defined, and they can then take advantage of the
custom_dir to make tweaks to your theme to better suit their needs.
The following layout is recommended for themes. Two files at the top level
directory called MANIFEST.in
and setup.py
beside the theme directory which
contains an empty __init__.py
file, a theme configuration file
(mkdocs_theme.yml
), and your template and media files.
.
|-- MANIFEST.in
|-- theme_name
| |-- __init__.py
| |-- mkdocs_theme.yml
| |-- main.html
| |-- styles.css
`-- setup.py
The MANIFEST.in
file should contain the following contents but with
theme_name updated and any extra file extensions added to the include.
recursive-include theme_name *.ico *.js *.css *.png *.html *.eot *.svg *.ttf *.woff
recursive-exclude * __pycache__
recursive-exclude * *.py[co]
The setup.py
should include the following text with the modifications
described below.
from setuptools import setup, find_packages
VERSION = '0.0.1'
setup(
name="mkdocs-themename",
version=VERSION,
url='',
license='',
description='',
author='',
author_email='',
packages=find_packages(),
include_package_data=True,
entry_points={
'mkdocs.themes': [
'themename = theme_name',
]
},
zip_safe=False
)
Fill in the URL, license, description, author and author email address.
The name should follow the convention mkdocs-themename
(like
mkdocs-bootstrap
and mkdocs-bootswatch
), starting with MkDocs, using
hyphens to separate words and including the name of your theme.
Most of the rest of the file can be left unedited. The last section we need to change is the entry_points. This is how MkDocs finds the theme(s) you are including in the package. The name on the left is the one that users will use in their mkdocs.yml and the one on the right is the directory containing your theme files.
The directory you created at the start of this section with the main.html file
should contain all of the other theme files. The minimum requirement is that
it includes a main.html
for the theme. It must also include a
__init__.py
file which should be empty, this file tells Python that the
directory is a package.
A packaged theme is required to include a configuration file named
mkdocs_theme.yml
which is placed in the root of your template files. The file
should contain default configuration options for the theme. However, if the
theme offers no configuration options, the file is still required and can be
left blank. A theme which is not packaged does not need a mkdocs_theme.yml
file as that file is not loaded from theme.custom_dir
.
The theme author is free to define any arbitrary options deemed necessary and
those options will be made available in the templates to control behavior.
For example, a theme might want to make a sidebar optional and include the
following in the mkdocs_theme.yml
file:
show_sidebar: true
Then in a template, that config option could be referenced:
{% if config.theme.show_sidebar %}
<div id="sidebar">...</div>
{% endif %}
And the user could override the default in their project's mkdocs.yml
config
file:
theme:
name: themename
show_sidebar: false
In addition to arbitrary options defined by the theme, MkDocs defines a few special options which alters its behavior:
BLOCK:
This option mirrors the theme config option of the same name. If this value is not defined in the
mkdocs_theme.yml
file and the user does not set it inmkdocs.yml
then it will default toen
(English). The value is expected to match the language used in the text provided by the theme (such a "next" and "previous" links) and should be used as the value of the<html>
tag'slang
attribute. See Supporting theme localization/ translation for more information.Note that during configuration validation, the provided string is converted to a
Locale
object. The object containsLocale.language
andLocale.territory
attributes and will resolve as a string from within a template. Therefore, the following will work fine:<html lang="{ config.theme.locale }">If the locale was set to
fr_CA
(Canadian French), then the above template would render as:<html lang="fr_CA">If you did not want the territory attribute to be included, then reference the
language
attribute directly:<html lang="{ config.theme.locale.language }">That would render as:
<html lang="fr">This option mirrors the theme config option of the same name and allows some defaults to be set by the theme. Note that while the user can add templates to this list, the user cannot remove templates included in the theme's config.
Defines a parent theme that this theme inherits from. The value should be the string name of the parent theme. Normal Jinja inheritance rules apply.
Plugins may also define some options which allow the theme to inform a plugin about which set of plugin options it expects. See the documentation for any plugins you may wish to support in your theme.
With the above changes, your theme should now be ready to install. This can be
done with pip, using pip install .
if you are still in the same directory as
the setup.py.
Most Python packages, including MkDocs, are distributed on PyPI. To do this, you should run the following command.
python setup.py register
If you don't have an account setup, you should be prompted to create one.
For a much more detailed guide, see the official Python packaging documentation for Packaging and Distributing Projects.
While the built-in themes provide support for localization/translation of
templates, custom themes and third-party themes may choose not to. Regardless,
the locale
setting of the theme
configuration option is always
present and is relied upon by other parts of the system. Therefore, it is
recommended that all third-party themes use the same setting for designating a
language regardless of the system they use for translation. In that way, users
will experience consistent behavior regardless of the theme they may choose.
The method for managing translations is up to the developers of a theme. However, if a theme developer chooses to use the same mechanisms used by the built-in themes, the sections below outline how to enable and make use of the same commands utilized by MkDocs.
WARNING:
As pybabel is not installed by default and most users will not have
pybabel installed, theme developers and/or translators should make sure to
have installed the necessary dependencies
(using pip install 'mkdocs[i18n]'
) in order for the commands to be
available for use.
The translation commands should be called from the root of your theme's working tree.
For an overview of the workflow used by MkDocs to translate the built-in themes, see the appropriate section of the Contributing Guide and the Translation Guide.
NOTE: If your theme inherits from an existing theme which already provides translation catalogs, your theme's translations will be merged with the parent theme's translations during a MkDocs build.
This means that you only need to concentrate on the added translations. Yet, you will still benefit from the translations of the parent theme. At the same time, you may override any of parent theme's translations!
Let's suppose that you're working on your own fork of the mkdocs-basic-theme and want to add translations to it.
Edit the templates by wrapping text in your HTML sources with
{% trans %}
and {% endtrans %}
as follows:
--- a/basic_theme/base.html
+++ b/basic_theme/base.html
@@ -88,7 +88,7 @@
<body>
- <h1>This is an example theme for MkDocs.</h1>
+ <h1>{% trans %}This is an example theme for MkDocs.{% endtrans %}</h1>
<p>
It is designed to be read by looking at the theme HTML which is heavily
Then you would follow the Translation Guide as usual to get your translations running.
While the Portable Object Template (pot
) file created by the
extract_messages
command and the Portable Object (po
) files created by the
init_catalog
and update_catalog
commands are useful for creating and
editing translations, they are not used by MkDocs directly and do not need to
be included in a packaged release of a theme. When MkDocs builds a site with
translations, it only makes use of the binary mo
files(s) for the specified
locale. Therefore, when packaging a theme, make sure to include it in the
"wheels", using a MANIFEST.in
file or otherwise.
Then, before building your Python package, you will want to ensure that the
binary mo
file for each locale is up-to-date by running the compile_catalog
command for each locale. MkDocs expects the binary mo
files to be located at
locales/<locale>/LC_MESSAGES/messages.mo
, which the compile_catalog
command automatically does for you. See Testing theme translations for
details.
NOTE:
As outlined in our Translation Guide, the MkDocs project has chosen to
include the pot
and po
files in our code repository, but not the
mo
files. This requires us to always run compile_catalog
before
packaging a new release regardless of whether any changes were made to a
translation or not. However, you may chose an alternate workflow for your
theme. At a minimum, you need to ensure that up-to-date mo
files are
included at the correct location in each release. However, you may use a
different process for generating those mo
files if you chose to do so.