mathml/mlist_to_mml.lua

238 lines
7.3 KiB
Lua
Raw Normal View History

2021-04-18 15:19:52 +02:00
local noad_t, accent_t, style_t, choice_t = node.id'noad', node.id'accent', node.id'style', node.id'choice'
local radical_t, fraction_t, fence_t = node.id'radical', node.id'fraction', node.id'fence'
local math_char_t, sub_box_t, sub_mlist_t = node.id'math_char', node.id'sub_box', node.id'sub_mlist'
local noad_sub = node.subtypes'noad'
local radical_sub = node.subtypes'radical'
local fence_sub = node.subtypes'fence'
local nodes_to_table
2021-04-19 13:30:54 +02:00
local function sub_style(s) return s//4*2+5 end
local function sup_style(s) return s//4*2+4+s%2 end
2021-04-18 15:19:52 +02:00
-- We ignore large_... since they aren't used for modern fonts
local function delim_to_table(delim)
if not delim then return end
local fam = delim.small_fam
return {[0] = 'mo', utf8.char(delim.small_char), ['tex:family'] = fam ~= 0 and fam or nil }
end
2021-04-19 13:30:54 +02:00
local function kernel_to_table(kernel, cur_style)
2021-04-18 15:19:52 +02:00
if not kernel then return end
local id = kernel.id
if id == math_char_t then
local char = kernel.char
local elem = char >= 0x30 and char < 0x39 and 'mn' or 'mi'
local fam = kernel.fam
return {[0] = elem, utf8.char(kernel.char), ['tex:family'] = fam ~= 0 and fam or nil }
elseif id == sub_box_t then
return {[0] = 'mi', {[0] = 'mglyph', ['tex:box'] = kernel.list}}
elseif id == sub_mlist_t then
2021-04-19 13:30:54 +02:00
return nodes_to_table(kernel.list, cur_style)
2021-04-18 15:19:52 +02:00
else
error'confusion'
end
end
2021-04-19 13:30:54 +02:00
local function do_sub_sup(t, n, cur_style)
local sub = kernel_to_table(n.sub, sub_style(cur_style))
local sup = kernel_to_table(n.sup, sup_style(cur_style))
2021-04-18 15:19:52 +02:00
if sub then
if sup then
return {[0] = 'msubsup', t, sub, sup}
else
return {[0] = 'msub', t, sub}
end
elseif sup then
return {[0] = 'msup', t, sup}
else
return t
end
end
2021-04-19 13:30:54 +02:00
local function noad_to_table(noad, sub, cur_style)
2021-04-18 15:19:52 +02:00
local class = noad_sub[sub]
2021-04-19 13:30:54 +02:00
local nucleus = kernel_to_table(noad.nucleus, class == 'over' and cur_style//2*2+1 or cur_style)
2021-04-18 15:19:52 +02:00
if class == 'ord' then
2021-04-19 13:30:54 +02:00
elseif class == 'opdisplaylimits' or class == 'oplimits' or class == 'opnolimits' or class == 'bin' or class == 'rel' or class == 'open'
2021-04-18 15:19:52 +02:00
or class == 'close' or class == 'punct' or class == 'inner' then
if nucleus[0] == 'mrow' then
-- TODO
else
nucleus[0] = 'mo'
end
nucleus['tex:class'] = class
2021-04-19 13:30:54 +02:00
if (noad.sup or noad.sub) and (class == 'opdisplaylimits' or class == 'oplimits') then
nucleus.movablelimits = class == 'opdisplaylimits'
local sub = kernel_to_table(noad.sub, sub_style(cur_style))
local sup = kernel_to_table(noad.sup, sup_style(cur_style))
return {[0] = sup and (sub and 'munderover' or 'mover') or 'munder',
nucleus,
sub or sup,
sub and sup,
}
end
elseif class == 'under' then
return {[0] = 'munder',
nucleus,
{[0] = 'mo', '_',},
}
elseif class == 'over' then
return {[0] = 'mover',
nucleus,
{[0] = 'mo', '\u{203E}',},
}
elseif class == 'vcenter' then
2021-04-18 15:19:52 +02:00
nucleus['tex:TODO'] = class
2021-04-19 13:30:54 +02:00
else
error[[confusion]]
end
return do_sub_sup(nucleus, noad, cur_style)
end
local function accent_to_table(accent, sub, cur_style)
local nucleus = kernel_to_table(accent.nucleus, cur_style//2*2+1)
local top_acc = kernel_to_table(accent.accent, cur_style)
local bot_acc = kernel_to_table(accent.bot_accent, cur_style)
if top_acc then
top_acc[0] = 'mo'
if sub & 1 == 1 then
top_acc.stretchy = 'false'
end
2021-04-18 15:19:52 +02:00
end
2021-04-19 13:30:54 +02:00
if bot_acc then
bot_acc[0] = 'mo'
if sub & 2 == 2 then
bot_acc.stretchy = 'false'
end
end
return {[0] = top_acc and (bot_acc and 'munderover' or 'mover') or 'munder',
nucleus,
bot_acc or top_acc,
bot_acc and top_acc,
}
2021-04-18 15:19:52 +02:00
end
2021-04-19 13:30:54 +02:00
local style_table = {
display = {displaystyle = "true", scriptlevel = "0"},
text = {displaystyle = "false", scriptlevel = "0"},
script = {displaystyle = "false", scriptlevel = "1"},
scriptscript = {displaystyle = "false", scriptlevel = "2"},
}
style_table.crampeddisplay, style_table.crampedtext,
style_table.crampedscript, style_table.crampedscriptscript =
style_table.display, style_table.text,
style_table.script, style_table.scriptscript
local function radical_to_table(radical, sub, cur_style)
2021-04-18 15:19:52 +02:00
local kind = radical_sub[sub]
2021-04-19 13:30:54 +02:00
local nucleus = kernel_to_table(radical.nucleus, cur_style//2*2+1)
2021-04-18 15:19:52 +02:00
local left = delim_to_table(radical.left)
local elem
if kind == 'radical' or kind == 'uradical' then
-- FIXME: Check that this is really a square root
elem = {[0] = 'msqrt', nucleus}
elseif kind == 'uroot' then
-- FIXME: Check that this is really a root
elem = {[0] = 'msqrt', nucleus, delim_to_table(radical.degree)}
elseif kind == 'uunderdelimiter' then
elem = {[0] = 'munder', left, nucleus}
elseif kind == 'uoverdelimiter' then
elem = {[0] = 'mover', left, nucleus}
elseif kind == 'udelimiterunder' then
elem = {[0] = 'munder', nucleus, left}
elseif kind == 'udelimiterover' then
elem = {[0] = 'mover', nucleus, left}
else
error[[confusion]]
end
2021-04-19 13:30:54 +02:00
return do_sub_sup(elem, radical, cur_style)
2021-04-18 15:19:52 +02:00
end
2021-04-19 13:30:54 +02:00
local function fraction_to_table(fraction, sub, cur_style)
local num = kernel_to_table(fraction.num, sup_style(cur_style))
local denom = kernel_to_table(fraction.denom, sub_style(cur_style))
2021-04-18 15:19:52 +02:00
local left = delim_to_table(fraction.left)
local right = delim_to_table(fraction.right)
local mfrac = {[0] = 'mfrac',
linethickness = fraction.width and fraction.width == 0 and 0 or nil,
bevelled = fraction.middle and "true" or nil,
num,
denom,
}
if left then
return {[0] = 'mrow',
left,
mfrac,
right, -- might be nil
}
elseif right then
return {[0] = 'mrow',
mfrac,
right,
}
else
return mfrac
end
end
2021-04-19 13:30:54 +02:00
local function fence_to_table(fence, sub, cur_style)
local delim = delim_to_table(fence.delimiter)
delim.stretchy = 'true'
delim.fence = 'true'
return delim
2021-04-18 15:19:52 +02:00
end
2021-04-19 13:30:54 +02:00
function nodes_to_table(head, cur_style)
2021-04-18 15:19:52 +02:00
local t = {[0] = "mrow"}
2021-04-19 13:30:54 +02:00
local result = t
2021-04-18 15:19:52 +02:00
for n, id, sub in node.traverse(head) do
if id == noad_t then
2021-04-19 13:30:54 +02:00
t[#t+1] = noad_to_table(n, sub, cur_style)
2021-04-18 15:19:52 +02:00
elseif id == accent_t then
2021-04-19 13:30:54 +02:00
t[#t+1] = accent_to_table(n, sub, cur_style)
2021-04-18 15:19:52 +02:00
elseif id == style_t then
2021-04-19 13:30:54 +02:00
if #t ~= 0 then
local new_t = {[0] = 'mstyle'}
t[#t+1] = new_t
t = new_t
end
if sub < 2 then
t.displaystyle, t.scriptlevel = true, 0
else
t.displaystyle, t.scriptlevel = false, sub//2 - 1
end
cur_style = sub
2021-04-18 15:19:52 +02:00
elseif id == choice_t then
2021-04-19 13:30:54 +02:00
local size = cur_style//2
t[#t+1] = nodes_to_table(n[size == 0 and 'display' or size == 1 and 'text'
or size == 2 and 'script'
or size == 3 and 'scriptscript' or assert(false)], 2*size)
2021-04-18 15:19:52 +02:00
elseif id == radical_t then
2021-04-19 13:30:54 +02:00
t[#t+1] = radical_to_table(n, sub, cur_style)
2021-04-18 15:19:52 +02:00
elseif id == fraction_t then
2021-04-19 13:30:54 +02:00
t[#t+1] = fraction_to_table(n, sub, cur_style)
2021-04-18 15:19:52 +02:00
elseif id == fence_t then
2021-04-19 13:30:54 +02:00
t[#t+1] = fence_to_table(n, sub, cur_style)
2021-04-18 15:19:52 +02:00
else
2021-04-19 13:30:54 +02:00
t[#t+1] = {[0] = 'tex:TODO', other = n}
2021-04-18 15:19:52 +02:00
end
end
2021-04-19 13:30:54 +02:00
return result
2021-04-18 15:19:52 +02:00
end
2021-04-19 13:30:54 +02:00
return function(head, style)
local result = nodes_to_table(head, style or 0)
2021-04-18 15:19:52 +02:00
result[0] = 'math'
result.xmlns = 'http://www.w3.org/1998/Math/MathML'
result['xmlns:tex'] = 'http://typesetting.eu/2021/LuaMathML'
2021-04-19 13:30:54 +02:00
if style == 2 then
result.display = 'block'
end
2021-04-18 15:19:52 +02:00
return result
end