module Text.TeXMath.Writers.OMML (writeOMML)
where
import Text.XML.Light
import Text.TeXMath.Types
import Data.Generics (everywhere, mkT)
import Data.Char (isSymbol, isPunctuation)
writeOMML :: DisplayType -> [Exp] -> Element
writeOMML dt = container . concatMap (showExp [])
. everywhere (mkT $ handleDownup dt)
. everywhere (mkT $ handleDownup' dt)
where container = case dt of
DisplayBlock -> \x -> mnode "oMathPara"
[ mnode "oMathParaPr"
$ mnodeA "jc" "center" ()
, mnode "oMath" x ]
DisplayInline -> mnode "oMath"
mnode :: Node t => String -> t -> Element
mnode s = node (QName s Nothing (Just "m"))
mnodeA :: Node t => String -> String -> t -> Element
mnodeA s v = add_attr (Attr (QName "val" Nothing (Just "m")) v) . mnode s
str :: [Element] -> String -> Element
str [] s = mnode "r" [ mnode "t" s ]
str props s = mnode "r" [ mnode "rPr" props
, mnode "t" s ]
showFraction :: [Element] -> FractionType -> Exp -> Exp -> Element
showFraction props ft x y =
case ft of
NormalFrac -> mnode "f" [ mnode "fPr" $
mnodeA "type" "bar" ()
, mnode "num" x'
, mnode "den" y']
DisplayFrac -> showFraction props NormalFrac x y
InlineFrac -> mnode "f" [ mnode "fPr" $
mnodeA "type" "lin" ()
, mnode "num" x'
, mnode "den" y']
NoLineFrac -> mnode "f" [ mnode "fPr" $
mnodeA "type" "noBar" ()
, mnode "num" x'
, mnode "den" y'
]
where x' = showExp props x
y' = showExp props y
maximum' :: [Int] -> Int
maximum' [] = 0
maximum' xs = maximum xs
makeArray :: [Element] -> [Alignment] -> [ArrayLine] -> Element
makeArray props as rs = mnode "m" $ mProps : map toMr rs
where mProps = mnode "mPr"
[ mnodeA "baseJc" "center" ()
, mnodeA "plcHide" "1" ()
, mnode "mcs" $ map toMc as' ]
as' = take (maximum' $ map length rs) $ as ++ cycle [AlignCenter]
toMr r = mnode "mr" $ map (mnode "e" . concatMap (showExp props)) r
toMc a = mnode "mc" $ mnode "mcPr"
[ mnodeA "mcJc" (toAlign a) ()
, mnodeA "count" "1" ()
]
toAlign AlignLeft = "left"
toAlign AlignRight = "right"
toAlign AlignCenter = "center"
makeText :: TextType -> String -> Element
makeText a s = str (setProps a) s
setProps :: TextType -> [Element]
setProps tt =
case tt of
TextNormal -> [sty "p"]
TextBold -> [sty "b"]
TextItalic -> [sty "i"]
TextMonospace -> [sty "p", scr "monospace"]
TextSansSerif -> [sty "p", scr "sans-serif"]
TextDoubleStruck -> [sty "p", scr "double-struck"]
TextScript -> [sty "p", scr "script"]
TextFraktur -> [sty "p", scr "fraktur"]
TextBoldItalic -> [sty "bi"]
TextSansSerifBold -> [sty "b", scr "sans-serif"]
TextBoldScript -> [sty "b", scr "script"]
TextBoldFraktur -> [sty "b", scr "fraktur"]
TextSansSerifItalic -> [sty "i", scr "sans-serif"]
TextSansSerifBoldItalic -> [sty "bi", scr "sans-serif"]
where sty x = mnodeA "sty" x ()
scr x = mnodeA "scr" x ()
handleDownup :: DisplayType -> [Exp] -> [Exp]
handleDownup dt (exp' : xs) =
case exp' of
EOver convertible x y
| isNary x ->
EGrouped [EUnderover convertible x emptyGroup y, next] : rest
| convertible && dt == DisplayInline -> ESuper x y : xs
EUnder convertible x y
| isNary x ->
EGrouped [EUnderover convertible x y emptyGroup, next] : rest
| convertible && dt == DisplayInline -> ESub x y : xs
EUnderover convertible x y z
| isNary x ->
EGrouped [EUnderover convertible x y z, next] : rest
| convertible && dt == DisplayInline -> ESubsup x y z : xs
ESub x y
| isNary x -> EGrouped [ESubsup x y emptyGroup, next] : rest
ESuper x y
| isNary x -> EGrouped [ESubsup x emptyGroup y, next] : rest
ESubsup x y z
| isNary x -> EGrouped [ESubsup x y z, next] : rest
_ -> exp' : xs
where (next, rest) = case xs of
(t:ts) -> (t,ts)
[] -> (emptyGroup, [])
emptyGroup = EGrouped []
handleDownup _ [] = []
handleDownup' :: DisplayType -> [InEDelimited] -> [InEDelimited]
handleDownup' dt ((Right exp') : xs) =
case exp' of
EOver convertible x y
| isNary x ->
Right (EGrouped [EUnderover convertible x emptyGroup y, next]) :
rest
| convertible && dt == DisplayInline -> Right (ESuper x y) : xs
EUnder convertible x y
| isNary x ->
Right (EGrouped [EUnderover convertible x y emptyGroup, next]) :
rest
| convertible && dt == DisplayInline -> Right (ESub x y) : xs
EUnderover convertible x y z
| isNary x ->
Right (EGrouped [EUnderover convertible x y z, next]) : rest
| convertible && dt == DisplayInline -> Right (ESubsup x y z) : xs
ESub x y
| isNary x -> Right (EGrouped [ESubsup x y emptyGroup, next]) : rest
ESuper x y
| isNary x -> Right (EGrouped [ESubsup x emptyGroup y, next]) : rest
ESubsup x y z
| isNary x -> Right (EGrouped [ESubsup x y z, next]) : rest
_ -> Right exp' : xs
where (next, rest) = case xs of
(Right t:ts) -> (t,ts)
_ -> (emptyGroup, xs)
emptyGroup = EGrouped []
handleDownup' _ xs = xs
showExp :: [Element] -> Exp -> [Element]
showExp props e =
case e of
ENumber x -> [str props x]
EGrouped [EUnderover _ (ESymbol Op s) y z, w] ->
[makeNary props "undOvr" s y z w]
EGrouped [ESubsup (ESymbol Op s) y z, w] ->
[makeNary props "subSup" s y z w]
EGrouped xs -> concatMap (showExp props) xs
EDelimited start end xs ->
[mnode "d" [ mnode "dPr"
[ mnodeA "begChr" start ()
, mnodeA "endChr" end ()
, mnode "grow" () ]
, mnode "e" $ concatMap
(either ((:[]) . str props) (showExp props)) xs
] ]
EIdentifier x -> [str props x]
EMathOperator x -> [makeText TextNormal x]
ESymbol _ [c]
| isSymbol c || isPunctuation c
-> [str props [c]]
ESymbol ty xs
| ty `elem` [Op, Bin, Rel]
-> [mnode "box"
[ mnode "boxPr"
[ mnodeA "opEmu" "1" () ]
, mnode "e"
[str props xs]
]]
ESymbol _ xs -> [str props xs]
ESpace n
| n > 0 && n <= 0.17 -> [str props "\x2009"]
| n > 0.17 && n <= 0.23 -> [str props "\x2005"]
| n > 0.23 && n <= 0.28 -> [str props "\x2004"]
| n > 0.28 && n <= 0.5 -> [str props "\x2004"]
| n > 0.5 && n <= 1.8 -> [str props "\x2001"]
| n > 1.8 -> [str props "\x2001\x2001"]
| otherwise -> []
EUnder _ x (ESymbol _ [c]) | isBarChar c ->
[mnode "bar" [ mnode "barPr" $
mnodeA "pos" "bot" ()
, mnode "e" $ showExp props x ]]
EOver _ x (ESymbol _ [c]) | isBarChar c ->
[mnode "bar" [ mnode "barPr" $
mnodeA "pos" "top" ()
, mnode "e" $ showExp props x ]]
EOver _ x (ESymbol st y) | st == Accent || st == TOver ->
[mnode "groupChr" [ mnode "groupChrPr"
[ mnodeA "chr" y ()
, mnodeA "pos" "top" ()
, mnodeA "vertJc" "bot" () ]
, mnode "e" $ showExp props x ]]
EUnder _ x (ESymbol st y) | st == Accent || st == TUnder ->
[mnode "groupChr" [ mnode "groupChrPr"
[ mnodeA "chr" y ()
, mnodeA "pos" "bot" ()
, mnodeA "vertJc" "top" () ]
, mnode "e" $ showExp props x ]]
ESub x y -> [mnode "sSub" [ mnode "e" $ showExp props x
, mnode "sub" $ showExp props y]]
ESuper x y -> [mnode "sSup" [ mnode "e" $ showExp props x
, mnode "sup" $ showExp props y]]
ESubsup x y z -> [mnode "sSubSup" [ mnode "e" $ showExp props x
, mnode "sub" $ showExp props y
, mnode "sup" $ showExp props z]]
EUnder _ x y -> [mnode "limLow" [ mnode "e" $ showExp props x
, mnode "lim" $ showExp props y]]
EOver _ x y -> [mnode "limUpp" [ mnode "e" $ showExp props x
, mnode "lim" $ showExp props y]]
EUnderover c x y z -> showExp props (EUnder c (EOver c x z) y)
ESqrt x -> [mnode "rad" [ mnode "radPr" $ mnodeA "degHide" "1" ()
, mnode "deg" ()
, mnode "e" $ showExp props x]]
ERoot i x -> [mnode "rad" [ mnode "deg" $ showExp props i
, mnode "e" $ showExp props x]]
EFraction ft x y -> [showFraction props ft x y]
EPhantom x -> [mnode "phant" [ mnode "phantPr"
[ mnodeA "show" "0" () ]
, mnode "e" $ showExp props x]]
EBoxed x -> [mnode "borderBox" [ mnode "e" $ showExp props x]]
EScaled _ x -> showExp props x
EArray as ls -> [makeArray props as ls]
EText a s -> [makeText a s]
EStyled a es -> concatMap (showExp (setProps a)) es
isBarChar :: Char -> Bool
isBarChar c = c == '\x203E' || c == '\x00AF' ||
c == '\x0304' || c == '\x0333'
isNary :: Exp -> Bool
isNary (ESymbol Op _) = True
isNary _ = False
makeNary :: [Element] -> String -> String -> Exp -> Exp -> Exp -> Element
makeNary props t s y z w =
mnode "nary" [ mnode "naryPr"
[ mnodeA "chr" s ()
, mnodeA "limLoc" t ()
, mnodeA "subHide"
(if y == EGrouped [] then "1" else "0") ()
, mnodeA "supHide"
(if z == EGrouped [] then "1" else "0") ()
]
, mnode "sub" $ showExp props y
, mnode "sup" $ showExp props z
, mnode "e" $ showExp props w ]