Skip to content

Commit

Permalink
feat(math): Support MathML movablelimits attribute
Browse files Browse the repository at this point in the history
  • Loading branch information
Omikhleia authored and Didier Willis committed Nov 9, 2024
1 parent 626fe7a commit 92c678a
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 31 deletions.
10 changes: 3 additions & 7 deletions packages/math/base-elements.lua
Original file line number Diff line number Diff line change
Expand Up @@ -718,17 +718,13 @@ function elements.underOver:_stretchyReshapeToBase (part)
end

function elements.underOver:shape ()
local isBaseLargeOp = SU.boolean(self.base and self.base.largeop, false)
if not (self.mode == mathMode.display or self.mode == mathMode.displayCramped) and isBaseLargeOp then
-- FIXME
-- Added the "largeop" condition, but it's kind of a workaround:
-- It should rather be the "moveablelimits" property in MathML, but we do not have that yet.
-- When the base is a moveable limit, the under/over scripts are not placed under/over the base,
local isMovableLimits = SU.boolean(self.base and self.base.movablelimits, false)
if not (self.mode == mathMode.display or self.mode == mathMode.displayCramped) and isMovableLimits then
-- When the base is a movable limit, the under/over scripts are not placed under/over the base,
-- but other to the right of it, when display mode is not used.
-- Notable effects:
-- Mozilla MathML test 19 (on "k times" > overbrace > base)
-- Maxwell's Equations in MathML3 Test Suite "complex1" (on the vectors in fractions)
-- For now, go with the "largeop" property, but this is not correct.
self.isUnderOver = true
elements.subscript.shape(self)
return
Expand Down
41 changes: 25 additions & 16 deletions packages/math/texlike.lua
Original file line number Diff line number Diff line change
Expand Up @@ -280,8 +280,17 @@ local function isOperatorKind (tree, typeOfAtom, typeOfSymbol)
return false
end

local function isBigOperator (tree)
return isOperatorKind(tree, "big", atomType.bigOperator)
local function isMoveableLimits (tree)
if tree.command ~= "mo" then
return false
end
if tree.options and SU.boolean(tree.options.movablelimits, false) then
return true
end
if tree[1] and symbolDefaults[tree[1]] and SU.boolean(symbolDefaults[tree[1]].movablelimits, false) then
return true
end
return false
end
local function isCloseOperator (tree)
return isOperatorKind(tree, "close", atomType.closeSymbol)
Expand Down Expand Up @@ -414,13 +423,13 @@ local function compileToMathML_aux (_, arg_env, tree)
tree.options = {}
-- Translate TeX-like sub/superscripts to `munderover` or `msubsup`,
-- depending on whether the base is a big operator
elseif tree.id == "sup" and isBigOperator(tree[1]) then
elseif tree.id == "sup" and isMoveableLimits(tree[1]) then
tree.command = "mover"
elseif tree.id == "sub" and isBigOperator(tree[1]) then
elseif tree.id == "sub" and isMoveableLimits(tree[1]) then
tree.command = "munder"
elseif tree.id == "subsup" and isBigOperator(tree[1]) then
elseif tree.id == "subsup" and isMoveableLimits(tree[1]) then
tree.command = "munderover"
elseif tree.id == "supsub" and isBigOperator(tree[1]) then
elseif tree.id == "supsub" and isMoveableLimits(tree[1]) then
tree.command = "munderover"
local tmp = tree[2]
tree[2] = tree[3]
Expand Down Expand Up @@ -589,20 +598,20 @@ compileToMathML(
\def{bi}{\mi[mathvariant=bold-italic]{#1}}
\def{dsi}{\mi[mathvariant=double-struck]{#1}}
\def{lim}{\mo[atom=big]{lim}}
\def{lim}{\mo[movablelimits=true]{lim}}
% From amsmath:
\def{to}{\mo[atom=bin]{→}}
\def{gcd}{\mo[atom=big]{gcd}}
\def{sup}{\mo[atom=big]{sup}}
\def{inf}{\mo[atom=big]{inf}}
\def{max}{\mo[atom=big]{max}}
\def{min}{\mo[atom=big]{min}}
\def{gcd}{\mo[movablelimits=true]{gcd}}
\def{sup}{\mo[movablelimits=true]{sup}}
\def{inf}{\mo[movablelimits=true]{inf}}
\def{max}{\mo[movablelimits=true]{max}}
\def{min}{\mo[movablelimits=true]{min}}
% Those use U+202F NARROW NO-BREAK SPACE in their names
\def{limsup}{\mo[atom=big]{lim sup}}
\def{liminf}{\mo[atom=big]{lim inf}}
\def{projlim}{\mo[atom=big]{proj lim}}
\def{injlim}{\mo[atom=big]{inj lim}}
\def{limsup}{\mo[movablelimits=true]{lim sup}}
\def{liminf}{\mo[movablelimits=true]{lim inf}}
\def{projlim}{\mo[movablelimits=true]{proj lim}}
\def{injlim}{\mo[movablelimits=true]{inj lim}}
% Standard spaces gleaned from plain TeX
\def{thinspace}{\mspace[width=thin]}
Expand Down
16 changes: 8 additions & 8 deletions packages/math/unicode-symbols.lua
Original file line number Diff line number Diff line change
Expand Up @@ -2628,14 +2628,14 @@ symbolDefaults["="] = { atom = atomType.relationalOperator }
symbolDefaults[""] = { atom = atomType.relationalOperator }
symbolDefaults[""] = { atom = atomType.relationalOperator }
symbolDefaults[""] = { atom = atomType.relationalOperator }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { atom = atomType.bigOperator, largeop = true, movablelimits = true }
symbolDefaults[""] = { largeop = true }
symbolDefaults[""] = { largeop = true }
symbolDefaults[""] = { largeop = true }
Expand Down

0 comments on commit 92c678a

Please sign in to comment.