The Unicode-Math Package
The Unicode-Math Package
T
E
Xand
LuaT
E
X typesetting engines. With this package, changing maths fonts will be
as easy as changing text fonts not that there are many Unicode maths fonts
yet. Maths input can also be simplied with Unicode since literal glyphs may
be entered instead of control sequences in your document source.
The package is fulled tested under X
T
E
X, but LuaT
E
X support is not yet
complete. User beware, but let me know of any troubles.
Alongside this documentationle, youshouldbe able to nda minimal ex-
ample demonstrating the use of the package, unimath-example.ltx. It also
comes with a separate document, unimath-symbols.pdf, containing a com-
plete listing of mathematical symbols dened by unicode-math.
Finally, while the STIXfonts may be usedwith this package, accessing their
alphabets in their private user area is not yet supported. (Of these additional
alphabets there is a separate caligraphic design distinct to the script design
already included.) Better support for the STIX fonts is planned for an upcom-
ing revision of the package after any problems have been ironed out with the
initial version.
1
Contents
1 Introduction 4
2 Acknowledgements 4
3 Getting started 4
3.1 Package options 5
3.2 Known issues 5
4 Unicode maths font setup 6
4.1 Using multiple fonts 6
4.2 Script and scriptscript
fonts/features 7
5 Maths input 8
5.1 Math style 8
5.2 Bold style 8
5.3 Sans serif style 10
5.4 All (the rest) of the math-
ematical alphabets 10
5.5 Miscellanea 12
I The unicode-math pack-
age 18
6 Things we need 19
6.1 Extras 20
6.2 Compatibility with LuaT
E
X 21
6.3 Alphabet Unicode posi-
tions 21
6.4 STIX fonts 27
6.5 Package options 31
6.6 Overcoming\@onlypreamble 37
7 Fundamentals 37
7.1 Enlarging the number of
maths families 37
7.2 Setting math chars, math
codes, etc. 37
7.3 The main \setmathfont
macro 40
7.4 (Big) operators 46
7.5 Radicals 47
7.6 Delimiters 47
7.7 Maths accents 47
8 Font features 47
8.1 OpenType maths font
features 48
8.2 Script and scriptscript
font options 48
8.3 Range processing 48
8.4 Resolving Greek symbol
name control sequences 52
9 Maths alphabets mapping def-
initions 52
9.1 Initialising math styles 53
9.2 Dening the math style
macros 54
9.3 Dening the mathalpha-
bets per style 55
9.4 Mapping naked math
characters 57
9.5 Mapping chars inside a
math style 59
9.6 Alphabets 61
10 A token list to contain the data
of the math table 75
11 Denitions of the active math
characters 76
12 Epilogue 78
12.1 Primes 78
12.2 Unicode radicals 83
12.3 Unicode sub- and super-
scripts 84
12.4 Synonyms and all the rest 89
12.5 Compatibility 90
13 Error messages 92
2
14 stix table data extraction 93
A Documenting maths support in
the NFSS 93
B X
T
E
X math font dimensions 95
3
1 Introduction
This document describes the unicode-math package, which is an experimental im-
plementation of a macro to Unicode glyph encoding for mathematical characters.
Its intended use is for X
T
E
X, although it is conjectured that some eect could be
spent to create a cross-format package that would also work with LuaT
E
X.
Users who desire to specify maths alphabets only (Greek and Latin letters,
and Arabic numerals) may wish to use Andrew Moschous mathspec package in-
stead.
2 Acknowledgements
Many thanks to: Microsoft for developing the mathematics extension to Open-
Type as part of Microsoft Oice 2007; Jonathan Kew for implementing Unicode
math support in X
T
E
X; Barbara Beeton for her prodigious eort compiling the
denitive list of Unicode math glyphs and their L
A
T
E
X names (inventing them
where necessary), and also for her thoughtful replies to my sometimes incessant
questions. Ross Moore and Chris Rowley have provided moral and technical sup-
port from the very early days with great insight into the issues we face trying to
extend and use T
E
X in the future. Apostolos Syropoulos, Joel Salomon, Khaled
Hosny, and Mariusz Wodzicki have been fantastic beta testers.
3 Getting started
Load unicode-math as a regular L
A
T
E
X package. It should be loaded after any other
maths or font-relatedpackage incase it needs to overwrite their denitions. Heres
an example:
\usepackage{amsmath} % if desired
\usepackage{unicode-math}
\setmathfont{Asana-Math.otf}
Two OpenType maths fonts are included by default in T
E
X Live 2010: Asana
Math and XITS Math. These can be loaded directly with their lename with both
X
L
A
T
E
X and LuaL
A
T
E
X; resp.,
\setmathfont{Asana-Math.otf}
\setmathfont{xits-math.otf}
Other OpenType maths fonts may be loaded in the usual way; please see the
fontspec documentation for more information.
4
Table 1: Package options.
Option Description See
math-style Style of letters section 5.1
bold-style Style of bold letters section 5.2
sans-style Style of sans serif letters section 5.3
nabla Style of the nabla symbol section 5.5.1
partial Style of the partial symbol section 5.5.2
vargreek-shape Style of phi and epsilon section 5.5.3
colon Behaviour of \colon section 5.5.6
slash-delimiter Glyph to use for stretchy slash section 5.5.7
3.1 Package options
Package options may be set when the package as loaded or at any later stage with
the \unimathsetup command. Therefore, the following two examples are equiv-
alent:
\usepackage[math-style=TeX]{unicode-math}
% OR
\usepackage{unicode-math}
\unimathsetup{math-style=TeX}
Note, however, that some package options aects how maths is initialised and
changing an option such as math-style will not take eect until a new maths
font is set up.
Package options may also be used when declaring new maths fonts, passed
via options to the \setmathfont command. Therefore, the following two exam-
ples are equivalent:
\unimathsetup{math-style=TeX}
\setmathfont{Cambria Math}
% OR
\setmathfont[math-style=TeX]{Cambria Math}
A short list of package options is shown in table 1. See following sections for
more information.
3.2 Known issues
In some cases, X
T
E
Xs math support is either missing or I have not discovered
how to access features for various types of maths construct. An example of this
are horizontal extensible symbols, such as underbraces, overbraces, and arrows
5
Table 2: Maths font options.
Option Description See
range Style of letters section 4.1
script-font Font to use for sub- and super-scripts section 4.2
script-features Font features for sub- and super-scripts section 4.2
sscript-font Font to use for nested sub- and super-scripts section 4.2
sscript-features Font features for nested sub- and super-scripts section 4.2
that can grow longer if necessary. Behaviour with such symbols is not necessarily
going to be consistent; please report problem areas to me.
L
A
T
E
Xs concept of math versions is not yet supported. The only way to get
bold maths is to add markup for it all. This is still an area that requires investiga-
tion.
Symbols for maths characters have been inherited from the STIX project and
may change slightly in the long term. We have tried to preserve backwards com-
patibility with L
A
T
E
X conventions as best as possible; again, please report areas of
concern.
4 Unicode maths font setup
In the ideal case, a single Unicode font will contain all maths glyphs we need.
The le unicode-math-table.tex (based on Barbara Beetons stix table) provides
the mapping between Unicode maths glyphs and macro names (all 3298 or
however many of them!). A single command
\setmathfont[font features]{font name}
implements this for every every symbol and alphabetic variant. That means x to
, \xi to , \leq to , etc., \mathscr{H} to and so on, all for Unicode glyphs
within a single font.
This package deals well with Unicode characters for maths input. This in-
cludes using literal Greek letters in formulae, resolving to upright or italic de-
pending on preference.
Font features specic to unicode-math are shown in table 2. Package options
(see table 1) may also be used. Other fontspec features are also valid.
4.1 Using multiple fonts
There will probably be fewcases where a single Unicode maths font suices (sim-
ply due to glyph coverage). The upcoming stix font comes to mind as a possible
exception. It will therefore be necessary to delegate specic Unicode ranges of
glyphs to separate fonts:
6
\setmathfont[range=unicode range,font features]{font name}
where unicode range is a comma-separated list of Unicode slots and ranges such
as {"27D0-"27EB,"27FF,"295B-"297F}. Youmay also use the macro for accessing
the glyph, such as \int, or whole collection of symbols with the same math type,
such as \mathopen, or complete math alphabets such as \mathbb. (Only numerical
slots, however, can be used in ranged declarations.)
X
T
E
X users only X
T
E
X uses the rst maths font selected for choosing various
parameters such as the thickness of fraction rules and so on. (In LuaT
E
X, they are
chosen automatically based on the current font.) To select a new font for these
parameters use \resetmathfont, which behaves identically to \setmathfont.
4.1.1 Control over maths alphabets
Exact control over maths alphabets can be somewhat involved. Here is the current
plan.
[range=\mathbb] to use the font for bb letters only.
[range=\mathbfsfit/{greek,Greek}] for Greek lowercase and uppercase
only (with latin, Latin, num as well for Latin lower-/upper-case and num-
bers).
[range=\mathsfit->\mathbfsfit] to map to dierent output alphabet(s)
(which is rather useless right nowbut will become less useless in the future).
And nowthe trick. If a particular math alphabet is not dened in the font, fall
back onto the lower-base plane (i.e., upright) glyphs. Therefore, to use an ascii-
encoded fractur font, for example, write
\setmathfont[range=\mathfrak]{SomeFracturFont}
and because the math plane fractur glyphs will be missing, unicode-math will
know to use the ascii ones instead. If necessary (but why?) this behaviour can
be forced with [range=\mathfrac->\mathup].
4.2 Script and scriptscript fonts/features
Cambria Math uses OpenType font features to activate smaller optical sizes for
scriptsize and scriptscriptsize symbols (the and , respectively, in
B
C
). Other
fonts will possibly use entirely separate fonts.
Not yet implemented: Bothof these options must be takeninto account. I hope
this will be mostly automatic fromthe users points of view. The +ssty feature can
be detected and applied automatically, and appropriate optical size information
embedded in the fonts will ensure this latter case. Fine tuning should be possible
automatically with fontspec options. We might have to wait until MnMath, for
example, before we really know.
7
5 Maths input
X
T
E
Xs Unicode support allows maths input through two methods. Like classical
T
E
X, macros such as \alpha, \sum, \pm, \leq, and so on, provide verbose access
to the entire repertoire of characters dened by Unicode. The literal characters
themselves may be used instead, for more readable input les.
5.1 Math style
Classically, T
E
X uses italic lowercase Greek letters and upright uppercase Greek
letters for variables in mathematics. This is contrary to the iso standards of us-
ing italic forms for both upper- and lowercase. Furthermore, the French (contrary
again, quelle surprise) have been known to use upright uppercase Latin letters as
well as upright upper- and lowercase Greek. Finally, it is not unknown to use
upright letters for all characters, as seen in the Euler fonts.
The unicode-math package accommodates these possibilities with an inter-
face heavily inspired by Walter Schmidts lucimatx package: a package option
math-style that takes one of four arguments: TeX, ISO, french, or upright.
The philosophy behind the interface to the mathematical alphabet symbols
lies in L
A
T
E
Xs attempt of separating content and formatting. Because input source
text may come froma variety of places, the upright and mathematical italic Latin
and Greek alphabets are unied fromthe point of viewof having a specied mean-
ing in the source text. That is, to get a mathematical , either the ascii (keyboard)
letter x may be typed, or the actual Unicode character may be used. Similarly for
Greek letters. The upright or italic forms are then chosen based on the math-style
package option.
If glyphs are desired that do not map as per the package option (for example,
an upright g is desired but typing $g$ yields ), markup is required to specify
this; to follow from the example: \mathup{g}. Maths alphabets commands such
as \mathup are detailed later.
Alternative interface However, some users may not like this convention of nor-
malising their input. For them, an upright x is an upright x and thats that. (This
will be the case when obtaining source text from copy/pasting PDF or Microsoft
Worddocuments, for example.) For these users, the literal optionto math-style
will eect this behaviour.
The math-style options eects are shown in brief in table 3.
5.2 Bold style
Similar as in the previous section, ISOstandards dier somewhat to T
E
Xs conven-
tions (and classical typesetting) for boldness in mathematics. In the past, it has
8
Table 3: Eects of the math-style package option.
Example
Package option Latin Greek
math-style=ISO (, , , ) (, , , )
math-style=TeX (, , , ) (, , , )
math-style=french (, , B, X) (, , , )
math-style=upright (a, z, B, X) (, , , )
Table 4: Eects of the bold-style package option.
Example
Package option Latin Greek
bold-style=ISO (, , , ) (, , , )
bold-style=TeX (, , , ) (, , , )
bold-style=upright (, , , ) (, , , )
been customary to use bold upright letters to denote things like vectors and ma-
trices. For example, = (
x
,
,
z
). Presumably, this was due to the relatively
scarcity of bold italic fonts in the pre-digital typesetting era. It has been suggested
that italic bold symbols are used nowadays instead.
Bold Greek letters have simply been bold variant glyphs of their regular
weight, as in = (
,
q
,
0
). Confusingly, the syntax in L
A
T
E
X has been dierent
for these two examples: \mathbf in the former (), and \bm (or \boldsymbol,
deprecated) in the latter ().
In unicode-math, the \mathbf command works directly with both Greek and
Latinmaths alphabet characters anddepending onpackage optioneither switches
to upright for Latin letters (bold-style=TeX) as well or keeps them italic (bold-
style=ISO).
To matchthe package options for non-boldcharacters, for bold-style=upright
all bold characters are upright, and bold-style=literal does not change the up-
right/italic shape of the letter.
Upright and italic bold mathematical letters input as direct Unicode charac-
ters are normalised with the same rules. For example, with bold-style=TeX, a
literal bold italic latin character will be typeset upright.
Note that bold-style is independent of math-style, although if the former
is not specied then sensible defaults are chosen based on the latter.
The bold-style options eects are shown in brief in table 4.
9
5.3 Sans serif style
Unicode contains upright and italic, medium and bold mathematical alphabet
characters. These may be explicitly selected with the \mathsfup, \mathsfit,
\mathbfsfup, and \mathbfsfit commands discussed in section 5.4.
Howshouldthe generic \mathsf behave? Unlike bold, sans serif is usedmuch
more sparingly in mathematics. Ive seen recommendations to typeset tensors in
sans serif italic or sans serif italic bold (e.g., examples in the isomath and mattens
packages). But L
A
T
E
Xs \mathsf is upright sans serif.
Therefore I reluctantly add the package options [sans-style=upright] and
[sans-style=italic] to control the behaviour of \mathsf. The upright style sets
up the command to use the seemingly-useless upright sans serif, including Greek;
the italic style switches to using italic inbothLatinandGreek alphabets. Inother
words, this option simply changes the meaning of \mathsf to either \mathsfup or
\mathsfit, respectively. Please let me know if more granular control is necessary
here.
There is also a [sans-style=literal] setting, set automatically with [math-
style=literal], which retains the uprightness of the input characters usedwhen
selecting the sans serif output.
5.3.1 What about bold sans serif?
While you might want your bold upright and your sans serif italic, I dont believe
youd also want your bold sans serif upright (or all vice versa, if thats even con-
ceivable). Therefore, bold sans serif follows from the setting for sans serif; it is
completely independent of the setting for bold.
In other words, \mathbfsf is \mathbfsfup or \mathbfsfit based on [sans-
style=upright] or [sans-style=italic], respectively. And[sans-style=literal]
causes \mathbfsf to retain the same italic or upright shape as the input, and turns
it bold sans serif.
Note well! There is no medium-weight sans serif Greek alphabet in Unicode;
therefore, \mathsf{\alpha} does not make sense (simply produces ) while
\mathbfsf{\alpha} gives .
5.4 All (the rest) of the mathematical alphabets
Unicode contains separate codepoints for most if not all variations of alphabet
shape one may wish to use in mathematical notation. The complete list is shown
in table 5. Some of these have been covered in the previous sections.
At present, the math font switching commands do not nest; therefore if you
want sans serif bold, youmust write \mathsfbf{...} rather than\mathbf{\mathsf{...}}.
This may change in the future.
10
Table 5: Mathematical alphabets dened in Unicode. Black dots indicate an al-
phabet exists in the font specied; grey dots indicate shapes that should always
be taken fromthe upright font even in the italic style. See main text for description
of \mathbbit.
Font Alphabet
Style Shape Series Switch Latin Greek Numerals
Serif Upright Normal \mathup
Bold \mathbfup
Italic Normal \mathit
Bold \mathbfit
Sans serif Upright Normal \mathsfup
Italic Normal \mathsfit
Upright Bold \mathbfsfup
Italic Bold \mathbfsfit
Typewriter Upright Normal \mathtt
Double-struck Upright Normal \mathbb
Italic Normal \mathbbit
Script Upright Normal \mathscr
Bold \matbfscr
Fraktur Upright Normal \mathfrak
Bold \mathbffrac
11
5.4.1 Double-struck
The double-struck alphabet (also known as blackboard bold) consists of upright
Latinletters {,], numerals , summationsymbol , andfour Greek letters
only: {H].
While \mathbb{\sum} does produce a double-struck summation symbol, its
limits arent properly aligned. Therefore, either the literal character or the control
sequence \Bbbsum are recommended instead.
There are also ve Latin italic double-struck letters: . These can be ac-
cessed(if not withtheir literal characters or control sequences) withthe \mathbbit
alphabet switch, but note that only those ve letters will give the expected output.
5.4.2 Caligraphic vs. Script variants
The Unicode maths encoding contains an alphabet style for Script letters, and
while by default \mathcal and \mathscr are synonyms, there are some situations
when a separate Caligraphic style is needed as well.
If a font contains alternate glyphs for a separat caligraphic style, they can be
selected explicitly as shown below. This feature is currently only supported by the
XITS Math font, where the caligraphic letters are accessed with the same glyph
slots as the script letters but with the rst stylistic set feature (ss01) applied.
\setmathfont[range={\mathcal,\mathbfcal},StylisticSet=1]{XITS Math}
An example is shown below.
The Script style (\mathscr) in XITS Math is:
The Caligraphic style (\mathcal) in XITS Math is:
5.5 Miscellanea
5.5.1 Nabla
The symbol comes in the six forms shown in table 6. We want an individual
option to specify whether we want upright or italic nabla by default (when ei-
ther upright or italic nabla is used in the source). T
E
X classically uses an up-
right nabla, and iso standards agree with this convention. The package op-
tions nabla=upright and nabla=italic switch between the two choices, and
nabla=literal respects the shape of the input character. This is then inherited
through \mathbf; \mathit and \mathup can be used to force one way or the other.
nabla=italic is the default. nabla=literal is activated automatically after
math-style=literal.
12
Table 6: The various forms of nabla.
Description Glyph
Upright Serif
Bold serif
Bold sans
Italic Serif
Bold serif
Bold sans
Table 7: The various forms of the partial dierential. Note that in the fonts used
to display these glyphs, the rst upright partial is incorrectly shown in an italic
style.
Description Glyph
Regular Upright
Italic
Bold Upright
Italic
Sans bold Upright
Italic
5.5.2 Partial
The same applies to the symbols u+2202 partial dierential and u+1D715 math
italic partial dierential.
At time of writing, both the Cambria Math and STIX fonts display these two
glyphs in the same italic style, but this is hopefully a bug that will be corrected in
the future the plain partial dierential should really have an upright shape.
Use the partial=upright or partial=italic package options to specify
which one you would like, or partial=literal to have the same character used
in the output as was used for the input. The default is (always, unless someone
requests and argues otherwise) partial=italic.
1
partial=literal is activated
following math-style=literal.
See table 7 for the variations on the partial dierential symbol.
1
A good argument would revolve around some international standards body recommending up-
right over italic. I just dont have the time right now to look it up.
13
5.5.3 Epsilon and phi: vs. and vs.
T
E
X denes \epsilon to look like and \varepsilon to look like . The Unicode
glyph directly after delta and before zeta is epsilon and looks like ; there is a
subsequent variant of epsilon that looks like . This creates a problem. People
who use Unicode input wont want their glyphs transforming; T
E
X users will be
confused that what they think as normal epsilon is actual the variant epsilon.
And the same problem exists for phi.
We have a package optionto control this behaviour. Withvargreek-shape=TeX,
\phi and \epsilon produce and and \varphi and \varepsilon produce
and . With vargreek-shape=unicode, these symbols are swapped. Note, how-
ever, that Unicode characters are not aected by this option. That is, no remapping
occurs of the characters/glyphs, only the control sequences.
The package default is to use vargreek-shape=TeX.
5.5.4 Primes
Primes (
i
) may be input in several ways. You may use any combination of ascii
straight quote ('), Unicode prime u+2032 (
i
), and \prime; when multiple primes
occur next to each other, they chain together to form double, triple, or quadruple
primes if the font contains pre-drawn glyphs. These may also be accessed with
\dprime, \trprime, and \qprime, respectively.
If the font does not contain the pre-drawn glyphs or more than four primes
are used, the single prime glyph is used multiple times with a negative kern to get
the spacing right. There is no user interface to adjust this negative kern yet (be-
cause I havent decided what it should look like); if you need to, write something
like this:
\ExplSyntaxOn
\muskip_gset:Nn \g_um_primekern_muskip { -\thinmuskip/2 }
\ExplySyntaxOff
Backwards or reverse primes behave in exactly the same way; use any of ascii
back tick (`), Unicode reverse prime u+2035 (), or \backprime to access it. Mul-
tiple backwards primes can also be called with \backdprime, \backtrprime, and
\backqprime.
If you ever need to enter the straight quote ' or the backtick ` in maths mode,
these glyphs can be accessed with \mathstraightquote and \mathbacktick.
5.5.5 Unicode subscripts and superscripts
You may, if you wish, use Unicode subscripts and superscripts in your source
document. For basic expressions, the use of these characters can make the input
14
A Z
Figure 1: The Unicode superscripts supported as input characters. These are the
literal glyphs from Charis SIL, not the output seen when used for maths input.
The A and Z are to provide context for the size and location of the superscript
glyphs.
A Z
Figure 2: The Unicode subscripts supported as input characters. See note from
gure 1.
more readable. Adjacent sub- or super-scripts will be concatenated into a single
expression.
The range of subscripts and superscripts supported by this package are
shown in gures 1 and 2. Please request more if you think it is appropriate.
5.5.6 Colon
The colon is one of the few confusing characters of Unicode maths. In T
E
X, : is
dened as a colon with relation spacing: . While \colon is dened as a
colon with punctuation spacing: .
In Unicode, u+003A colon is dened as a punctuation symbol, while u+2236
ratio is the colon-like symbol used in mathematics to denote ratios and other
things.
This breaks the usual straightforwardmapping fromcontrol sequence to Uni-
code input character to (the same) Unicode glyph.
To preserve input compatibility, we remap the ascii input character : to
u+2236. Typing a literal u+2236 char will result in the same output. If amsmath
is loaded, then the denition of \colon is inherited from there (it looks like a
punctuation colon with additional space around it). Otherwise, \colon is made
to output a colon with \mathpunct spacing.
The package option colon=literal forces ascii input : to be printed as
\mathcolon instead.
5.5.7 Slashes and backslashes
There are several slash-like symbols dened in Unicode. The complete list is
shown in table 8.
In regular L
A
T
E
X we can write \left\slash\right\backslash and so on
and obtain extensible delimiter-like symbols. Not all of the Unicode slashes are
suitable for this (and do not have the font support to do it).
15
Table 8: Slashes and backslashes.
Slot Name Glyph Command
u+002F solidus / \slash
u+2044 fraction slash \fracslash
u+2215 division slash \divslash
u+29F8 big solidus \xsol
u+005C reverse solidus \ \backslash
u+2216 set minus \smallsetminus
u+29F5 reverse solidus operator \setminus
u+29F9 big reverse solidus \xbsol
Slash Of u+2044 fraction slash, TR25 says that it is:
used to build up simple fractions in running texthowever parsers
of mathematical texts should be prepared to handle fraction slash
when it is received from other sources.
u+2215 division slash should be used when division is represented without
a built-up fraction; 227, for example.
u+29F8 big solidus is a big operator (like ).
Backslash The u+005C reverse solidus character \backslash is used for denot-
ing double cosets: \. (So Im led to believe.) It may be used as a stretchy de-
limiter if supported by the font.
MathML uses u+2216 set minus like this: .
2
The L
A
T
E
X command name
\smallsetminus is used for backwards compatibility.
Presumably, u+29F5 reverse solidus operator is intended to be used in a sim-
ilar way, but it could also (perhaps?) be used to represent inverse division:
7 \ 22.
3
The L
A
T
E
X name for this character is \setminus.
Finally, u+29F9 big reverse solidus is a big operator (like ).
How to use all of these things Unfortunately, font support for the above char-
acters/glyphs is rather inconsistent. In Cambria Math, the only slash that grows
(say when writing
_
___
1 1
1 u
_ )
2
4.4.5.11 http://www.w3.org/TR/MathML3/
3
This is valid syntax in the Octave and Matlab programming languages, in which it means matrix
inverse pre-multiplication. I.e., A \ B A
-1
B.
16
is the fraction slash, which we just established above is sort of only supposed to
be used in text.
Of the above characters, the following are allowed to be used after \left,
\middle, and \right:
\solidus;
\fracslash;
\slash; and,
\backslash (the only reverse slash).
However, we assume that there is only one stretchy slash in the font; this
is assumed by default to be u+002F solidus. Writing \left/ or \left\slash or
\leftfracslash will all result in the same stretchy delimiter being used.
The delimiter used can be changed with the slash-delimiter package op-
tion. Allowed values are ascii, frac, and div, corresponding to the respective
Unicode slots.
For example: as mentioned above, Cambria Maths stretchy slash is u+2044
fraction slash. When using Cambria Math, then unicode-math should be loaded
with the slash-delimiter=frac option. (This should be a font option rather than
a package option, but it will change soon.)
5.5.8 Pre-drawn fraction characters
Pre-drawn fractions u+00BCu+00BE, u+2150u+215E are not suitable for use in
mathematics output. However, they can be useful as input characters to abbre-
viate common fractions.
For example, instead of writing \tfrac12 x, its more readable to have x
in the source instead. (There are four missing glyphs above for uS, 17, 19, and
11u; I dont have a font that contains them.)
If the \tfrac command exists (i.e., if amsmath is loaded or you have spe-
cially dened \tfrac for this purpose), it will be used to typeset the fractions.
If not, regular \frac will be used. The command to use (\tfrac or \frac) can
be forced either way with the package option active-frac=small or active-
frac=normalsize, respectively.
5.5.9 Circles
Unicode denes a large number of dierent types of circles for a variety of math-
ematical purposes. There are thirteen alone just considering the all white and all
black ones, shown in table 9.
17
Slot Command Glyph
u+00B7 \cdotp
u+22C5 \cdot
u+2219 \vysmblkcircle
u+2022 \smblkcircle
u+2981 \mdsmblkcircle
u+26AB \mdblkcircle
u+25CF \mdlgblkcircle
u+2B24 \lgblkcircle
Glyph Command Slot
\vysmwhtcircle u+2218
\smwhtcircle u+25E6
\mdsmwhtcircle u+26AC
\mdwhtcircle u+26AA
\mdlgwhtcircle u+25CB
\lgwhtcircle u+25EF
Table 9: Filled and hollow Unicode circles.
Slot Command Glyph Class
u+25B5 \vartriangle binary
u+25B3 \bigtriangleup binary
u+25B3 \triangle ordinary
u+2206 \increment ordinary
u+0394 \mathup\Delta ordinary
Table 10: Dierent upwards pointing triangles.
L
A
T
E
X denes considerably fewer: \circ and csbigcirc for white; \bullet for
black. This package maps those commands to \vysmwhtcircle, \mdlgwhtcircle,
and \smblkcircle, respectively.
5.5.10 Triangles
While there arent as many dierent sizes of triangle as there are circle, theres
some important distinctions to make betweena fewsimilar characters. See table 10
for the full summary.
These triangles all have dierent intended meanings. Note for backwards
compatibility with T
E
X, u+25B3 has two dierent mappings in unicode-math. \big-
triangleup is intended as a binary operator whereas \triangle is intended to be
used as a letter-like symbol.
But youre better o if youre using the latter form to indicate an increment
to use the glyph intended for this purpose, u+2206: .
Finally, given that and are provided for you already, it is better o to only
use upright Greek Delta if youre actually using it as a symbolic entity such as
a variable on its own.
18
File I
The unicode-math package
<*preamble>
6 Things we need
1 \usepackage{ifxetex,ifluatex}
2 \ifxetex\else\ifluatex\else
3 \PackageError{unicode-math}{%
4 Cannot be run with pdfLaTeX!\MessageBreak
5 Use XeLaTeX or LuaLaTeX instead.%
6 }\@ehd
7 \fi\fi
Packages
8 \RequirePackage{expl3}[2009/08/12]
9 \RequirePackage{xparse}[2009/08/31]
10 \RequirePackage{l3keys2e}
11 \RequirePackage{fontspec}[2010/05/18]
12 \RequirePackage{catchfile}
Start using L
A
T
E
X3 nally!
13 \ExplSyntaxOn
14 \@ifclassloaded{memoir}{
15 \cs_set_eq:NN \um_after_pkg:nn \AtEndPackage
16 }{
17 \RequirePackage{scrlfile}
18 \cs_set_eq:NN \um_after_pkg:nn \AfterPackage
19 }
Extra expl3 variants
20 \cs_generate_variant:Nn \tl_put_right:Nn {cx}
21 \cs_generate_variant:Nn \seq_if_in:NnTF {NV}
22 \cs_generate_variant:Nn \prop_gput:Nnn {Nxn}
23 \cs_generate_variant:Nn \prop_get:NnN {cxN}
24 \cs_generate_variant:Nn \prop_if_in:NnTF {cx}
25 \cs_new:Npn \exp_args:NNcc #1#2#3#4 {
26 \exp_after:wN #1 \exp_after:wN #2
27 \cs:w #3 \exp_after:wN \cs_end:
28 \cs:w #4 \cs_end:
29 }
19
Conditionals
30 \bool_new:N \l_um_fontspec_feature_bool
31 \bool_new:N \l_um_ot_math_bool
32 \bool_new:N \l_um_init_bool
33 \bool_new:N \l_um_implicit_alph_bool
34 \bool_new:N \g_um_mainfont_set_bool
For math-style:
35 \bool_new:N \g_um_literal_bool
36 \bool_new:N \g_um_upLatin_bool
37 \bool_new:N \g_um_uplatin_bool
38 \bool_new:N \g_um_upGreek_bool
39 \bool_new:N \g_um_upgreek_bool
For bold-style:
40 \bool_new:N \g_um_bfliteral_bool
41 \bool_new:N \g_um_bfupLatin_bool
42 \bool_new:N \g_um_bfuplatin_bool
43 \bool_new:N \g_um_bfupGreek_bool
44 \bool_new:N \g_um_bfupgreek_bool
For sans-style:
45 \bool_new:N \g_um_upsans_bool
46 \bool_new:N \g_um_sfliteral_bool
For assorted package options:
47 \bool_new:N \g_um_upNabla_bool
48 \bool_new:N \g_um_uppartial_bool
49 \bool_new:N \g_um_literal_Nabla_bool
50 \bool_new:N \g_um_literal_partial_bool
51 \bool_new:N \g_um_texgreek_bool
52 \bool_new:N \l_um_smallfrac_bool
53 \bool_new:N \g_um_literal_colon_bool
Variables
54 \int_new:N \g_um_fam_int
55 \tl_set:Nn \g_um_math_alphabet_name_latin_tl {Latin,~lowercase}
56 \tl_set:Nn \g_um_math_alphabet_name_Latin_tl {Latin,~uppercase}
57 \tl_set:Nn \g_um_math_alphabet_name_greek_tl {Greek,~lowercase}
58 \tl_set:Nn \g_um_math_alphabet_name_Greek_tl {Greek,~uppercase}
59 \tl_set:Nn \g_um_math_alphabet_name_num_tl {Numerals}
60 \tl_set:Nn \g_um_math_alphabet_name_misc_tl {Misc.}
6.1 Extras
\um_glyph_if_exist:nTF : TODO: Generalise for arbitrary fonts! \_um_font is not always the one used for
a specic glyph!!
20
61 \prg_new_conditional:Nnn \um_glyph_if_exist:n {p,TF,T,F} {
62 \etex_iffontchar:D \l_um_font #1 \scan_stop:
63 \prg_return_true:
64 \else:
65 \prg_return_false:
66 \fi:
67 }
68 \cs_generate_variant:Nn \um_glyph_if_exist_p:n {c}
69 \cs_generate_variant:Nn \um_glyph_if_exist:nTF {c}
70 \cs_generate_variant:Nn \um_glyph_if_exist:nT {c}
71 \cs_generate_variant:Nn \um_glyph_if_exist:nF {c}
6.2 Compatibility with LuaT
E
X
72 \xetex_or_luatex:nnn { \cs_new:Npn \um_cs_compat:n #1 }
73 { \cs_set_eq:cc {U#1} {XeTeX#1} }
74 { \cs_set_eq:cc {U#1} {luatexU#1} }
75 \um_cs_compat:n {mathcode}
76 \um_cs_compat:n {delcode}
77 \um_cs_compat:n {mathcodenum}
78 \um_cs_compat:n {mathcharnum}
79 \um_cs_compat:n {mathchardef}
80 \um_cs_compat:n {radical}
81 \um_cs_compat:n {mathaccent}
82 \um_cs_compat:n {delimiter}
6.2.1 Function variants
83 \cs_generate_variant:Nn \fontspec_select:nn {x}
84
(Error messages and warning denitions go here fromthe msg chunk dened
in section 13 on page 92.)
85 *
6.3 Alphabet Unicode positions
Before we begin, lets dene the positions of the various Unicode alphabets so that
our code is a little more readable.
4
Rather than readable, in the end, this makes the code more extensible.
86 \cs_new:Npn \usv_set:nnn #1#2#3 {
87 \tl_set:cn { \um_to_usv:nn {#1}{#2} } {#3}
88 }
89 \cs_new:Npn \um_to_usv:nn #1#2 { g_um_#1_#2_usv }
4
u.s.v. stands for Unicode scalar value.
21
Alphabets
90 \usv_set:nnn {up}{num}{48}
91 \usv_set:nnn {up}{Latin}{65}
92 \usv_set:nnn {up}{latin}{97}
93 \usv_set:nnn {up}{Greek}{"391}
94 \usv_set:nnn {up}{greek}{"3B1}
95 \usv_set:nnn {it}{Latin}{"1D434}
96 \usv_set:nnn {it}{latin}{"1D44E}
97 \usv_set:nnn {it}{Greek}{"1D6E2}
98 \usv_set:nnn {it}{greek}{"1D6FC}
99 \usv_set:nnn {bb}{num}{"1D7D8}
100 \usv_set:nnn {bb}{Latin}{"1D538}
101 \usv_set:nnn {bb}{latin}{"1D552}
102 \usv_set:nnn {scr}{Latin}{"1D49C}
103 \usv_set:nnn {cal}{Latin}{"1D49C}
104 \usv_set:nnn {scr}{latin}{"1D4B6}
105 \usv_set:nnn {frak}{Latin}{"1D504}
106 \usv_set:nnn {frak}{latin}{"1D51E}
107 \usv_set:nnn {sf}{num}{"1D7E2}
108 \usv_set:nnn {sfup}{num}{"1D7E2}
109 \usv_set:nnn {sfit}{num}{"1D7E2}
110 \usv_set:nnn {sfup}{Latin}{"1D5A0}
111 \usv_set:nnn {sf}{Latin}{"1D5A0}
112 \usv_set:nnn {sfup}{latin}{"1D5BA}
113 \usv_set:nnn {sf}{latin}{"1D5BA}
114 \usv_set:nnn {sfit}{Latin}{"1D608}
115 \usv_set:nnn {sfit}{latin}{"1D622}
116 \usv_set:nnn {tt}{num}{"1D7F6}
117 \usv_set:nnn {tt}{Latin}{"1D670}
118 \usv_set:nnn {tt}{latin}{"1D68A}
Bold:
119 \usv_set:nnn {bf}{num}{"1D7CE}
120 \usv_set:nnn {bfup}{num}{"1D7CE}
121 \usv_set:nnn {bfit}{num}{"1D7CE}
122 \usv_set:nnn {bfup}{Latin}{"1D400}
123 \usv_set:nnn {bfup}{latin}{"1D41A}
124 \usv_set:nnn {bfup}{Greek}{"1D6A8}
125 \usv_set:nnn {bfup}{greek}{"1D6C2}
126 \usv_set:nnn {bfit}{Latin}{"1D468}
127 \usv_set:nnn {bfit}{latin}{"1D482}
128 \usv_set:nnn {bfit}{Greek}{"1D71C}
129 \usv_set:nnn {bfit}{greek}{"1D736}
130 \usv_set:nnn {bffrak}{Latin}{"1D56C}
131 \usv_set:nnn {bffrak}{latin}{"1D586}
132 \usv_set:nnn {bfscr}{Latin}{"1D4D0}
133 \usv_set:nnn {bfcal}{Latin}{"1D4D0}
22
134 \usv_set:nnn {bfscr}{latin}{"1D4EA}
135 \usv_set:nnn {bfsf}{num}{"1D7EC}
136 \usv_set:nnn {bfsfup}{num}{"1D7EC}
137 \usv_set:nnn {bfsfit}{num}{"1D7EC}
138 \usv_set:nnn {bfsfup}{Latin}{"1D5D4}
139 \usv_set:nnn {bfsfup}{latin}{"1D5EE}
140 \usv_set:nnn {bfsfup}{Greek}{"1D756}
141 \usv_set:nnn {bfsfup}{greek}{"1D770}
142 \usv_set:nnn {bfsfit}{Latin}{"1D63C}
143 \usv_set:nnn {bfsfit}{latin}{"1D656}
144 \usv_set:nnn {bfsfit}{Greek}{"1D790}
145 \usv_set:nnn {bfsfit}{greek}{"1D7AA}
146 \usv_set:nnn {bfsf}{Latin}{ \bool_if:NTF \g_um_upLatin_bool \g_um_bfsfup_Latin_usv \g_um_bfsfit_Latin_usv }
147 \usv_set:nnn {bfsf}{latin}{ \bool_if:NTF \g_um_uplatin_bool \g_um_bfsfup_latin_usv \g_um_bfsfit_latin_usv }
148 \usv_set:nnn {bfsf}{Greek}{ \bool_if:NTF \g_um_upGreek_bool \g_um_bfsfup_Greek_usv \g_um_bfsfit_Greek_usv }
149 \usv_set:nnn {bfsf}{greek}{ \bool_if:NTF \g_um_upgreek_bool \g_um_bfsfup_greek_usv \g_um_bfsfit_greek_usv }
150 \usv_set:nnn {bf}{Latin}{ \bool_if:NTF \g_um_bfupLatin_bool \g_um_bfup_Latin_usv \g_um_bfit_Latin_usv }
151 \usv_set:nnn {bf}{latin}{ \bool_if:NTF \g_um_bfuplatin_bool \g_um_bfup_latin_usv \g_um_bfit_latin_usv }
152 \usv_set:nnn {bf}{Greek}{ \bool_if:NTF \g_um_bfupGreek_bool \g_um_bfup_Greek_usv \g_um_bfit_Greek_usv }
153 \usv_set:nnn {bf}{greek}{ \bool_if:NTF \g_um_bfupgreek_bool \g_um_bfup_greek_usv \g_um_bfit_greek_usv }
Greek variants:
154 \usv_set:nnn {up}{varTheta}{"3F4}
155 \usv_set:nnn {up}{Digamma}{"3DC}
156 \usv_set:nnn {up}{varepsilon}{"3F5}
157 \usv_set:nnn {up}{vartheta}{"3D1}
158 \usv_set:nnn {up}{varkappa}{"3F0}
159 \usv_set:nnn {up}{varphi}{"3D5}
160 \usv_set:nnn {up}{varrho}{"3F1}
161 \usv_set:nnn {up}{varpi}{"3D6}
162 \usv_set:nnn {up}{digamma}{"3DD}
Bold:
163 \usv_set:nnn {bfup}{varTheta}{"1D6B9}
164 \usv_set:nnn {bfup}{Digamma}{"1D7CA}
165 \usv_set:nnn {bfup}{varepsilon}{"1D6DC}
166 \usv_set:nnn {bfup}{vartheta}{"1D6DD}
167 \usv_set:nnn {bfup}{varkappa}{"1D6DE}
168 \usv_set:nnn {bfup}{varphi}{"1D6DF}
169 \usv_set:nnn {bfup}{varrho}{"1D6E0}
170 \usv_set:nnn {bfup}{varpi}{"1D6E1}
171 \usv_set:nnn {bfup}{digamma}{"1D7CB}
Italic Greek variants:
172 \usv_set:nnn {it}{varTheta}{"1D6F3}
173 \usv_set:nnn {it}{varepsilon}{"1D716}
174 \usv_set:nnn {it}{vartheta}{"1D717}
23
175 \usv_set:nnn {it}{varkappa}{"1D718}
176 \usv_set:nnn {it}{varphi}{"1D719}
177 \usv_set:nnn {it}{varrho}{"1D71A}
178 \usv_set:nnn {it}{varpi}{"1D71B}
Bold italic:
179 \usv_set:nnn {bfit}{varTheta}{"1D72D}
180 \usv_set:nnn {bfit}{varepsilon}{"1D750}
181 \usv_set:nnn {bfit}{vartheta}{"1D751}
182 \usv_set:nnn {bfit}{varkappa}{"1D752}
183 \usv_set:nnn {bfit}{varphi}{"1D753}
184 \usv_set:nnn {bfit}{varrho}{"1D754}
185 \usv_set:nnn {bfit}{varpi}{"1D755}
Bold sans:
186 \usv_set:nnn {bfsfup}{varTheta}{"1D767}
187 \usv_set:nnn {bfsfup}{varepsilon}{"1D78A}
188 \usv_set:nnn {bfsfup}{vartheta}{"1D78B}
189 \usv_set:nnn {bfsfup}{varkappa}{"1D78C}
190 \usv_set:nnn {bfsfup}{varphi}{"1D78D}
191 \usv_set:nnn {bfsfup}{varrho}{"1D78E}
192 \usv_set:nnn {bfsfup}{varpi}{"1D78F}
Bold sans italic:
193 \usv_set:nnn {bfsfit}{varTheta} {"1D7A1}
194 \usv_set:nnn {bfsfit}{varepsilon}{"1D7C4}
195 \usv_set:nnn {bfsfit}{vartheta} {"1D7C5}
196 \usv_set:nnn {bfsfit}{varkappa} {"1D7C6}
197 \usv_set:nnn {bfsfit}{varphi} {"1D7C7}
198 \usv_set:nnn {bfsfit}{varrho} {"1D7C8}
199 \usv_set:nnn {bfsfit}{varpi} {"1D7C9}
Nabla:
200 \usv_set:nnn {up} {Nabla}{"02207}
201 \usv_set:nnn {it} {Nabla}{"1D6FB}
202 \usv_set:nnn {bfup} {Nabla}{"1D6C1}
203 \usv_set:nnn {bfit} {Nabla}{"1D735}
204 \usv_set:nnn {bfsfup}{Nabla}{"1D76F}
205 \usv_set:nnn {bfsfit}{Nabla}{"1D7A9}
Partial:
206 \usv_set:nnn {up} {partial}{"02202}
207 \usv_set:nnn {it} {partial}{"1D715}
208 \usv_set:nnn {bfup} {partial}{"1D6DB}
209 \usv_set:nnn {bfit} {partial}{"1D74F}
210 \usv_set:nnn {bfsfup}{partial}{"1D789}
211 \usv_set:nnn {bfsfit}{partial}{"1D7C3}
24
Exceptions These are need for mapping with the exceptions in other alphabets:
(coming up)
212 \usv_set:nnn {up}{B}{`\B}
213 \usv_set:nnn {up}{C}{`\C}
214 \usv_set:nnn {up}{D}{`\D}
215 \usv_set:nnn {up}{E}{`\E}
216 \usv_set:nnn {up}{F}{`\F}
217 \usv_set:nnn {up}{H}{`\H}
218 \usv_set:nnn {up}{I}{`\I}
219 \usv_set:nnn {up}{L}{`\L}
220 \usv_set:nnn {up}{M}{`\M}
221 \usv_set:nnn {up}{N}{`\N}
222 \usv_set:nnn {up}{P}{`\P}
223 \usv_set:nnn {up}{Q}{`\Q}
224 \usv_set:nnn {up}{R}{`\R}
225 \usv_set:nnn {up}{Z}{`\Z}
226 \usv_set:nnn {it}{B}{"1D435}
227 \usv_set:nnn {it}{C}{"1D436}
228 \usv_set:nnn {it}{D}{"1D437}
229 \usv_set:nnn {it}{E}{"1D438}
230 \usv_set:nnn {it}{F}{"1D439}
231 \usv_set:nnn {it}{H}{"1D43B}
232 \usv_set:nnn {it}{I}{"1D43C}
233 \usv_set:nnn {it}{L}{"1D43F}
234 \usv_set:nnn {it}{M}{"1D440}
235 \usv_set:nnn {it}{N}{"1D441}
236 \usv_set:nnn {it}{P}{"1D443}
237 \usv_set:nnn {it}{Q}{"1D444}
238 \usv_set:nnn {it}{R}{"1D445}
239 \usv_set:nnn {it}{Z}{"1D44D}
240 \usv_set:nnn {up}{d}{`\d}
241 \usv_set:nnn {up}{e}{`\e}
242 \usv_set:nnn {up}{g}{`\g}
243 \usv_set:nnn {up}{h}{`\h}
244 \usv_set:nnn {up}{i}{`\i}
245 \usv_set:nnn {up}{j}{`\j}
246 \usv_set:nnn {up}{o}{`\o}
247 \usv_set:nnn {it}{d}{"1D451}
248 \usv_set:nnn {it}{e}{"1D452}
249 \usv_set:nnn {it}{g}{"1D454}
250 \usv_set:nnn {it}{h}{"0210E}
251 \usv_set:nnn {it}{i}{"1D456}
252 \usv_set:nnn {it}{j}{"1D457}
253 \usv_set:nnn {it}{o}{"1D45C}
Latin h:
25
254 \usv_set:nnn {bb} {h}{"1D559}
255 \usv_set:nnn {tt} {h}{"1D691}
256 \usv_set:nnn {scr} {h}{"1D4BD}
257 \usv_set:nnn {frak} {h}{"1D525}
258 \usv_set:nnn {bfup} {h}{"1D421}
259 \usv_set:nnn {bfit} {h}{"1D489}
260 \usv_set:nnn {sfup} {h}{"1D5C1}
261 \usv_set:nnn {sfit} {h}{"1D629}
262 \usv_set:nnn {bffrak}{h}{"1D58D}
263 \usv_set:nnn {bfscr} {h}{"1D4F1}
264 \usv_set:nnn {bfsfup}{h}{"1D5F5}
265 \usv_set:nnn {bfsfit}{h}{"1D65D}
Dotless i and j:
266 \usv_set:nnn {up}{dotlessi}{"00131}
267 \usv_set:nnn {up}{dotlessj}{"00237}
268 \usv_set:nnn {it}{dotlessi}{"1D6A4}
269 \usv_set:nnn {it}{dotlessj}{"1D6A5}
Blackboard:
270 \usv_set:nnn {bb}{C}{"2102}
271 \usv_set:nnn {bb}{H}{"210D}
272 \usv_set:nnn {bb}{N}{"2115}
273 \usv_set:nnn {bb}{P}{"2119}
274 \usv_set:nnn {bb}{Q}{"211A}
275 \usv_set:nnn {bb}{R}{"211D}
276 \usv_set:nnn {bb}{Z}{"2124}
277 \usv_set:nnn {up}{Pi} {"003A0}
278 \usv_set:nnn {up}{pi} {"003C0}
279 \usv_set:nnn {up}{Gamma} {"00393}
280 \usv_set:nnn {up}{gamma} {"003B3}
281 \usv_set:nnn {up}{summation}{"02211}
282 \usv_set:nnn {it}{Pi} {"1D6F1}
283 \usv_set:nnn {it}{pi} {"1D70B}
284 \usv_set:nnn {it}{Gamma} {"1D6E4}
285 \usv_set:nnn {it}{gamma} {"1D6FE}
286 \usv_set:nnn {bb}{Pi} {"0213F}
287 \usv_set:nnn {bb}{pi} {"0213C}
288 \usv_set:nnn {bb}{Gamma} {"0213E}
289 \usv_set:nnn {bb}{gamma} {"0213D}
290 \usv_set:nnn {bb}{summation}{"02140}
Italic blackboard:
291 \usv_set:nnn {bbit}{D}{"2145}
292 \usv_set:nnn {bbit}{d}{"2146}
293 \usv_set:nnn {bbit}{e}{"2147}
294 \usv_set:nnn {bbit}{i}{"2148}
295 \usv_set:nnn {bbit}{j}{"2149}
26
Script exceptions:
296 \usv_set:nnn {scr}{B}{"212C}
297 \usv_set:nnn {scr}{E}{"2130}
298 \usv_set:nnn {scr}{F}{"2131}
299 \usv_set:nnn {scr}{H}{"210B}
300 \usv_set:nnn {scr}{I}{"2110}
301 \usv_set:nnn {scr}{L}{"2112}
302 \usv_set:nnn {scr}{M}{"2133}
303 \usv_set:nnn {scr}{R}{"211B}
304 \usv_set:nnn {scr}{e}{"212F}
305 \usv_set:nnn {scr}{g}{"210A}
306 \usv_set:nnn {scr}{o}{"2134}
307 \usv_set:nnn {cal}{B}{"212C}
308 \usv_set:nnn {cal}{E}{"2130}
309 \usv_set:nnn {cal}{F}{"2131}
310 \usv_set:nnn {cal}{H}{"210B}
311 \usv_set:nnn {cal}{I}{"2110}
312 \usv_set:nnn {cal}{L}{"2112}
313 \usv_set:nnn {cal}{M}{"2133}
314 \usv_set:nnn {cal}{R}{"211B}
Fractur exceptions:
315 \usv_set:nnn {frak}{C}{"212D}
316 \usv_set:nnn {frak}{H}{"210C}
317 \usv_set:nnn {frak}{I}{"2111}
318 \usv_set:nnn {frak}{R}{"211C}
319 \usv_set:nnn {frak}{Z}{"2128}
6.4 STIX fonts
Version 1.0.0 of the STIX fonts contains a number of alphabets in the private use
area of Unicode; i.e., it contains many math glyphs that have not (yet or if ever)
been accepted into the Unicode standard.
But we still want to be able to use them if possible.
320
321 *
Upright
322 \usv_set:nnn {stixsfup}{partial}{"E17C}
323 \usv_set:nnn {stixsfup}{Greek}{"E17D}
324 \usv_set:nnn {stixsfup}{greek}{"E196}
325 \usv_set:nnn {stixsfup}{varTheta}{"E18E}
326 \usv_set:nnn {stixsfup}{varepsilon}{"E1AF}
327 \usv_set:nnn {stixsfup}{vartheta}{"E1B0}
328 \usv_set:nnn {stixsfup}{varkappa}{0000} % ???
27
329 \usv_set:nnn {stixsfup}{varphi}{"E1B1}
330 \usv_set:nnn {stixsfup}{varrho}{"E1B2}
331 \usv_set:nnn {stixsfup}{varpi}{"E1B3}
332 \usv_set:nnn {stixupslash}{Greek}{"E2FC}
Italic
333 \usv_set:nnn {stixbbit}{A}{"E154}
334 \usv_set:nnn {stixbbit}{B}{"E155}
335 \usv_set:nnn {stixbbit}{E}{"E156}
336 \usv_set:nnn {stixbbit}{F}{"E157}
337 \usv_set:nnn {stixbbit}{G}{"E158}
338 \usv_set:nnn {stixbbit}{I}{"E159}
339 \usv_set:nnn {stixbbit}{J}{"E15A}
340 \usv_set:nnn {stixbbit}{K}{"E15B}
341 \usv_set:nnn {stixbbit}{L}{"E15C}
342 \usv_set:nnn {stixbbit}{M}{"E15D}
343 \usv_set:nnn {stixbbit}{O}{"E15E}
344 \usv_set:nnn {stixbbit}{S}{"E15F}
345 \usv_set:nnn {stixbbit}{T}{"E160}
346 \usv_set:nnn {stixbbit}{U}{"E161}
347 \usv_set:nnn {stixbbit}{V}{"E162}
348 \usv_set:nnn {stixbbit}{W}{"E163}
349 \usv_set:nnn {stixbbit}{X}{"E164}
350 \usv_set:nnn {stixbbit}{Y}{"E165}
351 \usv_set:nnn {stixbbit}{a}{"E166}
352 \usv_set:nnn {stixbbit}{b}{"E167}
353 \usv_set:nnn {stixbbit}{c}{"E168}
354 \usv_set:nnn {stixbbit}{f}{"E169}
355 \usv_set:nnn {stixbbit}{g}{"E16A}
356 \usv_set:nnn {stixbbit}{h}{"E16B}
357 \usv_set:nnn {stixbbit}{k}{"E16C}
358 \usv_set:nnn {stixbbit}{l}{"E16D}
359 \usv_set:nnn {stixbbit}{m}{"E16E}
360 \usv_set:nnn {stixbbit}{n}{"E16F}
361 \usv_set:nnn {stixbbit}{o}{"E170}
362 \usv_set:nnn {stixbbit}{p}{"E171}
363 \usv_set:nnn {stixbbit}{q}{"E172}
364 \usv_set:nnn {stixbbit}{r}{"E173}
365 \usv_set:nnn {stixbbit}{s}{"E174}
366 \usv_set:nnn {stixbbit}{t}{"E175}
367 \usv_set:nnn {stixbbit}{u}{"E176}
368 \usv_set:nnn {stixbbit}{v}{"E177}
369 \usv_set:nnn {stixbbit}{w}{"E178}
370 \usv_set:nnn {stixbbit}{x}{"E179}
371 \usv_set:nnn {stixbbit}{y}{"E17A}
372 \usv_set:nnn {stixbbit}{z}{"E17B}
28
373 \usv_set:nnn {stixsfit}{Numerals}{"E1B4}
374 \usv_set:nnn {stixsfit}{partial}{"E1BE}
375 \usv_set:nnn {stixsfit}{Greek}{"E1BF}
376 \usv_set:nnn {stixsfit}{greek}{"E1D8}
377 \usv_set:nnn {stixsfit}{varTheta}{"E1D0}
378 \usv_set:nnn {stixsfit}{varepsilon}{"E1F1}
379 \usv_set:nnn {stixsfit}{vartheta}{"E1F2}
380 \usv_set:nnn {stixsfit}{varkappa}{0000} % ???
381 \usv_set:nnn {stixsfit}{varphi}{"E1F3}
382 \usv_set:nnn {stixsfit}{varrho}{"E1F4}
383 \usv_set:nnn {stixsfit}{varpi}{"E1F5}
384 \usv_set:nnn {stixcal}{Latin}{"E22D}
385 \usv_set:nnn {stixcal}{num}{"E262}
386 \usv_set:nnn {scr}{num}{48}
387 \usv_set:nnn {it}{num}{48}
388 \usv_set:nnn {stixsfitslash}{Latin}{"E294}
389 \usv_set:nnn {stixsfitslash}{latin}{"E2C8}
390 \usv_set:nnn {stixsfitslash}{greek}{"E32C}
391 \usv_set:nnn {stixsfitslash}{varepsilon}{"E37A}
392 \usv_set:nnn {stixsfitslash}{vartheta}{"E35E}
393 \usv_set:nnn {stixsfitslash}{varkappa}{"E374}
394 \usv_set:nnn {stixsfitslash}{varphi}{"E360}
395 \usv_set:nnn {stixsfitslash}{varrho}{"E376}
396 \usv_set:nnn {stixsfitslash}{varpi}{"E362}
397 \usv_set:nnn {stixsfitslash}{digamma}{"E36A}
Bold
398 \usv_set:nnn {stixbfupslash}{Greek}{"E2FD}
399 \usv_set:nnn {stixbfupslash}{Digamma}{"E369}
400 \usv_set:nnn {stixbfbb}{A}{"E38A}
401 \usv_set:nnn {stixbfbb}{B}{"E38B}
402 \usv_set:nnn {stixbfbb}{E}{"E38D}
403 \usv_set:nnn {stixbfbb}{F}{"E38E}
404 \usv_set:nnn {stixbfbb}{G}{"E38F}
405 \usv_set:nnn {stixbfbb}{I}{"E390}
406 \usv_set:nnn {stixbfbb}{J}{"E391}
407 \usv_set:nnn {stixbfbb}{K}{"E392}
408 \usv_set:nnn {stixbfbb}{L}{"E393}
409 \usv_set:nnn {stixbfbb}{M}{"E394}
410 \usv_set:nnn {stixbfbb}{O}{"E395}
411 \usv_set:nnn {stixbfbb}{S}{"E396}
412 \usv_set:nnn {stixbfbb}{T}{"E397}
413 \usv_set:nnn {stixbfbb}{U}{"E398}
414 \usv_set:nnn {stixbfbb}{V}{"E399}
415 \usv_set:nnn {stixbfbb}{W}{"E39A}
29
416 \usv_set:nnn {stixbfbb}{X}{"E39B}
417 \usv_set:nnn {stixbfbb}{Y}{"E39C}
418 \usv_set:nnn {stixbfbb}{a}{"E39D}
419 \usv_set:nnn {stixbfbb}{b}{"E39E}
420 \usv_set:nnn {stixbfbb}{c}{"E39F}
421 \usv_set:nnn {stixbfbb}{f}{"E3A2}
422 \usv_set:nnn {stixbfbb}{g}{"E3A3}
423 \usv_set:nnn {stixbfbb}{h}{"E3A4}
424 \usv_set:nnn {stixbfbb}{k}{"E3A7}
425 \usv_set:nnn {stixbfbb}{l}{"E3A8}
426 \usv_set:nnn {stixbfbb}{m}{"E3A9}
427 \usv_set:nnn {stixbfbb}{n}{"E3AA}
428 \usv_set:nnn {stixbfbb}{o}{"E3AB}
429 \usv_set:nnn {stixbfbb}{p}{"E3AC}
430 \usv_set:nnn {stixbfbb}{q}{"E3AD}
431 \usv_set:nnn {stixbfbb}{r}{"E3AE}
432 \usv_set:nnn {stixbfbb}{s}{"E3AF}
433 \usv_set:nnn {stixbfbb}{t}{"E3B0}
434 \usv_set:nnn {stixbfbb}{u}{"E3B1}
435 \usv_set:nnn {stixbfbb}{v}{"E3B2}
436 \usv_set:nnn {stixbfbb}{w}{"E3B3}
437 \usv_set:nnn {stixbfbb}{x}{"E3B4}
438 \usv_set:nnn {stixbfbb}{y}{"E3B5}
439 \usv_set:nnn {stixbfbb}{z}{"E3B6}
440 \usv_set:nnn {stixbfsfup}{Numerals}{"E3B7}
Bold Italic
441 \usv_set:nnn {stixbfsfit}{Numerals}{"E1F6}
442 \usv_set:nnn {stixbfbbit}{A}{"E200}
443 \usv_set:nnn {stixbfbbit}{B}{"E201}
444 \usv_set:nnn {stixbfbbit}{E}{"E203}
445 \usv_set:nnn {stixbfbbit}{F}{"E204}
446 \usv_set:nnn {stixbfbbit}{G}{"E205}
447 \usv_set:nnn {stixbfbbit}{I}{"E206}
448 \usv_set:nnn {stixbfbbit}{J}{"E207}
449 \usv_set:nnn {stixbfbbit}{K}{"E208}
450 \usv_set:nnn {stixbfbbit}{L}{"E209}
451 \usv_set:nnn {stixbfbbit}{M}{"E20A}
452 \usv_set:nnn {stixbfbbit}{O}{"E20B}
453 \usv_set:nnn {stixbfbbit}{S}{"E20C}
454 \usv_set:nnn {stixbfbbit}{T}{"E20D}
455 \usv_set:nnn {stixbfbbit}{U}{"E20E}
456 \usv_set:nnn {stixbfbbit}{V}{"E20F}
457 \usv_set:nnn {stixbfbbit}{W}{"E210}
458 \usv_set:nnn {stixbfbbit}{X}{"E211}
30
459 \usv_set:nnn {stixbfbbit}{Y}{"E212}
460 \usv_set:nnn {stixbfbbit}{a}{"E213}
461 \usv_set:nnn {stixbfbbit}{b}{"E214}
462 \usv_set:nnn {stixbfbbit}{c}{"E215}
463 \usv_set:nnn {stixbfbbit}{e}{"E217}
464 \usv_set:nnn {stixbfbbit}{f}{"E218}
465 \usv_set:nnn {stixbfbbit}{g}{"E219}
466 \usv_set:nnn {stixbfbbit}{h}{"E21A}
467 \usv_set:nnn {stixbfbbit}{k}{"E21D}
468 \usv_set:nnn {stixbfbbit}{l}{"E21E}
469 \usv_set:nnn {stixbfbbit}{m}{"E21F}
470 \usv_set:nnn {stixbfbbit}{n}{"E220}
471 \usv_set:nnn {stixbfbbit}{o}{"E221}
472 \usv_set:nnn {stixbfbbit}{p}{"E222}
473 \usv_set:nnn {stixbfbbit}{q}{"E223}
474 \usv_set:nnn {stixbfbbit}{r}{"E224}
475 \usv_set:nnn {stixbfbbit}{s}{"E225}
476 \usv_set:nnn {stixbfbbit}{t}{"E226}
477 \usv_set:nnn {stixbfbbit}{u}{"E227}
478 \usv_set:nnn {stixbfbbit}{v}{"E228}
479 \usv_set:nnn {stixbfbbit}{w}{"E229}
480 \usv_set:nnn {stixbfbbit}{x}{"E22A}
481 \usv_set:nnn {stixbfbbit}{y}{"E22B}
482 \usv_set:nnn {stixbfbbit}{z}{"E22C}
483 \usv_set:nnn {stixbfcal}{Latin}{"E247}
484 \usv_set:nnn {stixbfitslash}{Latin}{"E295}
485 \usv_set:nnn {stixbfitslash}{latin}{"E2C9}
486 \usv_set:nnn {stixbfitslash}{greek}{"E32D}
487 \usv_set:nnn {stixsfitslash}{varepsilon}{"E37B}
488 \usv_set:nnn {stixsfitslash}{vartheta}{"E35F}
489 \usv_set:nnn {stixsfitslash}{varkappa}{"E375}
490 \usv_set:nnn {stixsfitslash}{varphi}{"E361}
491 \usv_set:nnn {stixsfitslash}{varrho}{"E377}
492 \usv_set:nnn {stixsfitslash}{varpi}{"E363}
493 \usv_set:nnn {stixsfitslash}{digamma}{"E36B}
494
495 *
6.5 Package options
\unimathsetup This macro can be used in lieu of or later to override options declared when the
package is loaded.
496 \DeclareDocumentCommand \unimathsetup {m} {
497 \clist_clear:N \l_um_unknown_keys_clist
31
498 \keys_set:nn {unicode-math} {#1}
499 }
math-style
500 \keys_define:nn {unicode-math} {
501 normal-style .choice_code:n =
502 {
503 \bool_set_false:N \g_um_literal_bool
504 \ifcase \l_keys_choice_int
505 \bool_set_false:N \g_um_upGreek_bool
506 \bool_set_false:N \g_um_upgreek_bool
507 \bool_set_false:N \g_um_upLatin_bool
508 \bool_set_false:N \g_um_uplatin_bool
509 \or
510 \bool_set_true:N \g_um_upGreek_bool
511 \bool_set_false:N \g_um_upgreek_bool
512 \bool_set_false:N \g_um_upLatin_bool
513 \bool_set_false:N \g_um_uplatin_bool
514 \or
515 \bool_set_true:N \g_um_upGreek_bool
516 \bool_set_true:N \g_um_upgreek_bool
517 \bool_set_true:N \g_um_upLatin_bool
518 \bool_set_false:N \g_um_uplatin_bool
519 \or
520 \bool_set_true:N \g_um_upGreek_bool
521 \bool_set_true:N \g_um_upgreek_bool
522 \bool_set_true:N \g_um_upLatin_bool
523 \bool_set_true:N \g_um_uplatin_bool
524 \or
525 \bool_set_true:N \g_um_literal_bool
526 \fi
527 } ,
528 normal-style .generate_choices:n = {ISO,TeX,french,upright,literal} ,
529 }
530 \keys_define:nn {unicode-math} {
531 math-style .choice_code:n =
532 {
533 \ifcase \l_keys_choice_int
534 \unimathsetup {
535 normal-style=ISO,
536 bold-style=ISO,
537 sans-style=italic,
538 nabla=upright,
539 partial=italic,
540 }
541 \or
32
542 \unimathsetup {
543 normal-style=TeX,
544 bold-style=TeX,
545 sans-style=upright,
546 nabla=upright,
547 partial=italic,
548 }
549 \or
550 \unimathsetup {
551 normal-style=french,
552 bold-style=upright,
553 sans-style=upright,
554 nabla=upright,
555 partial=upright,
556 }
557 \or
558 \unimathsetup {
559 normal-style=upright,
560 bold-style=upright,
561 sans-style=upright,
562 nabla=upright,
563 partial=upright,
564 }
565 \or
566 \unimathsetup {
567 normal-style=literal,
568 bold-style=literal,
569 sans-style=literal,
570 colon=literal,
571 nabla=literal,
572 partial=literal,
573 }
574 \fi
575 } ,
576 math-style .generate_choices:n = {ISO,TeX,french,upright,literal} ,
577 }
bold-style
578 \keys_define:nn {unicode-math} {
579 bold-style .choice_code:n = {
580 \bool_set_false:N \g_um_bfliteral_bool
581 \ifcase \l_keys_choice_int
582 \bool_set_false:N \g_um_bfupGreek_bool
583 \bool_set_false:N \g_um_bfupgreek_bool
584 \bool_set_false:N \g_um_bfupLatin_bool
585 \bool_set_false:N \g_um_bfuplatin_bool
33
586 \or
587 \bool_set_true:N \g_um_bfupGreek_bool
588 \bool_set_false:N \g_um_bfupgreek_bool
589 \bool_set_true:N \g_um_bfupLatin_bool
590 \bool_set_true:N \g_um_bfuplatin_bool
591 \or
592 \bool_set_true:N \g_um_bfupGreek_bool
593 \bool_set_true:N \g_um_bfupgreek_bool
594 \bool_set_true:N \g_um_bfupLatin_bool
595 \bool_set_true:N \g_um_bfuplatin_bool
596 \or
597 \bool_set_true:N \g_um_bfliteral_bool
598 \fi
599 } ,
600 bold-style .generate_choices:n = {ISO,TeX,upright,literal} ,
601 }
sans-style
602 \keys_define:nn {unicode-math} {
603 sans-style .choice_code:n = {
604 \ifcase \l_keys_choice_int
605 \bool_set_false:N \g_um_upsans_bool
606 \or
607 \bool_set_true:N \g_um_upsans_bool
608 \or
609 \bool_set_true:N \g_um_sfliteral_bool
610 \fi
611 } ,
612 sans-style .generate_choices:n = {italic,upright,literal} ,
613 }
Nabla and partial
614 \keys_define:nn {unicode-math} {
615 nabla .choice_code:n = {
616 \bool_set_false:N \g_um_literal_Nabla_bool
617 \ifcase \l_keys_choice_int
618 \bool_set_true:N \g_um_upNabla_bool
619 \or
620 \bool_set_false:N \g_um_upNabla_bool
621 \or
622 \bool_set_true:N \g_um_literal_Nabla_bool
623 \fi
624 } ,
625 nabla .generate_choices:n = {upright,italic,literal} ,
626 }
34
627 \keys_define:nn {unicode-math} {
628 partial .choice_code:n = {
629 \bool_set_false:N \g_um_literal_partial_bool
630 \ifcase \l_keys_choice_int
631 \bool_set_true:N \g_um_uppartial_bool
632 \or
633 \bool_set_false:N \g_um_uppartial_bool
634 \or
635 \bool_set_true:N \g_um_literal_partial_bool
636 \fi
637 } ,
638 partial .generate_choices:n = {upright,italic,literal} ,
639 }
Epsilon and phi shapes
640 \keys_define:nn {unicode-math} {
641 vargreek-shape .choice: ,
642 vargreek-shape / unicode .code:n = {
643 \bool_set_false:N \g_um_texgreek_bool
644 } ,
645 vargreek-shape / TeX .code:n = {
646 \bool_set_true:N \g_um_texgreek_bool
647 }
648 }
Colon style
649 \keys_define:nn {unicode-math} {
650 colon .choice: ,
651 colon / literal .code:n = {
652 \bool_set_true:N \g_um_literal_colon_bool
653 } ,
654 colon / TeX .code:n = {
655 \bool_set_false:N \g_um_literal_colon_bool
656 }
657 }
Slash delimiter style
658 \keys_define:nn {unicode-math} {
659 slash-delimiter .choice: ,
660 slash-delimiter / ascii .code:n = {
661 \tl_set:Nn \g_um_slash_delimiter_usv {"002F}
662 } ,
663 slash-delimiter / frac .code:n = {
664 \tl_set:Nn \g_um_slash_delimiter_usv {"2044}
665 } ,
35
666 slash-delimiter / div .code:n = {
667 \tl_set:Nn \g_um_slash_delimiter_usv {"2215}
668 }
669 }
Active fraction style
670 \keys_define:nn {unicode-math} {
671 active-frac .choice: ,
672 active-frac / small .code:n = {
673 \cs_if_exist:NTF \tfrac {
674 \bool_set_true:N \l_um_smallfrac_bool
675 }{
676 \um_warning:n {no-tfrac}
677 \bool_set_false:N \l_um_smallfrac_bool
678 }
679 \use:c{um_setup_active_frac:}
680 } ,
681 active-frac / normalsize .code:n = {
682 \bool_set_false:N \l_um_smallfrac_bool
683 \use:c{um_setup_active_frac:}
684 }
685 }
Debug/tracing
686 \keys_define:nn {unicode-math} {
687 trace .choice: ,
688 trace / debug .code:n = {
689 \msg_redirect_module:nnn { unicode-math } { trace } { warning }
690 } ,
691 trace / on .code:n = {
692 \msg_redirect_module:nnn { unicode-math } { trace } { trace }
693 } ,
694 trace / off .code:n = {
695 \msg_redirect_module:nnn { unicode-math } { trace } { none }
696 } ,
697 }
698 \clist_new:N \l_um_unknown_keys_clist
699 \keys_define:nn {unicode-math} {
700 unknown .code:n = {
701 \clist_put_right:No \l_um_unknown_keys_clist {
702 \l_keys_key_tl = {#1}
703 }
704 }
705 }
36
706 \unimathsetup {math-style=TeX}
707 \unimathsetup {slash-delimiter=ascii}
708 \unimathsetup {trace=off}
709 \cs_if_exist:NT \tfrac {
710 \unimathsetup {active-frac=small}
711 }
712 \ProcessKeysOptions {unicode-math}
6.6 Overcoming \@onlypreamble
The requirement of only setting up the maths fonts in the preamble is now re-
moved. The following list might be overly ambitious.
713 \tl_map_inline:nn {
714 \new@mathgroup\cdp@list\cdp@elt\DeclareMathSizes
715 \@DeclareMathSizes\newmathalphabet\newmathalphabet@@\newmathalphabet@@@
716 \DeclareMathVersion\define@mathalphabet\define@mathgroup\addtoversion
717 \version@list\version@elt\alpha@list\alpha@elt
718 \restore@mathversion\init@restore@version\dorestore@version\process@table
719 \new@mathversion\DeclareSymbolFont\group@list\group@elt
720 \new@symbolfont\SetSymbolFont\SetSymbolFont@\get@cdp
721 \DeclareMathAlphabet\new@mathalphabet\SetMathAlphabet\SetMathAlphabet@
722 \DeclareMathAccent\set@mathaccent\DeclareMathSymbol\set@mathchar
723 \set@mathsymbol\DeclareMathDelimiter\@xxDeclareMathDelimiter
724 \@DeclareMathDelimiter\@xDeclareMathDelimiter\set@mathdelimiter
725 \set@@mathdelimiter\DeclareMathRadical\mathchar@type
726 \DeclareSymbolFontAlphabet\DeclareSymbolFontAlphabet@
727 }{
728 \tl_remove_in:Nn \@preamblecmds {\do#1}
729 }
7 Fundamentals
7.1 Enlarging the number of maths families
To start with, weve got a power of two as many \fams as before. So (from
ltfssbas.dtx) we want to redene
730 \def\new@mathgroup{\alloc@8\mathgroup\chardef\@cclvi}
731 \let\newfam\new@mathgroup
This is suicient for L
A
T
E
Xs \DeclareSymbolFont-type commands to be able
to dene 256 named maths fonts.
7.2 Setting math chars, math codes, etc.
\um_set_mathsymbol:nNNn #1 : A L
A
T
E
X symbol font, e.g., operators
37
#2 : Symbol macro, e.g., \alpha
#3 : Type, e.g., \mathalpha
#4 : Slot, e.g., "221E
There are a bunch of tests to perform to process the various characters. The fol-
lowing assignments should all be fairly straightforward.
732 \cs_set:Npn \um_set_mathsymbol:nNNn #1#2#3#4 {
733 \prg_case_tl:Nnn #3 {
734 \mathop {
735 \um_set_big_operator:nnn {#1} {#2} {#4}
736 }
737 \mathopen {
738 \tl_if_in:NnTF \l_um_radicals_tl {#2} {
739 \cs_gset:cpx {\cs_to_str:N #2 sign} { \um_radical:nn {#1} {#4} }
740 \tl_set:cn {l_um_radical_\cs_to_str:N #2_tl} {\use:c{sym #1}~ #4}
741 }{
742 \um_set_delcode:nnn {#1} {#4} {#4}
743 \um_set_mathcode:nnn {#4} \mathopen {#1}
744 \cs_gset:Npx #2 { \um_delimiter:Nnn \mathopen {#1} {#4} }
745 }
746 }
747 \mathclose {
748 \um_set_delcode:nnn {#1} {#4} {#4}
749 \um_set_mathcode:nnn {#4} \mathclose {#1}
750 \cs_gset:Npx #2 { \um_delimiter:Nnn \mathclose {#1} {#4} }
751 }
752 \mathaccent {
753 \cs_gset:Npx #2 { \um_accent:Nnn #3 {#1} {#4} }
754 }
755 \mathfence {
756 \um_set_mathcode:nnn {#4} {#3} {#1}
757 \um_set_delcode:nnn {#1} {#4} {#4}
758 \cs_gset:cpx {l \cs_to_str:N #2} { \um_delimiter:Nnn \math-
open {#1} {#4} }
759 \cs_gset:cpx {r \cs_to_str:N #2} { \um_delimiter:Nnn \math-
close {#1} {#4} }
760 }
761 \mathover { % LuaTeX only
762 \cs_set:Npn #2 ##1 { \mathop { \um_overbrace:nnn {#1} {#4} {##1} } \lim-
its }
763 }
764 \mathunder { % LuaTeX only
765 \cs_set:Npn #2 ##1 { \mathop { \um_underbrace:nnn {#1} {#4} {##1} } \lim-
its }
766 }
767 }{
768 \um_set_mathcode:nnn {#4} {#3} {#1}
38
769 }
770 }
\um_set_big_operator:nnn #1 : Symbol font name
#2 : Macro to assign
#3 : Glyph slot
In the examples following, say were dening for the symbol \sum(). In order for
literal Unicode characters to be used in the source and still have the correct limits
behaviour, big operators are made math-active. This involves three steps:
The active math char is dened to expand to the macro \sum_sym. (Later, the
control sequence \sum will be assigned the math char.)
Declare the plain old mathchardef for the control sequence \sumop. (This
follows the convention of L
A
T
E
X/amsmath.)
Dene \sum_sym as \sumop, followed by \nolimits if necessary.
Whether the \nolimits suix is inserted is controlled by the token list \l_um_-
nolimits_tl, which contains a list of such characters. This list is checked dynam-
ically to allow it to be updated mid-document.
Examples of expansion, by default, for two big operators:
( \sum ) \sum_sym \sumop\nolimits
( \int ) \int_sym \intop
771 \cs_new:Npn \um_set_big_operator:nnn #1#2#3 {
772 \group_begin:
773 \char_make_active:n {#3}
774 \char_gmake_mathactive:n {#3}
775 \um_active_char_set:wc #3 \q_nil { \cs_to_str:N #2 _sym }
776 \group_end:
777 \um_set_mathchar:cNnn {\cs_to_str:N #2 op} \mathop {#1} {#3}
778 \cs_gset:cpx { \cs_to_str:N #2 _sym } {
779 \exp_not:c { \cs_to_str:N #2 op }
780 \exp_not:n { \tl_if_in:NnT \l_um_nolimits_tl {#2} \nolimits }
781 }
782 }
\um_set_mathcode:nnnn
\um_set_mathcode:nnn
\um_set_mathchar:NNnn
\um_set_mathchar:cNnn
\um_set_delcode:nnn
\um_radical:nn
\um_delimiter:Nnn
\um_accent:Nnn
These are all wrappers for the primitive commands that take numerical input only.
783 \cs_set:Npn \um_set_mathcode:nnnn #1#2#3#4 {
784 \Umathcode \intexpr_eval:n {#1} =
785 \mathchar@type#2 \csname sym#3\endcsname \intexpr_eval:n {#4} \scan_stop:
786 }
787 \cs_set:Npn \um_set_mathcode:nnn #1#2#3 {
788 \Umathcode \intexpr_eval:n {#1} =
789 \mathchar@type#2 \csname sym#3\endcsname \intexpr_eval:n {#1} \scan_stop:
39
790 }
791 \cs_set:Npn \um_set_mathchar:NNnn #1#2#3#4 {
792 \Umathchardef #1 =
793 \mathchar@type#2 \csname sym#3\endcsname \intexpr_eval:n {#4} \scan_stop:
794 }
795 \cs_new:Npn \um_set_delcode:nnn #1#2#3 {
796 \Udelcode#2 = \csname sym#1\endcsname #3
797 }
798 \cs_new:Npn \um_radical:nn #1#2 {
799 \Uradical \csname sym#1\endcsname #2 \scan_stop:
800 }
801 \cs_new:Npn \um_delimiter:Nnn #1#2#3 {
802 \Udelimiter \mathchar@type#1 \csname sym#2\endcsname #3 \scan_stop:
803 }
804 \cs_new:Npn \um_accent:Nnn #1#2#3 {
805 \Umathaccent \mathchar@type#1 \csname sym#2\endcsname #3 \scan_stop:
806 }
807 \cs_generate_variant:Nn \um_set_mathchar:NNnn {c}
\um_overbrace:nnn
\um_underbrace:nnn
LuaT
E
X functions for dening over/under-braces
808 \cs_set:Npn \um_overbrace:nnn #1#2#3 {
809 \luatexUdelimiterover \csname sym#1\endcsname #2 {#3}
810 }
811 \cs_set:Npn \um_underbrace:nnn #1#2#3 {
812 \luatexUdelimiterunder \csname sym#1\endcsname #2 {#3}
813 }
\char_gmake_mathactive:N
\char_gmake_mathactive:n 814 \cs_new:Npn \char_gmake_mathactive:N #1 {
815 \global\mathcode `#1 = "8000 \scan_stop:
816 }
817 \cs_new:Npn \char_gmake_mathactive:n #1 {
818 \global\mathcode #1 = "8000 \scan_stop:
819 }
7.3 The main \setmathfont macro
Using a range including large character sets such as \mathrel, \mathalpha, etc.,
is very slow! I hope to improve the performance somehow.
\setmathfont [#1]: font features
#2 : font name
820 \cs_new:Npn \um_init: {
Erase any conception L
A
T
E
X has of previously dened math symbol fonts;
this allows \DeclareSymbolFont at any point in the document.
40
821 \let\glb@currsize\relax
To start with, assume were dening the font for every math symbol char-
acter.
822 \bool_set_true:N \l_um_init_bool
823 \seq_clear:N \l_um_char_range_seq
824 \clist_clear:N \l_um_char_num_range_clist
825 \seq_clear:N \l_um_mathalph_seq
826 \clist_clear:N \l_um_unknown_keys_clist
827 \seq_clear:N \l_um_missing_alph_seq
828 }
829 \DeclareDocumentCommand \setmathfont { O{} m } {
830 \um_init:
Grab the current size information (is this robust enough? Maybe it should
be preceded by \normalsize).
831 \csname S@\f@size\endcsname
Set the name of the math version being dened. (obviously more needs to
be done here!)
832 \tl_set:Nn \l_um_mversion_tf {normal}
833 \DeclareMathVersion{\l_um_mversion_tf}
Dene default font features for the script and scriptscript font.
834 \tl_set:Nn \l_um_script_features_tl {ScriptStyle}
835 \tl_set:Nn \l_um_sscript_features_tl {ScriptScriptStyle}
836 \tl_set:Nn \l_um_script_font_tl {#2}
837 \tl_set:Nn \l_um_sscript_font_tl {#2}
Use fontspec to select a font to use. The macro \S@size contains the deni-
tions of the sizes used for maths letters, subscripts and subsubscripts in \tf@size,
\sf@size, and \ssf@size, respectively.
838 \keys_set:nn {unicode-math} {#1}
839 \um_fontspec_select_font:n {#2}
Check whether were using a real maths font:
840 \group_begin:
841 \fontfamily{\zf@family}\selectfont
842 \fontspec_if_script:nTF {math}
843 {\bool_gset_true:N \l_um_ot_math_bool}
844 {\bool_gset_false:N \l_um_ot_math_bool}
845 \group_end:
If were dening the full Unicode math repetoire, then we skip all the parsing
processing needed if were only dening a subset.
41
Math symbols are dened with \_um_sym:nnn; see section 7.3.1 for the
individual denitions
846 \bool_if:NTF \l_um_init_bool {
847 \tl_set:Nn \um_symfont_tl {um_allsym}
848 \msg_trace:nnx {unicode-math} {default-math-font} {#2}
849 \cs_set_eq:NN \_um_sym:nnn \um_process_symbol_noparse:nnn
850 \cs_set_eq:NN \um_set_mathalphabet_char:Nnn \um_mathmap_noparse:Nnn
851 \cs_set_eq:NN \um_remap_symbol:nnn \um_remap_symbol_noparse:nnn
852 \cs_set_eq:NN \um_maybe_init_alphabet:n \um_init_alphabet:n
853 \cs_set_eq:NN \um_map_char_single:nn \um_map_char_noparse:nn
854 \cs_set_eq:NN \um_assign_delcode:nn \um_assign_delcode_noparse:nn
855 }{
856 \int_incr:N \g_um_fam_int
857 \tl_set:Nx \um_symfont_tl {um_fam\int_use:N\g_um_fam_int}
858 \cs_set_eq:NN \_um_sym:nnn \um_process_symbol_parse:nnn
859 \cs_set_eq:NN \um_set_mathalphabet_char:Nnn \um_mathmap_parse:Nnn
860 \cs_set_eq:NN \um_remap_symbol:nnn \um_remap_symbol_parse:nnn
861 \cs_set_eq:NN \um_maybe_init_alphabet:n \use_none:n
862 \cs_set_eq:NN \um_map_char_single:nn \um_map_char_parse:nn
863 \cs_set_eq:NN \um_assign_delcode:nn \um_assign_delcode_parse:nn
864 }
Now dened \um_symfont_tl as the L
A
T
E
X math font to access everything:
865 \DeclareSymbolFont{\um_symfont_tl}
866 {\encodingdefault}{\zf@family}{\mddefault}{\updefault}
867 \bool_if:nT {\l_um_ot_math_bool && !\g_um_mainfont_set_bool} {
868 \bool_set_true:N \g_um_mainfont_set_bool
869 \DeclareSymbolFont{symbols}
870 {\encodingdefault}{\zf@family}{\mddefault}{\updefault}
871 \DeclareSymbolFont{largesymbols}
872 {\encodingdefault}{\zf@family}{\mddefault}{\updefault}
873 }
And now we input every single maths char.
874 \um_input_math_symbol_table:
Finally,
Remap symbols that dont take their natural mathcode
Activate any symbols that need to be math-active
Assign delimiter codes for symbols that need to grow
Setup the maths alphabets (\mathbf etc.)
875 \um_remap_symbols:
876 \um_setup_mathactives:
877 \um_setup_delcodes:
878 \um_setup_alphabets:
42
Prevent spaces:
879 \ignorespaces
880 }
\resetmathfont
881 \DeclareDocumentCommand \resetmathfont { O{} m } {
882 \bool_set_false:N \g_um_mainfont_set_bool
883 \setmathfont[#1]{#2}
884 }
\um_fontspec_select_font: Select the font with \fontspec and dene \l_um_font from it.
885 \cs_new:Npn \um_fontspec_select_font:n #1 {
886 \bool_set_true:N \l_um_fontspec_feature_bool
887 \fontspec_select:xn
888 {
889 \luatex_if_engine:T { Renderer = Basic, }
890 BoldFont = {}, ItalicFont = {},
891 Script = Math,
892 SizeFeatures = {
893 {Size = \tf@size-} ,
894 {Size = \sf@size-\tf@size ,
895 Font = \l_um_script_font_tl ,
896 \l_um_script_features_tl
897 } ,
898 {Size = -\sf@size ,
899 Font = \l_um_sscript_font_tl ,
900 \l_um_sscript_features_tl
901 }
902 },
903 \l_um_unknown_keys_clist
904 }
905 {#1}
906 \tl_set_eq:NN \l_um_font \zf@basefont
907 \bool_set_false:N \l_um_fontspec_feature_bool
908 }
7.3.1 Functions for setting up symbols with mathcodes
\um_process_symbol_noparse:nnn
\um_process_symbol_parse:nnn
If the range font feature has been used, then only a subset of the Unicode glyphs
are to be dened. See section 8.3 for the code that enables this.
909 \cs_set:Npn \um_process_symbol_noparse:nnn #1#2#3 {
910 \um_set_mathsymbol:nNNn {\um_symfont_tl} #2#3{#1}
911 }
912 \cs_set:Npn \um_process_symbol_parse:nnn #1#2#3 {
913 \um@parse@term{#1}{#2}{#3}{
43
914 \um_process_symbol_noparse:nnn {#1}{#2}{#3}
915 }
916 }
\um_remap_symbols:
\um_remap_symbol_noparse:nnn
\um_remap_symbol_parse:nnn
This function is used to dene the mathcodes for those chars which should be
mapped to a dierent glyph than themselves.
917 \cs_new:Npn \um_remap_symbols: {
918 \um_remap_symbol:nnn{`\-}{\mathbin}{"02212}% hyphen to minus
919 \um_remap_symbol:nnn{`\*}{\mathbin}{"02217}% text asterisk to "cen-
tred asterisk"
920 \bool_if:NF \g_um_literal_colon_bool {
921 \um_remap_symbol:nnn{`\:}{\mathrel}{"02236}% colon to ratio (i.e., punct to rel)
922 }
923 }
Where \um_remap_symbol:nnn is dened to be one of these two, depending on
the range setup:
924 \cs_new:Npn \um_remap_symbol_parse:nnn #1#2#3 {
925 \um@parse@term {#3} {\@nil} {#2} {
926 \um_remap_symbol_noparse:nnn {#1} {#2} {#3}
927 }
928 }
929 \cs_new:Npn \um_remap_symbol_noparse:nnn #1#2#3 {
930 \clist_map_inline:nn {#1} {
931 \um_set_mathcode:nnnn {##1} {#2} {\um_symfont_tl} {#3}
932 }
933 }
7.3.2 Active math characters
There are more math active chars later in the subscript/superscript section. But
they dont need to be able to be typeset directly.
\um_setup_mathactives:
934 \cs_new:Npn \um_setup_mathactives: {
935 \um_make_mathactive:nNN {"2032} \um_prime_single_mchar \mathord
936 \um_make_mathactive:nNN {"2033} \um_prime_double_mchar \mathord
937 \um_make_mathactive:nNN {"2034} \um_prime_triple_mchar \mathord
938 \um_make_mathactive:nNN {"2057} \um_prime_quad_mchar \mathord
939 \um_make_mathactive:nNN {"2035} \um_backprime_single_mchar \mathord
940 \um_make_mathactive:nNN {"2036} \um_backprime_double_mchar \mathord
941 \um_make_mathactive:nNN {"2037} \um_backprime_triple_mchar \mathord
942 \um_make_mathactive:nNN {`\'} \mathstraightquote \mathord
943 \um_make_mathactive:nNN {`\`} \mathbacktick \mathord
944 }
44
\um_make_mathactive:nNN : TODO : hook into range feature Makes #1 a mathactive char, and gives cs #2 the
meaning of mathchar #1 with class #3. You are responsible for giving active #1 a
particular meaning!
945 \cs_new:Npn \um_make_mathactive:nNN #1#2#3 {
946 \um_set_mathchar:NNnn #2 #3 {\um_symfont_tl} {#1}
947 \char_gmake_mathactive:n {#1}
948 }
7.3.3 Delimiter codes
\um_assign_delcode:nn : TODO : hook csnames into range feature
949 \cs_new:Npn \um_assign_delcode_noparse:nn #1#2 {
950 \um_set_delcode:nnn \um_symfont_tl {#1} {#2}
951 }
952 \cs_new:Npn \um_assign_delcode_parse:nn #1#2 {
953 \um@parse@term {#2}{\@nil}{\@nil} {
954 \um_assign_delcode_noparse:nn {#1} {#2}
955 }
956 }
\um_assign_delcode:n Shorthand.
957 \cs_new:Npn \um_assign_delcode:n #1 {
958 \um_assign_delcode:nn {#1} {#1}
959 }
Some symbols that arent mathopen/mathclose still need to have delimiter
codes assigned. The list of vertical arrows may be incomplete. On the other hand,
many fonts wont support themall being stretchy. And some of themare probably
not meant to stretch, either. But adding them here doesnt hurt.
\um_setup_delcodes:
960 \cs_new:Npn \um_setup_delcodes: {
961 \um_assign_delcode:nn {`\/} {\g_um_slash_delimiter_usv}
962 \um_assign_delcode:nn {"2044} {\g_um_slash_delimiter_usv} % fracslash
963 \um_assign_delcode:nn {"2215} {\g_um_slash_delimiter_usv} % divslash
964 \um_assign_delcode:n {"005C} % backslash
965 \um_assign_delcode:nn {`\<} {"27E8} % angle brackets with ascii notation
966 \um_assign_delcode:nn {`\>} {"27E9} % angle brackets with ascii notation
967 \um_assign_delcode:n {"2191} % up arrow
968 \um_assign_delcode:n {"2193} % down arrow
969 \um_assign_delcode:n {"2195} % updown arrow
970 \um_assign_delcode:n {"219F} % up arrow twohead
971 \um_assign_delcode:n {"21A1} % down arrow twohead
972 \um_assign_delcode:n {"21A5} % up arrow from bar
973 \um_assign_delcode:n {"21A7} % down arrow from bar
974 \um_assign_delcode:n {"21A8} % updown arrow from bar
45
975 \um_assign_delcode:n {"21BE} % up harpoon right
976 \um_assign_delcode:n {"21BF} % up harpoon left
977 \um_assign_delcode:n {"21C2} % down harpoon right
978 \um_assign_delcode:n {"21C3} % down harpoon left
979 \um_assign_delcode:n {"21C5} % arrows up down
980 \um_assign_delcode:n {"21F5} % arrows down up
981 \um_assign_delcode:n {"21C8} % arrows up up
982 \um_assign_delcode:n {"21CA} % arrows down down
983 \um_assign_delcode:n {"21D1} % double up arrow
984 \um_assign_delcode:n {"21D3} % double down arrow
985 \um_assign_delcode:n {"21D5} % double updown arrow
986 \um_assign_delcode:n {"21DE} % up arrow double stroke
987 \um_assign_delcode:n {"21DF} % down arrow double stroke
988 \um_assign_delcode:n {"21E1} % up arrow dashed
989 \um_assign_delcode:n {"21E3} % down arrow dashed
990 \um_assign_delcode:n {"21E7} % up white arrow
991 \um_assign_delcode:n {"21E9} % down white arrow
992 \um_assign_delcode:n {"21EA} % up white arrow from bar
993 \um_assign_delcode:n {"21F3} % updown white arrow
994 }
7.4 (Big) operators
Turns out that X
T
E
X is clever enough to deal with big operators for us automati-
cally with \Umathchardef. Amazing!
However, the limits arent set automatically; that is, we want to dene, a la
Plain T
E
Xetc., \def\int{\intop\nolimits}, so there needs to be a transformation
from \int to \intop during the expansion of \_um_sym:nnn in the appropriate
contexts.
\l_um_nolimits_tl This macro is a sequence containing those maths operators that require a \nolim-
its suix. This list is used when processing unicode-math-table.tex to dene
such commands automatically (see the macro \um_set_mathsymbol:nNNn). Ive
chosen essentially just the operators that look like integrals; hopefully a better
mathematician can help me out here. Ive a feeling that its more useful not to
include the multiple integrals such as H, but that might be a matter of preference.
995 \tl_new:Nn \l_um_nolimits_tl {
996 \int\iint\iiint\iiiint\oint\oiint\oiiint
997 \intclockwise\varointclockwise\ointctrclockwise\sumint
998 \intbar\intBar\fint\cirfnint\awint\rppolint
999 \scpolint\npolint\pointint\sqint\intlarhk\intx
1000 \intcap\intcup\upint\lowint
1001 }
\addnolimits This macro appends material to the macro containing the list of operators that
dont take limits.
46
1002 \DeclareDocumentCommand \addnolimits {m} {
1003 \tl_put_right:Nn \l_um_nolimits_tl {#1}
1004 }
\removenolimits Can this macro be given a better name? It removes an item from the nolimits list.
1005 \DeclareDocumentCommand \removenolimits {m} {
1006 \tl_remove_all_in:Nn \l_um_nolimits_tl {#1}
1007 }
7.5 Radicals
The radical for square root is organised in \um_set_mathsymbol:nNNn. I think its
the only radical ever. (Actually, there is also \cuberoot and \fourthroot, but they
dont seem to behave as proper radicals.)
Also, what about right-to-left square roots?
\um@radicals We organise radicals in the same way as nolimits-operators; that is, in a comma-
list.
1008 \tl_new:Nn \l_um_radicals_tl {\sqrt}
7.6 Delimiters
\left We redene the primitive to be preceded by \mathopen; this gives much better
spacing in cases such as \sin\left. Courtesy of Frank Mittelbach:
http://www.latex-project.org/cgi-bin/ltxbugs2html?pr=latex/3853&prlatex/
3754
1009 \let\left@primitive\left
1010 \def\left{\mathopen{}\left@primitive}
No re-denition is made for \right because its not necessary.
7.7 Maths accents
Maths accents should just work if they are available in the font.
8 Font features
\um@zf@feature Use the same method as fontspec for feature denition (i.e., using xkeyval) but
with a conditional to restrict the scope of these features to unicode-math com-
mands.
1011 \newcommand\um@zf@feature[2]{
1012 \define@key[zf]{options}{#1}[]{
1013 \bool_if:NTF \l_um_fontspec_feature_bool {
47
1014 #2
1015 }{
1016 \um_warning:n {maths-feature-only}
1017 }
1018 }
1019 }
8.1 OpenType maths font features
1020 \xetex_or_luatex:nnn { \um@zf@feature {ScriptStyle} }
1021 { \fontspec_update_featstr:n {+ssty=0} }
1022 { \fontspec_update_featstr:n {+ssty=1} }
1023 \xetex_or_luatex:nnn { \um@zf@feature {ScriptScriptStyle} }
1024 { \fontspec_update_featstr:n {+ssty=1} }
1025 { \fontspec_update_featstr:n {+ssty=2} }
8.2 Script and scriptscript font options
1026 \keys_define:nn {unicode-math}
1027 {
1028 script-features .tl_set:N = \l_um_script_features_tl ,
1029 sscript-features .tl_set:N = \l_um_sscript_features_tl ,
1030 script-font .tl_set:N = \l_um_script_font_tl ,
1031 sscript-font .tl_set:N = \l_um_sscript_font_tl ,
1032 }
8.3 Range processing
1033 \seq_new:N \l_um_mathalph_seq
1034 \seq_new:N \l_um_char_range_seq
1035 \keys_define:nn {unicode-math} {
1036 range .code:n = {
1037 \bool_set_false:N \l_um_init_bool
1038 \seq_clear:N \l_um_char_range_seq
1039 \seq_clear:N \l_um_mathalph_seq
1040 \clist_map_inline:nn {#1} {
1041 \um_if_mathalph_decl:nTF {##1} {
1042 \seq_put_right:Nx \l_um_mathalph_seq {
1043 { \exp_not:V \l_um_tmpa_tl }
1044 { \exp_not:V \l_um_tmpb_tl }
1045 { \exp_not:V \l_um_tmpc_tl }
1046 }
1047 }{
1048 \seq_put_right:Nn \l_um_char_range_seq {##1}
1049 }
1050 }
1051 }
1052 }
48
\um_if_mathalph_decl:nTF Possible forms of input:
\mathscr
\mathscr->\mathup
\mathscr/{Latin}
\mathscr/{Latin}->\mathup
Outputs:
tmpa: math style (e.g., \mathscr)
tmpb: alphabets (e.g., Latin)
tmpc: remap style (e.g., \mathup). Defaults to tmpa.
The remap style can also be \mathcal->stixcal, which I marginally prefer
in the general case.
1053 \prg_new_conditional:Nnn \um_if_mathalph_decl:n {TF} {
1054 \KV_remove_surrounding_spaces:nw {\tl_set:Nf\l_um_tmpa_tl} #1 \q_nil
1055 \tl_clear:N \l_um_tmpb_tl
1056 \tl_clear:N \l_um_tmpc_tl
1057 \tl_if_in:NnT \l_um_tmpa_tl {->} {
1058 \exp_after:wN \um_split_arrow:w \l_um_tmpa_tl \q_nil
1059 }
1060 \tl_if_in:NnT \l_um_tmpa_tl {/} {
1061 \exp_after:wN \um_split_slash:w \l_um_tmpa_tl \q_nil
1062 }
1063 \tl_if_empty:NT \l_um_tmpc_tl { \tl_set_eq:NN \l_um_tmpc_tl \l_um_tmpa_tl }
1064 \seq_if_in:NVTF \g_um_mathstyles_seq \l_um_tmpa_tl {
1065 \prg_return_true:
1066 }{
1067 \prg_return_false:
1068 }
1069 }
1070 \cs_set:Npn \um_split_arrow:w #1->#2 \q_nil {
1071 \tl_set:Nn \l_um_tmpa_tl {#1}
1072 \tl_if_single:nTF {#2}
1073 { \tl_set:Nn \l_um_tmpc_tl {#2} }
1074 { \exp_args:NNc \tl_set:Nn \l_um_tmpc_tl {math#2} }
1075 }
1076 \cs_set:Npn \um_split_slash:w #1/#2 \q_nil {
1077 \tl_set:Nn \l_um_tmpa_tl {#1}
1078 \tl_set:Nn \l_um_tmpb_tl {#2}
1079 }
Pretty basic comma separated range processing. Donald Arseneaus selectp
package has a cleverer technique.
\um@parse@term #1 : Unicode character slot
#2 : control sequence (character macro)
#3 : control sequence (math type)
#4 : code to execute
49
This macro expands to #4 if any of its arguments are contained in \l_um_char_-
range_seq. This list can contain either character ranges (for checking with #1) or
control sequences. These latter can either be the command name of a specic char-
acter, or the math type of one (e.g., \mathbin).
Character ranges are passed to \um@parse@range, which accepts input in the
form shown in table 11.
Table 11: Ranges accepted by \um@parse@range.
Input Range
x =
x-
-y
x-y
Start by iterating over the commalist, ignoring empties, and initialising the
scratch conditional:
1080 \newcommand\um@parse@term[4]{
1081 \seq_map_variable:NNn \l_um_char_range_seq \@ii {
1082 \unless\ifx\@ii\@empty
1083 \@tempswafalse
Match to either the character macro (\alpha) or the math type (\mathbin):
1084 \expandafter\um@firstchar\expandafter{\@ii}
1085 \ifx\@tempa\um@backslash
1086 \expandafter\ifx\@ii#2\relax
1087 \@tempswatrue
1088 \else
1089 \expandafter\ifx\@ii#3\relax
1090 \@tempswatrue
1091 \fi
1092 \fi
Otherwise, we have a number range, which is passed to another macro:
1093 \else
1094 \expandafter\um@parse@range\@ii-\@marker-\@nil#1\@nil
1095 \fi
If we have a match, execute the code! It also populates the \l_um_char_num_-
range_clist macro, which is used when dening \mathbf (etc.) \mathchar remap-
pings.
1096 \if@tempswa
1097 \clist_put_right:Nx \l_um_char_num_range_clist { \int-
expr_eval:n {#1} }
1098 #4
1099 \fi
50
1100 \fi
1101 }
1102 }
1103 \def\um@firstof#1#2\@nil{#1}
1104 \edef\um@backslash{\expandafter\um@firstof\string\string\@nil}
1105 \def\um@firstchar#1{\edef\@tempa{\expandafter\um@firstof\string#1\@nil}}
\um@parse@range Weird syntax. As shown previously in table 11, this macro can be passed four
dierent input types via \um@parse@term.
1106 \def\um@parse@range#1-#2-#3\@nil#4\@nil{
1107 \def\@tempa{#1}
1108 \def\@tempb{#2}
Range =
C-list input \@ii=X
Macro input \um@parse@range X-\@marker-\@nil#1\@nil
Arguments #1-#2-#3 = X-\@marker-{}
1109 \expandafter\ifx\expandafter\@marker\@tempb\relax
1110 \intexpr_compare:nT {#4=#1} \@tempswatrue
1111 \else
Range
C-list input \@ii=X-
Macro input \um@parse@range X--\@marker-\@nil#1\@nil
Arguments #1-#2-#3 = X-{}-\@marker-
1112 \ifx\@empty\@tempb
1113 \intexpr_compare:nT {#4>#1-1} \@tempswatrue
1114 \else
Range
C-list input \@ii=-Y
Macro input \um@parse@range -Y-\@marker-\@nil#1\@nil
Arguments #1-#2-#3 = {}-Y-\@marker-
1115 \ifx\@empty\@tempa
1116 \intexpr_compare:nT {#4<#2+1} \@tempswatrue
Range
C-list input \@ii=X-Y
Macro input \um@parse@range X-Y-\@marker-\@nil#1\@nil
Arguments #1-#2-#3 = X-Y-\@marker-
1117 \else
1118 \intexpr_compare:nT {#4>#1-1} {
1119 \intexpr_compare:nT {#4<#2+1} \@tempswatrue
1120 }
1121 \fi
1122 \fi
1123 \fi
1124 }
51
8.4 Resolving Greek symbol name control sequences
\um_resolve_greek: This macro denes \Alpha\omega as their corresponding Unicode (mathemati-
cal italic) character. Remember that the mapping to upright or italic happens with
the mathcode denitions, whereas these macros just stand for the literal Unicode
characters.
1125 \AtBeginDocument{\um_resolve_greek:}
1126 \cs_new:Npn \um_resolve_greek: {
1127 \clist_map_inline:nn {
1128 Alpha,Beta,Gamma,Delta,Epsilon,Zeta,Eta,Theta,Iota,Kappa,Lambda,
1129 alpha,beta,gamma,delta, zeta,eta,theta,iota,kappa,lambda,
1130 Mu,Nu,Xi,Omicron,Pi,Rho,Sigma,Tau,Upsilon,Phi,Chi,Psi,Omega,
1131 mu,nu,xi,omicron,pi,rho,sigma,tau,upsilon, chi,psi,omega,
1132 varTheta,
1133 varsigma,vartheta,varkappa,varrho,varpi
1134 }{
1135 \tl_set:cx {##1} { \exp_not:c { mit ##1 } }
1136 }
1137 \tl_set:Nn \epsilon {
1138 \bool_if:NTF \g_um_texgreek_bool \mitvarepsilon \mitepsilon
1139 }
1140 \tl_set:Nn \phi {
1141 \bool_if:NTF \g_um_texgreek_bool \mitvarphi \mitphi
1142 }
1143 \tl_set:Nn \varepsilon {
1144 \bool_if:NTF \g_um_texgreek_bool \mitepsilon \mitvarepsilon
1145 }
1146 \tl_set:Nn \varphi {
1147 \bool_if:NTF \g_um_texgreek_bool \mitphi \mitvarphi
1148 }
1149 }
9 Maths alphabets mapping denitions
Algorithm for setting alphabet fonts. By default, when range is empty, we are in
implicit mode. If range contains the name of the math alphabet, we are in explicit
mode and do things slightly dierently.
Implicit mode:
Try and set all of the alphabet shapes.
Check for the rst glyph of each alphabet to detect if the font supports each
alphabet shape.
For alphabets that do exist, overwrite whatevers already there.
52
For alphabets that are not supported, do nothing. (This includes leaving the
old alphabet denition in place.)
Explicit mode:
Only set the alphabets specied.
Check for the rst glyph of the alphabet to detect if the font contains the
alphabet shape in the Unicode math plane.
For Unicode math alphabets, overwrite whatevers already there.
Otherwise, use the ascii letters instead.
9.1 Initialising math styles
\um_new_mathstyle:N This function denes a new command like \mathfrak.
1150 \cs_new:Npn \um_new_mathstyle:N #1 {
1151 \um_prepare_mathstyle:f {\exp_after:wN \use_none:nnnnn \token_to_str:N #1}
1152 \seq_put_right:Nn \g_um_mathstyles_seq {#1}
1153 }
\g_um_default_mathalph_seq This sequence stores the alphabets in each math style.
1154 \seq_new:N \g_um_default_mathalph_seq
\g_um_mathstyles_seq This is every math style known to unicode-math.
1155 \seq_new:N \g_um_mathstyles_seq
1156 \AtEndOfPackage{
1157 \clist_map_inline:nn {
1158 {\mathup } {latin,Latin,greek,Greek,num,misc} {\mathup } ,
1159 {\mathit } {latin,Latin,greek,Greek,misc} {\mathit } ,
1160 {\mathbb } {latin,Latin,num,misc} {\mathbb } ,
1161 {\mathbbit } {misc} {\mathbbit } ,
1162 {\mathscr } {latin,Latin} {\mathscr } ,
1163 {\mathcal } {Latin} {\mathscr } ,
1164 {\mathbfcal } {Latin} {\mathbfscr } ,
1165 {\mathfrak } {latin,Latin} {\mathfrak } ,
1166 {\mathtt } {latin,Latin,num} {\mathtt } ,
1167 {\mathsfup } {latin,Latin,num} {\mathsfup } ,
1168 {\mathsfit } {latin,Latin} {\mathsfit } ,
1169 {\mathbfup } {latin,Latin,greek,Greek,num,misc} {\mathbfup } ,
1170 {\mathbfit } {latin,Latin,greek,Greek,misc} {\mathbfit } ,
1171 {\mathbfscr } {latin,Latin} {\mathbfscr } ,
1172 {\mathbffrak} {latin,Latin} {\mathbffrak} ,
1173 {\mathbfsfup} {latin,Latin,greek,Greek,num,misc} {\mathbfsfup} ,
1174 {\mathbfsfit} {latin,Latin,greek,Greek,misc} {\mathbfsfit}
53
1175 }{
1176 \seq_put_right:Nn \g_um_default_mathalph_seq {#1}
1177 \exp_after:wN \um_new_mathstyle:N \use_i:nnn #1
1178 }
These are false mathstyles that inherit other denitions:
1179 \um_new_mathstyle:N \mathsf
1180 \um_new_mathstyle:N \mathbf
1181 \um_new_mathstyle:N \mathbfsf
1182 }
9.2 Dening the math style macros
We call the dierent shapes that a math alphabet can be a math style. Note that
dierent alphabets can exist within the same math style. E.g., we call bold the
math style bf and within it there are upper and lower case Greek and Roman
alphabets and Arabic numerals.
\um_prepare_mathstyle:n #1 : math style name (e.g., it or bb)
Dene the high level math alphabet macros (\mathit, etc.) in terms of unicode-
math denitions. Use \bgroup/\egroup so sscripts scan the whole thing.
1183 \cs_new:Npn \um_prepare_mathstyle:n #1 {
1184 \um_init_alphabet:x {#1}
1185 \cs_set:cpn {_um_math#1_aux:n} ##1 {
1186 \use:c {um_switchto_math#1:} ##1 \egroup
1187 }
1188 \cs_set_protected:cpx {math#1} {
1189 \exp_not:n{
1190 \bgroup
1191 \mode_if_math:F {
1192 \egroup\expandafter
1193 \non@alpherr\expandafter{\csname math#1\endcsname\space}
1194 }
1195 }
1196 \exp_not:c {_um_math#1_aux:n}
1197 }
1198 }
1199 \cs_generate_variant:Nn \um_prepare_mathstyle:n {f}
\um_init_alphabet:n #1 : math alphabet name (e.g., it or bb)
This macro initialises the macros used to set up a math alphabet. First used with
the math alphabet macro is rst dened, but then used later when redening a
particular maths alphabet.
1200 \cs_set:Npn \um_init_alphabet:n #1 {
1201 \um_trace:nx {alph-initialise} {#1}
54
1202 \cs_set_eq:cN {um_switchto_math#1:} \prg_do_nothing:
1203 }
1204 \cs_generate_variant:Nn \um_init_alphabet:n {x}
Variants
1205 \cs_new:Npn \um_maybe_init_alphabet:V {
1206 \exp_args:NV \um_maybe_init_alphabet:n
1207 }
9.3 Dening the math alphabets per style
Variables:
1208 \seq_new:N \l_um_missing_alph_seq
\um_setup_alphabets: This function is called within \setmathfont to congure the mapping between
characters inside math styles.
1209 \cs_new:Npn \um_setup_alphabets: {
If range= has beenusedto congure styles, those choices will be in\l_um_mathalph_seq.
If not, set up the styles implicitly:
1210 \seq_if_empty:NTF \l_um_mathalph_seq {
1211 \um_trace:n {setup-implicit}
1212 \seq_set_eq:NN \l_um_mathalph_seq \g_um_default_mathalph_seq
1213 \bool_set_true:N \l_um_implicit_alph_bool
1214 \um_maybe_init_alphabet:n {sf}
1215 \um_maybe_init_alphabet:n {bf}
1216 \um_maybe_init_alphabet:n {bfsf}
1217 }
If range= has been used then were in explicit mode:
1218 {
1219 \um_trace:n {setup-explicit}
1220 \bool_set_false:N \l_um_implicit_alph_bool
1221 \cs_set_eq:NN \um_set_mathalphabet_char:Nnn \um_mathmap_noparse:Nnn
1222 \cs_set_eq:NN \um_map_char_single:nn \um_map_char_noparse:nn
1223 }
Now perform the mapping:
1224 \seq_map_inline:Nn \l_um_mathalph_seq {
1225 \tl_set:No \l_um_tmpa_tl { \use_i:nnn ##1 }
1226 \tl_set:No \l_um_tmpb_tl { \use_ii:nnn ##1 }
1227 \tl_set:No \l_um_remap_style_tl { \use_iii:nnn ##1 }
1228 \tl_set:Nx \l_um_remap_style_tl {
1229 \exp_after:wN \exp_after:wN \exp_after:wN \use_none:nnnnn
1230 \exp_after:wN \token_to_str:N \l_um_remap_style_tl
1231 }
1232 \tl_if_empty:NT \l_um_tmpb_tl {
1233 \cs_set_eq:NN \um_maybe_init_alphabet:n \um_init_alphabet:n
55
1234 \tl_set:Nn \l_um_tmpb_tl { latin,Latin,greek,Greek,num,misc }
1235 }
1236 \um_setup_math_alphabet:VVV
1237 \l_um_tmpa_tl \l_um_tmpb_tl \l_um_remap_style_tl
1238 }
1239 \um_warn_missing_alphabets:
1240 }
1241 \cs_new:Npn \um_warn_missing_alphabets: {
1242 \seq_if_empty:NF \l_um_missing_alph_seq {
1243 \typeout{
1244 Package~unicode-math~Warning:~
1245 missing~math~alphabets~in~font~ \fontname\l_um_font
1246 }
1247 \seq_map_inline:Nn \l_um_missing_alph_seq {
1248 \typeout{\space\space\space\space##1}
1249 }
1250 }
1251 }
\um_setup_math_alphabet:Nnn #1 : Math font style command (e.g., \mathbb)
#2 : Math alphabets, comma separated of {latin,Latin,greek,Greek,num}
#3 : Name of the output math style (usually same as input bb)
1252 \cs_new:Npn \um_setup_math_alphabet:Nnn #1#2#3 {
1253 \tl_set:Nx \l_um_style_tl {
1254 \exp_after:wN \use_none:nnnnn \token_to_str:N #1
1255 }
First check that at least one of the alphabets for the font shape is dened
1256 \clist_map_inline:nn {#2} {
1257 \cs_if_exist:cT {um_config_ \l_um_style_tl _##1:n} {
1258 \str_if_eq:nnTF {##1}{misc} {
1259 \um_maybe_init_alphabet:V \l_um_style_tl
1260 \clist_map_break:
1261 }{
1262 \um_glyph_if_exist:cT { \um_to_usv:nn {#3}{##1} }{
1263 \um_maybe_init_alphabet:V \l_um_style_tl
1264 \clist_map_break:
1265 }
1266 }
1267 }
1268 }
and then loop through them dening the individual ranges:
1269 \clist_map_inline:nn {#2} {
1270 \cs_if_exist:cT {um_config_ \l_um_style_tl _##1:n} {
1271 \str_if_eq:nnTF {##1}{misc} {
1272 \um_trace:nx {setup-alph} {math \l_um_style_tl~(##1)}
56
1273 \use:c {um_config_ \l_um_style_tl _##1:n} {#3}
1274 }{
1275 \um_glyph_if_exist:cTF { \um_to_usv:nn {#3}{##1} } {
1276 \um_trace:nx {setup-alph} {math \l_um_style_tl~(##1)}
1277 \use:c {um_config_ \l_um_style_tl _##1:n} {#3}
1278 }{
1279 \bool_if:NTF \l_um_implicit_alph_bool {
1280 \seq_put_right:Nx \l_um_missing_alph_seq {
1281 \@backslashchar math \l_um_style_tl \space
1282 (\tl_use:c{g_um_math_alphabet_name_##1_tl})
1283 }
1284 }{
1285 \use:c {um_config_ \l_um_style_tl _##1:n} {up}
1286 }
1287 }
1288 }
1289 }
1290 }
1291 }
1292 \cs_generate_variant:Nn \um_setup_math_alphabet:Nnn {VVV}
9.4 Mapping naked math characters
Before we show the denitions of the alphabet mappings using the functions
\um_config_\l_um_style_tl_##1:n, we rst want to dene some functions to
be used inside them to actually perform the character mapping.
9.4.1 Functions
\um_map_char_single:nn Wrapper for \um_map_char_noparse:nn or \um_map_char_parse:nn depending
on the context.
1293 \cs_new:Npn \um_map_char_single:cc { \exp_args:Ncc \um_map_char_single:nn }
\um_map_char_noparse:nn
\um_map_char_parse:nn 1294 \cs_new:Npn \um_map_char_noparse:nn #1#2 {
1295 \um_set_mathcode:nnnn {#1}{\mathalpha}{\um_symfont_tl}{#2}
1296 }
1297 \cs_new:Npn \um_map_char_parse:nn #1#2 {
1298 \um@parse@term {#1} {\@nil} {\mathalpha} {
1299 \um_map_char_noparse:nn {#1}{#2}
1300 }
1301 }
\um_map_single:nnn #1 : char name (dotlessi)
#2 : from alphabet(s)
57
#3 : to alphabet
1302 \cs_new:Npn \um_map_char_single:nnn #1#2#3 {
1303 \um_map_char_single:cc { \um_to_usv:nn {#1}{#3} }
1304 { \um_to_usv:nn {#2}{#3} }
1305 }
1306 \cs_set:Npn \um_map_single:nnn #1#2#3 {
1307 \cs_if_exist:cT { \um_to_usv:nn {#3} {#1} }
1308 {
1309 \clist_map_inline:nn {#2} {
1310 \um_map_char_single:nnn {##1} {#3} {#1}
1311 }
1312 }
1313 }
\um_map_chars_range:nnnn #1 : Number of chars (26)
#2 : From style, one or more (it)
#3 : To style (up)
#4 : Alphabet name (Latin)
First the function with numbers:
1314 \cs_set:Npn \um_map_chars_range:nnn #1#2#3 {
1315 \prg_stepwise_inline:nnnn {0}{1}{#1-1} {
1316 \um_map_char_single:nn {#2+##1}{#3+##1}
1317 }
1318 }
1319 \cs_generate_variant:Nn \um_map_chars_range:nnn {ncc}
And the wrapper with names:
1320 \cs_new:Npn \um_map_chars_range:nnnn #1#2#3#4 {
1321 \um_map_chars_range:ncc {#1} { \um_to_usv:nn {#2}{#4} }
1322 { \um_to_usv:nn {#3}{#4} }
1323 }
9.4.2 Functions for alphabets
1324 \cs_set:Npn \um_map_chars_Latin:nn #1#2 {
1325 \clist_map_inline:nn {#1} {
1326 \um_map_chars_range:nnnn {26} {##1} {#2} {Latin}
1327 }
1328 }
1329 \cs_set:Npn \um_map_chars_latin:nn #1#2 {
1330 \clist_map_inline:nn {#1} {
1331 \um_map_chars_range:nnnn {26} {##1} {#2} {latin}
1332 }
1333 }
1334 \cs_set:Npn \um_map_chars_greek:nn #1#2 {
1335 \clist_map_inline:nn {#1} {
58
1336 \um_map_chars_range:nnnn {25} {##1} {#2} {greek}
1337 \um_map_char_single:nnn {##1} {#2} {varepsilon}
1338 \um_map_char_single:nnn {##1} {#2} {vartheta}
1339 \um_map_char_single:nnn {##1} {#2} {varkappa}
1340 \um_map_char_single:nnn {##1} {#2} {varphi}
1341 \um_map_char_single:nnn {##1} {#2} {varrho}
1342 \um_map_char_single:nnn {##1} {#2} {varpi}
1343 }
1344 }
1345 \cs_set:Npn \um_map_chars_Greek:nn #1#2 {
1346 \clist_map_inline:nn {#1} {
1347 \um_map_chars_range:nnnn {25} {##1} {#2} {Greek}
1348 \um_map_char_single:nnn {##1} {#2} {varTheta}
1349 }
1350 }
1351 \cs_set:Npn \um_map_chars_numbers:nn #1#2 {
1352 \um_map_chars_range:nnnn {10} {#1} {#2} {num}
1353 }
9.5 Mapping chars inside a math style
9.5.1 Functions for setting up the maths alphabets
\um_set_mathalphabet_char:Nnn This is a wrapper for either \um_mathmap_noparse:Nnn or \um_mathmap_parse:Nnn,
depending on the context.
1354 \cs_new:Npn \um_set_mathalphabet_char:Ncc {
1355 \exp_args:NNcc \um_set_mathalphabet_char:Nnn
1356 }
\um_mathmap_noparse:Nnn #1 : Maths alphabet, e.g., \mathbb
#2 : Input slot(s), e.g., the slot for A (comma separated)
#3 : Output slot, e.g., the slot for
Adds \um_set_mathcode:nnnn declarations to the specied maths alphabets def-
inition.
1357 \cs_set:Npn \um_mathmap_noparse:Nnn #1#2#3 {
1358 \clist_map_inline:nn {#2} {
1359 \tl_put_right:cx {um_switchto_\cs_to_str:N #1:} {
1360 \um_set_mathcode:nnnn{##1}{\mathalpha}{\um_symfont_tl}{#3}
1361 }
1362 }
1363 }
\um_mathmap_parse:Nnn #1 : Maths alphabet, e.g., \mathbb
#2 : Input slot(s), e.g., the slot for A (comma separated)
#3 : Output slot, e.g., the slot for
59
When \um@parse@term is executed, it populates the \l_um_char_num_range_-
clist macro with slot numbers corresponding to the specied range. This range
is used to conditionally add \um_set_mathcode:nnnn declaractions to the maths
alphabet denition.
1364 \cs_set:Npn \um_mathmap_parse:Nnn #1#2#3 {
1365 \clist_if_in:NnT \l_um_char_num_range_clist {#3} {
1366 \um_mathmap_noparse:Nnn {#1}{#2}{#3}
1367 }
1368 }
\um_set_mathalphabet_char:Nnnn #1 : math style command
#2 : input math alphabet name
#3 : output math alphabet name
#4 : char name to map
1369 \cs_new:Npn \um_set_mathalphabet_char:Nnnn #1#2#3#4 {
1370 \um_set_mathalphabet_char:Ncc #1 { \um_to_usv:nn {#2} {#4} }
1371 { \um_to_usv:nn {#3} {#4} }
1372 }
\um_set_mathalph_range:nNnn #1 : Number of iterations
#2 : Maths alphabet
#3 : Starting input char (single)
#4 : Starting output char
Loops through character ranges setting \mathcode. First the version that uses
numbers:
1373 \cs_new:Npn \um_set_mathalph_range:nNnn #1#2#3#4 {
1374 \prg_stepwise_inline:nnnn {0}{1}{#1-1} {
1375 \um_set_mathalphabet_char:Nnn {#2} { ##1 + #3 } { ##1 + #4 }
1376 }
1377 }
1378 \cs_generate_variant:Nn \um_set_mathalph_range:nNnn {nNcc}
Then the wrapper version that uses names:
1379 \cs_new:Npn \um_set_mathalph_range:nNnnn #1#2#3#4#5 {
1380 \um_set_mathalph_range:nNcc {#1} #2 { \um_to_usv:nn {#3} {#5} }
1381 { \um_to_usv:nn {#4} {#5} }
1382 }
9.5.2 Individual mapping functions for dierent alphabets
1383 \cs_new:Npn \um_set_mathalphabet_pos:Nnnn #1#2#3#4 {
1384 \cs_if_exist:cT { \um_to_usv:nn {#4}{#2} } {
1385 \clist_map_inline:nn {#3} {
1386 \um_set_mathalphabet_char:Nnnn #1 {##1} {#4} {#2}
1387 }
1388 }
60
1389 }
1390 \cs_new:Npn \um_set_mathalphabet_numbers:Nnn #1#2#3 {
1391 \clist_map_inline:nn {#2} {
1392 \um_set_mathalph_range:nNnnn {10} #1 {##1} {#3} {num}
1393 }
1394 }
1395 \cs_new:Npn \um_set_mathalphabet_Latin:Nnn #1#2#3 {
1396 \clist_map_inline:nn {#2} {
1397 \um_set_mathalph_range:nNnnn {26} #1 {##1} {#3} {Latin}
1398 }
1399 }
1400 \cs_new:Npn \um_set_mathalphabet_latin:Nnn #1#2#3 {
1401 \clist_map_inline:nn {#2} {
1402 \um_set_mathalph_range:nNnnn {26} #1 {##1} {#3} {latin}
1403 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {h}
1404 }
1405 }
1406 \cs_new:Npn \um_set_mathalphabet_Greek:Nnn #1#2#3 {
1407 \clist_map_inline:nn {#2} {
1408 \um_set_mathalph_range:nNnnn {25} #1 {##1} {#3} {Greek}
1409 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varTheta}
1410 }
1411 }
1412 \cs_new:Npn \um_set_mathalphabet_greek:Nnn #1#2#3 {
1413 \clist_map_inline:nn {#2} {
1414 \um_set_mathalph_range:nNnnn {25} #1 {##1} {#3} {greek}
1415 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varepsilon}
1416 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {vartheta}
1417 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varkappa}
1418 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varphi}
1419 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varrho}
1420 \um_set_mathalphabet_char:Nnnn #1 {##1} {#3} {varpi}
1421 }
1422 }
9.6 Alphabets
9.6.1 Upright: \mathup
1423 \cs_new:Npn \um_config_up_num:n #1 {
1424 \um_map_chars_numbers:nn {up}{#1}
1425 \um_set_mathalphabet_numbers:Nnn \mathup {up}{#1}
1426 }
1427 \cs_new:Npn \um_config_up_Latin:n #1 {
1428 \bool_if:NTF \g_um_literal_bool {
1429 \um_map_chars_Latin:nn {up} {#1}
61
1430 }{
1431 \bool_if:NT \g_um_upLatin_bool {
1432 \um_map_chars_Latin:nn {up,it} {#1}
1433 }
1434 }
1435 \um_set_mathalphabet_Latin:Nnn \mathup {up,it}{#1}
1436 }
1437 \cs_new:Npn \um_config_up_latin:n #1 {
1438 \bool_if:NTF \g_um_literal_bool {
1439 \um_map_chars_latin:nn {up} {#1}
1440 }{
1441 \bool_if:NT \g_um_uplatin_bool {
1442 \um_map_chars_latin:nn {up,it} {#1}
1443 \um_map_single:nnn {h} {up,it} {#1}
1444 \um_map_single:nnn {dotlessi} {up,it} {#1}
1445 \um_map_single:nnn {dotlessj} {up,it} {#1}
1446 }
1447 }
1448 \um_set_mathalphabet_latin:Nnn \mathup {up,it}{#1}
1449 }
1450 \cs_new:Npn \um_config_up_Greek:n #1 {
1451 \bool_if:NTF \g_um_literal_bool {
1452 \um_map_chars_Greek:nn {up}{#1}
1453 }{
1454 \bool_if:NT \g_um_upGreek_bool {
1455 \um_map_chars_Greek:nn {up,it}{#1}
1456 }
1457 }
1458 \um_set_mathalphabet_Greek:Nnn \mathup {up,it}{#1}
1459 }
1460 \cs_new:Npn \um_config_up_greek:n #1 {
1461 \bool_if:NTF \g_um_literal_bool {
1462 \um_map_chars_greek:nn {up} {#1}
1463 }{
1464 \bool_if:NT \g_um_upgreek_bool {
1465 \um_map_chars_greek:nn {up,it} {#1}
1466 }
1467 }
1468 \um_set_mathalphabet_greek:Nnn \mathup {up,it} {#1}
1469 }
1470 \cs_new:Npn \um_config_up_misc:n #1 {
1471 \bool_if:NTF \g_um_literal_Nabla_bool {
1472 \um_map_single:nnn {Nabla}{up}{up}
1473 }{
1474 \bool_if:NT \g_um_upNabla_bool {
1475 \um_map_single:nnn {Nabla}{up,it}{up}
62
1476 }
1477 }
1478 \bool_if:NTF \g_um_literal_partial_bool {
1479 \um_map_single:nnn {partial}{up}{up}
1480 }{
1481 \bool_if:NT \g_um_uppartial_bool {
1482 \um_map_single:nnn {partial}{up,it}{up}
1483 }
1484 }
1485 \um_set_mathalphabet_pos:Nnnn \mathup {partial} {up,it} {#1}
1486 \um_set_mathalphabet_pos:Nnnn \mathup {Nabla} {up,it} {#1}
1487 \um_set_mathalphabet_pos:Nnnn \mathup {dotlessi} {up,it} {#1}
1488 \um_set_mathalphabet_pos:Nnnn \mathup {dotlessj} {up,it} {#1}
1489 }
9.6.2 Italic: \mathit
1490 \cs_new:Npn \um_config_it_Latin:n #1 {
1491 \bool_if:NTF \g_um_literal_bool {
1492 \um_map_chars_Latin:nn {it} {#1}
1493 }{
1494 \bool_if:NF \g_um_upLatin_bool {
1495 \um_map_chars_Latin:nn {up,it} {#1}
1496 }
1497 }
1498 \um_set_mathalphabet_Latin:Nnn \mathit {up,it}{#1}
1499 }
1500 \cs_new:Npn \um_config_it_latin:n #1 {
1501 \bool_if:NTF \g_um_literal_bool {
1502 \um_map_chars_latin:nn {it} {#1}
1503 \um_map_single:nnn {h}{it}{#1}
1504 }{
1505 \bool_if:NF \g_um_uplatin_bool {
1506 \um_map_chars_latin:nn {up,it} {#1}
1507 \um_map_single:nnn {h}{up,it}{#1}
1508 \um_map_single:nnn {dotlessi}{up,it}{#1}
1509 \um_map_single:nnn {dotlessj}{up,it}{#1}
1510 }
1511 }
1512 \um_set_mathalphabet_latin:Nnn \mathit {up,it} {#1}
1513 \um_set_mathalphabet_pos:Nnnn \mathit {dotlessi} {up,it} {#1}
1514 \um_set_mathalphabet_pos:Nnnn \mathit {dotlessj} {up,it} {#1}
1515 }
1516 \cs_new:Npn \um_config_it_Greek:n #1 {
1517 \bool_if:NTF \g_um_literal_bool {
1518 \um_map_chars_Greek:nn {it}{#1}
1519 }{
63
1520 \bool_if:NF \g_um_upGreek_bool {
1521 \um_map_chars_Greek:nn {up,it}{#1}
1522 }
1523 }
1524 \um_set_mathalphabet_Greek:Nnn \mathit {up,it}{#1}
1525 }
1526 \cs_new:Npn \um_config_it_greek:n #1 {
1527 \bool_if:NTF \g_um_literal_bool {
1528 \um_map_chars_greek:nn {it} {#1}
1529 }{
1530 \bool_if:NF \g_um_upgreek_bool {
1531 \um_map_chars_greek:nn {it,up} {#1}
1532 }
1533 }
1534 \um_set_mathalphabet_greek:Nnn \mathit {up,it} {#1}
1535 }
1536 \cs_new:Npn \um_config_it_misc:n #1 {
1537 \bool_if:NTF \g_um_literal_Nabla_bool {
1538 \um_map_single:nnn {Nabla}{it}{it}
1539 }{
1540 \bool_if:NF \g_um_upNabla_bool {
1541 \um_map_single:nnn {Nabla}{up,it}{it}
1542 }
1543 }
1544 \bool_if:NTF \g_um_literal_partial_bool {
1545 \um_map_single:nnn {partial}{it}{it}
1546 }{
1547 \bool_if:NF \g_um_uppartial_bool {
1548 \um_map_single:nnn {partial}{up,it}{it}
1549 }
1550 }
1551 \um_set_mathalphabet_pos:Nnnn \mathit {partial} {up,it}{#1}
1552 \um_set_mathalphabet_pos:Nnnn \mathit {Nabla} {up,it}{#1}
1553 }
9.6.3 Blackboard or double-struck: \mathbb and \mathbbit
1554 \cs_new:Npn \um_config_bb_latin:n #1 {
1555 \um_set_mathalphabet_latin:Nnn \mathbb {up,it}{#1}
1556 }
1557 \cs_new:Npn \um_config_bb_Latin:n #1 {
1558 \um_set_mathalphabet_Latin:Nnn \mathbb {up,it}{#1}
1559 \um_set_mathalphabet_pos:Nnnn \mathbb {C} {up,it} {#1}
1560 \um_set_mathalphabet_pos:Nnnn \mathbb {H} {up,it} {#1}
1561 \um_set_mathalphabet_pos:Nnnn \mathbb {N} {up,it} {#1}
1562 \um_set_mathalphabet_pos:Nnnn \mathbb {P} {up,it} {#1}
1563 \um_set_mathalphabet_pos:Nnnn \mathbb {Q} {up,it} {#1}
64
1564 \um_set_mathalphabet_pos:Nnnn \mathbb {R} {up,it} {#1}
1565 \um_set_mathalphabet_pos:Nnnn \mathbb {Z} {up,it} {#1}
1566 }
1567 \cs_new:Npn \um_config_bb_num:n #1 {
1568 \um_set_mathalphabet_numbers:Nnn \mathbb {up}{#1}
1569 }
1570 \cs_new:Npn \um_config_bb_misc:n #1 {
1571 \um_set_mathalphabet_pos:Nnnn \mathbb {Pi} {up,it} {#1}
1572 \um_set_mathalphabet_pos:Nnnn \mathbb {pi} {up,it} {#1}
1573 \um_set_mathalphabet_pos:Nnnn \mathbb {Gamma} {up,it} {#1}
1574 \um_set_mathalphabet_pos:Nnnn \mathbb {gamma} {up,it} {#1}
1575 \um_set_mathalphabet_pos:Nnnn \mathbb {summation} {up} {#1}
1576 }
1577 \cs_new:Npn \um_config_bbit_misc:n #1 {
1578 \um_set_mathalphabet_pos:Nnnn \mathbbit {D} {up,it} {#1}
1579 \um_set_mathalphabet_pos:Nnnn \mathbbit {d} {up,it} {#1}
1580 \um_set_mathalphabet_pos:Nnnn \mathbbit {e} {up,it} {#1}
1581 \um_set_mathalphabet_pos:Nnnn \mathbbit {i} {up,it} {#1}
1582 \um_set_mathalphabet_pos:Nnnn \mathbbit {j} {up,it} {#1}
1583 }
9.6.4 Script and caligraphic: \mathscr and \mathcal
1584 \cs_new:Npn \um_config_scr_Latin:n #1 {
1585 \um_set_mathalphabet_Latin:Nnn \mathscr {up,it}{#1}
1586 \um_set_mathalphabet_pos:Nnnn \mathscr {B}{up,it}{#1}
1587 \um_set_mathalphabet_pos:Nnnn \mathscr {E}{up,it}{#1}
1588 \um_set_mathalphabet_pos:Nnnn \mathscr {F}{up,it}{#1}
1589 \um_set_mathalphabet_pos:Nnnn \mathscr {H}{up,it}{#1}
1590 \um_set_mathalphabet_pos:Nnnn \mathscr {I}{up,it}{#1}
1591 \um_set_mathalphabet_pos:Nnnn \mathscr {L}{up,it}{#1}
1592 \um_set_mathalphabet_pos:Nnnn \mathscr {M}{up,it}{#1}
1593 \um_set_mathalphabet_pos:Nnnn \mathscr {R}{up,it}{#1}
1594 }
1595 \cs_new:Npn \um_config_scr_latin:n #1 {
1596 \um_set_mathalphabet_latin:Nnn \mathscr {up,it}{#1}
1597 \um_set_mathalphabet_pos:Nnnn \mathscr {e}{up,it}{#1}
1598 \um_set_mathalphabet_pos:Nnnn \mathscr {g}{up,it}{#1}
1599 \um_set_mathalphabet_pos:Nnnn \mathscr {o}{up,it}{#1}
1600 }
These are by default synonyms for the above, but with the STIX fonts we want to
use the alternate alphabet.
1601 \cs_new:Npn \um_config_cal_Latin:n #1 {
1602 \um_set_mathalphabet_Latin:Nnn \mathcal {up,it}{#1}
1603 \um_set_mathalphabet_pos:Nnnn \mathcal {B}{up,it}{#1}
1604 \um_set_mathalphabet_pos:Nnnn \mathcal {E}{up,it}{#1}
1605 \um_set_mathalphabet_pos:Nnnn \mathcal {F}{up,it}{#1}
65
1606 \um_set_mathalphabet_pos:Nnnn \mathcal {H}{up,it}{#1}
1607 \um_set_mathalphabet_pos:Nnnn \mathcal {I}{up,it}{#1}
1608 \um_set_mathalphabet_pos:Nnnn \mathcal {L}{up,it}{#1}
1609 \um_set_mathalphabet_pos:Nnnn \mathcal {M}{up,it}{#1}
1610 \um_set_mathalphabet_pos:Nnnn \mathcal {R}{up,it}{#1}
1611 }
9.6.5 Fractur or fraktur or blackletter: \mathfrak
1612 \cs_new:Npn \um_config_frak_Latin:n #1 {
1613 \um_set_mathalphabet_Latin:Nnn \mathfrak {up,it}{#1}
1614 \um_set_mathalphabet_pos:Nnnn \mathfrak {C}{up,it}{#1}
1615 \um_set_mathalphabet_pos:Nnnn \mathfrak {H}{up,it}{#1}
1616 \um_set_mathalphabet_pos:Nnnn \mathfrak {I}{up,it}{#1}
1617 \um_set_mathalphabet_pos:Nnnn \mathfrak {R}{up,it}{#1}
1618 \um_set_mathalphabet_pos:Nnnn \mathfrak {Z}{up,it}{#1}
1619 }
1620 \cs_new:Npn \um_config_frak_latin:n #1 {
1621 \um_set_mathalphabet_latin:Nnn \mathfrak {up,it}{#1}
1622 }
9.6.6 Sans serif upright: \mathsfup
1623 \cs_new:Npn \um_config_sfup_num:n #1 {
1624 \um_set_mathalphabet_numbers:Nnn \mathsf {up}{#1}
1625 \um_set_mathalphabet_numbers:Nnn \mathsfup {up}{#1}
1626 }
1627 \cs_new:Npn \um_config_sfup_Latin:n #1 {
1628 \bool_if:NTF \g_um_sfliteral_bool {
1629 \um_map_chars_Latin:nn {sfup} {#1}
1630 \um_set_mathalphabet_Latin:Nnn \mathsf {up}{#1}
1631 }{
1632 \bool_if:NT \g_um_upsans_bool {
1633 \um_map_chars_Latin:nn {sfup,sfit} {#1}
1634 \um_set_mathalphabet_Latin:Nnn \mathsf {up,it}{#1}
1635 }
1636 }
1637 \um_set_mathalphabet_Latin:Nnn \mathsfup {up,it}{#1}
1638 }
1639 \cs_new:Npn \um_config_sfup_latin:n #1 {
1640 \bool_if:NTF \g_um_sfliteral_bool {
1641 \um_map_chars_latin:nn {sfup} {#1}
1642 \um_set_mathalphabet_latin:Nnn \mathsf {up}{#1}
1643 }{
1644 \bool_if:NT \g_um_upsans_bool {
1645 \um_map_chars_latin:nn {sfup,sfit} {#1}
1646 \um_set_mathalphabet_latin:Nnn \mathsf {up,it}{#1}
1647 }
1648 }
66
1649 \um_set_mathalphabet_latin:Nnn \mathsfup {up,it}{#1}
1650 }
9.6.7 Sans serif italic: \mathsfit
1651 \cs_new:Npn \um_config_sfit_Latin:n #1 {
1652 \bool_if:NTF \g_um_sfliteral_bool {
1653 \um_map_chars_Latin:nn {sfit} {#1}
1654 \um_set_mathalphabet_Latin:Nnn \mathsf {it}{#1}
1655 }{
1656 \bool_if:NF \g_um_upsans_bool {
1657 \um_map_chars_Latin:nn {sfup,sfit} {#1}
1658 \um_set_mathalphabet_Latin:Nnn \mathsf {up,it}{#1}
1659 }
1660 }
1661 \um_set_mathalphabet_Latin:Nnn \mathsfit {up,it}{#1}
1662 }
1663 \cs_new:Npn \um_config_sfit_latin:n #1 {
1664 \bool_if:NTF \g_um_sfliteral_bool {
1665 \um_map_chars_latin:nn {sfit} {#1}
1666 \um_set_mathalphabet_latin:Nnn \mathsf {it}{#1}
1667 }{
1668 \bool_if:NF \g_um_upsans_bool {
1669 \um_map_chars_latin:nn {sfup,sfit} {#1}
1670 \um_set_mathalphabet_latin:Nnn \mathsf {up,it}{#1}
1671 }
1672 }
1673 \um_set_mathalphabet_latin:Nnn \mathsfit {up,it}{#1}
1674 }
9.6.8 Typewriter or monospaced: \mathtt
1675 \cs_new:Npn \um_config_tt_num:n #1 {
1676 \um_set_mathalphabet_numbers:Nnn \mathtt {up}{#1}
1677 }
1678 \cs_new:Npn \um_config_tt_Latin:n #1 {
1679 \um_set_mathalphabet_Latin:Nnn \mathtt {up,it}{#1}
1680 }
1681 \cs_new:Npn \um_config_tt_latin:n #1 {
1682 \um_set_mathalphabet_latin:Nnn \mathtt {up,it}{#1}
1683 }
9.6.9 Bold Italic: \mathbfit
1684 \cs_new:Npn \um_config_bfit_Latin:n #1 {
1685 \bool_if:NF \g_um_bfupLatin_bool {
1686 \um_map_chars_Latin:nn {bfup,bfit} {#1}
1687 }
1688 \um_set_mathalphabet_Latin:Nnn \mathbfit {up,it}{#1}
67
1689 \bool_if:NTF \g_um_bfliteral_bool {
1690 \um_map_chars_Latin:nn {bfit} {#1}
1691 \um_set_mathalphabet_Latin:Nnn \mathbf {it}{#1}
1692 }{
1693 \bool_if:NF \g_um_bfupLatin_bool {
1694 \um_map_chars_Latin:nn {bfup,bfit} {#1}
1695 \um_set_mathalphabet_Latin:Nnn \mathbf {up,it}{#1}
1696 }
1697 }
1698 }
1699 \cs_new:Npn \um_config_bfit_latin:n #1 {
1700 \bool_if:NF \g_um_bfuplatin_bool {
1701 \um_map_chars_latin:nn {bfup,bfit} {#1}
1702 }
1703 \um_set_mathalphabet_latin:Nnn \mathbfit {up,it}{#1}
1704 \bool_if:NTF \g_um_bfliteral_bool {
1705 \um_map_chars_latin:nn {bfit} {#1}
1706 \um_set_mathalphabet_latin:Nnn \mathbf {it}{#1}
1707 }{
1708 \bool_if:NF \g_um_bfuplatin_bool {
1709 \um_map_chars_latin:nn {bfup,bfit} {#1}
1710 \um_set_mathalphabet_latin:Nnn \mathbf {up,it}{#1}
1711 }
1712 }
1713 }
1714 \cs_new:Npn \um_config_bfit_Greek:n #1 {
1715 \um_set_mathalphabet_Greek:Nnn \mathbfit {up,it}{#1}
1716 \bool_if:NTF \g_um_bfliteral_bool {
1717 \um_map_chars_Greek:nn {bfit}{#1}
1718 \um_set_mathalphabet_Greek:Nnn \mathbf {it}{#1}
1719 }{
1720 \bool_if:NF \g_um_bfupGreek_bool {
1721 \um_map_chars_Greek:nn {bfup,bfit}{#1}
1722 \um_set_mathalphabet_Greek:Nnn \mathbf {up,it}{#1}
1723 }
1724 }
1725 }
1726 \cs_new:Npn \um_config_bfit_greek:n #1 {
1727 \um_set_mathalphabet_greek:Nnn \mathbfit {up,it} {#1}
1728 \bool_if:NTF \g_um_bfliteral_bool {
1729 \um_map_chars_greek:nn {bfit} {#1}
1730 \um_set_mathalphabet_greek:Nnn \mathbf {it} {#1}
1731 }{
1732 \bool_if:NF \g_um_bfupgreek_bool {
1733 \um_map_chars_greek:nn {bfit,bfup} {#1}
1734 \um_set_mathalphabet_greek:Nnn \mathbf {up,it} {#1}
68
1735 }
1736 }
1737 }
1738 \cs_new:Npn \um_config_bfit_misc:n #1 {
1739 \bool_if:NTF \g_um_literal_Nabla_bool {
1740 \um_map_single:nnn {Nabla}{bfit}{#1}
1741 }{
1742 \bool_if:NF \g_um_upNabla_bool {
1743 \um_map_single:nnn {Nabla}{bfup,bfit}{#1}
1744 }
1745 }
1746 \bool_if:NTF \g_um_literal_partial_bool {
1747 \um_map_single:nnn {partial}{bfit}{#1}
1748 }{
1749 \bool_if:NF \g_um_uppartial_bool {
1750 \um_map_single:nnn {partial}{bfup,bfit}{#1}
1751 }
1752 }
1753 \um_set_mathalphabet_pos:Nnnn \mathbfit {partial} {up,it}{#1}
1754 \um_set_mathalphabet_pos:Nnnn \mathbfit {Nabla} {up,it}{#1}
1755 \bool_if:NTF \g_um_literal_partial_bool {
1756 \um_set_mathalphabet_pos:Nnnn \mathbf {partial} {it}{#1}
1757 }{
1758 \bool_if:NF \g_um_uppartial_bool {
1759 \um_set_mathalphabet_pos:Nnnn \mathbf {partial} {up,it}{#1}
1760 }
1761 }
1762 \bool_if:NTF \g_um_literal_Nabla_bool {
1763 \um_set_mathalphabet_pos:Nnnn \mathbf {Nabla} {it}{#1}
1764 }{
1765 \bool_if:NF \g_um_upNabla_bool {
1766 \um_set_mathalphabet_pos:Nnnn \mathbf {Nabla} {up,it}{#1}
1767 }
1768 }
1769 }
9.6.10 Bold Upright: \mathbfup
1770 \cs_new:Npn \um_config_bfup_num:n #1 {
1771 \um_set_mathalphabet_numbers:Nnn \mathbf {up}{#1}
1772 \um_set_mathalphabet_numbers:Nnn \mathbfup {up}{#1}
1773 }
1774 \cs_new:Npn \um_config_bfup_Latin:n #1 {
1775 \bool_if:NT \g_um_bfupLatin_bool {
1776 \um_map_chars_Latin:nn {bfup,bfit} {#1}
1777 }
1778 \um_set_mathalphabet_Latin:Nnn \mathbfup {up,it}{#1}
69
1779 \bool_if:NTF \g_um_bfliteral_bool {
1780 \um_map_chars_Latin:nn {bfup} {#1}
1781 \um_set_mathalphabet_Latin:Nnn \mathbf {up}{#1}
1782 }{
1783 \bool_if:NT \g_um_bfupLatin_bool {
1784 \um_map_chars_Latin:nn {bfup,bfit} {#1}
1785 \um_set_mathalphabet_Latin:Nnn \mathbf {up,it}{#1}
1786 }
1787 }
1788 }
1789 \cs_new:Npn \um_config_bfup_latin:n #1 {
1790 \bool_if:NT \g_um_bfuplatin_bool {
1791 \um_map_chars_latin:nn {bfup,bfit} {#1}
1792 }
1793 \um_set_mathalphabet_latin:Nnn \mathbfup {up,it}{#1}
1794 \bool_if:NTF \g_um_bfliteral_bool {
1795 \um_map_chars_latin:nn {bfup} {#1}
1796 \um_set_mathalphabet_latin:Nnn \mathbf {up}{#1}
1797 }{
1798 \bool_if:NT \g_um_bfuplatin_bool {
1799 \um_map_chars_latin:nn {bfup,bfit} {#1}
1800 \um_set_mathalphabet_latin:Nnn \mathbf {up,it}{#1}
1801 }
1802 }
1803 }
1804 \cs_new:Npn \um_config_bfup_Greek:n #1 {
1805 \um_set_mathalphabet_Greek:Nnn \mathbfup {up,it}{#1}
1806 \bool_if:NTF \g_um_bfliteral_bool {
1807 \um_map_chars_Greek:nn {bfup}{#1}
1808 \um_set_mathalphabet_Greek:Nnn \mathbf {up}{#1}
1809 }{
1810 \bool_if:NT \g_um_bfupGreek_bool {
1811 \um_map_chars_Greek:nn {bfup,bfit}{#1}
1812 \um_set_mathalphabet_Greek:Nnn \mathbf {up,it}{#1}
1813 }
1814 }
1815 }
1816 \cs_new:Npn \um_config_bfup_greek:n #1 {
1817 \um_set_mathalphabet_greek:Nnn \mathbfup {up,it} {#1}
1818 \bool_if:NTF \g_um_bfliteral_bool {
1819 \um_map_chars_greek:nn {bfup} {#1}
1820 \um_set_mathalphabet_greek:Nnn \mathbf {up} {#1}
1821 }{
1822 \bool_if:NT \g_um_bfupgreek_bool {
1823 \um_map_chars_greek:nn {bfup,bfit} {#1}
1824 \um_set_mathalphabet_greek:Nnn \mathbf {up,it} {#1}
70
1825 }
1826 }
1827 }
1828 \cs_new:Npn \um_config_bfup_misc:n #1 {
1829 \bool_if:NTF \g_um_literal_Nabla_bool {
1830 \um_map_single:nnn {Nabla}{bfup}{#1}
1831 }{
1832 \bool_if:NT \g_um_upNabla_bool {
1833 \um_map_single:nnn {Nabla}{bfup,bfit}{#1}
1834 }
1835 }
1836 \bool_if:NTF \g_um_literal_partial_bool {
1837 \um_map_single:nnn {partial}{bfup}{#1}
1838 }{
1839 \bool_if:NT \g_um_uppartial_bool {
1840 \um_map_single:nnn {partial}{bfup,bfit}{#1}
1841 }
1842 }
1843 \um_set_mathalphabet_pos:Nnnn \mathbfup {partial} {up,it}{#1}
1844 \um_set_mathalphabet_pos:Nnnn \mathbfup {Nabla} {up,it}{#1}
1845 \um_set_mathalphabet_pos:Nnnn \mathbfup {digamma} {up}{#1}
1846 \um_set_mathalphabet_pos:Nnnn \mathbfup {Digamma} {up}{#1}
1847 \um_set_mathalphabet_pos:Nnnn \mathbf {digamma} {up}{#1}
1848 \um_set_mathalphabet_pos:Nnnn \mathbf {Digamma} {up}{#1}
1849 \bool_if:NTF \g_um_literal_partial_bool {
1850 \um_set_mathalphabet_pos:Nnnn \mathbf {partial} {up}{#1}
1851 }{
1852 \bool_if:NT \g_um_uppartial_bool {
1853 \um_set_mathalphabet_pos:Nnnn \mathbf {partial} {up,it}{#1}
1854 }
1855 }
1856 \bool_if:NTF \g_um_literal_Nabla_bool {
1857 \um_set_mathalphabet_pos:Nnnn \mathbf {Nabla} {up}{#1}
1858 }{
1859 \bool_if:NT \g_um_upNabla_bool {
1860 \um_set_mathalphabet_pos:Nnnn \mathbf {Nabla} {up,it}{#1}
1861 }
1862 }
1863 }
9.6.11 Bold fractur or fraktur or blackletter: \mathbffrak
1864 \cs_new:Npn \um_config_bffrak_Latin:n #1 {
1865 \um_set_mathalphabet_Latin:Nnn \mathbffrak {up,it}{#1}
1866 }
1867 \cs_new:Npn \um_config_bffrak_latin:n #1 {
1868 \um_set_mathalphabet_latin:Nnn \mathbffrak {up,it}{#1}
71
1869 }
9.6.12 Bold script or calligraphic: \mathbfscr
1870 \cs_new:Npn \um_config_bfscr_Latin:n #1 {
1871 \um_set_mathalphabet_Latin:Nnn \mathbfscr {up,it}{#1}
1872 }
1873 \cs_new:Npn \um_config_bfscr_latin:n #1 {
1874 \um_set_mathalphabet_latin:Nnn \mathbfscr {up,it}{#1}
1875 }
1876 \cs_new:Npn \um_config_bfcal_Latin:n #1 {
1877 \um_set_mathalphabet_Latin:Nnn \mathbfcal {up,it}{#1}
1878 }
9.6.13 Bold upright sans serif: \mathbfsfup
1879 \cs_new:Npn \um_config_bfsfup_num:n #1 {
1880 \um_set_mathalphabet_numbers:Nnn \mathbfsf {up}{#1}
1881 \um_set_mathalphabet_numbers:Nnn \mathbfsfup {up}{#1}
1882 }
1883 \cs_new:Npn \um_config_bfsfup_Latin:n #1 {
1884 \bool_if:NTF \g_um_sfliteral_bool {
1885 \um_map_chars_Latin:nn {bfsfup} {#1}
1886 \um_set_mathalphabet_Latin:Nnn \mathbfsf {up}{#1}
1887 }{
1888 \bool_if:NT \g_um_upsans_bool {
1889 \um_map_chars_Latin:nn {bfsfup,bfsfit} {#1}
1890 \um_set_mathalphabet_Latin:Nnn \mathbfsf {up,it}{#1}
1891 }
1892 }
1893 \um_set_mathalphabet_Latin:Nnn \mathbfsfup {up,it}{#1}
1894 }
1895 \cs_new:Npn \um_config_bfsfup_latin:n #1 {
1896 \bool_if:NTF \g_um_sfliteral_bool {
1897 \um_map_chars_latin:nn {bfsfup} {#1}
1898 \um_set_mathalphabet_latin:Nnn \mathbfsf {up}{#1}
1899 }{
1900 \bool_if:NT \g_um_upsans_bool {
1901 \um_map_chars_latin:nn {bfsfup,bfsfit} {#1}
1902 \um_set_mathalphabet_latin:Nnn \mathbfsf {up,it}{#1}
1903 }
1904 }
1905 \um_set_mathalphabet_latin:Nnn \mathbfsfup {up,it}{#1}
1906 }
1907 \cs_new:Npn \um_config_bfsfup_Greek:n #1 {
1908 \bool_if:NTF \g_um_sfliteral_bool {
1909 \um_map_chars_Greek:nn {bfsfup}{#1}
1910 \um_set_mathalphabet_Greek:Nnn \mathbfsf {up}{#1}
1911 }{
72
1912 \bool_if:NT \g_um_upsans_bool {
1913 \um_map_chars_Greek:nn {bfsfup,bfsfit}{#1}
1914 \um_set_mathalphabet_Greek:Nnn \mathbfsf {up,it}{#1}
1915 }
1916 }
1917 \um_set_mathalphabet_Greek:Nnn \mathbfsfup {up,it}{#1}
1918 }
1919 \cs_new:Npn \um_config_bfsfup_greek:n #1 {
1920 \bool_if:NTF \g_um_sfliteral_bool {
1921 \um_map_chars_greek:nn {bfsfup} {#1}
1922 \um_set_mathalphabet_greek:Nnn \mathbfsf {up} {#1}
1923 }{
1924 \bool_if:NT \g_um_upsans_bool {
1925 \um_map_chars_greek:nn {bfsfup,bfsfit} {#1}
1926 \um_set_mathalphabet_greek:Nnn \mathbfsf {up,it} {#1}
1927 }
1928 }
1929 \um_set_mathalphabet_greek:Nnn \mathbfsfup {up,it} {#1}
1930 }
1931 \cs_new:Npn \um_config_bfsfup_misc:n #1 {
1932 \bool_if:NTF \g_um_literal_Nabla_bool {
1933 \um_map_single:nnn {Nabla}{bfsfup}{#1}
1934 }{
1935 \bool_if:NT \g_um_upNabla_bool {
1936 \um_map_single:nnn {Nabla}{bfsfup,bfsfit}{#1}
1937 }
1938 }
1939 \bool_if:NTF \g_um_literal_partial_bool {
1940 \um_map_single:nnn {partial}{bfsfup}{#1}
1941 }{
1942 \bool_if:NT \g_um_uppartial_bool {
1943 \um_map_single:nnn {partial}{bfsfup,bfsfit}{#1}
1944 }
1945 }
1946 \um_set_mathalphabet_pos:Nnnn \mathbfsfup {partial} {up,it}{#1}
1947 \um_set_mathalphabet_pos:Nnnn \mathbfsfup {Nabla} {up,it}{#1}
1948 \bool_if:NTF \g_um_literal_partial_bool {
1949 \um_set_mathalphabet_pos:Nnnn \mathbfsf {partial} {up}{#1}
1950 }{
1951 \bool_if:NT \g_um_uppartial_bool {
1952 \um_set_mathalphabet_pos:Nnnn \mathbfsf {partial} {up,it}{#1}
1953 }
1954 }
1955 \bool_if:NTF \g_um_literal_Nabla_bool {
1956 \um_set_mathalphabet_pos:Nnnn \mathbfsf {Nabla} {up}{#1}
1957 }{
73
1958 \bool_if:NT \g_um_upNabla_bool {
1959 \um_set_mathalphabet_pos:Nnnn \mathbfsf {Nabla} {up,it}{#1}
1960 }
1961 }
1962 }
9.6.14 Bold italic sans serif: \mathbfsfit
1963 \cs_new:Npn \um_config_bfsfit_Latin:n #1 {
1964 \bool_if:NTF \g_um_sfliteral_bool {
1965 \um_map_chars_Latin:nn {bfsfit} {#1}
1966 \um_set_mathalphabet_Latin:Nnn \mathbfsf {it}{#1}
1967 }{
1968 \bool_if:NF \g_um_upsans_bool {
1969 \um_map_chars_Latin:nn {bfsfup,bfsfit} {#1}
1970 \um_set_mathalphabet_Latin:Nnn \mathbfsf {up,it}{#1}
1971 }
1972 }
1973 \um_set_mathalphabet_Latin:Nnn \mathbfsfit {up,it}{#1}
1974 }
1975 \cs_new:Npn \um_config_bfsfit_latin:n #1 {
1976 \bool_if:NTF \g_um_sfliteral_bool {
1977 \um_map_chars_latin:nn {bfsfit} {#1}
1978 \um_set_mathalphabet_latin:Nnn \mathbfsf {it}{#1}
1979 }{
1980 \bool_if:NF \g_um_upsans_bool {
1981 \um_map_chars_latin:nn {bfsfup,bfsfit} {#1}
1982 \um_set_mathalphabet_latin:Nnn \mathbfsf {up,it}{#1}
1983 }
1984 }
1985 \um_set_mathalphabet_latin:Nnn \mathbfsfit {up,it}{#1}
1986 }
1987 \cs_new:Npn \um_config_bfsfit_Greek:n #1 {
1988 \bool_if:NTF \g_um_sfliteral_bool {
1989 \um_map_chars_Greek:nn {bfsfit}{#1}
1990 \um_set_mathalphabet_Greek:Nnn \mathbfsf {it}{#1}
1991 }{
1992 \bool_if:NF \g_um_upsans_bool {
1993 \um_map_chars_Greek:nn {bfsfup,bfsfit}{#1}
1994 \um_set_mathalphabet_Greek:Nnn \mathbfsf {up,it}{#1}
1995 }
1996 }
1997 \um_set_mathalphabet_Greek:Nnn \mathbfsfit {up,it}{#1}
1998 }
1999 \cs_new:Npn \um_config_bfsfit_greek:n #1 {
2000 \bool_if:NTF \g_um_sfliteral_bool {
2001 \um_map_chars_greek:nn {bfsfit} {#1}
74
2002 \um_set_mathalphabet_greek:Nnn \mathbfsf {it} {#1}
2003 }{
2004 \bool_if:NF \g_um_upsans_bool {
2005 \um_map_chars_greek:nn {bfsfup,bfsfit} {#1}
2006 \um_set_mathalphabet_greek:Nnn \mathbfsf {up,it} {#1}
2007 }
2008 }
2009 \um_set_mathalphabet_greek:Nnn \mathbfsfit {up,it} {#1}
2010 }
2011 \cs_new:Npn \um_config_bfsfit_misc:n #1 {
2012 \bool_if:NTF \g_um_literal_Nabla_bool {
2013 \um_map_single:nnn {Nabla}{bfsfit}{#1}
2014 }{
2015 \bool_if:NF \g_um_upNabla_bool {
2016 \um_map_single:nnn {Nabla}{bfsfup,bfsfit}{#1}
2017 }
2018 }
2019 \bool_if:NTF \g_um_literal_partial_bool {
2020 \um_map_single:nnn {partial}{bfsfit}{#1}
2021 }{
2022 \bool_if:NF \g_um_uppartial_bool {
2023 \um_map_single:nnn {partial}{bfsfup,bfsfit}{#1}
2024 }
2025 }
2026 \um_set_mathalphabet_pos:Nnnn \mathbfsfit {partial} {up,it}{#1}
2027 \um_set_mathalphabet_pos:Nnnn \mathbfsfit {Nabla} {up,it}{#1}
2028 \bool_if:NTF \g_um_literal_partial_bool {
2029 \um_set_mathalphabet_pos:Nnnn \mathbfsf {partial} {it}{#1}
2030 }{
2031 \bool_if:NF \g_um_uppartial_bool {
2032 \um_set_mathalphabet_pos:Nnnn \mathbfsf {partial} {up,it}{#1}
2033 }
2034 }
2035 \bool_if:NTF \g_um_literal_Nabla_bool {
2036 \um_set_mathalphabet_pos:Nnnn \mathbfsf {Nabla} {it}{#1}
2037 }{
2038 \bool_if:NF \g_um_upNabla_bool {
2039 \um_set_mathalphabet_pos:Nnnn \mathbfsf {Nabla} {up,it}{#1}
2040 }
2041 }
2042 }
10 A token list to contain the data of the math table
Instead of \input-ing the unicode math table every time we want to re-read its
data, we save it within a macro. This has two advantages: 1. it should be slightly
75
faster, at the expense of memory; 2. we dont need to worry about catcodes later,
since theyre frozen at this point.
In time, the case statement inside set_mathsymbol will be moved in here to
avoid re-running it every time.
2043 \xetex_or_luatex:nnn { \cs_set:Npn \um_symbol_setup: }
2044 {
2045 \def\mathfence{\mathfence}
2046 \def\mathover{\mathover}
2047 \def\mathunder{\mathunder}
2048 \cs_set:Npn \UnicodeMathSymbol ##1##2##3##4 {
2049 \prg_case_tl:Nnn ##3 { \mathover {} \mathunder {} }
2050 {
2051 \exp_not:n {\_um_sym:nnn{##1}{##2}{##3}}
2052 }
2053 }
2054 }
2055 {
2056 \cs_set:Npn \UnicodeMathSymbol ##1##2##3##4 {
2057 \exp_not:n {\_um_sym:nnn{##1}{##2}{##3}}
2058 }
2059 }
2060 \CatchFileEdef \g_um_mathtable_tl {unicode-math-table.tex} {\um_symbol_setup:}
\um_input_math_symbol_table: This function simply expands to the token list containing all the data.
2061 \cs_new:Npn \um_input_math_symbol_table: {\g_um_mathtable_tl}
11 Denitions of the active math characters
Here we dene every Unicode math codepoint an equivalent macro name. The
two are equivalent, in a \let\xyz=^^^^1234 kind of way.
\um_cs_set_eq_active_char:Nw
\um_active_char_set:wc
We need to do some trickery to transform the \_um_sym:nnn argument "ABCDEF
into the X
T
E
X caret input form ^^^^^abcdef. It is very important that the argu-
ment has ve characters. Otherwise we need to change the number of ^ chars.
To do this, turn ^ into a regular other character and dene the macro to
performthe lowercasing and\let. \scantokens changes the carets back into their
original meaning after the group has ended and ^s catcode returns to normal.
2062 \begingroup
2063 \char_make_other:N \^
2064 \cs_gset:Npn \um_cs_set_eq_active_char:Nw #1 = "#2 \q_nil {
2065 \lowercase{
2066 \tl_rescan:nn {
2067 \char_make_other:N \{
2068 \char_make_other:N \}
76
2069 \char_make_other:N \&
2070 \char_make_other:N \%
2071 \char_make_other:N \$
2072 }{
2073 \global\let#1=^^^^^#2
2074 }
2075 }
2076 }
Making ^ the right catcode isnt strictly necessary right now but it helps to future
proof us with, e.g., breqn. Because were inside a \tl_rescan:nn, use plain old
T
E
X syntax to avoid any catcode problems.
2077 \cs_gnew:Npn \um_active_char_set:wc "#1 \q_nil #2 {
2078 \lowercase {
2079 \tl_rescan:nn {
2080 \catcode`\_=11\relax
2081 \catcode`\:=11\relax
2082 \catcode`\^=7\relax
2083 }{
2084 \protected\gdef^^^^^#1{\csname #2\endcsname}%
2085 }
2086 }
2087 }
2088 \endgroup
Now give \_um_sym:nnn a denition in terms of \um_cs_set_eq_active_-
char:Nw and were good to go.
Ensure catcodes are appropriate; make sure # is an other so that we dont
get confused with \mathoctothorpe.
2089 \AtBeginDocument{
2090 \group_begin:
2091 \char_make_math_superscript:N \^
2092 \cs_set:Npn \_um_sym:nnn #1#2#3 {
2093 \bool_if:nF { \cs_if_eq_p:NN #3 \mathaccent ||
2094 \cs_if_eq_p:NN #3 \mathopen ||
2095 \cs_if_eq_p:NN #3 \mathclose ||
2096 \cs_if_eq_p:NN #3 \mathover ||
2097 \cs_if_eq_p:NN #3 \mathunder } {
2098 \um_cs_set_eq_active_char:Nw #2 = #1 \q_nil \ignorespaces
2099 }
2100 }
2101 \char_make_other:N \#
2102 \um_input_math_symbol_table:
2103 \group_end:
2104 }
Fix \backslash, which is dened as the escape char character above:
77
2105 \group_begin:
2106 \lccode`\*=`\\
2107 \char_make_escape:N \|
2108 \char_make_other:N \\
2109 |lowercase{
2110 |AtBeginDocument{
2111 |let|backslash=*
2112 }
2113 }
2114 |group_end:
Fix \backslash:
12 Epilogue
Lots of little things to tidy up.
12.1 Primes
We need a new prime algorithm. Unicode math has four pre-drawn prime
glyphs.
u+2032 prime (\prime):
i
u+2033 double prime (\dprime):
u
u+2034 triple prime (\trprime):
~
u+2057 quadruple prime (\qprime):
~
As you can see, theyre all drawn at the correct height without being super-
scripted. However, in a correctly behaving OpenType font, we also see dierent
behaviour after the ssty feature is applied:
xi xu x~ x~
The glyphs are nowfull size so that when placed inside a superscript, their shape
will match the originally sized ones. Many thanks to Ross Mills of Tiro Typeworks
for originally pointing out this behaviour.
In regular L
A
T
E
X, primes can be entered with the straight quote character ',
and multiple straight quotes chain together to produce multiple primes. Better
results can be achieved in unicode-math by chaining multiple single primes into a
pre-drawn multi-prime glyph; consider
iii
vs.
~
.
For Unicode maths, we wish to conserve this behaviour and augment it with
the possibility of adding any combination of Unicode prime or any of the -prime
characters. E.g., the user might copy-paste a double prime from another source
and then later type another single prime after it; the output should be the triple
prime.
78
Our algorithm is:
Prime encountered; pcount=1.
Scan ahead; if prime: pcount:=pcount+1; repeat.
If not prime, stop scanning.
If pcount=1, \prime, end.
If pcount=2, check \dprime; if it exists, use it, end; if not, goto last step.
Ditto pcount=3 & \trprime.
Ditto pcount=4 & \qprime.
If pcount>4 or the glyph doesnt exist, insert pcount \primes with \primek-
ern between each.
This is a wrapper to insert a superscript; if there is a subsequent trailing su-
perscript, then it is included within the insertion.
2115 \cs_new:Nn \um_arg_i_before_egroup:n {#1\egroup}
2116 \cs_new:Nn \um_superscript:n {
2117 ^\bgroup #1
2118 \peek_meaning_remove:NTF ^
2119 \um_arg_i_before_egroup:n \egroup
2120 }
2121 \muskip_new:N \g_um_primekern_muskip
2122 \muskip_gset:Nn \g_um_primekern_muskip { -\thinmuskip/2 }% arbitrary
2123 \int_new:N \l_um_primecount_int
2124 \cs_new:Npn \um_nprimes:Nn #1#2 {
2125 \um_superscript:n {
2126 #1
2127 \prg_replicate:nn {#2-1} { \mskip \g_um_primekern_muskip #1 }
2128 }
2129 }
2130 \cs_new:Npn \um_nprimes_select:nn #1#2 {
2131 \prg_case_int:nnn {#2}{
2132 {1} { \um_superscript:n {#1} }
2133 {2} {
2134 \um_glyph_if_exist:nTF {"2033}
2135 { \um_superscript:n {\um_prime_double_mchar} }
2136 { \um_nprimes:Nn #1 {#2} }
2137 }
2138 {3} {
2139 \um_glyph_if_exist:nTF {"2034}
2140 { \um_superscript:n {\um_prime_triple_mchar} }
2141 { \um_nprimes:Nn #1 {#2} }
2142 }
2143 {4} {
2144 \um_glyph_if_exist:nTF {"2057}
2145 { \um_superscript:n {\um_prime_quad_mchar} }
2146 { \um_nprimes:Nn #1 {#2} }
2147 }
79
2148 }{
2149 \um_nprimes:Nn #1 {#2}
2150 }
2151 }
2152 \cs_new:Npn \um_nbackprimes_select:nn #1#2 {
2153 \prg_case_int:nnn {#2}{
2154 {1} { \um_superscript:n {#1} }
2155 {2} {
2156 \um_glyph_if_exist:nTF {"2036}
2157 { \um_superscript:n {\um_backprime_double_mchar} }
2158 { \um_nprimes:Nn #1 {#2} }
2159 }
2160 {3} {
2161 \um_glyph_if_exist:nTF {"2037}
2162 { \um_superscript:n {\um_backprime_triple_mchar} }
2163 { \um_nprimes:Nn #1 {#2} }
2164 }
2165 }{
2166 \um_nprimes:Nn #1 {#2}
2167 }
2168 }
Scanning is annoying because Im too lazy to do it for the general case.
2169 \cs_new:Npn \um_scan_prime: {
2170 \int_zero:N \l_um_primecount_int
2171 \um_scanprime_collect:N \um_prime_single_mchar
2172 }
2173 \cs_new:Npn \um_scan_dprime: {
2174 \int_set:Nn \l_um_primecount_int {1}
2175 \um_scanprime_collect:N \um_prime_single_mchar
2176 }
2177 \cs_new:Npn \um_scan_trprime: {
2178 \int_set:Nn \l_um_primecount_int {2}
2179 \um_scanprime_collect:N \um_prime_single_mchar
2180 }
2181 \cs_new:Npn \um_scan_qprime: {
2182 \int_set:Nn \l_um_primecount_int {3}
2183 \um_scanprime_collect:N \um_prime_single_mchar
2184 }
2185 \cs_new:Npn \um_scanprime_collect:N #1 {
2186 \int_incr:N \l_um_primecount_int
2187 \peek_meaning_remove:NTF ' {
2188 \um_scanprime_collect:N #1
2189 }{
2190 \peek_meaning_remove:NTF \um_scan_prime: {
2191 \um_scanprime_collect:N #1
2192 }{
80
2193 \peek_meaning_remove:NTF ^^^^2032 {
2194 \um_scanprime_collect:N #1
2195 }{
2196 \peek_meaning_remove:NTF \um_scan_dprime: {
2197 \int_incr:N \l_um_primecount_int
2198 \um_scanprime_collect:N #1
2199 }{
2200 \peek_meaning_remove:NTF ^^^^2033 {
2201 \int_incr:N \l_um_primecount_int
2202 \um_scanprime_collect:N #1
2203 }{
2204 \peek_meaning_remove:NTF \um_scan_trprime: {
2205 \int_add:Nn \l_um_primecount_int {2}
2206 \um_scanprime_collect:N #1
2207 }{
2208 \peek_meaning_remove:NTF ^^^^2034 {
2209 \int_add:Nn \l_um_primecount_int {2}
2210 \um_scanprime_collect:N #1
2211 }{
2212 \peek_meaning_remove:NTF \um_scan_qprime: {
2213 \int_add:Nn \l_um_primecount_int {3}
2214 \um_scanprime_collect:N #1
2215 }{
2216 \peek_meaning_remove:NTF ^^^^2057 {
2217 \int_add:Nn \l_um_primecount_int {3}
2218 \um_scanprime_collect:N #1
2219 }{
2220 \um_nprimes_select:nn {#1} {\l_um_primecount_int}
2221 }
2222 }
2223 }
2224 }
2225 }
2226 }
2227 }
2228 }
2229 }
2230 }
2231 \cs_new:Npn \um_scan_backprime: {
2232 \int_zero:N \l_um_primecount_int
2233 \um_scanbackprime_collect:N \um_backprime_single_mchar
2234 }
2235 \cs_new:Npn \um_scan_backdprime: {
2236 \int_set:Nn \l_um_primecount_int {1}
2237 \um_scanbackprime_collect:N \um_backprime_single_mchar
2238 }
81
2239 \cs_new:Npn \um_scan_backtrprime: {
2240 \int_set:Nn \l_um_primecount_int {2}
2241 \um_scanbackprime_collect:N \um_backprime_single_mchar
2242 }
2243 \cs_new:Npn \um_scanbackprime_collect:N #1 {
2244 \int_incr:N \l_um_primecount_int
2245 \peek_meaning_remove:NTF ` {
2246 \um_scanbackprime_collect:N #1
2247 }{
2248 \peek_meaning_remove:NTF \um_scan_backprime: {
2249 \um_scanbackprime_collect:N #1
2250 }{
2251 \peek_meaning_remove:NTF ^^^^2035 {
2252 \um_scanbackprime_collect:N #1
2253 }{
2254 \peek_meaning_remove:NTF \um_scan_backdprime: {
2255 \int_incr:N \l_um_primecount_int
2256 \um_scanbackprime_collect:N #1
2257 }{
2258 \peek_meaning_remove:NTF ^^^^2036 {
2259 \int_incr:N \l_um_primecount_int
2260 \um_scanbackprime_collect:N #1
2261 }{
2262 \peek_meaning_remove:NTF \um_scan_backtrprime: {
2263 \int_add:Nn \l_um_primecount_int {2}
2264 \um_scanbackprime_collect:N #1
2265 }{
2266 \peek_meaning_remove:NTF ^^^^2037 {
2267 \int_add:Nn \l_um_primecount_int {2}
2268 \um_scanbackprime_collect:N #1
2269 }{
2270 \um_nbackprimes_select:nn {#1} {\l_um_primecount_int}
2271 }
2272 }
2273 }
2274 }
2275 }
2276 }
2277 }
2278 }
2279 \AtBeginDocument {
2280 \cs_set_eq:NN \prime \um_scan_prime:
2281 \cs_set_eq:NN \drime \um_scan_dprime:
2282 \cs_set_eq:NN \trprime \um_scan_trprime:
2283 \cs_set_eq:NN \qprime \um_scan_qprime:
2284 \cs_set_eq:NN \backprime \um_scan_backprime:
82
2285 \cs_set_eq:NN \backdprime \um_scan_backdprime:
2286 \cs_set_eq:NN \backtrprime \um_scan_backtrprime:
2287 }
2288 \group_begin:
2289 \char_make_active:N \'
2290 \char_make_active:N \`
2291 \char_make_active:n {"2032}
2292 \char_make_active:n {"2033}
2293 \char_make_active:n {"2034}
2294 \char_make_active:n {"2057}
2295 \char_make_active:n {"2035}
2296 \char_make_active:n {"2036}
2297 \char_make_active:n {"2037}
2298 \AtBeginDocument{
2299 \cs_set_eq:NN ' \um_scan_prime:
2300 \cs_set_eq:NN ^^^^2032 \um_scan_prime:
2301 \cs_set_eq:NN ^^^^2033 \um_scan_dprime:
2302 \cs_set_eq:NN ^^^^2034 \um_scan_trprime:
2303 \cs_set_eq:NN ^^^^2057 \um_scan_qprime:
2304 \cs_set_eq:NN ` \um_scan_backprime:
2305 \cs_set_eq:NN ^^^^2035 \um_scan_backprime:
2306 \cs_set_eq:NN ^^^^2036 \um_scan_backdprime:
2307 \cs_set_eq:NN ^^^^2037 \um_scan_backtrprime:
2308 }
2309 \group_end:
12.2 Unicode radicals
\sqrt Redene this macro for LuaT
E
X, which provides us a nice primitive to use.
2310 \luatex_if_engine:T {
2311 \RenewDocumentCommand \sqrt { O{} m } {
2312 \luatexUroot \l_um_radical_sqrt_tl {#1} {#2}
2313 }
2314 \cs_set:Npn \root #1 \of #2 {
2315 \luatexUroot \l_um_radical_sqrt_tl {#1} {#2}
2316 }
2317 }
\r@@t #1 : A mathstyle (for \mathpalette)
#2 : Leading superscript for the sqrt sign
A re-implementation of L
A
T
E
Xs hard-coded n-root sign using the appropriate
\fontdimens.
2318 \cs_set_nopar:Npn \r@@t #1#2 {
2319 \setbox\z@\hbox{$\m@th #1\sqrtsign{#2}$}
2320 \um_mathstyle_scale:Nnn{#1}{\kern}{\fontdimen63\l_um_font}
2321 \raise \dimexpr(
83
2322 \um_fontdimen_to_percent:nn{65}{\l_um_font}\ht\z@-
2323 \um_fontdimen_to_percent:nn{65}{\l_um_font}\dp\z@
2324 )\relax
2325 \copy \rootbox
2326 \um_mathstyle_scale:Nnn{#1}{\kern}{\fontdimen64\l_um_font}
2327 \box \z@
2328 }
\um_fontdimen_to_percent:nn #1 : Font dimen number
#2 : Font variable
\fontdimens 10, 11, and 65 arent actually dimensions, theyre percentage values
given in units of sp. This macro takes a font dimension number and outputs the
decimal value of the associated parameter.
2329 \cs_new:Npn \um_fontdimen_to_percent:nn #1#2 {
2330 0.\strip@pt\dimexpr\fontdimen#1#2 *65536\relax
2331 }
\um_mathstyle_scale:Nnn #1 : A math style (\scriptstyle, say)
#2 : Macro that takes a non-delimited length argument (like \kern)
#3 : Length control sequence to be scaled according to the math style
This macro is used to scale the lengths reported by \fontdimen according to the
scale factor for script- and scriptscript-size objects.
2332 \cs_new:Npn \um_mathstyle_scale:Nnn #1#2#3 {
2333 \ifx#1\scriptstyle
2334 #2\um_fontdimen_to_percent:nn{10}\l_um_font#3
2335 \else
2336 \ifx#1\scriptscriptstyle
2337 #2\um_fontdimen_to_percent:nn{11}\l_um_font#3
2338 \else
2339 #2#3
2340 \fi
2341 \fi
2342 }
12.3 Unicode sub- and super-scripts
The idea here is to enter a scanning state after a superscript or subscript is encoun-
tered. If subsequent superscripts or subscripts (resp.) are found, they are lumped
together. Each sub/super has a corresponding regular size glyph which is used
by X
T
E
X to typeset the results; this means that the actual subscript/superscript
glyphs are never seen in the output document they are only used as input char-
acters.
Open question: should the superscript-like modiers (u+1D2C modier cap-
ital letter a and on) be included here?
2343 \prop_new:N \g_um_supers_prop
84
2344 \prop_new:N \g_um_subs_prop
2345 \group_begin:
Superscripts Populate a property list with superscript characters; their meaning
as their key, for reasons that will become apparent soon, and their replacement as
each keys value. Then make the superscript active and bind it to the scanning
function.
\scantokens makes this process much simpler since we can activate the char
and assign its meaning in one step.
2346 \cs_set:Npn \um_setup_active_superscript:nn #1#2 {
2347 \prop_gput:Nxn \g_um_supers_prop {\meaning #1} {#2}
2348 \char_make_active:N #1
2349 \char_gmake_mathactive:N #1
2350 \scantokens{
2351 \cs_gset:Npn #1 {
2352 \tl_set:Nn \l_um_ss_chain_tl {#2}
2353 \cs_set_eq:NN \um_sub_or_super:n \sp
2354 \tl_set:Nn \l_um_tmpa_tl {supers}
2355 \um_scan_sscript:
2356 }
2357 }
2358 }
Bam:
2359 \um_setup_active_superscript:nn {^^^^2070} {0}
2360 \um_setup_active_superscript:nn {^^^^00b9} {1}
2361 \um_setup_active_superscript:nn {^^^^00b2} {2}
2362 \um_setup_active_superscript:nn {^^^^00b3} {3}
2363 \um_setup_active_superscript:nn {^^^^2074} {4}
2364 \um_setup_active_superscript:nn {^^^^2075} {5}
2365 \um_setup_active_superscript:nn {^^^^2076} {6}
2366 \um_setup_active_superscript:nn {^^^^2077} {7}
2367 \um_setup_active_superscript:nn {^^^^2078} {8}
2368 \um_setup_active_superscript:nn {^^^^2079} {9}
2369 \um_setup_active_superscript:nn {^^^^207a} {+}
2370 \um_setup_active_superscript:nn {^^^^207b} {-}
2371 \um_setup_active_superscript:nn {^^^^207c} {=}
2372 \um_setup_active_superscript:nn {^^^^207d} {(}
2373 \um_setup_active_superscript:nn {^^^^207e} {)}
2374 \um_setup_active_superscript:nn {^^^^2071} {i}
2375 \um_setup_active_superscript:nn {^^^^207f} {n}
Subscripts Ditto above.
2376 \cs_set:Npn \um_setup_active_subscript:nn #1#2 {
2377 \prop_gput:Nxn \g_um_subs_prop {\meaning #1} {#2}
85
2378 \char_make_active:N #1
2379 \char_gmake_mathactive:N #1
2380 \scantokens{
2381 \cs_gset:Npn #1 {
2382 \tl_set:Nn \l_um_ss_chain_tl {#2}
2383 \cs_set_eq:NN \um_sub_or_super:n \sb
2384 \tl_set:Nn \l_um_tmpa_tl {subs}
2385 \um_scan_sscript:
2386 }
2387 }
2388 }
A few more subscripts than superscripts:
2389 \um_setup_active_subscript:nn {^^^^2080} {0}
2390 \um_setup_active_subscript:nn {^^^^2081} {1}
2391 \um_setup_active_subscript:nn {^^^^2082} {2}
2392 \um_setup_active_subscript:nn {^^^^2083} {3}
2393 \um_setup_active_subscript:nn {^^^^2084} {4}
2394 \um_setup_active_subscript:nn {^^^^2085} {5}
2395 \um_setup_active_subscript:nn {^^^^2086} {6}
2396 \um_setup_active_subscript:nn {^^^^2087} {7}
2397 \um_setup_active_subscript:nn {^^^^2088} {8}
2398 \um_setup_active_subscript:nn {^^^^2089} {9}
2399 \um_setup_active_subscript:nn {^^^^208a} {+}
2400 \um_setup_active_subscript:nn {^^^^208b} {-}
2401 \um_setup_active_subscript:nn {^^^^208c} {=}
2402 \um_setup_active_subscript:nn {^^^^208d} {(}
2403 \um_setup_active_subscript:nn {^^^^208e} {)}
2404 \um_setup_active_subscript:nn {^^^^2090} {a}
2405 \um_setup_active_subscript:nn {^^^^2091} {e}
2406 \um_setup_active_subscript:nn {^^^^1d62} {i}
2407 \um_setup_active_subscript:nn {^^^^2092} {o}
2408 \um_setup_active_subscript:nn {^^^^1d63} {r}
2409 \um_setup_active_subscript:nn {^^^^1d64} {u}
2410 \um_setup_active_subscript:nn {^^^^1d65} {v}
2411 \um_setup_active_subscript:nn {^^^^2093} {x}
2412 \um_setup_active_subscript:nn {^^^^1d66} {\beta}
2413 \um_setup_active_subscript:nn {^^^^1d67} {\gamma}
2414 \um_setup_active_subscript:nn {^^^^1d68} {\rho}
2415 \um_setup_active_subscript:nn {^^^^1d69} {\phi}
2416 \um_setup_active_subscript:nn {^^^^1d6a} {\chi}
2417 \group_end:
The scanning command, evident in its purpose:
2418 \cs_new:Npn \um_scan_sscript: {
2419 \um_scan_sscript:TF {
2420 \um_scan_sscript:
86
2421 }{
2422 \um_sub_or_super:n {\l_um_ss_chain_tl}
2423 }
2424 }
The main theme here is stolen from the source to the various \peek_ functions.
Consider this function as simply boilerplate:
2425 \cs_new:Npn \um_scan_sscript:TF #1#2 {
2426 \tl_set:Nx \l_peek_true_aux_tl { \exp_not:n{ #1 } }
2427 \tl_set_eq:NN \l_peek_true_tl \c_peek_true_remove_next_tl
2428 \tl_set:Nx \l_peek_false_tl {\exp_not:n{\group_align_safe_end: #2}}
2429 \group_align_safe_begin:
2430 \peek_after:NN \um_peek_execute_branches_ss:
2431 }
We do not skip spaces when scanning ahead, and we explicitly wish to bail out
on encountering a space or a brace.
2432 \cs_new:Npn \um_peek_execute_branches_ss: {
2433 \bool_if:nTF {
2434 \token_if_eq_catcode_p:NN \l_peek_token \c_group_begin_token ||
2435 \token_if_eq_catcode_p:NN \l_peek_token \c_group_end_token ||
2436 \token_if_eq_meaning_p:NN \l_peek_token \c_space_token
2437 }
2438 { \l_peek_false_tl }
2439 { \um_peek_execute_branches_ss_aux: }
2440 }
This is the actual comparison code. Because the peeking has already tokenised
the next token, its too late to extract its charcode directly. Instead, we look at its
meaning, which remains a character even though it is itself math-active. If the
character is ever made fully active, this will break our assumptions!
If the chars meaning exists as a property list key, we build up a chain of sub-
/superscripts and iterate. (If not, exit and typeset what weve already collected.)
2441 \cs_new:Npn \um_peek_execute_branches_ss_aux: {
2442 \prop_if_in:cxTF
2443 {g_um_\l_um_tmpa_tl _prop}
2444 {\meaning\l_peek_token}
2445 {
2446 \prop_get:cxN
2447 {g_um_\l_um_tmpa_tl _prop}
2448 {\meaning\l_peek_token}
2449 \l_um_tmpb_tl
2450 \tl_put_right:NV \l_um_ss_chain_tl \l_um_tmpb_tl
2451 \l_peek_true_tl
2452 }
2453 {\l_peek_false_tl}
2454 }
87
12.3.1 Active fractions
Active fractions can be setup independently of any maths font denition; all it
requires is a mapping from the Unicode input chars to the relevant L
A
T
E
X fraction
declaration.
2455 \cs_new:Npn \um_define_active_frac:Nw #1 #2/#3 {
2456 \char_make_active:N #1
2457 \char_gmake_mathactive:N #1
2458 \tl_rescan:nn {
2459 \catcode`\_=11\relax
2460 \catcode`\:=11\relax
2461 }{
2462 \cs_gset:Npx #1 {
2463 \bool_if:NTF \l_um_smallfrac_bool {\exp_not:N\tfrac} {\exp_not:N\frac}
2464 {#2} {#3}
2465 }
2466 }
2467 }
These are redened for each math font selection in case the active-frac feature
changes.
2468 \cs_new:Npn \um_setup_active_frac: {
2469 \group_begin:
2470 \um_define_active_frac:Nw ^^^^2189 0/3
2471 \um_define_active_frac:Nw ^^^^2152 1/{10}
2472 \um_define_active_frac:Nw ^^^^2151 1/9
2473 \um_define_active_frac:Nw ^^^^215b 1/8
2474 \um_define_active_frac:Nw ^^^^2150 1/7
2475 \um_define_active_frac:Nw ^^^^2159 1/6
2476 \um_define_active_frac:Nw ^^^^2155 1/5
2477 \um_define_active_frac:Nw ^^^^00bc 1/4
2478 \um_define_active_frac:Nw ^^^^2153 1/3
2479 \um_define_active_frac:Nw ^^^^215c 3/8
2480 \um_define_active_frac:Nw ^^^^2156 2/5
2481 \um_define_active_frac:Nw ^^^^00bd 1/2
2482 \um_define_active_frac:Nw ^^^^2157 3/5
2483 \um_define_active_frac:Nw ^^^^215d 5/8
2484 \um_define_active_frac:Nw ^^^^2154 2/3
2485 \um_define_active_frac:Nw ^^^^00be 3/4
2486 \um_define_active_frac:Nw ^^^^2158 4/5
2487 \um_define_active_frac:Nw ^^^^215a 5/6
2488 \um_define_active_frac:Nw ^^^^215e 7/8
2489 \group_end:
2490 }
2491 \um_setup_active_frac:
88
12.4 Synonyms and all the rest
These are symbols with multiple names. Eventually to be taken care of automati-
cally by the maths characters database.
2492 \def\to{\rightarrow}
2493 \def\overrightarrow{\vec}
2494 \def\le{\leq}
2495 \def\ge{\geq}
2496 \def\neq{\ne}
2497 \def\triangle{\mathord{\bigtriangleup}}
2498 \def\bigcirc{\mdlgwhtcircle}
2499 \def\circ{\vysmwhtcircle}
2500 \def\bullet{\smblkcircle}
2501 \def\mathyen{\yen}
2502 \def\mathsterling{\sterling}
2503 \def\diamond{\smwhtdiamond}
2504 \def\emptyset{\varnothing}
2505 \def\hbar{\hslash}
These are somewhat odd: (and their usual Unicode uprightness does not match
their amssymb glyphs)
2506 \def\backepsilon{\upbackepsilon}
2507 \def\eth{\matheth}
Due to the magic of OpenType math, big operators are automatically enlarged
when necessary. Since there isnt a separate unicode glyph for small integral, Im
not sure if there is a better way to do this:
2508 \def\smallint{{\textstyle\int}\limits}
\colon Dene \colon as a mathpunct :. This is wrong: it shouldbe u+003A coloninstead!
We hope no-one will notice.
2509 \@ifpackageloaded{amsmath}{
2510 % define their own colon, perhaps I should just steal it. (It does look much bet-
ter.)
2511 }{
2512 \cs_set_protected:Npn \colon {
2513 \bool_if:NTF \g_um_literal_colon_bool {:} { \mathpunct{:} }
2514 }
2515 }
\mathrm
2516 \def\mathrm{\mathup}
2517 \let\mathfence\mathord
\overline is a LuaT
E
X primitive:
2518 \luatex_if_engine:T {
2519 \cs_set:Npn \overline {\pdfprimitive\overline}
2520 }
89
\digamma
\Digamma
I might end up just changing these in the table.
2521 \def\digamma{\updigamma}
2522 \def\Digamma{\upDigamma}
12.5 Compatibility
We need to change L
A
T
E
Xs idea of the font used to typeset things like \sin and
\cos:
2523 \def\operator@font{\um_switchto_mathup:}
\um_patch_pkg:nn #1 : package
#2 : code
If package is loaded either already or later in the preamble, code is executed
(after the package is loaded in the latter case).
2524 \cs_new:Npn \um_patch_pkg:nn #1#2 {
2525 \@ifpackageloaded {#1} {
2526 #2
2527 }{
2528 \um_after_pkg:nn {#1} {#2}
2529 }
2530 }
url Simply need to get url in a state such that when it switches to math mode
and enters ascii characters, the maths setup (i.e., unicode-math) doesnt remap the
symbols into Plane 1. Which is, of course, what \mathup is doing.
This is the same as writing, e.g., \def\UrlFont{\ttfamily\um_switchto_mathup:}
but activates automatically so old documents that might change the \url font still
work correctly.
2531 \um_patch_pkg:nn {url} {
2532 \tl_put_left:Nn \Url@FormatString { \um_switchto_mathup: }
2533 \tl_put_right:Nn \UrlSpecials {
2534 \do\`{\mathchar`\`}
2535 \do\'{\mathchar`\'}
2536 \do\${\mathchar`\$}
2537 \do\&{\mathchar`\&}
2538 }
2539 }
amsmath Since the mathcode of `\- is greater than eight bits, this piece of
\AtBeginDocument code from amsmath dies if we try and set the maths font in
the preamble:
2540 \um_patch_pkg:nn {amsmath} {
2541 \tl_remove_in:Nn \@begindocumenthook {
90
2542 \mathchardef\std@minus\mathcode`\-\relax
2543 \mathchardef\std@equal\mathcode`\=\relax
2544 }
2545 \def\std@minus{\Umathcharnum\Umathcodenum`\-\relax}
2546 \def\std@equal{\Umathcharnum\Umathcodenum`\=\relax}
2547 \cs_set:Npn \@cdots {\mathinner{\cdots}}
2548 \cs_set_eq:NN \dotsb@ \cdots
This isnt as clever as the amsmath denition but I think it works:
2549 \def \resetMathstrut@ {%
2550 \setbox\z@\hbox{$($}%
2551 \ht\Mathstrutbox@\ht\z@ \dp\Mathstrutbox@\dp\z@
2552 }
2553 }
amsopn This code is to improve the output of analphabetic symbols in text of
operator names (\sin, \cos, etc.). Just comment out the oending lines for now:
2554 \um_patch_pkg:nn {amsopn} {
2555 \cs_set:Npn \newmcodes@ {
2556 \mathcode`\'39\scan_stop:
2557 \mathcode`\*42\scan_stop:
2558 \mathcode`\."613A\scan_stop:
2559 %% \ifnum\mathcode`\-=45 \else
2560 %% \mathchardef\std@minus\mathcode`\-\relax
2561 %% \fi
2562 \mathcode`\-45\scan_stop:
2563 \mathcode`\/47\scan_stop:
2564 \mathcode`\:"603A\scan_stop:
2565 }
2566 }
Symbols
2567 \cs_set:Npn \| {\Vert}
\mathinner items:
2568 \cs_set:Npn \mathellipsis {\mathinner{\unicodeellipsis}}
2569 \cs_set:Npn \cdots {\mathinner{\unicodecdots}}
Accents
2570 \AtBeginDocument{
2571 \def\widehat{\hat}
2572 \def\widetilde{\tilde}
2573 }
91
2574 \cs_set_eq:NN \um_text_slash: \slash
2575 \cs_set:Npn \slash {
2576 \mode_if_math:TF {\mathslash} {\um_text_slash:}
2577 }
beamer At end of the package so the warnings are dened.
2578 \AtEndOfPackage{
2579 \@ifclassloaded{beamer}{
2580 \ifbeamer@suppressreplacements\else
2581 \um_warning:n {disable-beamer}
2582 \beamer@suppressreplacementstrue
2583 \fi
2584 }{}
2585 }
2586 \ExplSyntaxOff
2587
13 Error messages
These are dened at the beginning of the package, but we leave their denition
until now in the source to keep them out of the way.
2588 *
Wrapper functions:
2589 \cs_new:Npn \um_warning:n { \msg_warning:nn {unicode-math} }
2590 \cs_new:Npn \um_trace:n { \msg_trace:nn {unicode-math} }
2591 \cs_new:Npn \um_trace:nx { \msg_trace:nnx {unicode-math} }
2592 \msg_new:nnn {unicode-math} {maths-feature-only}
2593 {
2594 The~ '#1'~ font~ feature~ can~ only~ be~ used~ for~ maths~ fonts.
2595 }
2596 \msg_new:nnn {unicode-math} {disable-beamer}
2597 {
2598 Disabling~ beamer's~ math~ setup.\\
2599 Please~ load~ beamer~ with~ the~ [professionalfonts]~ class~ option.
2600 }
2601 \msg_new:nnn {unicode-math} {no-tfrac}
2602 {
2603 Small~ fraction~ command~ \protect\tfrac\ not~ defined.\\
2604 Load~ amsmath~ or~ define~ it~ manually~ before~ loading~ unicode-math.
2605 }
2606 \msg_new:nnn {unicode-math} {default-math-font}
2607 {
92
2608 Defining~ the~ default~ maths~ font~ as~ '#1'.
2609 }
2610 \msg_new:nnn {unicode-math} {setup-implicit}
2611 {
2612 Setup~ alphabets:~ implicit~ mode.
2613 }
2614 \msg_new:nnn {unicode-math} {setup-explicit}
2615 {
2616 Setup~ alphabets:~ explicit~ mode.
2617 }
2618 \msg_new:nnn {unicode-math} {alph-initialise}
2619 {
2620 Initialising~ \@backslashchar math#1.
2621 }
2622 \msg_new:nnn {unicode-math} {setup-alph}
2623 {
2624 Setup~ alphabet:~ #1.
2625 }
2626
The end.
14 stix table data extraction
The source for the T
E
X names for the very large number of mathematical glyphs
are provided via Barbara Beetons table le for the stix project (ams.org/STIX).
A version is located at http://www.ams.org/STIX/bnb/stix-tbl.asc but check
http://www.ams.org/STIX/ for more up-to-date info.
This table is converted into a form suitable for reading by X
T
E
X. A single le
is produced containing all (more than 3298) symbols. Future optimisations might
include generating various (possibly overlapping) subsets so not all denitions
must be readjust to redene a small range of symbols. Performance for nowseems
to be acceptable without such measures.
This le is currently developed outside this DTX le. It will be incorporated
when the nal version is ready. (I know this is not how things are supposed to
work!)
2627 < See stix-extract.sh for now. >
A Documenting maths support in the NFSS
In the following, NFSS decl. stands for something like {T1}{lmr}{m}{n}.
Maths symbol fonts Fonts for symbols: , ,
93
\DeclareSymbolFont{name}NFSS decl.
Declares a named maths font such as operators from which symbols are
dened with \DeclareMathSymbol.
Maths alphabet fonts Fonts for ABC xyz, ABC XYZ, etc.
\DeclareMathAlphabet{cmd}NFSS decl.
For commands such as \mathbf, accessed through maths mode that are un-
aected by the current text font, and which are used for alphabetic symbols
in the ascii range.
\DeclareSymbolFontAlphabet{cmd}{name}
Alternative (and optimisation) for \DeclareMathAlphabet if a single font is
being used for both alphabetic characters (as above) and symbols.
Maths versions Dierent maths weights can be dened with the following,
switched in text with the \mathversion{maths version} command.
\SetSymbolFont{name}{maths version}NFSS decl.
\SetMathAlphabet{cmd}{maths version}NFSS decl.
Maths symbols Symbol denitions in maths for both characters (=) and macros
(\eqdef): \DeclareMathSymbol{symbol}{type}{named font}{slot} This
is the macro that actually denes which font each symbol comes from and
how they behave.
Delimiters and radicals use wrappers around T
E
Xs \delimiter/\radical primi-
tives, which are re-designed in X
T
E
X. The syntax used in L
A
T
E
Xs NFSS is therefore
not so relevant here.
Delimiters A special class of maths symbol which enlarge themselves in certain
contexts.
\DeclareMathDelimiter{symbol}{type}{sym. font}{slot}{sym. font}{slot}
Radicals Similar to delimiters (\DeclareMathRadical takes the same syntax) but
behave weirdly. \sqrt might very well be the only one.
In those cases, glyph slots in two symbol fonts are required; one for the small
(regular) case, the other for situations when the glyph is larger. This is not the
case in X
T
E
X.
Accents are not included yet.
Summary For symbols, something like:
\def\DeclareMathSymbol#1#2#3#4{
\global\mathchardef#1"\mathchar@type#2
\expandafter\hexnumber@\csname sym#2\endcsname
{\hexnumber@{\count\z@}\hexnumber@{\count\tw@}}}
94
For characters, something like:
\def\DeclareMathSymbol#1#2#3#4{
\global\mathcode`#1"\mathchar@type#2
\expandafter\hexnumber@\csname sym#2\endcsname
{\hexnumber@{\count\z@}\hexnumber@{\count\tw@}}}
B X
T
E
X math font dimensions
These are the extended \fontdimens available for suitable fonts in X
T
E
X. Note
that LuaT
E
X takes an alternative route, and this package will eventually provide
a wrapper interface to the two (I hope).
\fontdimen Dimension name Description
10 ScriptPercentScaleDown Percentage of scaling down for script level 1.
Suggested value: 80%.
11 ScriptScriptPercentScale-
Down
Percentage of scaling down for script level 2
(ScriptScript). Suggested value: 60%.
12 DelimitedSubFormulaMin-
Height
Minimum height required for a delimited
expression to be treated as a subformula.
Suggested value: normal line height 1.5.
13 DisplayOperatorMinHeight Minimum height of n-ary operators (such as
integral and summation) for formulas in
display mode.
14 MathLeading White space to be left between math
formulas to ensure proper line spacing. For
example, for applications that treat line gap
as a part of line ascender, formulas with ink
going above (os2.sTypoAscender +
os2.sTypoLineGap MathLeading) or with
ink going below os2.sTypoDescender will
result in increasing line height.
15 AxisHeight Axis height of the font.
16 AccentBaseHeight Maximum (ink) height of accent base that
does not require raising the accents.
Suggested: x-height of the font
(os2.sxHeight) plus any possible overshots.
95
\fontdimen Dimension name Description
17 FlattenedAccentBaseHeight Maximum (ink) height of accent base that
does not require attening the accents.
Suggested: cap height of the font
(os2.sCapHeight).
18 SubscriptShiftDown The standard shift down applied to subscript
elements. Positive for moving in the
downward direction. Suggested:
os2.ySubscriptYOset.
19 SubscriptTopMax Maximum allowed height of the (ink) top of
subscripts that does not require moving
subscripts further down. Suggested: /5
x-height.
20 SubscriptBaselineDropMin Minimum allowed drop of the baseline of
subscripts relative to the (ink) bottom of the
base. Checked for bases that are treated as a
box or extended shape. Positive for subscript
baseline dropped below the base bottom.
21 SuperscriptShiftUp Standard shift up applied to superscript
elements. Suggested:
os2.ySuperscriptYOset.
22 SuperscriptShiftUpCramped Standard shift of superscripts relative to the
base, in cramped style.
23 SuperscriptBottomMin Minimum allowed height of the (ink) bottom
of superscripts that does not require moving
subscripts further up. Suggested: x-height.
24 SuperscriptBaselineDrop-
Max
Maximum allowed drop of the baseline of
superscripts relative to the (ink) top of the
base. Checked for bases that are treated as a
box or extended shape. Positive for
superscript baseline below the base top.
25 SubSuperscriptGapMin Minimum gap between the superscript and
subscript ink. Suggested: 4default rule
thickness.
26 SuperscriptBottomMax-
WithSubscript
The maximum level to which the (ink)
bottom of superscript can be pushed to
increase the gap between superscript and
subscript, before subscript starts being
moved down. Suggested: /5 x-height.
96
\fontdimen Dimension name Description
27 SpaceAfterScript Extra white space to be added after each
subscript and superscript. Suggested: 0.5pt
for a 12 pt font.
28 UpperLimitGapMin Minimum gap between the (ink) bottom of
the upper limit, and the (ink) top of the base
operator.
29 UpperLimitBaselineRiseMin Minimum distance between baseline of
upper limit and (ink) top of the base
operator.
30 LowerLimitGapMin Minimum gap between (ink) top of the lower
limit, and (ink) bottom of the base operator.
31 LowerLimitBaselineDrop-
Min
Minimum distance between baseline of the
lower limit and (ink) bottom of the base
operator.
32 StackTopShiftUp Standard shift up applied to the top element
of a stack.
33 StackTopDisplayStyleShift-
Up
Standard shift up applied to the top element
of a stack in display style.
34 StackBottomShiftDown Standard shift down applied to the bottom
element of a stack. Positive for moving in the
downward direction.
35 StackBottomDisplayStyle-
ShiftDown
Standard shift down applied to the bottom
element of a stack in display style. Positive
for moving in the downward direction.
36 StackGapMin Minimum gap between (ink) bottom of the
top element of a stack, and the (ink) top of
the bottom element. Suggested: 3default
rule thickness.
37 StackDisplayStyleGapMin Minimum gap between (ink) bottom of the
top element of a stack, and the (ink) top of
the bottom element in display style.
Suggested: 7default rule thickness.
38 StretchStackTopShiftUp Standard shift up applied to the top element
of the stretch stack.
39 StretchStackBottomShift-
Down
Standard shift down applied to the bottom
element of the stretch stack. Positive for
moving in the downward direction.
97
\fontdimen Dimension name Description
40 StretchStackGapAboveMin Minimum gap between the ink of the
stretched element, and the (ink) bottom of
the element above. Suggested:
UpperLimitGapMin
41 StretchStackGapBelowMin Minimum gap between the ink of the
stretched element, and the (ink) top of the
element below. Suggested:
LowerLimitGapMin.
42 FractionNumeratorShiftUp Standard shift up applied to the numerator.
43 FractionNumeratorDisplay-
StyleShiftUp
Standard shift up applied to the numerator
in display style. Suggested:
StackTopDisplayStyleShiftUp.
44 FractionDenominatorShift-
Down
Standard shift down applied to the
denominator. Positive for moving in the
downward direction.
45 FractionDenominator-
DisplayStyleShiftDown
Standard shift down applied to the
denominator in display style. Positive for
moving in the downward direction.
Suggested:
StackBottomDisplayStyleShiftDown.
46 FractionNumeratorGapMin Minimum tolerated gap between the (ink)
bottom of the numerator and the ink of the
fraction bar. Suggested: default rule
thickness
47 FractionNumDisplayStyle-
GapMin
Minimum tolerated gap between the (ink)
bottom of the numerator and the ink of the
fraction bar in display style. Suggested:
3default rule thickness.
48 FractionRuleThickness Thickness of the fraction bar. Suggested:
default rule thickness.
49 FractionDenominatorGap-
Min
Minimum tolerated gap between the (ink)
top of the denominator and the ink of the
fraction bar. Suggested: default rule
thickness
50 FractionDenomDisplay-
StyleGapMin
Minimum tolerated gap between the (ink)
top of the denominator and the ink of the
fraction bar in display style. Suggested:
3default rule thickness.
98
\fontdimen Dimension name Description
51 SkewedFractionHorizontal-
Gap
Horizontal distance between the top and
bottom elements of a skewed fraction.
52 SkewedFractionVerticalGap Vertical distance between the ink of the top
and bottom elements of a skewed fraction.
53 OverbarVerticalGap Distance between the overbar and the (ink)
top of he base. Suggested: 3default rule
thickness.
54 OverbarRuleThickness Thickness of overbar. Suggested: default rule
thickness.
55 OverbarExtraAscender Extra white space reserved above the
overbar. Suggested: default rule thickness.
56 UnderbarVerticalGap Distance between underbar and (ink) bottom
of the base. Suggested: 3default rule
thickness.
57 UnderbarRuleThickness Thickness of underbar. Suggested: default
rule thickness.
58 UnderbarExtraDescender Extra white space reserved below the
underbar. Always positive. Suggested:
default rule thickness.
59 RadicalVerticalGap Space between the (ink) top of the expression
and the bar over it. Suggested: 1 default
rule thickness.
60 RadicalDisplayStyleVertical-
Gap
Space between the (ink) top of the expression
and the bar over it. Suggested: default rule
thickness + x-height.
61 RadicalRuleThickness Thickness of the radical rule. This is the
thickness of the rule in designed or
constructed radical signs. Suggested: default
rule thickness.
62 RadicalExtraAscender Extra white space reserved above the radical.
Suggested: RadicalRuleThickness.
63 RadicalKernBeforeDegree Extra horizontal kern before the degree of a
radical, if such is present. Suggested: 5/18 of
em.
64 RadicalKernAfterDegree Negative kern after the degree of a radical, if
such is present. Suggested: 10/18 of em.
99
\fontdimen Dimension name Description
65 RadicalDegreeBottomRaise-
Percent
Height of the bottom of the radical degree, if
such is present, in proportion to the ascender
of the radical sign. Suggested: 60%.
100