Blocks Extension API
Block Structure
The various different block types are created via the Block
class. The Block
class can define all the various parts and handling of the various block parts. The basic structure of a block is shown below:
/// name | argument
options: per block options
Markdown content.
///
Block Extension Anatomy
Normally with Python Markdown, you'd create a processor derived from the various processor types available in the library. You'd then derive an extension from markdown.Extension
that would register the processor. Block extensions are very similar.
A Block extension is comprised of two parts: the Block
object and the BlocksExtension
. It should be noted that we do not use markdown.Extension
, but BlocksExtension
which is derived from it. This is done so we can abstract away the management of all the various registered Block
extensions. It is important to note though that when using the BlocksExtension
that we do not override the extendMarkdown
method, but instead override extendMarkdownBlocks
. In all other respects, BlocksExtension
is just like markdown.Extension
and you can register traditional processors via the md
object or register Block
objects via the block_mgr
object.
Below we have the very bare minimum required to create an extension.
from pymdownx.blocks import BlocksExtension
from pymdownx.blocks.block import Block
import xml.etree.ElementTree as etree
class MyBlock(Block):
NAME = 'my-block'
def on_create(self, parent):
return etree.SubElement(parent, 'div')
class MyBlockExtension(BlocksExtension):
def extendMarkdownBlocks(self, md, block_mgr):
block_mgr.register(MyBlock, self.getConfigs())
def makeExtension(*args, **kwargs):
"""Return extension."""
return MyBlockExtension(*args, **kwargs)
Then we can register and run it:
import markdown
MD = """
/// my-block
content
///
"""
print(markdown.markdown(MD, extensions=[MyBlockExtension()]))
<div>
<p>content</p>
</div>
The Block Object
The block object allows us to define the name of a block, whether an argument and/or options are allowed, and how we generally handle the content.
Global Options
Global options are often set in the BlocksExtension
object like traditional extensions. They are meant to globally control the behavior of a specific block type:
class AdmonitionExtension(BlocksExtension):
"""Admonition Blocks Extension."""
def __init__(self, *args, **kwargs):
"""Initialize."""
self.config = {
"types": [
['note', 'attention', 'caution', 'danger', 'error', 'tip', 'hint', 'warning'],
"Generate Admonition block extensions for the given types."
]
}
super().__init__(*args, **kwargs)
These options are available in the instantiated Block
object via the self.config
attribute.
Tracking Data Across Blocks
There are times when it can be useful to store data across multiple blocks. Each block instance has access to a tracker that is specific to a specific block type and persists across all blocks. It is only cleared when reset
is called on the Markdown
object.
The tracker is accessed by a given Block
extension via the self.tracker
attribute. The attribute contains a dictionary where various keys and values can be stored. This can be used to count blocks on a page or anything else you can think of.
Accessing the Markdown Object
Some plugins occasionally need access to the current Markdown object. If this is needed, it can be accessed via the class attribute self.md
.
Argument
The argument is used to declare a common block specific input for a particular block type. This is often, but not exclusively, used for things like titles. It is specified on the same line as the initial block deceleration.
Blocks are not required to use an argument and it is not required by default and must be declared as either optional or required in order for the block to accept an argument. An argument is declared by setting ARGUMENT
to True
if it is required, None
if it is optionally allowed, or False
if it is not allowed.
The argument is always parsed as a single string, if it is desired to validate the format of the argument or even to process it as multiple arguments, this can be done in the on_validate
event.
class MyBlock(Block):
# Name used for the block
NAME = 'my-block'
ARGUMENT = True
Options
Options is how a block specifies any per block features via an indented YAML block immediately after the block declaration. The YAML indented block is considered a part of the header and is great for options that don't make sense as part of the first line declaration.
An option consists of a keyword to specify the option name, and then a list containing the default value and a validator callback. The callback function should take the input and validate the type and/or coerce the value to an appropriate value. If the input, for whatever reason, is deemed invalid, the callback function should raise an error.
After processing, all options will be available as a dictionary via the instance attribute self.options
. Options will be accessible via the keyword and will return the resolved value.
Built-in validators
A number of Built-in validators are provided. Check out Built-in Validators to learn more, or feel free to write your own.
class MyBlock(Block):
# Name used for the block
NAME = 'my-block'
OPTIONS = {
'tag_name': ['default', type_html_indentifier]
}
Warning
attrs
is a reserved option that is automatically applied to all Block
extensions. This should not be overridden. attrs
takes a dictionary of str
keys and str
values describing the attributes to apply to the outer element of the block as returned by the on_create
.
The attrs
input is sent through type_html_attribute_dict
and is accessible to developers via self.options['attrs']
. The result is a dictionary of key/value pairs where the key is a str
and the value is a str
(or list[str]
in the special case of class
).
is_raw
def is_raw(self, tag: Element) -> bool:
...
This method, given a tag will determine if the block should be considered a "raw" tag based on the Blocks extension's internal logic.
is_block
def is_block(self, tag: Element) -> bool:
...
This method, given a tag will determine if the block should be considered a "block" tag based on the Blocks extension's internal logic.
html_escape
def html_escape(self, text: str) -> str:
...
Takes a string intended for an HTML tag's content and returns it after applying HTML escaping on it. Escapes &
, <
, and >
.
on_init
Event
def on_init(self) -> None:
...
The on_init
event is run every time a new block class is instantiated. This is usually where a specific block type would handle global options and initialize class variables that are needed. If the specified block name in Markdown matches the name of a registered block, that block class will be instantiated, triggering the on_init
event to execute. Each block in a document that is encountered generates its own, new instance.
Only the global config
is available at this time via self.config
. The Markdown
object is also available via self.md
.
The can be a good way to perform setup based on global or local options.
on_validate
Event
def on_validate(self, parent: Element) -> bool:
...
Executed right after the per block argument and option parsing occurs. The argument and options are accessible via self.argument
and self.options
. parent
is the current parent element.
on_validate
is a hook meant to allow the developer to invalidate a block if the options, argument, or even the parent element do not meet some arbitrary criteria. This hook can also be used to make adjustments variables and even do some initialization of class variables based on the results of specific options, arguments, or even the parent element.
If validation fails, False
should be returned and the block will not be parsed as a generic block.
on_create
Event
def on_create(self, parent: Element) -> Element:
...
Called when a block is initially found and initialized. The on_create
method should create the container for the block under the parent element. Other child elements can be created on the root of the container, but outer element of the created container should be returned.
on_add
Event
def on_add(self, block: Element) -> Element:
...
When any calls occur to process new content, on_add
is called. This gives the block a chance to return the element where the content is desired.
This can be useful if the outer element is not the element where the content should go. Keep in mind that content can also be rearranged if needed in the on_end
event.
on_markdown
Event
def on_markdown(self) -> str:
"""Check how element should be treated by the Markdown parser."""
...
The on_markdown
event is used to declare how the content of the block should be handled by the Markdown parser. A string with one of the following values must be returned. All content is treated as HTML content and is stored under the etree element returned via the on_add
event.
Only during the on_end
event will all the content be fully accumulated and processed by relevant block processors, and only during the on_inline_end
event will both block and inline processing be completed.
Result Value | Description |
---|---|
block | Parsed block content will be handled by the Markdown parser as content under a block element. |
inline | Parsed block content will be handled by the Markdown parser as content under an inline element. |
raw | Parsed block content will be preserved as is. No additional Markdown parsing will be applied. Content is expected to be indented and should be documented as such. |
auto | Depending on whether the wrapping parent is a block element, inline element, or something like a code element, Blocks will choose the best approach for the content. Decision is made based on the element returned by the on_add event. |
When using raw
mode, all text will be accumulated under the specified element as an AtomicString
. If nothing is done with the content during the on_end
event, all the content will be HTML escaped by the Python Markdown parser. If desired, the content can be placed into the Python Markdown HTML stash which will protect it from any other rouge Treeprocessors. Keep in mind, if the content is stashed, HTML escaping will not be applied automatically, so HTML escape if it is required.
Indent Raw Content
Because Python Markdown implements HTML processing as a preprocessor, content for a raw
block must be indented 4 spaces to avoid the HTML processing step. The content will not be indented when it reaches the on_end
event. Failure to indent will still allow the code to be processed, but it may not process as expected. An extension that uses raw
should make clear that this is a requirement to avoid unexpected results.
on_end
Event
def on_end(self, block: Element) -> None:
...
When a block is parsed to completion, the on_end
event is executed. This allows an extension to perform any post processing on the elements. You could save the data as raw text and then parse it special at the end or you could walk the HTML elements and move content around, add attributes, or whatever else is needed.
on_inline_end
Event
def on_inline_end(self, block: Element) -> None:
...
When a block is parsed to completion and all inline parsing has been applied, the on_inline_end
event is executed. It is the very last event for a block. This allows an extension to perform any post processing on an element after inline processing.
Built-in Validators
A number of validators are provided via for the purpose of validating YAML option inputs. If what you need is not present, feel free to write your own. All validators are imported from pymdownx.blocks.block
.
type_any
def type_any(value: Any) -> Any:
...
This takes a YAML input and simply passes it through. If you do not want to validate the input because it does not need to be checked, or if you just want to do it manually in the on_validate
event, then this is what you'd want to use.
class Block:
OPTIONS = {'name': [{}, type_any]}
type_none
def type_none(value: Any) -> None:
...
This takes a YAML input and ensures it is None
(or null
) in YAML. This is most useful paired with other types to indicate the option is "unset". See type_multi
to learn how to combine multiple existing types.
class Block:
OPTIONS = {'name': [none, type_multi(type_none, type_string)]}
type_number
def type_number(value: Any) -> int | float:
...
Takes a YAML input value and verifies that it is a float
or int
.
Returns the valid number (float
or int
) or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [0.0, type_number]}
type_integer
def type_integer(value: Any) -> int:
...
Takes a YAML input value and verifies that it is an int
.
Returns the valid int
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [0, type_integer]}
type_ranged_number
def type_ranged_number(minimum: int | float = None, maximum: int | float = None) -> Callable[[Any], int | float]:
Takes a minimum
and/or maximum
and returns a type function that accepts an input and validates that it is a number (float
or int
) that is within the specified range. If None
is provided for either minimum
or maximum
, they will be unbounded.
Returns the valid number (float
or int
) or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [0.0, type_ranged_number(0.0, 100.0)]}
type_ranged_integer
def type_ranged_integer(minimum: int = None, maximum: int = None) -> Callable[[Any], int]:
...
Takes a minimum
and/or maximum
and returns a type function that accepts an input and validates that it is an int
that is within the specified range. If None
is provided for either minimum
or maximum
, they will be unbounded.
Returns the valid int
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [0, type_ranged_integer(0, 100)]}
type_boolean
def type_boolean(value: Any) -> bool:
...
Takes a YAML input and validates that it is a boolean value.
Returns the valid boolean or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [False, type_boolean]}
type_ternary
def type_ternary(value: Any) -> bool | None:
...
Takes a YAML input and validates that it is a bool
value or None
.
Returns the valid bool
or None
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': [None, type_ternary]}
type_string
def type_string(value: Any) -> str:
...
Takes a YAML input and validates that it is a str
value.
Returns the valid str
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': ['default', type_string]}
type_insensitive_string
def type_insensitive_string(value: Any) -> str:
...
Takes a YAML input and validates that it is a str
value and normalizes it by lower casing it.
Returns the valid, lowercase str
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': ['default', type_insensitive_string]}
type_string_in
def type_string_in(value: list[str], insensitive: bool = True) -> Callable[[Any], str]:
...
Takes a list of acceptable string inputs and a boolean indicating whether comparison should be case insensitive. Returns a type function that takes an input and then validates that it is a str
and that the str
value is found in the acceptable string list.
Returns the valid str
or raises a ValueError
.
class Block:
OPTIONS = {'keyword': ['this', type_string_in(['this', 'that'], type_insensitive_string)]}
type_string_delimiter
def type_string_delimiter(value: str, string_type: Callable[[Any], str] = type_string) -> str:
...
Takes a delimiter and string type callback and returns a function that takes an input, verifies that it is a str
, splits it by the delimiter, and ensures that each part validates with the given string type callback.
Returns a list of valid str
values or raises a ValueError
.
class Block:
OPTIONS = {'keyword': ['default', type_string_delimiter(',' type_insensitive_string)]}
type_html_identifier
def type_html_identifier(value: Any) -> str:
...
Tests that a string is an "identifier" as described in CSS. This would normally match tag names, IDs, classes, and attribute names. This is useful if you'd like to validate such HTML constructs.
Returns a str
that is a valid identifier or raises ValueError
.
class Block:
OPTIONS = {'keyword': ['default', type_html_indentifier]}
type_html_classes
def type_html_classes(value: Any) -> list[str]:
...
Takes a YAML input value and verifies that it is a str
and treats it as a space delimited input. The input will be split by spaces and each part will be run through type_html_identifier
.
Returns a list of str
that are valid CSS classes or raises ValueError
.
class Block:
OPTIONS = {'keyword': ['default', type_html_classes]}
type_html_attribute_dict
def type_html_classes(value: Any) -> dict[str, Any]:
...
Note
The returned dictionary will have all values set to string except classes which will be a list of strings. The class
attribute is processed with type_html_classes
.
The id
attribute is also run through type_html_identifier
to ensure a good ID that can be targeted with traditional CSS selectors: #id
.
Takes a YAML input value and verifies that it is a dict
. Keys will be verified to be HTML identifiers and the values to be strings.
Returns a dict[str, Any]
where the values will either be str
or list[str]
as previously noted or raises ValueError
.
class Block:
OPTIONS = {'attributes': [{}, type_html_attribute_dict]}
type_multi
def type_multi(*args: Any) -> Callable[[Any], Any]:
...
Takes a multiple type functions and returns a single type function that takes a YAML input and validates it with all the provided type functions. If the input fails all the validation functions, a ValueError
is raised.