Source code for sphinx_immaterial.custom_admonitions

"""This module inherits from the generic ``admonition`` directive and makes the
title optional."""

from abc import ABC
from pathlib import PurePath
import re
from typing import List, Dict, Any, Tuple, Optional, Type, cast
from docutils import nodes
from docutils.parsers.rst import directives, Directive
import jinja2
import pydantic
from pydantic_extra_types.color import Color
import sphinx
import sphinx.addnodes
from sphinx.application import Sphinx
from sphinx.config import Config
from sphinx.domains.changeset import VersionChange
from sphinx.environment import BuildEnvironment
import sphinx.ext.todo
from sphinx.locale import admonitionlabels, _
from sphinx.util.logging import getLogger
from sphinx.writers.html5 import HTML5Translator
from .css_and_javascript_bundles import add_global_css
from .inline_icons import load_svg_into_builder_env, get_custom_icons
from . import html_translator_mixin

logger = getLogger(__name__)

# treat the todo directive from the sphinx extension as a built-in directive
admonitionlabels["todo"] = _("Todo")

INHERITED_ADMONITIONS = (
    "abstract",
    "info",
    "success",
    "question",
    "failure",
    "bug",
    "example",
    "quote",
)

_CUSTOM_ADMONITIONS_KEY = "sphinx_immaterial_custom_admonitions"


# defaults used for version directives re-styling
VERSION_DIR_STYLE = {
    "versionadded": {
        "icon": "material/alert-circle",
        "color": (72, 138, 87),
        "classes": [],
    },
    "versionchanged": {
        "icon": "material/alert-circle",
        "color": (238, 144, 64),
        "classes": [],
    },
    "deprecated": {"icon": "material/delete", "color": (203, 70, 83), "classes": []},
}
if sphinx.version_info >= (7, 3):
    # re-use deprecated style for versionremoved directive except with different icon
    VERSION_DIR_STYLE["versionremoved"] = {
        "icon": "material/close",
        "color": (203, 70, 83),
        "classes": [],
    }


