| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (C) 2007, 2010 Rob Buis <buis@kde.org> | 2 * Copyright (C) 2007, 2010 Rob Buis <buis@kde.org> |
| 3 * | 3 * |
| 4 * This library is free software; you can redistribute it and/or | 4 * This library is free software; you can redistribute it and/or |
| 5 * modify it under the terms of the GNU Library General Public | 5 * modify it under the terms of the GNU Library General Public |
| 6 * License as published by the Free Software Foundation; either | 6 * License as published by the Free Software Foundation; either |
| 7 * version 2 of the License, or (at your option) any later version. | 7 * version 2 of the License, or (at your option) any later version. |
| 8 * | 8 * |
| 9 * This library is distributed in the hope that it will be useful, | 9 * This library is distributed in the hope that it will be useful, |
| 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 12 * Library General Public License for more details. | 12 * Library General Public License for more details. |
| 13 * | 13 * |
| 14 * You should have received a copy of the GNU Library General Public License | 14 * You should have received a copy of the GNU Library General Public License |
| 15 * along with this library; see the file COPYING.LIB. If not, write to | 15 * along with this library; see the file COPYING.LIB. If not, write to |
| 16 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, | 16 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| 17 * Boston, MA 02110-1301, USA. | 17 * Boston, MA 02110-1301, USA. |
| 18 */ | 18 */ |
| 19 | 19 |
| 20 #include "config.h" | 20 #include "config.h" |
| 21 #include "core/svg/SVGViewSpec.h" | 21 #include "core/svg/SVGViewSpec.h" |
| 22 | 22 |
| 23 #include "SVGNames.h" | 23 #include "SVGNames.h" |
| 24 #include "core/dom/Document.h" | 24 #include "core/dom/Document.h" |
| 25 #include "core/svg/SVGAnimatedTransformList.h" | 25 #include "core/svg/SVGAnimatedTransformList.h" |
| 26 #include "core/svg/SVGParserUtilities.h" | 26 #include "core/svg/SVGParserUtilities.h" |
| 27 | 27 |
| 28 namespace WebCore { | 28 namespace WebCore { |
| 29 | 29 |
| 30 // Define custom non-animated property 'transform'. | |
| 31 const SVGPropertyInfo* SVGViewSpec::transformPropertyInfo() | |
| 32 { | |
| 33 static const SVGPropertyInfo* s_propertyInfo = 0; | |
| 34 if (!s_propertyInfo) { | |
| 35 s_propertyInfo = new SVGPropertyInfo(AnimatedTransformList, | |
| 36 PropertyIsReadOnly, | |
| 37 SVGNames::transformAttr, | |
| 38 transformIdentifier(), | |
| 39 0, | |
| 40 0); | |
| 41 } | |
| 42 return s_propertyInfo; | |
| 43 } | |
| 44 | |
| 45 SVGViewSpec::SVGViewSpec(SVGSVGElement* contextElement) | 30 SVGViewSpec::SVGViewSpec(SVGSVGElement* contextElement) |
| 46 // Note: We make |viewBox| and |preserveAspectRatio|'s contextElement the ta
rget element of SVGViewSpec. | 31 // Note: addToPropertyMap is not needed, as SVGViewSpec do not correspond to
an element. |
| 32 // Note: We make tear-offs' contextElement the target element of SVGViewSpec
. |
| 47 // This contextElement will be only used for keeping this alive from the tea
roff. | 33 // This contextElement will be only used for keeping this alive from the tea
roff. |
| 48 // SVGSVGElement holds a strong-ref to this SVGViewSpec, so this is kept ali
ve as: | 34 // SVGSVGElement holds a strong-ref to this SVGViewSpec, so this is kept ali
ve as: |
| 49 // AnimatedProperty tearoff -(contextElement)-> SVGSVGElement -(RefPtr)-> SV
GViewSpec. | 35 // AnimatedProperty tearoff -(contextElement)-> SVGSVGElement -(RefPtr)-> SV
GViewSpec. |
| 50 : SVGFitToViewBox(contextElement, PropertyMapPolicySkip) // Note: addToPrope
rtyMap is not needed, as SVGViewSpec do not correspond to an element. | 36 : SVGFitToViewBox(contextElement, PropertyMapPolicySkip) |
| 51 , m_contextElement(contextElement) | 37 , m_contextElement(contextElement) |
| 38 , m_transform(SVGAnimatedTransformList::create(contextElement, SVGNames::tra
nsformAttr, SVGTransformList::create())) |
| 52 { | 39 { |
| 53 ASSERT(m_contextElement); | 40 ASSERT(m_contextElement); |
| 54 ScriptWrappable::init(this); | 41 ScriptWrappable::init(this); |
| 55 | 42 |
| 56 viewBox()->setReadOnly(); | 43 viewBox()->setReadOnly(); |
| 57 preserveAspectRatio()->setReadOnly(); | 44 preserveAspectRatio()->setReadOnly(); |
| 58 } | 45 m_transform->setReadOnly(); |
| 59 | 46 // Note: addToPropertyMap is not needed, as SVGViewSpec do not correspond to
an element. |
| 60 const AtomicString& SVGViewSpec::transformIdentifier() | |
| 61 { | |
| 62 DEFINE_STATIC_LOCAL(AtomicString, s_identifier, ("SVGViewSpecTransformAttrib
ute", AtomicString::ConstructFromLiteral)); | |
| 63 return s_identifier; | |
| 64 } | 47 } |
| 65 | 48 |
| 66 String SVGViewSpec::preserveAspectRatioString() const | 49 String SVGViewSpec::preserveAspectRatioString() const |
| 67 { | 50 { |
| 68 return preserveAspectRatio()->baseValue()->valueAsString(); | 51 return preserveAspectRatio()->baseValue()->valueAsString(); |
| 69 } | 52 } |
| 70 | 53 |
| 71 void SVGViewSpec::setTransformString(const String& transform) | |
| 72 { | |
| 73 if (!m_contextElement) | |
| 74 return; | |
| 75 | |
| 76 SVGTransformList newList; | |
| 77 newList.parse(transform); | |
| 78 | |
| 79 if (SVGAnimatedProperty* wrapper = SVGAnimatedProperty::lookupWrapper<SVGEle
ment, SVGAnimatedTransformList>(m_contextElement, transformPropertyInfo())) | |
| 80 static_cast<SVGAnimatedTransformList*>(wrapper)->detachListWrappers(newL
ist.size()); | |
| 81 | |
| 82 m_transform = newList; | |
| 83 } | |
| 84 | |
| 85 String SVGViewSpec::transformString() const | 54 String SVGViewSpec::transformString() const |
| 86 { | 55 { |
| 87 return SVGPropertyTraits<SVGTransformList>::toString(m_transform); | 56 if (!m_transform) |
| 57 return String(); |
| 58 |
| 59 return m_transform->baseValue()->valueAsString(); |
| 88 } | 60 } |
| 89 | 61 |
| 90 String SVGViewSpec::viewBoxString() const | 62 String SVGViewSpec::viewBoxString() const |
| 91 { | 63 { |
| 92 return viewBox()->currentValue()->valueAsString(); | 64 return viewBox()->currentValue()->valueAsString(); |
| 93 } | 65 } |
| 94 | 66 |
| 95 SVGElement* SVGViewSpec::viewTarget() const | 67 SVGElement* SVGViewSpec::viewTarget() const |
| 96 { | 68 { |
| 97 if (!m_contextElement) | 69 if (!m_contextElement) |
| 98 return 0; | 70 return 0; |
| 99 Element* element = m_contextElement->treeScope().getElementById(AtomicString
(m_viewTargetString)); | 71 Element* element = m_contextElement->treeScope().getElementById(AtomicString
(m_viewTargetString)); |
| 100 if (!element || !element->isSVGElement()) | 72 if (!element || !element->isSVGElement()) |
| 101 return 0; | 73 return 0; |
| 102 return toSVGElement(element); | 74 return toSVGElement(element); |
| 103 } | 75 } |
| 104 | 76 |
| 105 SVGTransformListPropertyTearOff* SVGViewSpec::transform() | |
| 106 { | |
| 107 if (!m_contextElement) | |
| 108 return 0; | |
| 109 // Return the animVal here, as its readonly by default - which is exactly wh
at we want here. | |
| 110 return static_cast<SVGTransformListPropertyTearOff*>(static_pointer_cast<SVG
AnimatedTransformList>(lookupOrCreateTransformWrapper(this))->animVal()); | |
| 111 } | |
| 112 | |
| 113 PassRefPtr<SVGAnimatedProperty> SVGViewSpec::lookupOrCreateTransformWrapper(SVGV
iewSpec* ownerType) | |
| 114 { | |
| 115 ASSERT(ownerType); | |
| 116 ASSERT(ownerType->contextElement()); | |
| 117 return SVGAnimatedProperty::lookupOrCreateWrapper<SVGElement, SVGAnimatedTra
nsformList, SVGTransformList>(ownerType->contextElement(), transformPropertyInfo
(), ownerType->m_transform); | |
| 118 } | |
| 119 | |
| 120 void SVGViewSpec::detachContextElement() | 77 void SVGViewSpec::detachContextElement() |
| 121 { | 78 { |
| 79 m_transform = 0; |
| 122 clearViewBox(); | 80 clearViewBox(); |
| 123 clearPreserveAspectRatio(); | 81 clearPreserveAspectRatio(); |
| 124 m_contextElement = 0; | 82 m_contextElement = 0; |
| 125 } | 83 } |
| 126 | 84 |
| 127 void SVGViewSpec::reset() | 85 void SVGViewSpec::reset() |
| 128 { | 86 { |
| 129 resetZoomAndPan(); | 87 resetZoomAndPan(); |
| 130 m_transform.clear(); | 88 m_transform->baseValue()->clear(); |
| 131 updateViewBox(FloatRect()); | 89 updateViewBox(FloatRect()); |
| 132 ASSERT(preserveAspectRatio()); | 90 ASSERT(preserveAspectRatio()); |
| 133 preserveAspectRatio()->baseValue()->setAlign(SVGPreserveAspectRatio::SVG_PRE
SERVEASPECTRATIO_XMIDYMID); | 91 preserveAspectRatio()->baseValue()->setAlign(SVGPreserveAspectRatio::SVG_PRE
SERVEASPECTRATIO_XMIDYMID); |
| 134 preserveAspectRatio()->baseValue()->setMeetOrSlice(SVGPreserveAspectRatio::S
VG_MEETORSLICE_MEET); | 92 preserveAspectRatio()->baseValue()->setMeetOrSlice(SVGPreserveAspectRatio::S
VG_MEETORSLICE_MEET); |
| 135 m_viewTargetString = emptyString(); | 93 m_viewTargetString = emptyString(); |
| 136 } | 94 } |
| 137 | 95 |
| 138 static const LChar svgViewSpec[] = {'s', 'v', 'g', 'V', 'i', 'e', 'w'}; | 96 static const LChar svgViewSpec[] = {'s', 'v', 'g', 'V', 'i', 'e', 'w'}; |
| 139 static const LChar viewBoxSpec[] = {'v', 'i', 'e', 'w', 'B', 'o', 'x'}; | 97 static const LChar viewBoxSpec[] = {'v', 'i', 'e', 'w', 'B', 'o', 'x'}; |
| 140 static const LChar preserveAspectRatioSpec[] = {'p', 'r', 'e', 's', 'e', 'r', 'v
', 'e', 'A', 's', 'p', 'e', 'c', 't', 'R', 'a', 't', 'i', 'o'}; | 98 static const LChar preserveAspectRatioSpec[] = {'p', 'r', 'e', 's', 'e', 'r', 'v
', 'e', 'A', 's', 'p', 'e', 'c', 't', 'R', 'a', 't', 'i', 'o'}; |
| (...skipping 28 matching lines...) Expand all Loading... |
| 169 return false; | 127 return false; |
| 170 ptr++; | 128 ptr++; |
| 171 } else if (skipString(ptr, end, viewTargetSpec, WTF_ARRAY_LENGTH(vie
wTargetSpec))) { | 129 } else if (skipString(ptr, end, viewTargetSpec, WTF_ARRAY_LENGTH(vie
wTargetSpec))) { |
| 172 if (ptr >= end || *ptr != '(') | 130 if (ptr >= end || *ptr != '(') |
| 173 return false; | 131 return false; |
| 174 const CharType* viewTargetStart = ++ptr; | 132 const CharType* viewTargetStart = ++ptr; |
| 175 while (ptr < end && *ptr != ')') | 133 while (ptr < end && *ptr != ')') |
| 176 ptr++; | 134 ptr++; |
| 177 if (ptr >= end) | 135 if (ptr >= end) |
| 178 return false; | 136 return false; |
| 179 setViewTargetString(String(viewTargetStart, ptr - viewTargetStar
t)); | 137 m_viewTargetString = String(viewTargetStart, ptr - viewTargetSta
rt); |
| 180 ptr++; | 138 ptr++; |
| 181 } else | 139 } else |
| 182 return false; | 140 return false; |
| 183 } else if (*ptr == 'z') { | 141 } else if (*ptr == 'z') { |
| 184 if (!skipString(ptr, end, zoomAndPanSpec, WTF_ARRAY_LENGTH(zoomAndPa
nSpec))) | 142 if (!skipString(ptr, end, zoomAndPanSpec, WTF_ARRAY_LENGTH(zoomAndPa
nSpec))) |
| 185 return false; | 143 return false; |
| 186 if (ptr >= end || *ptr != '(') | 144 if (ptr >= end || *ptr != '(') |
| 187 return false; | 145 return false; |
| 188 ptr++; | 146 ptr++; |
| 189 if (!parseZoomAndPan(ptr, end)) | 147 if (!parseZoomAndPan(ptr, end)) |
| (...skipping 11 matching lines...) Expand all Loading... |
| 201 return false; | 159 return false; |
| 202 if (ptr >= end || *ptr != ')') | 160 if (ptr >= end || *ptr != ')') |
| 203 return false; | 161 return false; |
| 204 ptr++; | 162 ptr++; |
| 205 } else if (*ptr == 't') { | 163 } else if (*ptr == 't') { |
| 206 if (!skipString(ptr, end, transformSpec, WTF_ARRAY_LENGTH(transformS
pec))) | 164 if (!skipString(ptr, end, transformSpec, WTF_ARRAY_LENGTH(transformS
pec))) |
| 207 return false; | 165 return false; |
| 208 if (ptr >= end || *ptr != '(') | 166 if (ptr >= end || *ptr != '(') |
| 209 return false; | 167 return false; |
| 210 ptr++; | 168 ptr++; |
| 211 parseTransformAttribute(m_transform, ptr, end, DoNotClearList); | 169 m_transform->baseValue()->parse(ptr, end); |
| 212 if (ptr >= end || *ptr != ')') | 170 if (ptr >= end || *ptr != ')') |
| 213 return false; | 171 return false; |
| 214 ptr++; | 172 ptr++; |
| 215 } else | 173 } else |
| 216 return false; | 174 return false; |
| 217 | 175 |
| 218 if (ptr < end && *ptr == ';') | 176 if (ptr < end && *ptr == ';') |
| 219 ptr++; | 177 ptr++; |
| 220 } | 178 } |
| 221 | 179 |
| (...skipping 11 matching lines...) Expand all Loading... |
| 233 const LChar* ptr = spec.characters8(); | 191 const LChar* ptr = spec.characters8(); |
| 234 const LChar* end = ptr + spec.length(); | 192 const LChar* end = ptr + spec.length(); |
| 235 return parseViewSpecInternal(ptr, end); | 193 return parseViewSpecInternal(ptr, end); |
| 236 } | 194 } |
| 237 const UChar* ptr = spec.characters16(); | 195 const UChar* ptr = spec.characters16(); |
| 238 const UChar* end = ptr + spec.length(); | 196 const UChar* end = ptr + spec.length(); |
| 239 return parseViewSpecInternal(ptr, end); | 197 return parseViewSpecInternal(ptr, end); |
| 240 } | 198 } |
| 241 | 199 |
| 242 } | 200 } |
| OLD | NEW |