The Julia to Typst Interface
This guide illustrates how to implement Typst formatting for custom types.
Setup
julia> import Base: show
julia> import Typstry: TypstContext, show_typst
julia> using Typstry
Implementation
Consider this custom type.
julia> struct Hi end
Implement a show_typst
method to specify its Typst formatting. Remember to Annotate values taken from untyped locations.
julia> show_typst(io::IO, tc::TypstContext, ::Hi) = print(
io, "Hi", '!' ^ tc[:excitement]::Int
);
Although custom formatting may be handled in show_typst
with get(::TypstContext, ::Symbol, default)
, this may be repetitive when specifying defaults for multiple methods. There is also no way to tell if the value has been specified by the user or if it is a default. Instead, implement a custom TypstContext
which overrides default, but not user specifications.
julia> TypstContext(::Hi) = TypstContext(; excitement = 0);
Those two methods are a complete implementation of the Julia to Typst interface. The following method is optional, but enables interoperability with packages that do not know about Typstry.jl.
julia> show(io::IO, m::Union{
MIME"application/pdf",
MIME"image/png",
MIME"image/svg+xml",
MIME"text/typst"
}, h::Hi) = show(io, m, Typst(h));
Now, Hi
is fully supported by Typstry.jl and implements the show
interface.
julia> h = Hi();
julia> show_typst(h)
Hi
julia> TypstString(h; excitement = 1)
typst"Hi!"
julia> typst"\(h; excitement = 2)"
typst"Hi!!"
julia> show(IOContext(stdout, TypstContext(; excitement = 3)), "text/typst", h)
Hi!!!
Guidelines
While implementing the interface only requires two methods, it may be more challenging to determine how a Julia value should be represented in a Typst source file and its corresponding compiled document. Julia and Typst are distinct languages that differ in both syntax and semantics, so there may be multiple meaningful formats to choose from.
Make the obvious choice, if available
- There is a clear correspondence between these Julia and Typst values
julia> show_typst(1)
$1$
julia> show_typst(nothing)
#none
julia> show_typst(r"[a-z]")
#regex("[a-z]")
Consider both the Typst source text and compiled document formatting
- A
Docs.Text
is documented to "render [its value] as plain text", and therefore corresponds to text in a rendered Typst document - A
TypstString
represents Typst source text, and is printed directly
julia> show_typst(text"[\"a\"]")
#"[\"a\"]"
julia> show_typst(typst"[\"a\"]")
["a"]
Try to generate valid Typst source text
- A
TypstString
represents Typst source text, which may be invalid - A
UnitRange{Int}
is formatted differently for eachMode
, but is always valid
julia> show_typst(1:4)
#range(1, 5)
julia> show_typst(1:4; mode = code)
range(1, 5)
Test for edge cases
$1 / 2$
is not ambiguous inmarkup
mode1 / 2
may be ambiguous inmath
mode expressions, and should be parenthesized
julia> show_typst(1 // 2)
$1 / 2$
julia> show_typst(1 // 2; mode = math)
(1 / 2)
Format values in containers using show_typst
- Values may require their
TypstContext
- The
AbstractVector
method- Encloses source text in dollar signs, so it changes its
Mode
tomath
- Formats its elements with an indent, so it increments its
depth
- Encloses source text in dollar signs, so it changes its
julia> show_typst([true, Any[1, 1.2]])
$vec(
#true, vec(
1, 1.2
)
)$
Check parametric and abstract types
- Similar Julia types may not be representable in the same Typst format
julia> show_typst(0:2:6)
#range(0, 7, step: 2)
julia> show_typst(0:2.0:6)
$vec(
0.0, 2.0, 4.0, 6.0
)$