[docs] class CustomAdmonitionConfig(pydantic.BaseModel): """This data class validates the user's configuration value(s) in :confval:`sphinx_immaterial_custom_admonitions`. Each validated object corresponds to a generated custom :rst:dir:`admonition` directive tailored with theme specific options. """ name: str """The required name of the directive. This will be also used as a CSS class name. This value shall have characters that match the regular expression pattern ``[a-zA-Z0-9_-]``.""" title: Optional[str] = pydantic.Field(default=None, validate_default=True) """The default title to use when rendering the custom admonition. If this is not specified, then the `name` value is converted and used.""" icon: Optional[str] = None """The relative path to an icon that will be used in the admonition's title. If specified, this path shall be relative to - a SVG file placed in the documentation project's :confval:`sphinx_immaterial_icon_path` (this takes precedence). - the ``.icons`` folder that has `all of the icons bundled with this theme <https://github.com/squidfunk/mkdocs-material/tree/master/material/.icons>`_.""" color: Optional[Color] = None """The base color to be used for the admonition's styling. If specified, this must be defined via: - `name <http://www.w3.org/TR/SVG11/types.html#ColorKeywords>`_ (e.g. :python:`"Black"`, :python:`"azure"`) - `hexadecimal value <https://en.wikipedia.org/wiki/Web_colors#Hex_triplet>`_ (e.g. :python:`"0x000"`, :python:`"#FFFFFF"`, :python:`"7fffd4"`) - RGB/RGBA tuples (e.g. :python:`(255, 255, 255)`, :python:`(255, 255, 255, 0.5)`) - `RGB/RGBA strings <https://developer.mozilla.org/en-US/docs/Web/CSS/color_value>`_ (e.g. :python:`"rgb(255, 255, 255)"`, :python:`"rgba(255, 255, 255, 0.5)"`) - `HSL strings <https://developer.mozilla.org/en-US/docs/Web/CSS/color_value>`_ (e.g. :python:`"hsl(270, 60%, 70%)"`, :python:`"hsl(270, 60%, 70%, .5)"`) .. note:: Any specified transparency (alpha value) is ignored. """ classes: List[str] = [] """If specified, this list of qualified names will be added to every rendered admonition (specific to the generated directive) element's ``class`` attribute. To adopt the styling of pre-existing admonition, include the desired admonition directive's name in this list. .. code-block:: python :caption: Adopting the style of an :dutree:`hint` admonition: sphinx_immaterial_custom_admonitions = [ { "name": "my-admonition", "classes": ["hint"], }, ] """ override: bool = False """Can be used to override an existing directive (default is :python:`False`). Only set this to :python:`True` if the directive being overridden is an existing admonition :ref:`defined by rST and Sphinx <predefined_admonitions>` or an admonition :ref:`inherited from the mkdocs-material theme <inherited_admonitions>`. """ @pydantic.field_validator("name", mode="before") @classmethod def validate_name(cls, val): illegal = re.findall(r"([^a-zA-Z0-9\-_])", val) if illegal: raise ValueError( f"The following characters are illegal for directive names: {illegal}" ) return val @pydantic.field_validator("title") @classmethod def validate_title(cls, val, info: pydantic.ValidationInfo): if val is None: val = " ".join( re.split(r"[\-_]+", cast(str, info.data.get("name"))) ).title() return val @pydantic.field_validator("classes") @classmethod def validate_classes(cls, val): validated = [] for c in val: validated.append(nodes.make_id(c)) return validated
def visit_collapsible(self: HTML5Translator, node: nodes.Element, flag: str): tag_extra_args: Dict[str, Any] = {"CLASS": "admonition"} if flag.lower() == "open": tag_extra_args["open"] = "" self.body.append(self.starttag(node, "details", **tag_extra_args)) title = cast(nodes.Element, node[0]) self.body.append( self.starttag(title, "summary", suffix="", CLASS="admonition-title") ) for child in title.children: child.walkabout(self) self.body.append("</summary>") del node[0] def patch_visit_admonition(): orig_func = HTML5Translator.visit_admonition def visit_admonition(self: HTML5Translator, node: nodes.Element, name: str = ""): collapsible: Optional[str] = node.get("collapsible", None) if collapsible is not None: assert isinstance(node[0], nodes.title) visit_collapsible(self, node, collapsible) else: orig_func(self, node, name) HTML5Translator.visit_admonition = visit_admonition # type: ignore[assignment] def patch_depart_admonition(): orig_func = HTML5Translator.depart_admonition def depart_admonition(self: HTML5Translator, node: Optional[nodes.Element] = None): if node is None or node.get("collapsible", None) is None: orig_func(self, node) else: self.body.append("</details>\n") HTML5Translator.depart_admonition = depart_admonition # type: ignore[assignment] @html_translator_mixin.override def visit_versionmodified( self: html_translator_mixin.HTMLTranslatorMixin, node: sphinx.addnodes.versionmodified, super_func: html_translator_mixin.BaseVisitCallback[ sphinx.addnodes.versionmodified ], ) -> None: # do compatibility check for changes in Sphinx assert ( len(node) >= 1 and isinstance(node[0], nodes.paragraph) and node.get("type", None) is not None and node["type"] in VERSION_DIR_STYLE ) if VERSION_DIR_STYLE[node["type"]]["classes"]: node["classes"].extend(VERSION_DIR_STYLE[node["type"]]["classes"]) if node["type"] not in node["classes"]: node["classes"].append(node["type"]) collapsible: Optional[str] = node.get("collapsible", None) if collapsible is not None: visit_collapsible(self, node, collapsible) else: # similar to what the OG visitor does but with an added admonition class self.body.append(self.starttag(node, "div", CLASSES=["admonition"])) # add admonition-title class to first paragraph node[0]["classes"].append("admonition-title") @html_translator_mixin.override def depart_versionmodified( self: html_translator_mixin.HTMLTranslatorMixin, node: sphinx.addnodes.versionmodified, super_func: html_translator_mixin.BaseVisitCallback[ sphinx.addnodes.versionmodified ], ) -> None: collapsible: Optional[str] = node.get("collapsible", None) if collapsible is not None: self.body.append("</details>") else: super_func(self, node) patch_visit_admonition() patch_depart_admonition() class CustomVersionChange(VersionChange): """Derivative of the original version directives to add theme-specific admonition options""" option_spec = { # type: ignore[misc] "collapsible": directives.unchanged, "class": directives.class_option, "name": directives.unchanged, } def run(self): ret = super().run() assert len(ret) and isinstance(ret[0], sphinx.addnodes.versionmodified) if "collapsible" in self.options: if len(self.arguments) < 2: raise self.error( "Expected 2 arguments before content in %s directive" % self.name ) self.assert_has_content() ret[0]["collapsible"] = self.options["collapsible"] if "class" in self.options: ret[0]["classes"].extend(self.options["class"]) self.add_name(ret[0]) return ret class CustomAdmonitionDirective(Directive, ABC): """A base class to define custom admonition directives. .. warning:: Do not instantiate an object directly from this class as it could mess up the argument parsing for other derivative classes. """ node_class: Type[nodes.admonition] = nodes.admonition optional_arguments: int final_argument_whitespace: bool = True has_content = True default_title: str = "" classes: List[str] = [] option_spec = { "class": directives.class_option, "name": directives.unchanged, "collapsible": directives.unchanged, "no-title": directives.flag, "title": directives.unchanged, } def run(self): # docutils.parsers.rst.roles.set_classes() is deprecated & # its replacement is not available in older versions, so # manually convert key from "class" to "classes" if "class" in self.options: if "classes" not in self.options: self.options["classes"] = [] self.options["classes"].extend(self.options["class"]) del self.options["class"] title_text = "" if not self.arguments else self.arguments[0] if "title" in self.options: # this option can be combined with the directive argument used as a title. title_text += (" " if title_text else "") + self.options["title"] # don't auto-assert `:no-title:` if value is blank; just use default if not title_text: # title_text must be an explicit string for renderers like MyST title_text = str(self.default_title) self.assert_has_content() admonition_node = self.node_class("\n".join(self.content), **self.options) # type: ignore[call-arg] ( admonition_node.source, # type: ignore[attr-defined] admonition_node.line, # type: ignore[attr-defined] ) = self.state_machine.get_source_and_line(self.lineno) if isinstance(admonition_node, sphinx.ext.todo.todo_node): # todo admonitions need extra info for the todolist directive admonition_node["docname"] = admonition_node.source self.state.document.note_explicit_target(admonition_node) else: self.add_name(admonition_node) if "collapsible" in self.options: admonition_node["collapsible"] = self.options["collapsible"] if "no-title" in self.options and "collapsible" in self.options: logger.error( "title is needed for collapsible admonitions", location=admonition_node, ) del self.options["no-title"] # force-disable option textnodes, messages = self.state.inline_text(title_text, self.lineno) if "no-title" not in self.options and title_text: title = nodes.title(title_text, "", *textnodes) title.source, title.line = self.state_machine.get_source_and_line( self.lineno ) admonition_node += title admonition_node += messages admonition_node["classes"] += self.classes self.state.nested_parse(self.content, self.content_offset, admonition_node) return [admonition_node] def get_directive_class(name, title, classes=None) -> Type[CustomAdmonitionDirective]: """A helper function to produce a admonition directive's class.""" # alias upstream-deprecated CSS classes for pre-defined admonitions in sphinx class_list = [nodes.make_id(name)] if classes: class_list.extend(classes) # uncomment this block when we merge v9.x from upstream # if name in ("caution", "attention"): # class_list.append("warning") # elif name == "error": # class_list.append("danger") # elif name in ("important", "hint"): # class_list.append("tip") # elif name == "todo": # class_list.append("info") class CustomizedAdmonition(CustomAdmonitionDirective): default_title = title classes = class_list optional_arguments = int(name not in admonitionlabels) node_class = cast( Type[nodes.admonition], nodes.admonition if name != "todo" else sphinx.ext.todo.todo_node, ) return CustomizedAdmonition def on_builder_inited(app: Sphinx): """register the directives for the custom admonitions and build the CSS.""" custom_admonitions = [ x.model_copy() for x in cast( List[CustomAdmonitionConfig], getattr(app.config, "sphinx_immaterial_custom_admonitions"), ) ] builtin_css_classes = list(admonitionlabels.keys()) + list(INHERITED_ADMONITIONS) custom_admonition_names = [] for admonition in custom_admonitions: custom_admonition_names.append(admonition.name) if admonition.name in VERSION_DIR_STYLE: # if specific to version directives inheriting_style = any( filter(lambda x: x in builtin_css_classes, admonition.classes or []) ) if admonition.classes: cast(List[str], VERSION_DIR_STYLE[admonition.name]["classes"]).extend( admonition.classes ) if not inheriting_style or admonition.icon: admonition.icon = load_svg_into_builder_env( app.builder, admonition.icon or cast(str, VERSION_DIR_STYLE[admonition.name]["icon"]), ) if admonition.color is None and not inheriting_style: admonition.color = Color( cast( Tuple[int, int, int], VERSION_DIR_STYLE[admonition.name]["color"], ) ) continue # don't override the version directives app.add_directive( name=admonition.name, cls=get_directive_class( admonition.name, admonition.title, admonition.classes ), override=admonition.override, ) # set variables for CSS template to match HTML output from generated directives admonition.name = nodes.make_id(admonition.name) if admonition.icon is not None: admonition.icon = load_svg_into_builder_env(app.builder, admonition.icon) # add styles for sphinx directives versionadded, versionchanged, and deprecated for name, style in VERSION_DIR_STYLE.items(): if name in custom_admonition_names: continue # already handled above # add entries for default style of version directives version_dir_style = CustomAdmonitionConfig( name=name, icon=load_svg_into_builder_env(app.builder, cast(str, style["icon"])), color=style["color"], ) custom_admonitions.append(version_dir_style) setattr(app.builder.env, "sphinx_immaterial_custom_admonitions", custom_admonitions) def on_config_inited(app: Sphinx, config: Config): """Add admonitions based on CSS classes inherited from mkdocs-material theme.""" # override the generic admonition directive if getattr(config, "sphinx_immaterial_override_generic_admonitions"): app.add_directive("admonition", get_directive_class("admonition", ""), True) # generate directives for inherited admonitions from upstream CSS if getattr(config, "sphinx_immaterial_generate_extra_admonitions"): for admonition in INHERITED_ADMONITIONS: app.add_directive( admonition, get_directive_class(admonition, _(admonition.title())), ) confval_name = "sphinx_immaterial_custom_admonitions" # validate user defined config for custom admonitions user_defined_admonitions: List[CustomAdmonitionConfig] = pydantic.TypeAdapter( List[CustomAdmonitionConfig] ).validate_python(getattr(config, confval_name)) setattr(config, confval_name, user_defined_admonitions) user_defined_dir_names = [directive.name for directive in user_defined_admonitions] # override the specific admonitions defined in sphinx and docutils # these are the admonitions that have translated titles in sphinx.locale if getattr(config, "sphinx_immaterial_override_builtin_admonitions"): for admonition, title in admonitionlabels.items(): if admonition in user_defined_dir_names: continue app.add_directive(admonition, get_directive_class(admonition, title), True) if getattr(config, "sphinx_immaterial_override_version_directives"): # override original version directives with custom derivatives for name, __ in VERSION_DIR_STYLE.items(): app.add_directive(name, CustomVersionChange, override=True) def add_admonition_and_icon_css(app: Sphinx, env: BuildEnvironment): """Generates the CSS for icons and admonitions, then appends that to the theme's bundled CSS.""" custom_admonitions = getattr(env, _CUSTOM_ADMONITIONS_KEY) custom_icons = get_custom_icons(env) jinja_env = jinja2.Environment( loader=jinja2.FileSystemLoader(str(PurePath(__file__).parent)) ) template = jinja_env.get_template("custom_admonitions.css") generated = template.render( icons=custom_icons, admonitions=custom_admonitions, ) # append the generated CSS for icons and admonitions add_global_css(app, generated.replace("\n", "")) def setup(app: Sphinx): """register our custom directive.""" app.add_config_value( name="sphinx_immaterial_custom_admonitions", default=[], rebuild="env", types=[List[CustomAdmonitionConfig]], ) app.add_config_value( name="sphinx_immaterial_generate_extra_admonitions", default=True, rebuild="html", types=bool, ) app.add_config_value( name="sphinx_immaterial_override_generic_admonitions", default=True, rebuild="html", types=bool, ) app.add_config_value( name="sphinx_immaterial_override_builtin_admonitions", default=True, rebuild="html", types=bool, ) app.add_config_value( name="sphinx_immaterial_override_version_directives", default=True, rebuild="env", types=bool, ) app.connect("builder-inited", on_builder_inited) app.connect("env-check-consistency", add_admonition_and_icon_css) app.connect("config-inited", on_config_inited) return { "parallel_read_safe": True, "parallel_write_safe": True, }

Last update: Nov 16, 2024