cmarker
This package enables you to write CommonMark Markdown, and import it directly into Typst.
simple.typ |
simple.md |
---|---|
#import "@preview/cmarker:0.1.0"
#cmarker.render(read("simple.md"))
|
# We can write Markdown!
*Using* __lots__ ~of~ `fancy` [features](https://example.org/).
|
simple.pdf |
---|
This document is available in Markdown and rendered PDF formats.
API
We offer a single function:
render(
markdown,
smart-punctuation: true,
blockquote: none,
h1-level: 1,
raw-typst: true,
scope: (:),
show-source: false,
) -> content
The parameters are as follows:
-
markdown
: The CommonMark Markdown string to be processed. Parsed with the pulldown-cmark Rust library. You can set this toread("somefile.md")
to import an external markdown file; see the documentation for the read function.- Accepted values: All strings.
- Required.
-
smart-punctuation
: Automatically convert ASCII punctuation to Unicode equivalents:- nondirectional quotations (" and ') become directional (“” and ‘’);
- three consecutive full stops (...) become ellipses (…);
- two and three consecutive hypen-minus signs (-- and ---) become en and em dashes (– and —).
Note that although Typst also offers this functionality, this conversion is done through the Markdown parser rather than Typst.
- Accepted values: Booleans.
- Default value:
true
.
-
blockquote
: A callback to be used when a blockquote is encountered in the Markdown, ornone
if blockquotes should be treated as normal text. Because Typst does not support blockquotes natively, the user must configure this.- Accepted values: Functions accepting content and returning content, or
none
. - Default value:
none
.
For example, to display a black border to the left of the text one can use:
box.with(stroke: (left: 1pt + black), inset: (left: 5pt, y: 6pt))
- Accepted values: Functions accepting content and returning content, or
-
h1-level
: The level that top-level headings in Markdown should get in Typst. When set to zero, top-level headings are treated as text,##
headings become=
headings,###
headings become==
headings, et cetera; when set to2
,#
headings become==
headings,##
headings become===
headings, et cetera.- Accepted values: Integers in the range [0, 255].
- Default value: 1.
-
raw-typst
: Whether to allow raw Typst code to be injected into the document via HTML comments. If disabled, the comments will act as regular HTML comments.- Accepted values: Booleans.
- Default value:
true
.
For example, when this is enabled,
<!--raw-typst #circle(radius: 10pt) -->
will result in a circle in the document (but only when rendered through Typst). See also<!--typst-begin-exclude-->
and<!--typst-end-exclude-->
, which is the inverse of this. -
scope
: Whenraw-typst
is enabled, this is a dictionary providing the context in which the evaluated Typst code runs. It is useful to pass values in to code inside<!--raw-typst-->
blocks.- Accepted values: Any dictionary.
- Default value:
(:)
.
-
show-source
: A debugging tool. When set totrue
, the Typst code that would otherwise have been displayed will be instead rendered in a code block.- Accepted values: Booleans.
- Default value:
false
.
This function returns the rendered content
.
Supported Markdown Syntax
We support CommonMark with a couple extensions.
- Paragraph breaks: Two newlines, i.e. one blank line.
- Hard line breaks (used more in poetry than prose): Put two spaces at the end of the line.
*emphasis*
or_emphasis_
: emphasis**strong**
or__strong__
: strong~strikethrough~
:strikethrough[links](https://example.org)
: links### Headings
, where#
is a top-level heading,##
a subheading,###
a sub-subheading, etc`inline code blocks`
:inline code blocks
-
``` out of line code blocks ```
```rust let x = 5; ```
let x = 5;
---
, making a horizontal rule:
-
- Unordered - lists
- Unordered
- Lists
-
1. Ordered 1. Lists
- Ordered
- Lists
> blockquotes
, if theblockquote
parameter is set.- Images:
![Some tiled hexagons](examples/hexagons.png)
, giving
Interleaving Markdown and Typst
Sometimes, you might want to render a certain section of the document only when viewed as Markdown, or only when viewed through Typst. To achieve the former, you can simply wrap the section in <!--typst-begin-exclude-->
and <!--typst-end-exclude-->
:
<!--typst-begin-exclude-->
Hello from not Typst!
<!--typst-end-exclude-->
Most Markdown parsers support HTML comments, so from their perspective this is no different to just writing out the Markdown directly; but cmarker.typ
knows to search for those comments and avoid rendering the content in between.
Note that when the opening comment is followed by the end of an element, cmarker.typ
will close the block for you. For example:
> <!--typst-begin-exclude-->
> One
Two
In this code, “Two” will be given no matter where the document is rendered. This is done to prevent us from generating invalid Typst code.
Conversely, one can put Typst code inside a HTML comment of the form <!--raw-typst […]-->
to have it evaluated directly as Typst code (but only if the raw-typst
option to render
is set to true
, otherwise it will just be seen as a regular comment and removed):
<!--raw-typst Hello from #text(fill:blue)[Typst]!-->
Markdown–Typst Polyglots
This project has a manual as a PDF and a README as a Markdown document, but by the power of this library they are in fact the same thing! Furthermore, one can read the README.md
in a markdown viewer and it will display correctly, but one can also run typst compile README.md
to generate the Typst-typeset README.pdf
.
How does this work? We just have to be clever about how we write the README:
<picture>
(Typst preamble content)
#"</picture>
<!--".slice(0,0)
#import "@preview/cmarker:0.1.0"
#let markdown = read("README.md")
#markdown = markdown.slice(markdown.position("</picture>") + "</picture>".len())
#cmarker.render(markdown, h1-level: 0)
/*-->
Regular Markdown goes here…
<!--*///-->
The same code but syntax-highlighted as Typst code helps to illuminate it:
<picture>
(Typst preamble content)
#"</picture>
<!--".slice(0,0)
#import "@preview/cmarker:0.1.0"
#let markdown = read("README.md")
#markdown = markdown.slice(markdown.position("</picture>") + "</picture>".len())
#cmarker.render(markdown, h1-level: 0)
/*-->
Regular Markdown goes here…
<!--*///-->
Limitations
- We do not currently support HTML tags, and they will be stripped from the output; for example, GitHub supports writing
<sub>text</sub>
to get subscript text, but we will render that as simply “text”. In future it would be nice to support a subset of HTML tags. - We do not currently support Markdown tables and footnotes. In future it would be good to.
- Although I tried my best to escape everything correctly, I won’t provide a hard guarantee that everything is fully sandboxed even if you set
raw-typst: false
. That said, Typst itself is well-sandboxed anyway.
Development
- Build the plugin with
./build.sh
, which produces theplugin.wasm
necessary to use this. - Compile examples with
typst compile examples/{name}.typ --root .
. - Compile this README to PDF with
typst compile README.md
.