Use something like math.attach() in text mode?

LaTeX lets you do crazy things like placing superscripts and subscripts on both sides of something, applying arbitrary text styles to the different parts of the expression, e.g.:

Screenshot 2025-02-15 at 16.01.26

When I try to do something similar using #attach() in Typst, I can’t apply any text styles, apparently because of how Typst implements math styles.

Is there a way to attach text blocks all around another block while remaining in text mode? In geochemistry one would like to typeset things like this:

In other words, is there a way to duplicate #attach() in text mode?

The current situation is not ideal and will probably change in the future. One of the difficulties is that it’s often not clear what should be typeset with a math font (which can include seperate glyphs for upright, italic, sans serif…) and what should be typeset with the text font (which can look different from the upright math font).

Anyway currently, to use the characters from the math font you can do this:

$
  attach(
    upright(O),
    tl: sans("xyz"),
    bl: upright("abc"),
    tr: italic("sample"),
    br: upright("VSMOW"),
  )
$

image

And to use the text font (and the monospace font for sans serif), as described in this issue on GitHub you can define a helper function to include normal text elements in math mode:

#let mtext = text.with(font: "Libertinus Serif")

$
  attach(
    upright(O),
    tl: #`xyz`,
    bl: #mtext[abc],
    tr: #mtext[_sample_],
    br: #mtext[VSMOW],
  )
$

image

Note a related change in Typts 0.13 (the next release): single letter words written in double quotes (for example "a") will be typeset with the upright math instead of italic math font, i.e. single letter words will be treated the same as multi-letter words.

1 Like

Thanks, that sounds great. Is there a way to define the text font (here: Libertinus Serif) dynamically (by introspection), defining it as whatever is used in the current document/context?

Maybe like this:

// Do the following wherever you want to capture the current font
#context state("font").update(text.font)

#let mtext(it) = context text(font: state("font").get(), it)
1 Like