our
@EXPORT
= (
qw( &DefMathML )
,
qw( &pmml &pmml_scriptsize &pmml_smaller
&pmml_mi &pmml_mo &pmml_mn &pmml_bigop
&pmml_punctuate &pmml_parenthesize
&pmml_infix &pmml_script &pmml_summation)
,
qw( &cmml &cmml_share &cmml_shared &cmml_leaf
&cmml_or_compose &cmml_synth_not &cmml_synth_complement)
,
);
sub
preprocess {
my
(
$self
,
$doc
,
@nodes
) =
@_
;
$$self
{hackplane1} = 0
unless
$$self
{hackplane1};
$$self
{plane1} = 1
if
$$self
{hackplane1} || !
defined
$$self
{plane1};
$$self
{nestmath} = 0
unless
$$self
{nestmath};
$doc
->adjust_latexml_doctype(
'MathML'
);
$doc
->addNamespace(
$mmlURI
,
'm'
);
return
; }
sub
outerWrapper {
my
(
$self
,
$doc
,
$xmath
,
$mml
) =
@_
;
my
$math
=
$xmath
->parentNode;
my
$mode
=
$math
->getAttribute(
'mode'
) ||
'inline'
;
my
@img
= ();
if
(
my
$src
=
$math
->getAttribute(
'imagesrc'
)) {
my
$depth
=
$math
->getAttribute(
'imagedepth'
);
@img
= (
altimg
=>
$src
,
'altimg-width'
=>
$math
->getAttribute(
'imagewidth'
) .
'px'
,
'altimg-height'
=>
$math
->getAttribute(
'imageheight'
) .
'px'
,
'altimg-valign'
=> (
$depth
? -
$depth
.
'px'
:
undef
)); }
my
@rdfa
=
map
{
my
$val
= (
$math
->getAttribute(
$_
) ||
$xmath
->getAttribute(
$_
));
$val
? (
$_
=>
$val
) : () }
qw(about resource property rel rev typeof datatype content)
;
my
$wrapped
= [
'm:math'
, {
display
=> (
$mode
eq
'display'
?
'block'
:
'inline'
),
class
=>
$math
->getAttribute(
'class'
),
alttext
=>
$math
->getAttribute(
'tex'
),
@rdfa
,
@img
},
$mml
];
$self
->associateNode(
$wrapped
,
$xmath
, 1);
return
$wrapped
; }
our
%ENCODINGS
= (
'application/mathml-presentation+xml'
=>
'MathML-Presentation'
,
'application/mathml-content+xml'
=>
'MathML-Content'
,
'image/svg+xml'
=>
'SVG1.1'
,
);
sub
rawIDSuffix {
return
'.msvg'
; }
sub
combineParallel {
my
(
$self
,
$doc
,
$xmath
,
$primary
,
@secondaries
) =
@_
;
my
$id
=
$xmath
->getAttribute(
'fragid'
);
my
@alt
= ();
foreach
my
$secondary
(
@secondaries
) {
my
$mimetype
=
$$secondary
{mimetype} ||
'unknown'
;
my
$encoding
=
$ENCODINGS
{
$mimetype
} ||
$mimetype
;
if
(
$mimetype
=~ /^application\/mathml/) {
push
(
@alt
, [
'm:annotation-xml'
, {
encoding
=>
$encoding
},
$$secondary
{xml}]); }
elsif
(
my
$xml
=
$$secondary
{xml}) {
push
(
@alt
, [
'm:annotation-xml'
, {
encoding
=>
$encoding
},
$$secondary
{processor}->outerWrapper(
$doc
,
$xmath
,
$xml
)]); }
elsif
(
my
$src
=
$$secondary
{src}) {
push
(
@alt
, [
'm:annotation'
, {
encoding
=>
$encoding
,
src
=>
$src
}]); }
elsif
(
my
$string
=
$$secondary
{string}) {
push
(
@alt
, [
'm:annotation'
, {
encoding
=>
$encoding
},
$string
]); }
}
return
{
processor
=>
$self
,
mimetype
=>
$$primary
{mimetype},
xml
=> (
@alt
? [
'm:semantics'
, {},
$$primary
{xml},
@alt
] :
$$primary
{xml}) }; }
sub
getQName {
my
(
$node
) =
@_
;
return
$LaTeXML::Post::DOCUMENT
->getQName(
$node
); }
sub
addCrossref {
my
(
$self
,
$node
,
$id
) =
@_
;
$node
->setAttribute(
xref
=>
$id
);
return
; }
sub
realize {
my
(
$node
,
$branch
) =
@_
;
return
(
ref
$node
) ?
$LaTeXML::Post::DOCUMENT
->realizeXMNode(
$node
,
$branch
) :
$node
; }
my
%EMBELLISHING_ROLE
= (
SUPERSCRIPTOP
=> 1,
SUBSCRIPTOP
=> 1,
OVERACCENT
=> 1,
UNDERACCENT
=> 1,
MODIFIER
=> 1,
MODIFIEROP
=> 1);
sub
getOperatorRole {
my
(
$node
) =
@_
;
if
(!
$node
) {
return
; }
elsif
(
my
$role
=
$node
->getAttribute(
'role'
)) {
return
$role
; }
elsif
(getQName(
$node
) eq
'ltx:XMApp'
) {
my
(
$op
,
$base
) = element_nodes(
$node
);
return
(
$EMBELLISHING_ROLE
{
$op
->getAttribute(
'role'
) ||
''
}
? getOperatorRole(
$base
)
:
undef
); } }
our
$MMLTable_P
= {};
our
$MMLTable_C
= {};
sub
DefMathML {
my
(
$key
,
$presentation
,
$content
) =
@_
;
$$MMLTable_P
{
$key
} =
$presentation
if
$presentation
;
$$MMLTable_C
{
$key
} =
$content
if
$content
;
return
; }
sub
lookupPresenter {
my
(
$mode
,
$role
,
$name
) =
@_
;
$name
=
'?'
unless
$name
;
$role
=
'?'
unless
$role
;
return
$$MMLTable_P
{
"$mode:$role:$name"
} ||
$$MMLTable_P
{
"$mode:?:$name"
}
||
$$MMLTable_P
{
"$mode:$role:?"
} ||
$$MMLTable_P
{
"$mode:?:?"
}; }
sub
lookupContent {
my
(
$mode
,
$role
,
$name
) =
@_
;
return
$name
? ((
$role
&&
$$MMLTable_C
{
"$mode:$role:$name"
}) ||
$$MMLTable_C
{
"$mode:?:$name"
} ||
$$MMLTable_C
{
"$mode:?:?"
}) : (
(
$role
&&
$$MMLTable_C
{
"$mode:$role:?"
}) ||
$$MMLTable_C
{
"$mode:?:?"
}); }
my
%stylestep
= (
display
=>
'text'
,
text
=>
'script'
,
script
=>
'scriptscript'
,
scriptscript
=>
'scriptscript'
);
my
%stylesize
= (
display
=>
'100%'
,
text
=>
'100%'
,
script
=>
'70%'
,
scriptscript
=>
'50%'
);
my
%style_script_step
= (
display
=>
'script'
,
text
=>
'script'
,
script
=>
'scriptscript'
,
scriptscript
=>
'scriptscript'
);
my
%stylemap
= (
display
=> {
text
=> {
displaystyle
=>
'false'
},
script
=> {
displaystyle
=>
'false'
,
scriptlevel
=>
'+1'
},
scriptscript
=> {
displaystyle
=>
'false'
,
scriptlevel
=>
'+2'
} },
text
=> {
display
=> {
displaystyle
=>
'true'
},
script
=> {
scriptlevel
=>
'+1'
},
scriptscript
=> {
scriptlevel
=>
'+2'
} },
script
=> {
display
=> {
displaystyle
=>
'true'
,
scriptlevel
=>
'-1'
},
text
=> {
scriptlevel
=>
'-1'
},
scriptscript
=> {
scriptlevel
=>
'+1'
} },
scriptscript
=> {
display
=> {
displaystyle
=>
'true'
,
scriptlevel
=>
'-2'
},
text
=> {
scriptlevel
=>
'-2'
},
script
=> {
scriptlevel
=>
'-1'
} });
my
%stylemap2
= (
display
=> {
text
=> {},
script
=> {
scriptlevel
=>
'+1'
},
scriptscript
=> {
scriptlevel
=>
'+2'
} },
text
=> {
display
=> {},
script
=> {
scriptlevel
=>
'+1'
},
scriptscript
=> {
scriptlevel
=>
'+2'
} },
script
=> {
display
=> {
displaystyle
=>
'true'
,
scriptlevel
=>
'-1'
},
text
=> {
scriptlevel
=>
'-1'
},
scriptscript
=> {
scriptlevel
=>
'+1'
} },
scriptscript
=> {
display
=> {
displaystyle
=>
'true'
,
scriptlevel
=>
'-2'
},
text
=> {
scriptlevel
=>
'-2'
},
script
=> {
scriptlevel
=>
'-1'
} });
sub
pmml_top {
my
(
$self
,
$node
,
$style
) =
@_
;
local
$LaTeXML::MathML::STYLE
=
$style
||
'text'
;
local
$LaTeXML::MathML::FONT
= find_inherited_attribute(
$node
,
'font'
);
local
$LaTeXML::MathML::SIZE
= find_inherited_attribute(
$node
,
'fontsize'
) ||
'100%'
;
local
$LaTeXML::MathML::COLOR
= find_inherited_attribute(
$node
,
'color'
);
local
$LaTeXML::MathML::BGCOLOR
= find_inherited_attribute(
$node
,
'backgroundcolor'
);
local
$LaTeXML::MathML::OPACITY
= find_inherited_attribute(
$node
,
'opacity'
);
local
$LaTeXML::MathML::DESIRED_SIZE
=
$LaTeXML::MathML::SIZE
;
my
@result
=
map
{ pmml(
$_
) } element_nodes(
$node
);
my
$result
=
scalar
(
@result
) > 1 ? [
'm:mrow'
, {},
@result
] :
$result
[0];
$self
->adjust_spacing(
$result
);
return
$result
; }
sub
find_inherited_attribute {
my
(
$node
,
$attribute
) =
@_
;
if
(
my
$value
=
$node
->getAttribute(
$attribute
)) {
return
$value
; }
$node
=
$node
->parentNode; }
return
; }
sub
pmml_smaller {
my
(
$node
) =
@_
;
local
$LaTeXML::MathML::STYLE
=
$stylestep
{
$LaTeXML::MathML::STYLE
};
local
$LaTeXML::MathML::SIZE
=
$stylesize
{
$LaTeXML::MathML::STYLE
};
return
pmml(
$node
); }
sub
pmml_scriptsize {
my
(
$script
) =
@_
;
local
$LaTeXML::MathML::STYLE
=
$style_script_step
{
$LaTeXML::MathML::STYLE
};
local
$LaTeXML::MathML::SIZE
=
$stylesize
{
$LaTeXML::MathML::STYLE
};
return
(
$script
? pmml(
$script
) : [
'm:mrow'
]); }
sub
pmml {
my
(
$node
) =
@_
;
return
unless
$node
;
my
$refr
;
if
(getQName(
$node
) eq
'ltx:XMRef'
) {
$refr
=
$node
;
$node
= realize(
$node
); }
local
$LaTeXML::MathML::DESIRED_SIZE
= _getattr(
$refr
,
$node
,
'fontsize'
) ||
$LaTeXML::MathML::DESIRED_SIZE
;
local
$LaTeXML::MathML::COLOR
= _getattr(
$refr
,
$node
,
'color'
) ||
$LaTeXML::MathML::COLOR
;
local
$LaTeXML::MathML::BGCOLOR
= _getattr(
$refr
,
$node
,
'backgroundcolor'
)
||
$LaTeXML::MathML::BGCOLOR
;
local
$LaTeXML::MathML::OPACITY
= _getattr(
$refr
,
$node
,
'opacity'
) ||
$LaTeXML::MathML::OPACITY
;
my
$result
= pmml_internal(
$node
);
my
$e
= _getattr(
$refr
,
$node
,
'enclose'
);
my
$cl
=
join
(
' '
,
grep
{
$_
}
$refr
&&
$refr
->getAttribute(
'class'
),
$node
->getAttribute(
'class'
));
$result
= [
'm:menclose'
, {
notation
=>
$e
},
$result
]
if
$e
;
if
(
ref
$result
eq
'ARRAY'
) {
my
$l
= _getspace(
$refr
,
$node
,
'lpadding'
);
my
$r
= _getspace(
$refr
,
$node
,
'rpadding'
);
$$result
[1]{_lpadding} =
$l
if
$l
;
$$result
[1]{_rpadding} =
$r
if
$r
; }
if
(
$cl
&& ((
ref
$result
) eq
'ARRAY'
)) {
my
$ocl
=
$$result
[1]{class};
$$result
[1]{class} = (!
$ocl
|| (
$ocl
eq
$cl
) ?
$cl
:
"$ocl $cl"
); }
if
(
my
$role
= _getattr(
$refr
,
$node
,
'role'
)) {
$$result
[1]{_role} =
$role
; }
$LaTeXML::Post::MATHPROCESSOR
->associateNode(
$result
,
$node
);
return
$result
; }
sub
first_element {
my
(
$node
) =
@_
;
my
$c
=
$node
->firstChild;
while
(
$c
) {
return
$c
if
$c
->nodeType == XML_ELEMENT_NODE;
$c
=
$c
->nextSibling; }
return
; }
sub
_getattr {
my
(
$refr
,
$node
,
$attribute
) =
@_
;
return
(
$refr
&&
$refr
->getAttribute(
$attribute
)) ||
$node
&&
$node
->getAttribute(
$attribute
); }
sub
_getspace {
my
(
$refr
,
$node
,
$attribute
) =
@_
;
my
$refspace
=
$refr
&&
$refr
->getAttribute(
$attribute
);
my
$nodespace
=
$node
&&
$node
->getAttribute(
$attribute
);
return
(
$refspace
? getXMHintSpacing(
$refspace
) : 0)
+ (
$nodespace
? getXMHintSpacing(
$nodespace
) : 0); }
sub
getXMHintSpacing {
my
(
$width
) =
@_
;
if
(
$width
&& (
$width
=~ /^([\d\.\+\-]+)(pt|mu|em)(\s+plus\s+[\d\.]+pt)?(\s+minus\s+[\d\.]+pt)?$/)) {
return
($2 eq
'em'
? $1 : ($2 eq
'mu'
? $1 / 18.0 : $1 / 10.0)); }
else
{
return
0; } }
my
$NBSP
=
pack
(
'U'
, 0xA0);
sub
pmml_internal {
my
(
$node
) =
@_
;
return
[
'm:merror'
, {}, [
'm:mtext'
, {},
"Missing Subexpression"
]]
unless
$node
;
my
$self
=
$LaTeXML::Post::MATHPROCESSOR
;
my
$doc
=
$LaTeXML::Post::DOCUMENT
;
my
$tag
= getQName(
$node
);
my
$role
=
$node
->getAttribute(
'role'
);
if
(
$tag
eq
'ltx:XMath'
) {
return
pmml_row(
map
{ pmml(
$_
) } element_nodes(
$node
)); }
elsif
(
$tag
eq
'ltx:XMDual'
) {
my
(
$content
,
$presentation
) = element_nodes(
$node
);
return
pmml(
$presentation
); }
elsif
((
$tag
eq
'ltx:XMWrap'
) || (
$tag
eq
'ltx:XMArg'
)) {
return
pmml_maybe_resize(
$node
, pmml_row(
map
{ pmml(
$_
) } element_nodes(
$node
))); }
elsif
(
$tag
eq
'ltx:XMApp'
) {
my
(
$op
,
@args
) = element_nodes(
$node
);
if
(!
$op
) {
return
[
'm:merror'
, {}, [
'm:mtext'
, {},
"Missing Operator"
]]; }
elsif
(
$role
&& (
$role
=~ /^(FLOAT|POST)(SUB|SUPER)SCRIPT$/)) {
return
[($2 eq
'SUB'
?
'm:msub'
:
'm:msup'
), {}, [
'm:mi'
],
pmml_scriptsize(
$op
)]; }
else
{
my
$rop
= realize(
$op
);
my
$style
=
$rop
->getAttribute(
'mathstyle'
) ||
$op
->getAttribute(
'mathstyle'
);
my
$ostyle
=
$LaTeXML::MathML::STYLE
;
local
$LaTeXML::MathML::STYLE
= (
$style
&&
$stylestep
{
$style
} ?
$style
:
$LaTeXML::MathML::STYLE
);
my
$result
= &{ lookupPresenter(
'Apply'
, getOperatorRole(
$rop
),
$rop
->getAttribute(
'meaning'
))
}(
$op
,
@args
);
$result
= pmml_maybe_resize(
$node
,
$result
);
my
$needsmathstyle
= needsMathstyle(
$result
);
my
%styleattr
= %{ (
$style
&& (
$needsmathstyle
?
$stylemap
{
$ostyle
}{
$style
}
:
$stylemap2
{
$ostyle
}{
$style
})) || {} };
$result
= [
'm:mstyle'
, {
%styleattr
},
$result
]
if
keys
%styleattr
;
return
$result
; } }
elsif
(
$tag
eq
'ltx:XMTok'
) {
return
&{ lookupPresenter(
'Token'
,
$role
,
$node
->getAttribute(
'meaning'
)) }(
$node
); }
elsif
(
$tag
eq
'ltx:XMHint'
) {
return
&{ lookupPresenter(
'Hint'
,
$role
,
$node
->getAttribute(
'meaning'
)) }(
$node
); }
elsif
(
$tag
eq
'ltx:XMArray'
) {
my
$width
=
$node
->getAttribute(
'width'
);
my
$style
=
$node
->getAttribute(
'mathstyle'
);
my
$vattach
=
$node
->getAttribute(
'vattach'
);
my
$rowsep
=
$node
->getAttribute(
'rowsep'
) ||
'0pt'
;
my
$colsep
=
$node
->getAttribute(
'colsep'
) ||
'5pt'
;
$vattach
=
'axis'
if
!
$vattach
|| (
$vattach
eq
'middle'
);
$vattach
=
'bottom1'
if
$vattach
&& (
$vattach
eq
'top'
);
my
$ostyle
=
$LaTeXML::MathML::STYLE
;
local
$LaTeXML::MathML::STYLE
= (
$style
&&
$stylestep
{
$style
} ?
$style
:
$LaTeXML::MathML::STYLE
);
my
@rows
= ();
my
$nrows
= 0;
my
$ncols
= 0;
my
@spanned
= ();
foreach
my
$row
(element_nodes(
$node
)) {
my
@cols
= ();
my
$nc
= 0;
$nrows
++;
foreach
my
$col
(element_nodes(
$row
)) {
$nc
++;
$spanned
[
$nc
- 1]--
if
$spanned
[
$nc
- 1];
next
if
$spanned
[
$nc
- 1];
my
$a
=
$col
->getAttribute(
'align'
);
my
$b
=
$col
->getAttribute(
'border'
);
my
$bc
= (
$b
?
join
(
' '
,
map
{
'ltx_border_'
.
$_
}
split
(/\s/,
$b
)) :
$b
);
my
$th
=
$col
->getAttribute(
'thead'
);
my
$hc
= (
$th
?
join
(
' '
,
map
{
'ltx_th_'
.
$_
}
split
(/\s/,
$th
)) :
''
);
my
$cl
=
$col
->getAttribute(
'class'
);
my
$c
= (
$bc
? (
$hc
?
"$bc $hc"
:
$bc
) :
$hc
);
my
$cs
=
$col
->getAttribute(
'colspan'
);
my
$rs
=
$col
->getAttribute(
'rowspan'
);
my
@cell
= filter_row(
map
{ pmml(
$_
) } element_nodes(
$col
));
if
(
$rs
||
$cs
) {
for
(
my
$i
= 0 ;
$i
< (
$cs
|| 1) ;
$i
++) {
$spanned
[
$nc
- 1 +
$i
] = (
$rs
|| 1); } }
push
(
@cols
, [
'm:mtd'
, { (
$a
&& (
$a
ne
'center'
)
? (
columnalign
=>
$a
,
class
=>
'ltx_align_'
.
$a
) : ()),
(
$c
||
$cl
? (
class
=> (
$c
&&
$cl
?
"$c $cl"
:
$c
||
$cl
)) : ()),
(
$cs
? (
columnspan
=>
$cs
) : ()),
(
$rs
? (
rowspan
=>
$rs
) : ()) },
@cell
]); }
$ncols
=
$nc
if
$nc
>
$ncols
;
push
(
@rows
, [
'm:mtr'
, {},
@cols
]); }
$rowsep
=
undef
if
$nrows
< 2;
$colsep
=
undef
if
$ncols
< 2;
my
$result
= [
'm:mtable'
, { (
$vattach
ne
'axis'
? (
align
=>
$vattach
) : ()),
(
$rowsep
? (
rowspacing
=>
$rowsep
) : ()),
(
$colsep
? (
columnspacing
=>
$colsep
) : ()),
(
$width
? (
width
=>
$width
) : ()),
(
$LaTeXML::MathML::STYLE
eq
'display'
? (
displaystyle
=>
'true'
) : ()) },
@rows
];
my
$needsmathstyle
= needsMathstyle(
$result
);
my
%styleattr
= %{ (
$style
&& (
$needsmathstyle
?
$stylemap
{
$ostyle
}{
$style
}
:
$stylemap2
{
$ostyle
}{
$style
})) || {} };
$result
= [
'm:mstyle'
, {
%styleattr
},
$result
]
if
keys
%styleattr
;
$result
= pmml_maybe_resize(
$node
,
$result
);
return
$result
; }
elsif
(
$tag
eq
'ltx:XMText'
) {
my
@c
=
$node
->childNodes;
my
$result
;
if
(!
$$self
{nestmath}) {
$result
= pmml_row(
map
{ pmml_text_aux(
$_
) }
@c
); }
else
{
$result
= [
'm:mtext'
, {},
$self
->convertXMTextContent(
$doc
, 1,
@c
)]; }
return
pmml_maybe_resize(
$node
,
$result
); }
elsif
(
$tag
eq
'ltx:ERROR'
) {
my
$cl
=
$node
->getAttribute(
'class'
);
return
[
'm:merror'
, {
class
=>
join
(
' '
,
grep
{
$_
}
'ltx_ERROR'
,
$cl
) },
[
'm:mtext'
, {},
$node
->textContent]]; }
else
{
my
$text
=
$node
->textContent;
$text
=~ s/^\s+/
$NBSP
/;
$text
=~ s/\s+$/
$NBSP
/;
return
[
'm:mtext'
, {},
$text
]; } }
sub
needsMathstyle {
my
(
$node
) =
@_
;
if
(
ref
$node
eq
'ARRAY'
) {
my
(
$tag
,
$attr
,
@children
) =
@$node
;
return
1
if
$tag
eq
'm:mfrac'
;
return
1
if
$$attr
{_largeop};
return
0
if
(
$tag
eq
'm:mstyle'
) &&
defined
$$attr
{displaystyle};
return
1
if
grep
{ needsMathstyle(
$_
) }
@children
; }
return
; }
sub
pmml_maybe_resize {
my
(
$node
,
$result
) =
@_
;
return
$result
unless
ref
$node
;
my
$parent
;
if
((
ref
$node
) && (
$node
->nodeType == XML_ELEMENT_NODE)
&& (
$parent
=
$node
->parentNode) && (getQName(
$parent
) eq
'ltx:XMDual'
)) { }
else
{
$parent
=
undef
; }
my
$width
=
$node
->getAttribute(
'width'
) || (
$parent
&&
$parent
->getAttribute(
'width'
));
my
$height
=
$node
->getAttribute(
'height'
) || (
$parent
&&
$parent
->getAttribute(
'height'
));
my
$depth
=
$node
->getAttribute(
'depth'
) || (
$parent
&&
$parent
->getAttribute(
'depth'
));
my
$xoff
=
$node
->getAttribute(
'xoffset'
) || (
$parent
&&
$parent
->getAttribute(
'xoffset'
));
my
$yoff
=
$node
->getAttribute(
'yoffset'
) || (
$parent
&&
$parent
->getAttribute(
'yoffset'
));
if
(
$width
||
$height
||
$depth
||
$xoff
||
$yoff
) {
if
(
$$result
[0] eq
'm:mpadded'
) { }
elsif
(
$$result
[0] eq
'm:mrow'
) {
$$result
[0] =
'm:mpadded'
; }
else
{
$result
= [
'm:mpadded'
, {},
$result
]; }
my
$attr
=
$$result
[1];
if
(
$yoff
) {
if
(!
$height
) {
if
(
$yoff
=~ /^-/) {
$height
=
$yoff
; }
else
{
$height
=
"+"
.
$yoff
; } }
if
(!
$depth
) {
if
(
$yoff
=~ /^-/) {
$depth
=
$yoff
;
$depth
=~ s/^-/+/; }
else
{
$depth
=
"-"
.
$yoff
; } } }
$$attr
{width} =
$width
if
$width
;
$$attr
{height} =
$height
if
$height
;
$$attr
{depth} =
$depth
if
$depth
;
$$attr
{lspace} =
$xoff
if
$xoff
;
$$attr
{voffset} =
$yoff
if
$yoff
; }
if
(
my
$frame
=
$node
->getAttribute(
'framed'
)) {
my
$attr
=
$$result
[1];
my
$c
=
$$attr
{class};
my
$class
=
'ltx_framed_'
.
$frame
;
$$attr
{class} = (
$c
?
$c
.
' '
.
$class
:
$class
);
if
(
my
$color
=
$node
->getAttribute(
'framecolor'
)) {
my
$s
=
$$attr
{style};
my
$style
=
'border-color: '
.
$color
;
$$attr
{style} = (
$s
?
$s
.
'; '
.
$style
:
$style
); } }
return
$result
; }
sub
filter_row {
my
(
@items
) =
@_
;
return
grep
{ !
$$_
[1]{_ignorable}; }
grep
{
$_
}
@items
; }
sub
pmml_row {
my
(
@items
) =
@_
;
@items
= filter_row(
@items
);
return
(
scalar
(
@items
) == 1 ?
$items
[0] : [
'm:mrow'
, {},
@items
]); }
sub
pmml_unrow {
my
(
$mml
) =
@_
;
if
(
$mml
&& (
ref
$mml
) && (
$$mml
[0] eq
'm:mrow'
) && !
scalar
(
keys
%{
$$mml
[1] })) {
my
(
$tag
,
$attr
,
@children
) =
@$mml
;
return
@children
; }
else
{
return
(
$mml
); } }
sub
pmml_parenthesize {
my
(
$item
,
$open
,
$close
) =
@_
;
if
(!
$open
&& !
$close
) {
return
$item
; }
elsif
(
$$LaTeXML::Post::MATHPROCESSOR
{usemfenced}) {
return
[
'm:mfenced'
, {
open
=> (
$open
||
''
),
close
=> (
$close
||
''
) },
$item
]; }
else
{
return
[
'm:mrow'
, {},
(
$open
? (pmml_mo(
$open
,
role
=>
'OPEN'
)) : ()),
$item
,
(
$close
? (pmml_mo(
$close
,
role
=>
'CLOSE'
)) : ())]; } }
sub
pmml_punctuate {
my
(
$separators
,
@items
) =
@_
;
$separators
=
''
unless
defined
$separators
;
my
$lastsep
=
', '
;
my
@arglist
;
if
(
@items
) {
push
(
@arglist
,
shift
(
@items
));
while
(
@items
) {
if
(
$separators
=~ s/^(.*?)( |$)//) {
$lastsep
= $1
if
$1; }
push
(
@arglist
, pmml_mo(
$lastsep
,
role
=>
'PUNCT'
),
shift
(
@items
)); } }
return
pmml_row(
@arglist
); }
sub
pmml_infix {
my
(
$op
,
@args
) =
@_
;
$op
= realize(
$op
)
if
ref
$op
;
return
[
'm:mrow'
, {}]
unless
$op
&&
@args
;
my
@items
= ();
if
(
scalar
(
@args
) == 1) {
push
(
@items
,
(
ref
$op
&& (getQName(
$op
) ne
'ltx:XMTok'
) ? pmml(
$op
) : pmml_mo(
$op
,
role
=>
'OPERATOR'
)),
pmml(
$args
[0])); }
else
{
my
$role
= (
ref
$op
? getOperatorRole(
$op
) :
'none'
) ||
'none'
;
my
$arg1
= realize(
shift
(
@args
));
if
((
$role
eq
'ADDOP'
)
&& (getQName(
$arg1
) eq
'ltx:XMApp'
)
&& ((getOperatorRole((element_nodes(
$arg1
))[0]) ||
'none'
) eq
$role
)) {
push
(
@items
, pmml_unrow(pmml(
$arg1
))); }
else
{
push
(
@items
, pmml(
$arg1
)); }
while
(
@args
) {
push
(
@items
, (
ref
$op
? pmml(
$op
) : pmml_mo(
$op
)));
push
(
@items
, pmml(
shift
(
@args
))); } }
return
pmml_row(
@items
); }
my
%default_token_content
= (
MULOP
=>
"\x{2062}"
,
ADDOP
=>
"\x{2064}"
,
PUNCT
=>
"\x{2063}"
);
my
%plane1hackable
= (
script
=>
'script'
,
'bold-script'
=>
'script'
,
fraktur
=>
'fraktur'
,
'bold-fraktur'
=>
'fraktur'
,
'double-struck'
=>
'double-struck'
);
sub
stylizeContent {
my
(
$item
,
$tag
,
%attr
) =
@_
;
my
$iselement
= (
ref
$item
) eq
'XML::LibXML::Element'
;
my
$role
=
$attr
{role} || (
$iselement
&&
$item
->getAttribute(
'role'
));
my
$font
=
$attr
{font} || (
$iselement
&&
$item
->getAttribute(
'font'
))
||
$LaTeXML::MathML::FONT
;
my
$size
=
$attr
{fontsize} || (
$iselement
&&
$item
->getAttribute(
'fontsize'
))
||
$LaTeXML::MathML::DESIRED_SIZE
;
my
$color
=
$attr
{color} || (
$iselement
&&
$item
->getAttribute(
'color'
))
||
$LaTeXML::MathML::COLOR
;
my
$bgcolor
=
$attr
{backgroundcolor} && (
$iselement
&&
$item
->getAttribute(
'backgroundcolor'
))
||
$LaTeXML::MathML::BGCOLOR
;
my
$opacity
=
$attr
{opacity} || (
$iselement
&&
$item
->getAttribute(
'opacity'
))
||
$LaTeXML::MathML::OPACITY
;
my
$class
=
join
(
' '
,
grep
{
$_
; }
$attr
{class}, (
$iselement
&&
$item
->getAttribute(
'class'
)));
my
$cssstyle
=
join
(
'; '
,
grep
{
$_
; }
$attr
{ccsstyle}, (
$iselement
&&
$item
->getAttribute(
'cssstyle'
)));
my
$variant
= (
$font
? unicode_mathvariant(
$font
) :
''
);
my
$istoken
=
$tag
=~ /^m:(?:mi|mo|mn)$/;
my
$href
=
$istoken
&& (
$attr
{href} || (
$iselement
&&
$item
->getAttribute(
'href'
)));
my
$title
=
$istoken
&& (
$attr
{title} || (
$iselement
&&
$item
->getAttribute(
'title'
)));
my
$text
= (
ref
$item
?
$item
->textContent :
$item
);
my
$stretchy
= ((
defined
$attr
{stretchy} ?
$attr
{stretchy} : (
$iselement
&&
$item
->getAttribute(
'stretchy'
)))
||
'false'
) eq
'true'
;
my
$isfence
=
$role
&& (
$role
=~ /^(OPEN|CLOSE|MIDDLE)$/);
my
$issep
=
$role
&& (
$role
eq
'PUNCT'
);
my
$islargeop
=
$role
&& (
$role
=~ /^(SUMOP|INTOP)$/);
my
$ismoveop
=
$role
&& (
$role
=~ /^(SUMOP|INTOP|BIGOP|LIMITOP)$/);
my
$issymm
=
$islargeop
|| (
$text
eq
'/'
);
my
$pos
= (
$iselement
&&
$item
->getAttribute(
'scriptpos'
)) ||
'post'
;
if
((!
defined
$text
) || (
$text
eq
''
)) {
if
(
my
$default
=
$role
&&
$default_token_content
{
$role
}) {
$text
=
$default
; }
else
{
$text
= (
$iselement
?
$item
->getAttribute(
'name'
) ||
$item
->getAttribute(
'meaning'
) ||
$role
:
'?'
);
$color
=
'red'
; } }
elsif
((
$text
eq
'-'
) &&
$role
&& ((
$role
eq
'ADDOP'
) || (
$role
eq
'OPERATOR'
))) {
$text
=
"\x{2212}"
; }
if
((
$tag
eq
'm:mi'
) && (
$text
=~ /^.$/)) {
if
(
$variant
eq
'italic'
) {
$variant
=
undef
; }
elsif
(!
$variant
) {
$variant
=
'normal'
; } }
elsif
(
$font
&& !
$variant
) {
Warn(
'unexpected'
,
$font
,
undef
,
"Unrecognized font variant '$font'"
);
$variant
=
''
; }
my
$plane1
=
$$LaTeXML::Post::MATHPROCESSOR
{plane1};
my
$plane1hack
=
$$LaTeXML::Post::MATHPROCESSOR
{hackplane1};
my
$u_variant
=
$variant
&& (
$plane1hack
?
$plane1hackable
{
$variant
}
: (
$plane1
?
$variant
:
undef
));
my
$u_text
=
$u_variant
&& unicode_convert(
$text
,
$u_variant
);
if
((
defined
$u_text
) && (
$u_text
ne
''
)) {
$text
=
$u_text
;
$variant
= (
$plane1hack
&& (
$variant
ne
$u_variant
) && (
$variant
=~ /^bold/)
?
'bold'
:
undef
); }
if
(!
$font
) { }
elsif
(
$font
=~ /caligraphic/) {
$class
= (
$class
?
$class
.
' '
:
''
) .
'ltx_font_mathcaligraphic'
; }
elsif
(
$font
=~ /script/) {
$class
= (
$class
?
$class
.
' '
:
''
) .
'ltx_font_mathscript'
; }
elsif
((
$font
=~ /fraktur/) && (
$text
=~ /^[\+\-\d\.]*$/)) {
$class
= (
$class
?
$class
.
' '
:
''
) .
'ltx_font_oldstyle'
; }
elsif
(
$font
=~ /smallcaps/) {
$class
= (
$class
?
$class
.
' '
:
''
) .
'ltx_font_smallcaps'
; }
if
(
$opacity
) {
$cssstyle
= (
$cssstyle
?
$cssstyle
.
';'
:
''
) .
"opacity:$opacity"
; }
my
%props
= (
$tag
eq
'm:mo'
? opdict_lookup(
$text
,
$role
) : ());
$stretchy
=
undef
if
(
$tag
ne
'm:mo'
);
$size
=
undef
if
$stretchy
;
my
$stretchyhack
=
undef
;
if
(
$text
=~ /^[\x{2061}\x{2062}\x{2063}]*$/) {
$stretchy
=
$size
=
undef
; }
if
(
$size
) {
if
(
$size
eq (
$LaTeXML::MathML::SIZE
||
'text'
)) {
$size
=
undef
; }
elsif
((
$size
=~ /%$/) && (
$LaTeXML::MathML::STYLE
and
$LaTeXML::MathML::STYLE
=~ /script/)) {
my
$req
=
$size
;
$req
=~ s/%$//;
my
$ex
=
$stylesize
{
$LaTeXML::MathML::STYLE
};
$ex
=~ s/%$//;
$size
=
int
(100 *
$req
/
$ex
) .
'%'
if
$ex
; }
if
(
$size
) {
if
(
$issymm
||
$props
{symmetric}) {
$stretchyhack
= 1;
$stretchy
= 1; }
elsif
(
$tag
eq
'm:mo'
) {
$stretchy
=
undef
; } } }
return
(
$text
,
(
$variant
? (
mathvariant
=>
$variant
) : ()),
(
$size
? (
$stretchyhack
? (
minsize
=>
$size
,
maxsize
=>
$size
)
: (
mathsize
=>
$size
))
: ()),
(
$color
? (
mathcolor
=>
$color
) : ()),
(
$bgcolor
? (
mathbackground
=>
$bgcolor
) : ()),
(
$cssstyle
? (
style
=>
$cssstyle
) : ()),
(
$class
? (
class
=>
$class
) : ()),
(
$href
? (
href
=>
$href
) : ()),
(
$title
? (
title
=>
$title
) : ()),
((
$stretchy
xor
$props
{stretchy}) ? (
stretchy
=> (
$stretchy
?
'true'
:
'false'
)) : ()),
((
$isfence
xor
$props
{fence}) ? (
fence
=> (
$isfence
?
'true'
:
'false'
)) : ()),
((
$issep
xor
$props
{separator}) ? (
separator
=> (
$issep
?
'true'
:
'false'
)) : ()),
((
$islargeop
xor
$props
{largeop}) ? (
largeop
=> (
$islargeop
?
'true'
:
'false'
)) : ()),
(
$islargeop
? (
_largeop
=> 1) : ()),
(
$issymm
&& !
$props
{symmetric} && (
$stretchy
||
$props
{stretchy})
? (
symmetric
=>
'true'
) : ()),
(
$ismoveop
&& ((
$pos
=~ /mid/) ||
$LaTeXML::MathML::NOMOVABLELIMITS
) ? (
movablelimits
=>
'false'
) : ()),
(
defined
$props
{lspace} ? (
_lspace
=>
$props
{lspace}) : ()),
(
defined
$props
{rspace} ? (
_rspace
=>
$props
{rspace}) : ()),
); }
sub
pmml_mi {
my
(
$item
,
%attr
) =
@_
;
my
(
$text
,
%mmlattr
) = stylizeContent(
$item
,
'm:mi'
,
%attr
);
return
pmml_maybe_resize(
$item
, [
'm:mi'
, {
%mmlattr
},
$text
]); }
sub
pmml_mn {
my
(
$item
,
%attr
) =
@_
;
my
(
$text
,
%mmlattr
) = stylizeContent(
$item
,
'm:mn'
,
%attr
);
return
pmml_maybe_resize(
$item
, [
'm:mn'
, {
%mmlattr
},
$text
]); }
sub
pmml_mo {
my
(
$item
,
%attr
) =
@_
;
my
(
$text
,
%mmlattr
) = stylizeContent(
$item
,
'm:mo'
,
%attr
);
return
pmml_maybe_resize(
$item
, [
'm:mo'
, {
%mmlattr
, },
$text
]); }
sub
pmml_bigop {
my
(
$op
) =
@_
;
my
$style
=
$op
->getAttribute(
'mathstyle'
);
my
%styleattr
= %{ (
$style
&& (
$style
ne
$LaTeXML::MathML::STYLE
)
&&
$stylemap
{
$LaTeXML::MathML::STYLE
}{
$style
}) || {} };
local
$LaTeXML::MathML::STYLE
= (
$style
&&
$stylestep
{
$style
} ?
$style
:
$LaTeXML::MathML::STYLE
);
my
$mml
= pmml_mo(
$op
);
$mml
= [
'm:mstyle'
, {
%styleattr
},
$mml
]
if
keys
%styleattr
;
return
$mml
; }
sub
pmml_script {
my
(
$op
,
$base
,
$script
) =
@_
;
my
(
$innerbase
,
$prescripts
,
$midscripts
,
$postscripts
,
$emb_left
,
$emb_right
)
= pmml_script_decipher(
$op
,
$base
,
$script
);
my
$style
=
$innerbase
->getAttribute(
'mathstyle'
);
if
(
$style
&& (
$style
ne
$LaTeXML::MathML::STYLE
)) {
return
[
'm:mstyle'
, {
displaystyle
=> (
$style
eq
'display'
?
'true'
:
'false'
) },
pmml_script_multi_layout(
pmml_script_mid_layout(
$innerbase
,
$midscripts
,
$emb_left
,
$emb_right
),
$prescripts
,
$postscripts
)]; }
else
{
return
pmml_script_multi_layout(
pmml_script_mid_layout(
$innerbase
,
$midscripts
,
$emb_left
,
$emb_right
),
$prescripts
,
$postscripts
); } }
sub
pmml_script_mid_layout {
my
(
$base
,
$midscripts
,
$emb_left
,
$emb_right
) =
@_
;
if
(
scalar
(
@$midscripts
) == 0) {
{
local
$LaTeXML::MathML::STYLE
=
$base
->getAttribute(
'mathstyle'
) ||
$LaTeXML::MathML::STYLE
;
$base
= pmml(
$base
); }
return
$base
; }
else
{
{
local
$LaTeXML::MathML::NOMOVABLELIMITS
= 1;
local
$LaTeXML::MathML::STYLE
=
$base
->getAttribute(
'mathstyle'
) ||
$LaTeXML::MathML::STYLE
;
$base
= pmml(
$base
); }
my
$result
=
$base
;
for
my
$midscript
(
@$midscripts
) {
my
$under
= (!
defined
$$midscript
[0] ?
undef
: pmml_scriptsize_padded(
$$midscript
[0],
$emb_left
,
$emb_right
));
my
$over
= (!
defined
$$midscript
[1] ?
undef
: pmml_scriptsize_padded(
$$midscript
[1],
$emb_left
,
$emb_right
));
if
(!
defined
$over
) {
$result
= [
'm:munder'
, {},
$result
,
$under
]; }
elsif
(!
defined
$under
) {
$result
= [
'm:mover'
, {},
$result
,
$over
]; }
else
{
$result
= [
'm:munderover'
, {},
$result
,
$under
,
$over
]; } }
return
$result
; } }
sub
pmml_scriptsize_padded {
my
(
$script
,
$emb_left
,
$emb_right
) =
@_
;
return
(
$emb_left
||
$emb_right
? [
'm:mrow'
, {},
(
$emb_left
? ([
'm:mphantom'
, {}, pmml_scriptsize(
$emb_left
)]) : ()),
pmml_scriptsize(
$script
),
(
$emb_right
? ([
'm:mphantom'
, {}, pmml_scriptsize(
$emb_right
)]) : ())]
: pmml_scriptsize(
$script
)); }
sub
pmml_script_multi_layout {
my
(
$base
,
$prescripts
,
$postscripts
) =
@_
;
if
(
scalar
(
@$prescripts
) > 0) {
return
[
'm:mmultiscripts'
, {},
$base
,
(
map
{ (pmml_scriptsize(
$_
->[0]), pmml_scriptsize(
$_
->[1])) }
@$postscripts
),
[
'm:mprescripts'
],
(
map
{ (pmml_scriptsize(
$_
->[0]), pmml_scriptsize(
$_
->[1])) }
@$prescripts
)]; }
elsif
(
scalar
(
@$postscripts
) > 1) {
return
[
'm:mmultiscripts'
, {},
$base
,
(
map
{ (pmml_scriptsize(
$_
->[0]), pmml_scriptsize(
$_
->[1])) }
@$postscripts
)]; }
elsif
(
scalar
(
@$postscripts
) == 0) {
return
$base
; }
elsif
(!
defined
$$postscripts
[0][1]) {
return
[
'm:msub'
, {},
$base
, pmml_scriptsize(
$$postscripts
[0][0])]; }
elsif
(!
defined
$$postscripts
[0][0]) {
return
[
'm:msup'
, {},
$base
, pmml_scriptsize(
$$postscripts
[0][1])]; }
else
{
return
[
'm:msubsup'
, {},
$base
,
pmml_scriptsize(
$$postscripts
[0][0]), pmml_scriptsize(
$$postscripts
[0][1])]; } }
sub
pmml_script_decipher {
my
(
$op
,
$base
,
$script
) =
@_
;
my
(
@pres
,
@mids
,
@posts
);
my
(
$prelevel
,
$midlevel
,
$postlevel
) = (0, 0, 0);
my
$sawmid
= 0;
my
(
$emb_left
,
$emb_right
) = (
undef
,
undef
);
my
(
$y
) = (
$op
->getAttribute(
'role'
) ||
''
) =~ /^(SUPER|SUB)SCRIPTOP$/;
my
(
$pos
,
$level
) = (
$op
->getAttribute(
'scriptpos'
) ||
'post0'
)
=~ /^(pre|mid|post)?(\d+)?$/;
if
(
$pos
eq
'pre'
) {
if
(
$y
eq
'SUB'
) {
push
(
@pres
, [
$script
,
undef
]);
$prelevel
=
$level
; }
elsif
(
$y
eq
'SUPER'
) {
push
(
@pres
, [
undef
,
$script
]);
$prelevel
=
$level
; } }
elsif
(
$pos
eq
'mid'
) {
$sawmid
= 1;
if
(
$y
eq
'SUB'
) {
push
(
@mids
, [
$script
,
undef
]);
$midlevel
=
$level
; }
elsif
(
$y
eq
'SUPER'
) {
push
(
@mids
, [
undef
,
$script
]);
$midlevel
=
$level
; } }
else
{
if
(
$y
eq
'SUB'
) {
push
(
@posts
, [
$script
,
undef
]);
$postlevel
=
$level
; }
elsif
(
$y
eq
'SUPER'
) {
push
(
@posts
, [
undef
,
$script
]);
$postlevel
=
$level
; } }
while
(1) {
$base
= realize(
$base
,
'presentation'
);
last
unless
getQName(
$base
) eq
'ltx:XMApp'
;
my
(
$xop
,
$xbase
,
$xscript
) = element_nodes(
$base
);
last
unless
(getQName(
$xop
) eq
'ltx:XMTok'
);
my
(
$ny
) = (
$xop
->getAttribute(
'role'
) ||
''
) =~ /^(SUPER|SUB)SCRIPTOP$/;
last
unless
$ny
;
my
(
$nx
,
$nl
) = (
$xop
->getAttribute(
'scriptpos'
) ||
'post0'
)
=~ /^(pre|mid|post)?(\d+)?$/;
my
$spos
= (
$ny
eq
'SUB'
? 0 : 1);
if
(
$nx
eq
'pre'
) {
push
(
@pres
, [
undef
,
undef
])
if
(
$prelevel
ne
$nl
) ||
$pres
[-1][
$spos
];
$pres
[-1][
$spos
] =
$xscript
;
$prelevel
=
$nl
; }
elsif
(
$nx
eq
'mid'
) {
$sawmid
= 1;
unshift
(
@mids
, [
undef
,
undef
])
if
(
$midlevel
ne
$nl
) ||
$mids
[0][
$spos
];
$mids
[0][
$spos
] =
$xscript
;
$midlevel
=
$nl
; }
else
{
if
(
$sawmid
) {
$emb_right
=
$xscript
;
last
; }
unshift
(
@posts
, [
undef
,
undef
])
if
(
$postlevel
ne
$nl
) ||
$posts
[0][
$spos
];
$posts
[0][
$spos
] =
$xscript
;
$postlevel
=
$nl
; }
$base
=
$xbase
;
}
return
(
$base
, [
@pres
], [
@mids
], [
@posts
],
$emb_left
,
$emb_right
); }
sub
pmml_text_aux {
my
(
$node
,
%attr
) =
@_
;
return
()
unless
$node
;
my
$type
=
$node
->nodeType;
if
(
$type
== XML_TEXT_NODE) {
my
(
$string
,
%mmlattr
) = stylizeContent(
$node
,
'm:mtext'
,
%attr
);
$string
=~ s/^\s+/
$NBSP
/;
$string
=~ s/\s+$/
$NBSP
/;
return
[
'm:mtext'
, {
%mmlattr
},
$string
]; }
elsif
(
$type
== XML_DOCUMENT_FRAG_NODE) {
return
map
{ pmml_text_aux(
$_
,
%attr
) }
$node
->childNodes; }
elsif
(
$type
== XML_ELEMENT_NODE) {
if
(
my
$font
=
$node
->getAttribute(
'font'
)) {
$attr
{font} =
$font
; }
if
(
my
$size
=
$node
->getAttribute(
'fontsize'
)) {
$attr
{fontsize} =
$size
; }
if
(
my
$color
=
$node
->getAttribute(
'color'
)) {
$attr
{color} =
$color
; }
if
(
my
$bgcolor
=
$node
->getAttribute(
'backgroundcolor'
)) {
$attr
{backgroundcolor} =
$bgcolor
; }
if
(
my
$opacity
=
$node
->getAttribute(
'opacity'
)) {
$attr
{opacity} =
$opacity
; }
my
$tag
= getQName(
$node
);
if
(
$tag
eq
'ltx:Math'
) {
if
(
my
$xmath
=
$LaTeXML::Post::DOCUMENT
->findnode(
'ltx:XMath'
,
$node
)) {
return
pmml(
$xmath
); }
elsif
(
my
$mml
=
$LaTeXML::Post::DOCUMENT
->findnode(
'm:math'
,
$node
)) {
return
$mml
->childNodes; }
else
{
return
(); } }
elsif
((
$tag
eq
'ltx:text'
)
&& (!
grep
{
$node
->hasAttribute(
$_
) }
qw(framed framecolor)
)) {
return
pmml_maybe_resize(
$node
, pmml_row(
map
{ pmml_text_aux(
$_
,
%attr
) }
$node
->childNodes)); }
else
{
my
(
$ignore
,
%mmlattr
) = stylizeContent(
$node
,
'm:mtext'
,
%attr
);
delete
$mmlattr
{stretchy};
Warn(
'unexpected'
,
'nested-math'
,
$node
,
"We're getting m:math nested within an m:mtext"
)
if
$LaTeXML::Post::DOCUMENT
->findnodes(
'.//ltx:Math'
,
$node
);
return
[
'm:mtext'
, {
%mmlattr
},
$LaTeXML::Post::DOCUMENT
->cloneNode(
$node
,
'nest'
)]; } }
else
{
return
(); } }
sub
adjust_spacing {
my
(
$self
,
$mml
) =
@_
;
space_walk(
$self
,
$mml
);
return
; }
our
%tag_arg_pattern
= (
(
map
{
$_
=>
'atom'
; }
qw(m:mi m:mo m:mn m:ms m:mtext)
),
(
map
{
$_
=>
'mrow'
; }
qw(m:mrow m:mpadded m:msqrt m:mstyle m:merror m:mphantom m:mtd)
),
);
sub
space_walk {
my
(
$self
,
$node
) =
@_
;
return
unless
$node
;
my
(
$tag
,
$attr
,
@children
) =
@$node
;
my
$type
=
$tag_arg_pattern
{
$tag
} ||
'other'
;
if
(
$type
eq
'atom'
) { }
elsif
(
$type
eq
'mrow'
) {
my
@nodes
=
@children
;
my
$prev
=
shift
(
@nodes
);
while
(
$prev
&& (
$$prev
[0] eq
'm:mrow'
)) {
unshift
(
@nodes
,
@$prev
[2 ..
$#$prev
]);
$prev
=
shift
(
@nodes
); }
space_walk(
$self
,
$prev
);
while
(
my
$next
=
shift
(
@nodes
)) {
my
$invisop
;
if
((
$$next
[0] eq
'm:mo'
) &&
$$next
[2] && (
$$next
[2] =~ /^[\x{2061}\x{2062}\x{2063}]*$/)) {
$invisop
=
$next
;
$next
=
shift
(
@nodes
);
last
unless
$next
; }
if
(
$$next
[0] eq
'm:mrow'
) {
unshift
(
@nodes
,
@$next
[2 ..
$#$next
]);
unshift
(
@nodes
,
$invisop
)
if
$invisop
;
next
; }
if
(
$$next
[0] =~ /^m:(msup|msub|munder|mover|mmultiscripts)/) {
unshift
(
@nodes
,
$$next
[2]);
foreach
my
$script
(
@$next
[3 ..
$#$next
]) { # but recurse on the scripts
space_walk(
$self
,
$script
); }
unshift
(
@nodes
,
$invisop
)
if
$invisop
;
next
; }
space_walk(
$self
,
$next
);
adjust_pair(
$self
,
$prev
,
$next
,
$invisop
);
$prev
=
$next
; } }
else
{
map
{ space_walk(
$self
,
$_
); }
@children
; }
return
; }
sub
compute_size {
my
(
$node
) =
@_
;
my
(
$tag
,
$attr
,
@children
) =
@$node
;
my
$type
=
$tag_arg_pattern
{
$tag
} ||
'other'
;
if
(
$type
eq
'atom'
) {
my
$font
= LaTeXML::Common::Font->mathDefault();
my
(
$w
,
$h
,
$d
) =
$font
->computeStringSize(
$children
[0]);
if
(
$w
&&
$$attr
{class} && (
$$attr
{class} =~ /mathscript/)) {
$w
=
$w
->larger(LaTeXML::Common::Dimension->new(10 * 65535)); }
return
(
$w
,
$h
,
$d
); }
else
{
return
; } }
our
$role_atomtype
= {
ATOM
=>
'Ord'
,
UNKNOWN
=>
'Ord'
,
ID
=>
'Ord'
,
NUMBER
=>
'Ord'
,
ARRAY
=>
'Inner'
,
RELOP
=>
'Rel'
,
OPEN
=>
'Open'
,
CLOSE
=>
'Close'
,
MIDDLE
=>
'Ord'
,
PUNCT
=>
'Punct'
,
VERTBAR
=>
'Punct'
,
PERIOD
=>
'Punct'
,
METARELOP
=>
'Rel'
,
MODIFIEROP
=>
'Rel'
,
MODIFIER
=>
'Rel'
,
ARROW
=>
'Rel'
,
ADDOP
=>
'Bin'
,
MULOP
=>
'Bin'
,
BINOP
=>
'Bin'
,
POSTFIX
=>
'Ord'
,
FUNCTION
=>
'Ord'
,
OPFUNCTION
=>
'Op'
,
TRIGFUNCTION
=>
'Op'
,
APPLYOP
=>
'Bin'
,
COMPOSEOP
=>
'Bin'
,
SUPOP
=>
'Ord'
,
BIGOP
=>
'Op'
,
SUMOP
=>
'Op'
,
INTOP
=>
'Op'
,
LIMITOP
=>
'Op'
,
DIFFOP
=>
'Ord'
,
OPERATOR
=>
'Op'
,
POSTSUBSCRIPT
=>
'Inner'
,
POSTSUPERSCRIPT
=>
'Inner'
,
FLOATSUPERSCRIPT
=>
'Inner'
,
FLOATSUBSCRIPT
=>
'Inner'
, };
our
$atomtype_form
= {
Op
=>
'prefix'
,
Bin
=>
'infix'
,
Rel
=>
'infix'
,
Open
=>
'prefix'
,
Close
=>
'postfix'
,
Punct
=>
'postfix'
};
our
$atompair_spacing
= {
Ord
=> {
Ord
=> 0,
Op
=> 1,
Bin
=> -2,
Rel
=> -3,
Open
=> 0,
Close
=> 0,
Punct
=> 0,
Inner
=> -1 },
Op
=> {
Ord
=> 1,
Op
=> 1,
Bin
=> 0,
Rel
=> -3,
Open
=> 0,
Close
=> 0,
Punct
=> 0,
Inner
=> -1 },
Bin
=> {
Ord
=> -2,
Op
=> -2,
Bin
=> 0,
Rel
=> 0,
Open
=> -2,
Close
=> 0,
Punct
=> 0,
Inner
=> -2 },
Rel
=> {
Ord
=> -3,
Op
=> -3,
Bin
=> 0,
Rel
=> 0,
Open
=> -3,
Close
=> 0,
Punct
=> 0,
Inner
=> -3 },
Open
=> {
Ord
=> 0,
Op
=> 0,
Bin
=> 0,
Rel
=> 0,
Open
=> 0,
Close
=> 0,
Punct
=> 0,
Inner
=> 0 },
Close
=> {
Ord
=> 0,
Op
=> 1,
Bin
=> -2,
Rel
=> -3,
Open
=> 0,
Close
=> 0,
Punct
=> 0,
Inner
=> -1 },
Punct
=> {
Ord
=> -1,
Op
=> -1,
Bin
=> 0,
Rel
=> -1,
Open
=> -1,
Close
=> -1,
Punct
=> -1,
Inner
=> -1 },
Inner
=> {
Ord
=> -1,
Op
=> 1,
Bin
=> -2,
Rel
=> -3,
Open
=> -1,
Close
=> 0,
Punct
=> -1,
Inner
=> -1 },
};
our
@tex_spacing
= (0, 0.167, 0.222, 0.2778);
our
%embellisher
=
map
{
$_
=> 1; }
qw(m:msub m:msup m:msubsup m:munder m:mover m:munderover)
;
our
%m_atomtype
= (
'm:mfrac'
=>
'Ord'
,
'm:marray'
=>
'Inner'
,
'm:mspace'
=>
'Ord'
);
our
$epsilon
= 0.01;
our
$fudge
= 0.3;
sub
adjust_pair {
my
(
$self
,
$prev
,
$next
,
$invisop
) =
@_
;
my
(
$iprev
,
$inext
) = (
$prev
,
$next
);
while
(
$iprev
&&
$embellisher
{
$$iprev
[0] }) {
$iprev
=
$$iprev
[2]; }
while
(
$inext
&&
$embellisher
{
$$inext
[0] }) {
$inext
=
$$inext
[2]; }
my
$prev_req_right
=
$$prev
[1]{_rpadding} // 0;
my
$next_req_left
=
$$next
[1]{_lpadding} // 0;
my
$prev_role
=
$$iprev
[1]{_role} //
'ATOM'
;
my
$next_role
=
$$inext
[1]{_role} //
'ATOM'
;
my
$prev_type
=
$m_atomtype
{
$$iprev
[0] } ||
$$role_atomtype
{
$prev_role
} ||
'Ord'
;
my
$next_type
=
$m_atomtype
{
$$inext
[0] } ||
$$role_atomtype
{
$next_role
} ||
'Ord'
;
my
$tex_code
=
$$atompair_spacing
{
$prev_type
}{
$next_type
} // 0;
my
$tex_space
=
$tex_spacing
[
abs
(
$tex_code
)];
my
$req_space
=
$prev_req_right
+
$next_req_left
;
my
$target
=
$req_space
+
$tex_space
;
my
$prev_dict_right
=
$$iprev
[1]{_rspace} // 0;
my
$next_dict_left
=
$$inext
[1]{_lspace} // 0;
my
$default
= (
$prev_dict_right
+
$next_dict_left
);
my
$needs_adjustment
=
abs
(
$target
-
$default
) >
$epsilon
;
Debug(
"SPACEWALK: TeX $tex_space + $req_space vs OpDict $default : "
. (
$needs_adjustment
? colorizeString(
"ADJUSTMENT $target"
,
'warning'
) :
" NO adjustment"
)
. LaTeXML::Post::MathProcessor::shownode(
$prev
) .
" $prev_type "
. LaTeXML::Post::MathProcessor::shownode(
$next
) .
" $next_type "
)
if
$LaTeXML::DEBUG
{mathspacing};
return
unless
$needs_adjustment
;
if
(
$target
< 0) {
my
(
$w
,
$h
,
$d
) = compute_size(
$prev
);
if
(
$w
) {
my
$reqw
=
$w
->ptValue / 10.0 +
$target
;
$reqw
= 0
if
$reqw
< 0;
splice
(
@$prev
, 0,
$#$prev
+ 1,
'm:mpadded'
, {
width
=> fmt_em(
$reqw
) }, [
@$prev
]); } }
elsif
(
$$prev
[0] eq
'm:mspace'
) {
$$prev
[1]{width} = fmt_em(
$target
+ getXMHintSpacing(
$$prev
[1]{width})); }
elsif
(
$$next
[0] eq
'm:mspace'
) {
$$next
[1]{width} = fmt_em(
$target
+ getXMHintSpacing(
$$next
[1]{width})); }
elsif
(
$invisop
) {
$$invisop
[1]{lspace} = fmt_em(
$target
); }
elsif
((
$$prev
[0] eq
'm:mo'
) && (
$$next
[0] eq
'm:mo'
)) {
my
$p
=
$$prev
[1]{_rspace} // 0;
my
$n
=
$$next
[1]{_lspace} // 0;
my
$rem
;
if
((
$rem
=
$target
-
$n
) >= 0) {
$$prev
[1]{rspace} = fmt_em(
$rem
>
$epsilon
?
$rem
: 0); }
elsif
((
$rem
=
$target
-
$p
) >= 0) {
$$next
[1]{lspace} = fmt_em(
$rem
>
$epsilon
?
$rem
: 0); }
else
{
$rem
=
$target
/ 2;
$$prev
[1]{rspace} =
$rem
.
'em'
if
$rem
!=
$p
;
$$next
[1]{lspace} =
$rem
.
'em'
if
$rem
!=
$n
; } }
elsif
(
$$prev
[0] eq
'm:mo'
) {
$$prev
[1]{rspace} = fmt_em(
$target
); }
elsif
(
$$next
[0] eq
'm:mo'
) {
$$next
[1]{lspace} = fmt_em(
$target
); }
elsif
(
abs
(
$target
-
$default
) >
$fudge
) {
Info(
'ignored'
,
'spacing'
,
undef
,
"No place to set spacing to $target (default $default) between"
. LaTeXML::Post::MathProcessor::shownode(
$prev
)
.
" and "
. LaTeXML::Post::MathProcessor::shownode(
$next
)); }
return
; }
sub
fmt_em {
return
(
$_
[0] ?
sprintf
(
"%.3fem"
,
$_
[0]) :
'0em'
); }
sub
cmml_top {
my
(
$self
,
$node
) =
@_
;
local
$LaTeXML::MathML::STYLE
=
'text'
;
local
$LaTeXML::MathML::FONT
= find_inherited_attribute(
$node
,
'font'
);
local
$LaTeXML::MathML::SIZE
= find_inherited_attribute(
$node
,
'fontsize'
) ||
'100%'
;
local
$LaTeXML::MathML::COLOR
= find_inherited_attribute(
$node
,
'color'
);
local
$LaTeXML::MathML::BGCOLOR
= find_inherited_attribute(
$node
,
'backgroundcolor'
);
local
$LaTeXML::MathML::OPACITY
= find_inherited_attribute(
$node
,
'opacity'
);
local
$LaTeXML::MathML::DESIRED_SIZE
=
$LaTeXML::MathML::SIZE
;
return
cmml_contents(
$node
); }
sub
cmml {
my
(
$node
) =
@_
;
if
(getQName(
$node
) eq
'ltx:XMRef'
) {
$node
= realize(
$node
); }
my
$result
= cmml_internal(
$node
);
$LaTeXML::Post::MATHPROCESSOR
->associateNode(
$result
,
$node
);
return
$result
; }
sub
cmml_internal {
my
(
$node
) =
@_
;
return
[
'm:merror'
, {}, [
'm:mtext'
, {},
"Missing Subexpression"
]]
unless
$node
;
$node
= realize(
$node
)
if
getQName(
$node
) eq
'ltx:XMRef'
;
my
$tag
= getQName(
$node
);
if
(
$tag
eq
'ltx:XMDual'
) {
my
(
$content
,
$presentation
) = element_nodes(
$node
);
return
cmml(
$content
); }
elsif
((
$tag
eq
'ltx:XMWrap'
) || (
$tag
eq
'ltx:XMArg'
)) {
return
cmml_contents(
$node
); }
elsif
(
$tag
eq
'ltx:XMApp'
) {
if
(
my
$meaning
=
$node
->getAttribute(
'meaning'
)) {
return
&{ lookupContent(
'Token'
,
$node
->getAttribute(
'role'
),
$meaning
) }(
$node
); }
if
((
$node
->getAttribute(
'role'
) ||
''
) eq
'ID'
) {
return
cmml_decoratedSymbol(
$node
); }
else
{
my
(
$op
,
@args
) = element_nodes(
$node
);
my
$rop
=
$op
;
if
(!
$op
|| !(
$rop
= realize(
$op
))) {
return
[
'm:merror'
, {}, [
'm:mtext'
, {},
"Missing Operator"
]]; }
else
{
return
&{ lookupContent(
'Apply'
,
$rop
->getAttribute(
'role'
),
$rop
->getAttribute(
'meaning'
)) }(
$op
,
@args
); } } }
elsif
(
$tag
eq
'ltx:XMTok'
) {
return
&{ lookupContent(
'Token'
,
$node
->getAttribute(
'role'
),
$node
->getAttribute(
'meaning'
)) }(
$node
); }
elsif
(
$tag
eq
'ltx:XMHint'
) {
return
&{ lookupContent(
'Hint'
,
$node
->getAttribute(
'role'
),
$node
->getAttribute(
'meaning'
)) }(
$node
); }
elsif
(
$tag
eq
'ltx:XMArray'
) {
return
&{ lookupContent(
'Array'
,
$node
->getAttribute(
'role'
),
$node
->getAttribute(
'meaning'
)) }(
$node
); }
elsif
(
$tag
eq
'ltx:XMText'
) {
return
cmml_decoratedSymbol(
$node
); }
else
{
return
cmml_decoratedSymbol(
$node
); } }
sub
cmml_contents {
my
(
$node
) =
@_
;
my
(
$item
,
@rest
) = element_nodes(
$node
);
if
(!
$item
) {
return
[
'm:cerror'
, {}, [
'm:csymbol'
, {
cd
=>
'ambiguous'
},
'missing-subexpression'
]]; }
elsif
(
@rest
) {
return
cmml_unparsed(
$item
,
@rest
); }
else
{
return
cmml(
$item
); } }
sub
cmml_unparsed {
my
(
@nodes
) =
@_
;
my
@results
= ();
foreach
my
$node
(
@nodes
) {
if
((getQName(
$node
) eq
'ltx:XMTok'
)
&& ((
$node
->getAttribute(
'role'
) ||
'UNKNOWN'
) eq
'UNKNOWN'
)) {
my
$result
= [
'm:csymbol'
, {
cd
=>
'unknown'
},
$node
->textContent];
$LaTeXML::Post::MATHPROCESSOR
->associateNode(
$result
,
$node
);
push
(
@results
,
$result
); }
else
{
push
(
@results
, cmml(
$node
)); } }
return
[
'm:cerror'
, {},
[
'm:csymbol'
, {
cd
=>
'ambiguous'
},
'fragments'
],
@results
]; }
sub
cmml_leaf {
my
(
$item
) =
@_
;
if
(
my
$meaning
= (
ref
$item
) &&
$item
->getAttribute(
'meaning'
)) {
if
(
my
$cd
=
$item
->getAttribute(
'omcd'
)) {
return
[
'm:csymbol'
, {
cd
=>
$cd
},
$meaning
]; }
elsif
((
$item
->getAttribute(
'role'
) ||
''
) eq
'NUMBER'
) {
return
[
'm:cn'
, {
type
=> (
$meaning
=~ /^[+-]?\d+$/ ?
'integer'
:
'float'
) },
$meaning
]; }
else
{
return
[
'm:csymbol'
, {
cd
=>
'latexml'
},
$meaning
]; } }
else
{
my
(
$content
,
%mmlattr
) = stylizeContent(
$item
,
'm:ci'
);
if
(
my
$mv
=
$mmlattr
{mathvariant}) {
$content
=
$mv
.
"-"
.
$content
; }
return
[
'm:ci'
, {},
$content
]; } }
sub
cmml_decoratedSymbol {
my
(
$item
) =
@_
;
if
(
my
$meaning
= (
ref
$item
) &&
$item
->getAttribute(
'meaning'
)) {
my
$cd
=
$item
->getAttribute(
'omcd'
) ||
'latexml'
;
return
[
'm:csymbol'
, {
cd
=>
$cd
},
$meaning
]; }
else
{
return
[
'm:ci'
, {}, pmml(
$item
)]; } }
sub
cmml_not {
my
(
$arg
) =
@_
;
return
[
'm:apply'
, {}, [
'm:not'
, {}], cmml(
$arg
)]; }
sub
cmml_synth_not {
my
(
$op
,
@args
) =
@_
;
return
[
'm:apply'
, {}, [
'm:not'
, {}], [
'm:apply'
, {}, [
$op
, {}],
map
{ cmml(
$_
) }
@args
]]; }
sub
cmml_synth_complement {
my
(
$op
,
@args
) =
@_
;
return
[
'm:apply'
, {}, [
$op
, {}],
map
{ cmml(
$_
) }
reverse
(
@args
)]; }
sub
cmml_shared {
my
(
$node
) =
@_
;
$LaTeXML::Post::DOCUMENT
->generateNodeID(
$node
,
'sh'
);
return
cmml(
$node
); }
sub
cmml_share {
my
(
$node
) =
@_
;
my
$fragid
=
$node
->getAttribute(
'fragid'
);
if
(
$fragid
) {
return
[
'm:share'
, {
href
=>
'#'
.
$fragid
.
$LaTeXML::Post::MATHPROCESSOR
->IDSuffix }]; }
else
{
Warn(
'expected'
,
'fragid'
,
$node
,
"Shared node is missing fragid"
);
return
[
'm:share'
]; } }
sub
cmml_or_compose {
my
(
$operators
,
@args
) =
@_
;
my
@operators
=
@$operators
;
if
(
scalar
(
@operators
) == 1) {
return
[
'm:apply'
, {}, [
shift
(
@operators
), {}],
map
{ cmml(
$_
) }
@args
]; }
else
{
my
@parts
= ([
'm:apply'
, {}, [
shift
(
@operators
), {}],
map
{ cmml_shared(
$_
) }
@args
]);
foreach
my
$op
(
@operators
) {
push
(
@parts
, [
'm:apply'
, {}, [
shift
(
@operators
), {}],
map
{ cmml_share(
$_
) }
@args
]); }
return
[
'm:or'
, {},
@parts
]; } }
DefMathML(
"Token:?:?"
, \
&pmml_mi
, \
&cmml_leaf
);
DefMathML(
"Token:PUNCT:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:PERIOD:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:OPEN:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:CLOSE:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:MIDDLE:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:VERTBAR:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:ARROW:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:OVERACCENT:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:UNDERACCENT:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:NUMBER:?"
, \
&pmml_mn
,
sub
{
my
$n
=
$_
[0]->textContent;
return
[
'm:cn'
, {
type
=> (
$n
=~ /^[+-]?\d+$/ ?
'integer'
:
'float'
) },
$n
]; });
DefMathML(
"Token:?:absent"
,
sub
{
return
[
'm:mi'
, {}] });
DefMathML(
'Hint:?:?'
,
sub
{
my
(
$node
) =
@_
;
my
$w
= getXMHintSpacing(
$node
->getAttribute(
'width'
));
[
'm:mspace'
, { (
$w
? (
width
=>
$w
.
'em'
) : (
_ignorable
=> 1)) }]; },
sub
{
undef
; });
DefMathML(
'Apply:OVERACCENT:?'
,
sub
{
my
(
$accent
,
$base
) =
@_
;
if
(getQName(
$base
) eq
'ltx:XMApp'
) {
my
(
$xaccent
,
$xbase
) = element_nodes(
$base
);
if
((
$xaccent
->getAttribute(
'role'
) ||
''
) eq
'UNDERACCENT'
) {
return
[
'm:munderover'
, {
accent
=>
'true'
,
accentunder
=>
'true'
},
pmml(
$xbase
), pmml_scriptsize(
$xaccent
), pmml(
$accent
)]; } }
return
[
'm:mover'
, {
accent
=>
'true'
}, pmml(
$base
), pmml(
$accent
)]; });
DefMathML(
'Apply:UNDERACCENT:?'
,
sub
{
my
(
$accent
,
$base
) =
@_
;
if
(getQName(
$base
) eq
'ltx:XMApp'
) {
my
(
$xaccent
,
$xbase
) = element_nodes(
$base
);
if
((
$xaccent
->getAttribute(
'role'
) ||
''
) eq
'OVERACCENT'
) {
return
[
'm:munderover'
, {
accent
=>
'true'
,
accentunder
=>
'true'
},
pmml(
$xbase
), pmml_scriptsize(
$accent
), pmml(
$xaccent
)]; } }
return
[
'm:munder'
, {
accentunder
=>
'true'
}, pmml(
$base
), pmml(
$accent
)]; });
DefMathML(
'Apply:ENCLOSE:?'
,
sub
{
my
(
$op
,
$base
) =
@_
;
my
$enclosure
=
$op
->getAttribute(
'enclose'
);
my
$color
=
$op
->getAttribute(
'color'
) ||
$LaTeXML::MathML::COLOR
;
return
[
'm:menclose'
, {
notation
=>
$enclosure
,
mathcolor
=>
$color
},
(
$color
? [
'm:mstyle'
, {
mathcolor
=>
$LaTeXML::MathML::COLOR
||
'black'
}, pmml(
$base
)]
: pmml(
$base
))]; });
DefMathML(
"Token:APPLYOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:OPERATOR:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:?:?'
,
sub
{
my
(
$op
,
@args
) =
@_
;
my
$pop
= pmml(
$op
);
my
$inner
=
$pop
;
while
(
$inner
&& (
$$inner
[0] ne
'm:mo'
)) {
last
unless
$$inner
[0] =~ /^m:(?:msub|msup|munder|mover|mprescripts)/;
$inner
=
$$inner
[2]; }
my
$is_mo
=
$inner
&& (
$$inner
[0] eq
'm:mo'
);
return
[
'm:mrow'
, {},
$pop
, (
$is_mo
? () : pmml_mo(
"\x{2061}"
)),
map
{ pmml(
$_
) }
@args
]; },
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:apply'
, {}, cmml(
$op
),
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
'Apply:COMPOSEOP:?'
, \
&pmml_infix
,
undef
);
DefMathML(
"Token:COMPOSEOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:DIFFOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Apply:?:open-interval"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:interval'
, {
closure
=>
"open"
},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Apply:?:closed-interval"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:interval'
, {
closure
=>
"closed"
},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Apply:?:closed-open-interval"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:interval'
, {
closure
=>
"closed-open"
},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Apply:?:open-closed-interval"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:interval'
, {
closure
=>
"open-closed"
},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Token:?:inverse"
,
undef
,
sub
{
return
[
'm:inverse'
]; });
DefMathML(
"Token:?:lambda"
,
undef
,
sub
{
return
[
'm:lambda'
]; });
DefMathML(
"Token:?:compose"
,
undef
,
sub
{
return
[
'm:compose'
]; });
DefMathML(
"Token:?:identity"
,
undef
,
sub
{
return
[
'm:ident'
]; });
DefMathML(
"Token:?:domain"
,
undef
,
sub
{
return
[
'm:domain'
]; });
DefMathML(
"Token:?:codomain"
,
undef
,
sub
{
return
[
'm:codomain'
]; });
DefMathML(
"Token:?:image"
,
undef
,
sub
{
return
[
'm:image'
]; });
DefMathML(
"Array:?:cases"
,
undef
,
sub
{
my
(
$node
) =
@_
;
my
@rows
= ();
my
@otherwises
;
foreach
my
$row
(element_nodes(
$node
)) {
my
@items
= element_nodes(
$row
);
my
$n
=
scalar
(
@items
);
if
(
$n
== 0) { }
elsif
(
$n
== 1) {
push
(
@otherwises
,
$items
[0]); }
elsif
(
$items
[1]->textContent eq
'otherwise'
) {
push
(
@otherwises
,
$items
[0]); }
else
{
push
(
@rows
, [
'm:piece'
, {}, cmml_contents(
$items
[0]), cmml_contents(
$items
[1])]); } }
if
(
@otherwises
) {
if
(
@otherwises
> 1) {
Warn(
'unexpected'
,
'otherwise'
,
$node
,
"Cases statement seems to have multiple otherwise clauses"
,
@otherwises
); }
push
(
@rows
, [
'm:otherwise'
, {}, cmml_contents(
$otherwises
[0])]); }
return
[
'm:piecewise'
, {},
@rows
]; });
DefMathML(
"Token:ADDOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
"Token:ADDOP:plus"
,
undef
,
sub
{
return
[
'm:plus'
]; });
DefMathML(
"Token:ADDOP:minus"
,
undef
,
sub
{
return
[
'm:minus'
]; });
DefMathML(
'Apply:ADDOP:?'
, \
&pmml_infix
,
undef
);
DefMathML(
"Token:MULOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:MULOP:?'
, \
&pmml_infix
,
undef
);
DefMathML(
"Token:BINOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:BINOP:?'
, \
&pmml_infix
,
undef
);
DefMathML(
'Apply:FRACOP:?'
,
sub
{
my
(
$op
,
$num
,
$den
,
@more
) =
@_
;
my
$thickness
=
$op
->getAttribute(
'thickness'
);
my
$color
=
$op
->getAttribute(
'color'
) ||
$LaTeXML::MathML::COLOR
;
my
$bevelled
=
grep
{
$_
eq
'ltx_bevelled'
}
split
(/\s+/,
$op
->getAttribute(
'class'
) ||
''
);
return
[
'm:mfrac'
, { (
defined
$thickness
? (
linethickness
=>
$thickness
) : ()),
(
$color
? (
mathcolor
=>
$color
) : ()),
(
$bevelled
? (
bevelled
=>
'true'
) : ()) },
pmml_smaller(
$num
), pmml_smaller(
$den
)]; });
DefMathML(
'Apply:MODIFIEROP:?'
, \
&pmml_infix
,
undef
);
DefMathML(
"Token:MODIFIEROP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:MIDDLE:?'
, \
&pmml_infix
,
undef
);
DefMathML(
"Token:SUPOP:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:SUPERSCRIPTOP:?'
, \
&pmml_script
,
undef
);
DefMathML(
'Apply:SUBSCRIPTOP:?'
, \
&pmml_script
,
undef
);
DefMathML(
'Token:SUPERSCRIPTOP:?'
,
undef
,
sub
{
return
[
'm:csymbol'
, {
cd
=>
'ambiguous'
},
'superscript'
]; });
DefMathML(
'Token:SUBSCRIPTOP:?'
,
undef
,
sub
{
return
[
'm:csymbol'
, {
cd
=>
'ambiguous'
},
'subscript'
]; });
DefMathML(
'Apply:POSTFIX:?'
,
sub
{
return
[
'm:mrow'
, {}, pmml(
$_
[1]), pmml(
$_
[0])]; });
DefMathML(
"Token:POSTFIX:?"
, \
&pmml_mo
,
undef
);
DefMathML(
'Apply:?:square-root'
,
sub
{
my
$color
=
$_
[0]->getAttribute(
'color'
) ||
$LaTeXML::MathML::COLOR
;
return
[
'm:msqrt'
, { (
$color
? (
mathcolor
=>
$color
) : ()) }, pmml(
$_
[1])]; },
sub
{
return
[
'm:apply'
, {}, [
'm:root'
, {}], cmml(
$_
[1])]; });
DefMathML(
'Apply:?:nth-root'
,
sub
{
my
$color
=
$_
[0]->getAttribute(
'color'
) ||
$LaTeXML::MathML::COLOR
;
return
[
'm:mroot'
, { (
$color
? (
mathcolor
=>
$color
) : ()) }, pmml(
$_
[2]), pmml_scriptsize(
$_
[1])]; },
sub
{
return
[
'm:apply'
, {}, [
'm:root'
, {}], [
'm:degree'
, {}, cmml(
$_
[1])], cmml(
$_
[2])]; });
DefMathML(
"Token:?:quotient"
,
undef
,
sub
{
return
[
'm:quotient'
]; });
DefMathML(
"Token:?:factorial"
,
undef
,
sub
{
return
[
'm:factorial'
]; });
DefMathML(
"Token:?:divide"
,
undef
,
sub
{
return
[
'm:divide'
]; });
DefMathML(
"Token:?:maximum"
,
undef
,
sub
{
return
[
'm:max'
]; });
DefMathML(
"Token:?:minimum"
,
undef
,
sub
{
return
[
'm:min'
]; });
DefMathML(
"Token:?:minus"
,
undef
,
sub
{
return
[
'm:minus'
]; });
DefMathML(
"Token:?:uminus"
,
undef
,
sub
{
return
[
'm:uminus'
]; });
DefMathML(
"Token:?:plus"
,
undef
,
sub
{
return
[
'm:plus'
]; });
DefMathML(
"Token:?:power"
,
undef
,
sub
{
return
[
'm:power'
]; });
DefMathML(
"Token:?:remainder"
,
undef
,
sub
{
return
[
'm:rem'
]; });
DefMathML(
"Token:?:times"
,
undef
,
sub
{
return
[
'm:times'
]; });
DefMathML(
"Token:?:gcd"
,
undef
,
sub
{
return
[
'm:gcd'
]; });
DefMathML(
"Token:?:and"
,
undef
,
sub
{
return
[
'm:and'
]; });
DefMathML(
"Token:?:or"
,
undef
,
sub
{
return
[
'm:or'
]; });
DefMathML(
"Token:?:xor"
,
undef
,
sub
{
return
[
'm:xor'
]; });
DefMathML(
"Token:?:not"
,
undef
,
sub
{
return
[
'm:not'
]; });
DefMathML(
"Token:?:implies"
,
undef
,
sub
{
return
[
'm:implies'
]; });
DefMathML(
"Token:?:forall"
,
undef
,
sub
{
return
[
'm:forall'
]; });
DefMathML(
"Token:?:exists"
,
undef
,
sub
{
return
[
'm:exists'
]; });
DefMathML(
"Token:?:absolute-value"
,
undef
,
sub
{
return
[
'm:abs'
]; });
DefMathML(
"Token:?:conjugate"
,
undef
,
sub
{
return
[
'm:conjugate'
]; });
DefMathML(
"Token:?:argument"
,
undef
,
sub
{
return
[
'm:arg'
]; });
DefMathML(
"Token:?:real-part"
,
undef
,
sub
{
return
[
'm:real'
]; });
DefMathML(
"Token:?:imaginary-part"
,
undef
,
sub
{
return
[
'm:imaginary'
]; });
DefMathML(
"Token:?:lcm"
,
undef
,
sub
{
return
[
'm:lcm'
]; });
DefMathML(
"Token:?:floor"
,
undef
,
sub
{
return
[
'm:floor'
]; });
DefMathML(
"Token:?:ceiling"
,
undef
,
sub
{
return
[
'm:ceiling'
]; });
DefMathML(
"Token:RELOP:?"
, \
&pmml_mo
);
DefMathML(
"Token:?:equals"
,
undef
,
sub
{
return
[
'm:eq'
]; });
DefMathML(
"Token:?:not-equals"
,
undef
,
sub
{
return
[
'm:neq'
]; });
DefMathML(
"Token:?:greater-than"
,
undef
,
sub
{
return
[
'm:gt'
]; });
DefMathML(
"Token:?:less-than"
,
undef
,
sub
{
return
[
'm:lt'
]; });
DefMathML(
"Token:?:greater-than-or-equals"
,
undef
,
sub
{
return
[
'm:geq'
]; });
DefMathML(
"Token:?:less-than-or-equals"
,
undef
,
sub
{
return
[
'm:leq'
]; });
DefMathML(
"Token:?:equivalent-to"
,
undef
,
sub
{
return
[
'm:equivalent'
]; });
DefMathML(
"Token:?:approximately-equals"
,
undef
,
sub
{
return
[
'm:approx'
]; });
DefMathML(
"Apply:?:not-approximately-equals"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_not(
'm:approx'
,
@args
); });
DefMathML(
"Apply:?:less-than-or-approximately-equals"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_or_compose([
'm:lt'
,
'm:approx'
],
@args
); });
DefMathML(
"Token:?:factor-of"
,
undef
,
sub
{
return
[
'm:factorof'
]; });
DefMathML(
"Token:METARELOP:?"
, \
&pmml_mo
);
DefMathML(
'Apply:RELOP:?'
, \
&pmml_infix
);
DefMathML(
'Apply:METARELOP:?'
, \
&pmml_infix
);
DefMathML(
'Apply:?:formulae'
,
sub
{
my
(
$op
,
@elements
) =
@_
;
return
pmml_row(
map
{ pmml(
$_
) }
@elements
); },
sub
{
my
(
$op
,
@elements
) =
@_
;
return
[
'm:apply'
, {},
[
'm:csymbol'
, {
cd
=>
'ambiguous'
},
'formulae-sequence'
],
map
{ cmml(
$_
) }
@elements
];
});
DefMathML(
'Apply:?:multirelation'
,
sub
{
my
(
$op
,
@elements
) =
@_
;
return
pmml_row(
map
{ pmml(
$_
) }
@elements
); },
sub
{
my
(
$op
,
@elements
) =
@_
;
my
$lhs
= cmml(
shift
(
@elements
));
return
$lhs
unless
@elements
;
my
@relations
= ();
while
(
@elements
) {
my
$rel
=
shift
(
@elements
);
my
$rhs
=
shift
(
@elements
);
push
(
@relations
, [
'm:apply'
, {}, cmml(
$rel
),
$lhs
, cmml_shared(
$rhs
)]);
$lhs
= cmml_share(
$rhs
); }
return
(
scalar
(
@relations
) > 1 ? [
'm:apply'
, {}, [
'm:and'
, {}],
@relations
] :
$relations
[0]); }
);
DefMathML(
"Token:INTOP:?"
, \
&pmml_bigop
);
DefMathML(
"Token:LIMITOP:?"
, \
&pmml_mo
);
DefMathML(
'Apply:ARROW:?'
, \
&pmml_infix
);
DefMathML(
"Token:?:integral"
,
undef
,
sub
{
return
[
'm:int'
]; });
DefMathML(
"Token:?:differential"
,
undef
,
sub
{
return
[
'm:diff'
]; });
DefMathML(
"Token:?:partial-differential"
,
undef
,
sub
{
return
[
'm:partialdiff'
]; });
DefMathML(
"Token:?:divergence"
,
undef
,
sub
{
return
[
'm:divergence'
]; });
DefMathML(
"Token:?:gradient"
,
undef
,
sub
{
return
[
'm:grad'
]; });
DefMathML(
"Token:?:curl"
,
undef
,
sub
{
return
[
'm:curl'
]; });
DefMathML(
"Token:?:laplacian"
,
undef
,
sub
{
return
[
'm:laplacian'
]; });
DefMathML(
"Apply:?:set"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:set'
, {},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Apply:?:list"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:list'
, {},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Token:?:union"
,
undef
,
sub
{
return
[
'm:union'
]; });
DefMathML(
"Token:?:intersection"
,
undef
,
sub
{
return
[
'm:intersect'
]; });
DefMathML(
"Token:?:element-of"
,
undef
,
sub
{
return
[
'm:in'
]; });
DefMathML(
"Token:?:not-element-of"
,
undef
,
sub
{
return
[
'm:notin'
]; });
DefMathML(
"Apply:?:contains"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_complement(
'm:in'
,
@args
); });
DefMathML(
"Apply:?:not-contains"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_complement(
'm:notin'
,
@args
); });
DefMathML(
"Token:?:subset-of"
,
undef
,
sub
{
return
[
'm:subset'
]; });
DefMathML(
"Token:?:subset-of-or-equals"
,
undef
,
sub
{
return
[
'm:subset'
]; });
DefMathML(
"Token:?:subset-of-and-not-equals"
,
undef
,
sub
{
return
[
'm:prsubset'
]; });
DefMathML(
"Apply:?:superset-of"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_complement(
'm:subset'
,
@args
); });
DefMathML(
"Apply:?:superset-of-or-equals"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_complement(
'm:subset'
,
@args
); });
DefMathML(
"Apply:?:superset-of-and-not-equals"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
cmml_synth_complement(
'm:prsubset'
,
@args
); });
DefMathML(
"Token:?:set-minus"
,
undef
,
sub
{
return
[
'm:setdiff'
]; });
DefMathML(
"Token:?:cardinality"
,
undef
,
sub
{
return
[
'm:card'
]; });
DefMathML(
"Token:?:cartesian-product"
,
undef
,
sub
{
return
[
'm:cartesianproduct'
]; });
DefMathML(
"Token:BIGOP:?"
, \
&pmml_bigop
);
DefMathML(
"Token:SUMOP:?"
, \
&pmml_bigop
);
sub
pmml_summation {
my
(
$op
,
$body
) =
@_
;
return
[
'm:mrow'
, {}, pmml(
$op
), pmml(
$body
)]; }
DefMathML(
'Apply:BIGOP:?'
, \
&pmml_summation
);
DefMathML(
'Apply:INTOP:?'
, \
&pmml_summation
);
DefMathML(
'Apply:SUMOP:?'
, \
&pmml_summation
);
DefMathML(
'Apply:?:limit-from'
,
sub
{
my
(
$op
,
$arg
,
$dir
) =
@_
;
[
'm:mrow'
, {}, pmml(
$arg
), pmml(
$dir
)]; });
DefMathML(
'Apply:?:annotated'
,
sub
{
my
(
$op
,
$var
,
$annotation
) =
@_
;
return
[
'm:mrow'
, {}, pmml(
$var
),
[
'm:mspace'
, {
width
=>
'0.3888888888888889em'
}],
pmml(
$annotation
)]; });
DefMathML(
"Token:?:sum"
,
undef
,
sub
{
return
[
'm:sum'
]; });
DefMathML(
"Token:?:prod"
,
undef
,
sub
{
return
[
'm:prod'
]; });
DefMathML(
"Token:?:limit"
,
undef
,
sub
{
return
[
'm:limit'
]; });
DefMathML(
"Token:?:tends-to"
,
undef
,
sub
{
return
[
'm:tendsto'
]; });
DefMathML(
"Token:?:exponential"
,
undef
,
sub
{
return
[
'm:exp'
]; });
DefMathML(
"Token:?:natural-logarithm"
,
undef
,
sub
{
return
[
'm:ln'
]; });
DefMathML(
"Token:?:logarithm"
,
undef
,
sub
{
return
[
'm:log'
]; });
DefMathML(
"Token:?:sine"
,
undef
,
sub
{
return
[
'm:sin'
]; });
DefMathML(
"Token:?:cosine"
,
undef
,
sub
{
return
[
'm:cos'
]; });
DefMathML(
"Token:?:tangent"
,
undef
,
sub
{
return
[
'm:tan'
]; });
DefMathML(
"Token:?:secant"
,
undef
,
sub
{
return
[
'm:sec'
]; });
DefMathML(
"Token:?:cosecant"
,
undef
,
sub
{
return
[
'm:csc'
]; });
DefMathML(
"Token:?:cotangent"
,
undef
,
sub
{
return
[
'm:cot'
]; });
DefMathML(
"Token:?:hyperbolic-sine"
,
undef
,
sub
{
return
[
'm:sinh'
]; });
DefMathML(
"Token:?:hyperbolic-cosine"
,
undef
,
sub
{
return
[
'm:cosh'
]; });
DefMathML(
"Token:?:hyperbolic-tangent"
,
undef
,
sub
{
return
[
'm:tanh'
]; });
DefMathML(
"Token:?:hyperbolic-secant"
,
undef
,
sub
{
return
[
'm:sech'
]; });
DefMathML(
"Token:?:hyperbolic-cosecant"
,
undef
,
sub
{
return
[
'm:csch'
]; });
DefMathML(
"Token:?:hyperbolic-cotantent"
,
undef
,
sub
{
return
[
'm:coth'
]; });
DefMathML(
"Token:?:inverse-sine"
,
undef
,
sub
{
return
[
'm:arcsin'
]; });
DefMathML(
"Token:?:inverse-cosine"
,
undef
,
sub
{
return
[
'm:arccos'
]; });
DefMathML(
"Token:?:inverse-tangent"
,
undef
,
sub
{
return
[
'm:arctan'
]; });
DefMathML(
"Token:?:inverse-secant"
,
undef
,
sub
{
return
[
'm:arcsec'
]; });
DefMathML(
"Token:?:inverse-cosecant"
,
undef
,
sub
{
return
[
'm:arccsc'
]; });
DefMathML(
"Token:?:inverse-cotangent"
,
undef
,
sub
{
return
[
'm:arccot'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-sine"
,
undef
,
sub
{
return
[
'm:arcsinh'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-cosine"
,
undef
,
sub
{
return
[
'm:arccosh'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-tangent"
,
undef
,
sub
{
return
[
'm:arctanh'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-secant"
,
undef
,
sub
{
return
[
'm:arcsech'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-cosecant"
,
undef
,
sub
{
return
[
'm:arccsch'
]; });
DefMathML(
"Token:?:inverse-hyperbolic-cotangent"
,
undef
,
sub
{
return
[
'm:arccoth'
]; });
DefMathML(
"Token:?:mean"
,
undef
,
sub
{
return
[
'm:mean'
]; });
DefMathML(
"Token:?:standard-deviation"
,
undef
,
sub
{
return
[
'm:sdev'
]; });
DefMathML(
"Token:?:variance"
,
undef
,
sub
{
return
[
'm:var'
]; });
DefMathML(
"Token:?:median"
,
undef
,
sub
{
return
[
'm:median'
]; });
DefMathML(
"Token:?:mode"
,
undef
,
sub
{
return
[
'm:mode'
]; });
DefMathML(
"Token:?:moment"
,
undef
,
sub
{
return
[
'm:moment'
]; });
DefMathML(
"Apply:?:vector"
,
undef
,
sub
{
my
(
$op
,
@args
) =
@_
;
return
[
'm:vector'
, {},
map
{ cmml(
$_
) }
@args
]; });
DefMathML(
"Token:?:determinant"
,
undef
,
sub
{
return
[
'm:determinant'
]; });
DefMathML(
"Token:?:transpose"
,
undef
,
sub
{
return
[
'm:transpose'
]; });
DefMathML(
"Token:?:selector"
,
undef
,
sub
{
return
[
'm:selector'
]; });
DefMathML(
"Token:?:vector-product"
,
undef
,
sub
{
return
[
'm:vectorproduct'
]; });
DefMathML(
"Token:?:scalar-product"
,
undef
,
sub
{
return
[
'm:scalarproduct'
]; });
DefMathML(
"Token:?:outer-product"
,
undef
,
sub
{
return
[
'm:outerproduct'
]; });
DefMathML(
"Array:?:?"
,
undef
,
sub
{
my
(
$node
) =
@_
;
return
[
'm:matrix'
, {},
map
{ [
'm:matrixrow'
, {},
map
{ cmml_contents(
$_
) } element_nodes(
$_
)] }
element_nodes(
$node
)]; });
DefMathML(
"Token:ID:integers"
,
undef
,
sub
{
return
[
'm:integers'
]; });
DefMathML(
"Token:ID:reals"
,
undef
,
sub
{
return
[
'm:reals'
]; });
DefMathML(
"Token:ID:rationals"
,
undef
,
sub
{
return
[
'm:rationals'
]; });
DefMathML(
"Token:ID:numbers"
,
undef
,
sub
{
return
[
'm:naturalnumbers'
]; });
DefMathML(
"Token:ID:complexes"
,
undef
,
sub
{
return
[
'm:complexes'
]; });
DefMathML(
"Token:ID:primes"
,
undef
,
sub
{
return
[
'm:primes'
]; });
DefMathML(
"Token:ID:exponential-e"
,
undef
,
sub
{
return
[
'm:exponentiale'
]; });
DefMathML(
"Token:ID:imaginary-i"
,
undef
,
sub
{
return
[
'm:imaginaryi'
]; });
DefMathML(
"Token:ID:notanumber"
,
undef
,
sub
{
return
[
'm:notanumber'
]; });
DefMathML(
"Token:ID:true"
,
undef
,
sub
{
return
[
'm:true'
]; });
DefMathML(
"Token:ID:false"
,
undef
,
sub
{
return
[
'm:false'
]; });
DefMathML(
"Token:ID:empty-set"
,
undef
,
sub
{
return
[
'm:emptyset'
]; });
DefMathML(
"Token:ID:circular-pi"
,
undef
,
sub
{
return
[
'm:pi'
]; });
DefMathML(
"Token:ID:Euler-constant"
,
undef
,
sub
{
return
[
'm:eulergamma'
]; });
DefMathML(
"Token:ID:infinity"
,
undef
,
sub
{
return
[
'm:infinity'
]; });
sub
do_cfrac {
my
(
$numer
,
$denom
) =
@_
;
if
(getQName(
$denom
) eq
'ltx:XMApp'
) {
my
(
$denomop
,
@denomargs
) = element_nodes(
$denom
);
if
(((
$denomop
->getAttribute(
'role'
) ||
''
) eq
'ADDOP'
)
|| ((
$denomop
->textContent ||
''
) eq
"\x{22EF}"
)) {
my
$last
=
pop
(
@denomargs
);
my
$curr
= [
'm:mfrac'
, {}, pmml_smaller(
$numer
),
[
'm:mrow'
, {},
(
@denomargs
> 1 ? pmml_infix(
$denomop
,
@denomargs
) : pmml_smaller(
$denomargs
[0])),
pmml_smaller(
$denomop
)]];
if
((
$last
->textContent ||
''
) eq
"\x{22EF}"
) {
return
(
$curr
, pmml_smaller(
$last
)); }
elsif
(getQName(
$last
) eq
'ltx:XMApp'
) {
my
(
$lastop
,
@lastargs
) = element_nodes(
$last
);
if
((
$lastop
->getAttribute(
'meaning'
) ||
''
) eq
'continued-fraction'
) {
return
(
$curr
, do_cfrac(
@lastargs
)); }
elsif
(((
$lastop
->textContent ||
''
) eq
"\x{2062}"
)
&& (
scalar
(
@lastargs
) == 2) && ((
$lastargs
[0]->textContent ||
''
) eq
"\x{22EF}"
)) {
return
(
$curr
, pmml_smaller(
$lastargs
[0]), pmml_smaller(
$lastargs
[1])); } } } }
return
[
'm:mfrac'
, {}, pmml_smaller(
$numer
), pmml_smaller(
$denom
)]; }
DefMathML(
'Apply:?:continued-fraction'
,
sub
{
my
(
$op
,
$numer
,
$denom
) =
@_
;
my
$style
= ((
$op
->getAttribute(
'name'
) ||
''
) eq
'cfrac-inline'
?
'inline'
:
'display'
);
if
(
$style
eq
'inline'
) {
return
pmml_row(do_cfrac(
$numer
,
$denom
)); }
else
{
return
[
'm:mfrac'
, {}, pmml_smaller(
$numer
), pmml_smaller(
$denom
)]; } });
DefMathML(
'Apply:?:hack-definite-integral'
,
undef
,
sub
{
my
(
$op
,
$lower
,
$upper
,
$integrand
,
$variable
) =
@_
;
return
[
'm:apply'
, {},
[
'm:int'
],
[
'm:bvar'
, {}, cmml(
$variable
)],
[
'm:lowlimit'
, {}, cmml(
$lower
)],
[
'm:uplimit'
, {}, cmml(
$upper
)],
cmml(
$integrand
)]; });
1;