1 /* 2 * Copyright (C) 2003 Lars Knoll (knoll (at) kde.org) 3 * Copyright (C) 2004, 2005, 2006, 2008, 2009, 2010 Apple Inc. All rights reserved. 4 * 5 * This library is free software; you can redistribute it and/or 6 * modify it under the terms of the GNU Library General Public 7 * License as published by the Free Software Foundation; either 8 * version 2 of the License, or (at your option) any later version. 9 * 10 * This library is distributed in the hope that it will be useful, 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13 * Library General Public License for more details. 14 * 15 * You should have received a copy of the GNU Library General Public License 16 * along with this library; see the file COPYING.LIB. If not, write to 17 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, 18 * Boston, MA 02110-1301, USA. 19 */ 20 21 #ifndef CSSParserValues_h 22 #define CSSParserValues_h 23 24 #include "CSSValueKeywords.h" 25 #include "core/css/CSSPrimitiveValue.h" 26 #include "core/css/CSSSelector.h" 27 #include "core/css/CSSValueList.h" 28 #include "wtf/text/AtomicString.h" 29 #include "wtf/text/WTFString.h" 30 31 namespace WebCore { 32 33 class CSSValue; 34 class QualifiedName; 35 36 struct CSSParserString { 37 void init(const LChar* characters, unsigned length) 38 { 39 m_data.characters8 = characters; 40 m_length = length; 41 m_is8Bit = true; 42 } 43 44 void init(const UChar* characters, unsigned length) 45 { 46 m_data.characters16 = characters; 47 m_length = length; 48 m_is8Bit = false; 49 } 50 51 void init(const String& string) 52 { 53 init(string, 0, string.length()); 54 } 55 56 void init(const String& string, unsigned startOffset, unsigned length) 57 { 58 m_length = length; 59 if (!m_length) { 60 m_data.characters8 = 0; 61 m_is8Bit = true; 62 return; 63 } 64 if (string.is8Bit()) { 65 m_data.characters8 = const_cast<LChar*>(string.characters8()) + startOffset; 66 m_is8Bit = true; 67 } else { 68 m_data.characters16 = const_cast<UChar*>(string.characters16()) + startOffset; 69 m_is8Bit = false; 70 } 71 } 72 73 void clear() 74 { 75 m_data.characters8 = 0; 76 m_length = 0; 77 m_is8Bit = true; 78 } 79 80 void trimTrailingWhitespace(); 81 82 bool is8Bit() const { return m_is8Bit; } 83 const LChar* characters8() const { ASSERT(is8Bit()); return m_data.characters8; } 84 const UChar* characters16() const { ASSERT(!is8Bit()); return m_data.characters16; } 85 template <typename CharacterType> 86 const CharacterType* characters() const; 87 88 unsigned length() const { return m_length; } 89 void setLength(unsigned length) { m_length = length; } 90 91 UChar operator[](unsigned i) const 92 { 93 ASSERT_WITH_SECURITY_IMPLICATION(i < m_length); 94 if (is8Bit()) 95 return m_data.characters8[i]; 96 return m_data.characters16[i]; 97 } 98 99 bool equalIgnoringCase(const char* str) const 100 { 101 if (is8Bit()) 102 return WTF::equalIgnoringCase(str, characters8(), length()); 103 return WTF::equalIgnoringCase(str, characters16(), length()); 104 } 105 106 template <size_t strLength> 107 bool startsWithIgnoringCase(const char (&str)[strLength]) const 108 { 109 return startsWithIgnoringCase(str, strLength - 1); 110 } 111 112 bool startsWithIgnoringCase(const char* str, size_t strLength) const 113 { 114 if (length() < strLength) 115 return false; 116 return is8Bit() ? WTF::equalIgnoringCase(str, characters8(), strLength) : WTF::equalIgnoringCase(str, characters16(), strLength); 117 } 118 119 operator String() const { return is8Bit() ? String(m_data.characters8, m_length) : StringImpl::create8BitIfPossible(m_data.characters16, m_length); } 120 operator AtomicString() const { return is8Bit() ? AtomicString(m_data.characters8, m_length) : AtomicString(m_data.characters16, m_length); } 121 122 AtomicString atomicSubstring(unsigned position, unsigned length) const; 123 124 bool isFunction() const { return length() > 0 && (*this)[length() - 1] == '('; } 125 126 union { 127 const LChar* characters8; 128 const UChar* characters16; 129 } m_data; 130 unsigned m_length; 131 bool m_is8Bit; 132 }; 133 134 template <> 135 inline const LChar* CSSParserString::characters<LChar>() const { return characters8(); } 136 137 template <> 138 inline const UChar* CSSParserString::characters<UChar>() const { return characters16(); } 139 140 struct CSSParserFunction; 141 142 struct CSSParserValue { 143 CSSValueID id; 144 bool isInt; 145 union { 146 double fValue; 147 int iValue; 148 CSSParserString string; 149 CSSParserFunction* function; 150 CSSParserValueList* valueList; 151 }; 152 enum { 153 Operator = 0x100000, 154 Function = 0x100001, 155 ValueList = 0x100002, 156 Q_EMS = 0x100003, 157 }; 158 int unit; 159 160 inline void setFromNumber(double value, int unit = CSSPrimitiveValue::CSS_NUMBER); 161 inline void setFromFunction(CSSParserFunction*); 162 inline void setFromValueList(PassOwnPtr<CSSParserValueList>); 163 164 PassRefPtr<CSSValue> createCSSValue(); 165 }; 166 167 class CSSParserValueList { 168 WTF_MAKE_FAST_ALLOCATED; 169 public: 170 CSSParserValueList() 171 : m_current(0) 172 { 173 } 174 ~CSSParserValueList(); 175 176 void addValue(const CSSParserValue&); 177 void insertValueAt(unsigned, const CSSParserValue&); 178 void deleteValueAt(unsigned); 179 void extend(CSSParserValueList&); 180 181 unsigned size() const { return m_values.size(); } 182 unsigned currentIndex() { return m_current; } 183 CSSParserValue* current() { return m_current < m_values.size() ? &m_values[m_current] : 0; } 184 CSSParserValue* next() { ++m_current; return current(); } 185 CSSParserValue* previous() 186 { 187 if (!m_current) 188 return 0; 189 --m_current; 190 return current(); 191 } 192 193 CSSParserValue* valueAt(unsigned i) { return i < m_values.size() ? &m_values[i] : 0; } 194 195 void clear() { m_values.clear(); } 196 197 private: 198 unsigned m_current; 199 Vector<CSSParserValue, 4> m_values; 200 }; 201 202 struct CSSParserFunction { 203 WTF_MAKE_FAST_ALLOCATED; 204 public: 205 CSSParserString name; 206 OwnPtr<CSSParserValueList> args; 207 }; 208 209 class CSSParserSelector { 210 WTF_MAKE_NONCOPYABLE(CSSParserSelector); WTF_MAKE_FAST_ALLOCATED; 211 public: 212 CSSParserSelector(); 213 explicit CSSParserSelector(const QualifiedName&); 214 ~CSSParserSelector(); 215 216 PassOwnPtr<CSSSelector> releaseSelector() { return m_selector.release(); } 217 218 CSSSelector::Relation relation() const { return m_selector->relation(); } 219 void setValue(const AtomicString& value) { m_selector->setValue(value); } 220 void setAttribute(const QualifiedName& value) { m_selector->setAttribute(value); } 221 void setArgument(const AtomicString& value) { m_selector->setArgument(value); } 222 void setMatch(CSSSelector::Match value) { m_selector->m_match = value; } 223 void setRelation(CSSSelector::Relation value) { m_selector->m_relation = value; } 224 void setForPage() { m_selector->setForPage(); } 225 void setRelationIsAffectedByPseudoContent() { m_selector->setRelationIsAffectedByPseudoContent(); } 226 bool relationIsAffectedByPseudoContent() const { return m_selector->relationIsAffectedByPseudoContent(); } 227 228 void adoptSelectorVector(Vector<OwnPtr<CSSParserSelector> >& selectorVector); 229 230 CSSParserSelector* functionArgumentSelector() const { return m_functionArgumentSelector; } 231 void setFunctionArgumentSelector(CSSParserSelector* selector) { m_functionArgumentSelector = selector; } 232 bool isDistributedPseudoElement() const { return m_selector->isDistributedPseudoElement(); } 233 CSSParserSelector* findDistributedPseudoElementSelector() const; 234 bool isContentPseudoElement() const { return m_selector->isContentPseudoElement(); } 235 236 CSSSelector::PseudoType pseudoType() const { return m_selector->pseudoType(); } 237 bool isCustomPseudoElement() const { return m_selector->isCustomPseudoElement(); } 238 bool needsCrossingTreeScopeBoundary() const { return isCustomPseudoElement() || pseudoType() == CSSSelector::PseudoCue; } 239 240 bool isSimple() const; 241 bool hasShadowPseudo() const; 242 243 CSSParserSelector* tagHistory() const { return m_tagHistory.get(); } 244 void setTagHistory(PassOwnPtr<CSSParserSelector> selector) { m_tagHistory = selector; } 245 void clearTagHistory() { m_tagHistory.clear(); } 246 void insertTagHistory(CSSSelector::Relation before, PassOwnPtr<CSSParserSelector>, CSSSelector::Relation after); 247 void appendTagHistory(CSSSelector::Relation, PassOwnPtr<CSSParserSelector>); 248 void prependTagSelector(const QualifiedName&, bool tagIsForNamespaceRule = false); 249 250 private: 251 OwnPtr<CSSSelector> m_selector; 252 OwnPtr<CSSParserSelector> m_tagHistory; 253 CSSParserSelector* m_functionArgumentSelector; 254 }; 255 256 inline bool CSSParserSelector::hasShadowPseudo() const 257 { 258 return m_selector->relation() == CSSSelector::ShadowPseudo; 259 } 260 261 inline void CSSParserValue::setFromNumber(double value, int unit) 262 { 263 id = CSSValueInvalid; 264 isInt = false; 265 if (std::isfinite(value)) 266 fValue = value; 267 else 268 fValue = 0; 269 this->unit = unit; 270 } 271 272 inline void CSSParserValue::setFromFunction(CSSParserFunction* function) 273 { 274 id = CSSValueInvalid; 275 this->function = function; 276 unit = Function; 277 } 278 279 inline void CSSParserValue::setFromValueList(PassOwnPtr<CSSParserValueList> valueList) 280 { 281 id = CSSValueInvalid; 282 this->valueList = valueList.leakPtr(); 283 unit = ValueList; 284 } 285 286 } 287 288 #endif 289