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 TypstryImplementation
Consider this custom type.
julia> struct Hi endImplement 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.Textis documented to "render [its value] as plain text", and therefore corresponds to text in a rendered Typst document - A
TypstStringrepresents 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
TypstStringrepresents 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 inmarkupmode1 / 2may be ambiguous inmathmode 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
AbstractVectormethod- Encloses source text in dollar signs, so it changes its
Modetomath - 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
)$