42fe25d8a40ed241a130fecc1bd8b659286b1e72
[WebKit-https.git] / Source / WebCore / mathml / MathMLElement.cpp
1 /*
2  * Copyright (C) 2009 Alex Milowski (alex@milowski.com). All rights reserved.
3  * Copyright (C) 2010 Apple Inc. All rights reserved.
4  * Copyright (C) 2010 Fran├žois Sausset (sausset@gmail.com). All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
18  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
19  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
21  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27
28 #include "config.h"
29
30 #if ENABLE(MATHML)
31
32 #include "MathMLElement.h"
33
34 #include "MathMLNames.h"
35 #include "RenderTableCell.h"
36
37 namespace WebCore {
38     
39 using namespace MathMLNames;
40     
41 MathMLElement::MathMLElement(const QualifiedName& tagName, Document& document)
42     : StyledElement(tagName, document, CreateMathMLElement)
43 {
44 }
45     
46 PassRefPtr<MathMLElement> MathMLElement::create(const QualifiedName& tagName, Document& document)
47 {
48     return adoptRef(new MathMLElement(tagName, document));
49 }
50
51 bool MathMLElement::isPresentationMathML() const
52 {
53     return hasTagName(MathMLNames::mtrTag) || hasTagName(MathMLNames::mtdTag) || hasTagName(MathMLNames::maligngroupTag) || hasTagName(MathMLNames::malignmarkTag) || hasTagName(MathMLNames::mencloseTag) || hasTagName(MathMLNames::mglyphTag) || hasTagName(MathMLNames::mlabeledtrTag) || hasTagName(MathMLNames::mlongdivTag) || hasTagName(MathMLNames::mpaddedTag) || hasTagName(MathMLNames::msTag) || hasTagName(MathMLNames::mscarriesTag) || hasTagName(MathMLNames::mscarryTag) || hasTagName(MathMLNames::msgroupTag) || hasTagName(MathMLNames::mslineTag) || hasTagName(MathMLNames::msrowTag) || hasTagName(MathMLNames::mstackTag);
54 }
55
56 int MathMLElement::colSpan() const
57 {
58     if (!hasTagName(mtdTag))
59         return 1;
60     const AtomicString& colSpanValue = fastGetAttribute(columnspanAttr);
61     return std::max(1, colSpanValue.toInt());
62 }
63
64 int MathMLElement::rowSpan() const
65 {
66     if (!hasTagName(mtdTag))
67         return 1;
68     const AtomicString& rowSpanValue = fastGetAttribute(rowspanAttr);
69     return std::max(1, rowSpanValue.toInt());
70 }
71
72 void MathMLElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
73 {
74     if (name == rowspanAttr) {
75         if (renderer() && renderer()->isTableCell() && hasTagName(mtdTag))
76             toRenderTableCell(renderer())->colSpanOrRowSpanChanged();
77     } else if (name == columnspanAttr) {
78         if (renderer() && renderer()->isTableCell() && hasTagName(mtdTag))
79             toRenderTableCell(renderer())->colSpanOrRowSpanChanged();
80     } else
81         StyledElement::parseAttribute(name, value);
82 }
83
84 bool MathMLElement::isPresentationAttribute(const QualifiedName& name) const
85 {
86     if (name == backgroundAttr || name == colorAttr || name == dirAttr || name == fontfamilyAttr || name == fontsizeAttr || name == fontstyleAttr || name == fontweightAttr || name == mathbackgroundAttr || name == mathcolorAttr || name == mathsizeAttr)
87         return true;
88     return StyledElement::isPresentationAttribute(name);
89 }
90
91 void MathMLElement::collectStyleForPresentationAttribute(const QualifiedName& name, const AtomicString& value, MutableStyleProperties& style)
92 {
93     if (name == mathbackgroundAttr)
94         addPropertyToPresentationAttributeStyle(style, CSSPropertyBackgroundColor, value);
95     else if (name == mathsizeAttr) {
96         // The following three values of mathsize are handled in WebCore/css/mathml.css
97         if (value != "normal" && value != "small" && value != "big")
98             addPropertyToPresentationAttributeStyle(style, CSSPropertyFontSize, value);
99     } else if (name == mathcolorAttr)
100         addPropertyToPresentationAttributeStyle(style, CSSPropertyColor, value);
101     // FIXME: deprecated attributes that should loose in a conflict with a non deprecated attribute
102     else if (name == fontsizeAttr)
103         addPropertyToPresentationAttributeStyle(style, CSSPropertyFontSize, value);
104     else if (name == backgroundAttr)
105         addPropertyToPresentationAttributeStyle(style, CSSPropertyBackgroundColor, value);
106     else if (name == colorAttr)
107         addPropertyToPresentationAttributeStyle(style, CSSPropertyColor, value);
108     else if (name == fontstyleAttr)
109         addPropertyToPresentationAttributeStyle(style, CSSPropertyFontStyle, value);
110     else if (name == fontweightAttr)
111         addPropertyToPresentationAttributeStyle(style, CSSPropertyFontWeight, value);
112     else if (name == fontfamilyAttr)
113         addPropertyToPresentationAttributeStyle(style, CSSPropertyFontFamily, value);
114     else if (name == dirAttr) {
115         if (hasTagName(mathTag) || hasTagName(mrowTag) || hasTagName(mstyleTag) || isMathMLToken())
116             addPropertyToPresentationAttributeStyle(style, CSSPropertyDirection, value);
117     }  else {
118         ASSERT(!isPresentationAttribute(name));
119         StyledElement::collectStyleForPresentationAttribute(name, value
120         , style);
121     }
122 }
123
124 bool MathMLElement::childShouldCreateRenderer(const Node& child) const
125 {
126     if (hasTagName(annotationTag))
127         return child.isTextNode();
128     if (hasTagName(annotation_xmlTag))
129         return StyledElement::childShouldCreateRenderer(child);
130
131     // Only create renderers for MathML elements or text. MathML prohibits non-MathML markup inside a <math> element.
132     return child.isTextNode() || child.isMathMLElement();
133 }
134
135 void MathMLElement::attributeChanged(const QualifiedName& name, const AtomicString& oldValue, const AtomicString& newValue, AttributeModificationReason reason)
136 {
137     if (isSemanticAnnotation() && (name == MathMLNames::srcAttr || name == MathMLNames::encodingAttr)) {
138         Element* parent = parentElement();
139         if (parent && parent->isMathMLElement() && parent->hasTagName(semanticsTag))
140             toMathMLElement(parent)->updateSelectedChild();
141     }
142     StyledElement::attributeChanged(name, oldValue, newValue, reason);
143 }
144
145 }
146
147 #endif // ENABLE(MATHML)