| 1 | // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 | // Copyright (C) 2016 Apple Inc. All rights reserved. |
| 3 | // |
| 4 | // Redistribution and use in source and binary forms, with or without |
| 5 | // modification, are permitted provided that the following conditions are |
| 6 | // met: |
| 7 | // |
| 8 | // * Redistributions of source code must retain the above copyright |
| 9 | // notice, this list of conditions and the following disclaimer. |
| 10 | // * Redistributions in binary form must reproduce the above |
| 11 | // copyright notice, this list of conditions and the following disclaimer |
| 12 | // in the documentation and/or other materials provided with the |
| 13 | // distribution. |
| 14 | // * Neither the name of Google Inc. nor the names of its |
| 15 | // contributors may be used to endorse or promote products derived from |
| 16 | // this software without specific prior written permission. |
| 17 | // |
| 18 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 19 | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 20 | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 21 | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 22 | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 23 | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 24 | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 25 | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 26 | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 27 | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 28 | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 29 | |
| 30 | #include "config.h" |
| 31 | #include "CSSParserImpl.h" |
| 32 | |
| 33 | #include "CSSAtRuleID.h" |
| 34 | #include "CSSCustomPropertyValue.h" |
| 35 | #include "CSSDeferredParser.h" |
| 36 | #include "CSSKeyframeRule.h" |
| 37 | #include "CSSKeyframesRule.h" |
| 38 | #include "CSSParserObserver.h" |
| 39 | #include "CSSParserObserverWrapper.h" |
| 40 | #include "CSSParserSelector.h" |
| 41 | #include "CSSPropertyParser.h" |
| 42 | #include "CSSSelectorParser.h" |
| 43 | #include "CSSStyleSheet.h" |
| 44 | #include "CSSSupportsParser.h" |
| 45 | #include "CSSTokenizer.h" |
| 46 | #include "CSSVariableParser.h" |
| 47 | #include "Document.h" |
| 48 | #include "Element.h" |
| 49 | #include "MediaList.h" |
| 50 | #include "MediaQueryParser.h" |
| 51 | #include "MediaQueryParserContext.h" |
| 52 | #include "StyleProperties.h" |
| 53 | #include "StyleRuleImport.h" |
| 54 | #include "StyleSheetContents.h" |
| 55 | |
| 56 | #include <bitset> |
| 57 | #include <memory> |
| 58 | |
| 59 | namespace WebCore { |
| 60 | |
| 61 | CSSParserImpl::CSSParserImpl(const CSSParserContext& context, StyleSheetContents* styleSheet) |
| 62 | : m_context(context) |
| 63 | , m_styleSheet(styleSheet) |
| 64 | { |
| 65 | |
| 66 | } |
| 67 | |
| 68 | CSSParserImpl::CSSParserImpl(CSSDeferredParser& deferredParser) |
| 69 | : m_context(deferredParser.context()) |
| 70 | , m_styleSheet(deferredParser.styleSheet()) |
| 71 | , m_deferredParser(&deferredParser) |
| 72 | { |
| 73 | } |
| 74 | |
| 75 | CSSParserImpl::CSSParserImpl(const CSSParserContext& context, const String& string, StyleSheetContents* styleSheet, CSSParserObserverWrapper* wrapper, CSSParser::RuleParsing ruleParsing) |
| 76 | : m_context(context) |
| 77 | , m_styleSheet(styleSheet) |
| 78 | , m_observerWrapper(wrapper) |
| 79 | { |
| 80 | m_tokenizer = wrapper ? std::make_unique<CSSTokenizer>(string, *wrapper) : std::make_unique<CSSTokenizer>(string); |
| 81 | if (context.deferredCSSParserEnabled && !wrapper && styleSheet && ruleParsing == CSSParser::RuleParsing::Deferred) |
| 82 | m_deferredParser = CSSDeferredParser::create(context, string, *styleSheet); |
| 83 | } |
| 84 | |
| 85 | CSSParser::ParseResult CSSParserImpl::parseValue(MutableStyleProperties* declaration, CSSPropertyID propertyID, const String& string, bool important, const CSSParserContext& context) |
| 86 | { |
| 87 | CSSParserImpl parser(context, string); |
| 88 | StyleRule::Type ruleType = StyleRule::Style; |
| 89 | #if ENABLE(CSS_DEVICE_ADAPTATION) |
| 90 | if (declaration->cssParserMode() == CSSViewportRuleMode) |
| 91 | ruleType = StyleRule::Viewport; |
| 92 | #endif |
| 93 | parser.consumeDeclarationValue(parser.tokenizer()->tokenRange(), propertyID, important, ruleType); |
| 94 | if (parser.m_parsedProperties.isEmpty()) |
| 95 | return CSSParser::ParseResult::Error; |
| 96 | return declaration->addParsedProperties(parser.m_parsedProperties) ? CSSParser::ParseResult::Changed : CSSParser::ParseResult::Unchanged; |
| 97 | } |
| 98 | |
| 99 | CSSParser::ParseResult CSSParserImpl::parseCustomPropertyValue(MutableStyleProperties* declaration, const AtomicString& propertyName, const String& string, bool important, const CSSParserContext& context) |
| 100 | { |
| 101 | CSSParserImpl parser(context, string); |
| 102 | parser.consumeCustomPropertyValue(parser.tokenizer()->tokenRange(), propertyName, important); |
| 103 | if (parser.m_parsedProperties.isEmpty()) |
| 104 | return CSSParser::ParseResult::Error; |
| 105 | return declaration->addParsedProperties(parser.m_parsedProperties) ? CSSParser::ParseResult::Changed : CSSParser::ParseResult::Unchanged; |
| 106 | } |
| 107 | |
| 108 | static inline void filterProperties(bool important, const ParsedPropertyVector& input, ParsedPropertyVector& output, size_t& unusedEntries, std::bitset<numCSSProperties>& seenProperties, HashSet<AtomicString>& seenCustomProperties) |
| 109 | { |
| 110 | // Add properties in reverse order so that highest priority definitions are reached first. Duplicate definitions can then be ignored when found. |
| 111 | for (size_t i = input.size(); i--; ) { |
| 112 | const CSSProperty& property = input[i]; |
| 113 | if (property.isImportant() != important) |
| 114 | continue; |
| 115 | const unsigned propertyIDIndex = property.id() - firstCSSProperty; |
| 116 | |
| 117 | if (property.id() == CSSPropertyCustom) { |
| 118 | auto& name = downcast<CSSCustomPropertyValue>(*property.value()).name(); |
| 119 | if (!seenCustomProperties.add(name).isNewEntry) |
| 120 | continue; |
| 121 | output[--unusedEntries] = property; |
| 122 | continue; |
| 123 | } |
| 124 | |
| 125 | // FIXME-NEWPARSER: We won't support @apply yet. |
| 126 | /*else if (property.id() == CSSPropertyApplyAtRule) { |
| 127 | // FIXME: Do we need to do anything here? |
| 128 | } */ |
| 129 | |
| 130 | if (seenProperties.test(propertyIDIndex)) |
| 131 | continue; |
| 132 | seenProperties.set(propertyIDIndex); |
| 133 | |
| 134 | output[--unusedEntries] = property; |
| 135 | } |
| 136 | } |
| 137 | |
| 138 | Ref<DeferredStyleProperties> CSSParserImpl::createDeferredStyleProperties(const CSSParserTokenRange& propertyRange) |
| 139 | { |
| 140 | ASSERT(m_deferredParser); |
| 141 | return DeferredStyleProperties::create(propertyRange, *m_deferredParser); |
| 142 | } |
| 143 | |
| 144 | static Ref<ImmutableStyleProperties> createStyleProperties(ParsedPropertyVector& parsedProperties, CSSParserMode mode) |
| 145 | { |
| 146 | std::bitset<numCSSProperties> seenProperties; |
| 147 | size_t unusedEntries = parsedProperties.size(); |
| 148 | ParsedPropertyVector results(unusedEntries); |
| 149 | HashSet<AtomicString> seenCustomProperties; |
| 150 | |
| 151 | filterProperties(true, parsedProperties, results, unusedEntries, seenProperties, seenCustomProperties); |
| 152 | filterProperties(false, parsedProperties, results, unusedEntries, seenProperties, seenCustomProperties); |
| 153 | |
| 154 | Ref<ImmutableStyleProperties> result = ImmutableStyleProperties::create(results.data() + unusedEntries, results.size() - unusedEntries, mode); |
| 155 | parsedProperties.clear(); |
| 156 | return result; |
| 157 | } |
| 158 | |
| 159 | Ref<ImmutableStyleProperties> CSSParserImpl::parseInlineStyleDeclaration(const String& string, const Element* element) |
| 160 | { |
| 161 | CSSParserContext context(element->document()); |
| 162 | context.mode = strictToCSSParserMode(element->isHTMLElement() && !element->document().inQuirksMode()); |
| 163 | |
| 164 | CSSParserImpl parser(context, string); |
| 165 | parser.consumeDeclarationList(parser.tokenizer()->tokenRange(), StyleRule::Style); |
| 166 | return createStyleProperties(parser.m_parsedProperties, context.mode); |
| 167 | } |
| 168 | |
| 169 | Ref<ImmutableStyleProperties> CSSParserImpl::parseDeferredDeclaration(CSSParserTokenRange tokenRange, const CSSParserContext& context, StyleSheetContents* styleSheet) |
| 170 | { |
| 171 | if (!styleSheet) { |
| 172 | ParsedPropertyVector properties; |
| 173 | return createStyleProperties(properties, context.mode); |
| 174 | } |
| 175 | CSSParserImpl parser(context, styleSheet); |
| 176 | parser.consumeDeclarationList(tokenRange, StyleRule::Style); |
| 177 | return createStyleProperties(parser.m_parsedProperties, context.mode); |
| 178 | } |
| 179 | |
| 180 | void CSSParserImpl::parseDeferredRuleList(CSSParserTokenRange tokenRange, CSSDeferredParser& deferredParser, Vector<RefPtr<StyleRuleBase>>& childRules) |
| 181 | { |
| 182 | if (!deferredParser.styleSheet()) |
| 183 | return; |
| 184 | CSSParserImpl parser(deferredParser); |
| 185 | parser.consumeRuleList(tokenRange, RegularRuleList, [&childRules](const RefPtr<StyleRuleBase>& rule) { |
| 186 | childRules.append(rule); |
| 187 | }); |
| 188 | childRules.shrinkToFit(); |
| 189 | } |
| 190 | |
| 191 | void CSSParserImpl::parseDeferredKeyframeList(CSSParserTokenRange tokenRange, CSSDeferredParser& deferredParser, StyleRuleKeyframes& keyframeRule) |
| 192 | { |
| 193 | if (!deferredParser.styleSheet()) |
| 194 | return; |
| 195 | CSSParserImpl parser(deferredParser); |
| 196 | parser.consumeRuleList(tokenRange, KeyframesRuleList, [&keyframeRule](const RefPtr<StyleRuleBase>& keyframe) { |
| 197 | keyframeRule.parserAppendKeyframe(downcast<const StyleRuleKeyframe>(keyframe.get())); |
| 198 | }); |
| 199 | } |
| 200 | |
| 201 | bool CSSParserImpl::parseDeclarationList(MutableStyleProperties* declaration, const String& string, const CSSParserContext& context) |
| 202 | { |
| 203 | CSSParserImpl parser(context, string); |
| 204 | StyleRule::Type ruleType = StyleRule::Style; |
| 205 | #if ENABLE(CSS_DEVICE_ADAPTATION) |
| 206 | if (declaration->cssParserMode() == CSSViewportRuleMode) |
| 207 | ruleType = StyleRule::Viewport; |
| 208 | #endif |
| 209 | parser.consumeDeclarationList(parser.tokenizer()->tokenRange(), ruleType); |
| 210 | if (parser.m_parsedProperties.isEmpty()) |
| 211 | return false; |
| 212 | |
| 213 | std::bitset<numCSSProperties> seenProperties; |
| 214 | size_t unusedEntries = parser.m_parsedProperties.size(); |
| 215 | ParsedPropertyVector results(unusedEntries); |
| 216 | HashSet<AtomicString> seenCustomProperties; |
| 217 | filterProperties(true, parser.m_parsedProperties, results, unusedEntries, seenProperties, seenCustomProperties); |
| 218 | filterProperties(false, parser.m_parsedProperties, results, unusedEntries, seenProperties, seenCustomProperties); |
| 219 | if (unusedEntries) |
| 220 | results.remove(0, unusedEntries); |
| 221 | return declaration->addParsedProperties(results); |
| 222 | } |
| 223 | |
| 224 | RefPtr<StyleRuleBase> CSSParserImpl::parseRule(const String& string, const CSSParserContext& context, StyleSheetContents* styleSheet, AllowedRulesType allowedRules) |
| 225 | { |
| 226 | CSSParserImpl parser(context, string, styleSheet); |
| 227 | CSSParserTokenRange range = parser.tokenizer()->tokenRange(); |
| 228 | range.consumeWhitespace(); |
| 229 | if (range.atEnd()) |
| 230 | return nullptr; // Parse error, empty rule |
| 231 | RefPtr<StyleRuleBase> rule; |
| 232 | if (range.peek().type() == AtKeywordToken) |
| 233 | rule = parser.consumeAtRule(range, allowedRules); |
| 234 | else |
| 235 | rule = parser.consumeQualifiedRule(range, allowedRules); |
| 236 | if (!rule) |
| 237 | return nullptr; // Parse error, failed to consume rule |
| 238 | range.consumeWhitespace(); |
| 239 | if (!rule || !range.atEnd()) |
| 240 | return nullptr; // Parse error, trailing garbage |
| 241 | return rule; |
| 242 | } |
| 243 | |
| 244 | void CSSParserImpl::parseStyleSheet(const String& string, const CSSParserContext& context, StyleSheetContents* styleSheet, CSSParser::RuleParsing ruleParsing) |
| 245 | { |
| 246 | CSSParserImpl parser(context, string, styleSheet, nullptr, ruleParsing); |
| 247 | bool firstRuleValid = parser.consumeRuleList(parser.tokenizer()->tokenRange(), TopLevelRuleList, [&styleSheet](RefPtr<StyleRuleBase> rule) { |
| 248 | if (rule->isCharsetRule()) |
| 249 | return; |
| 250 | styleSheet->parserAppendRule(rule.releaseNonNull()); |
| 251 | }); |
| 252 | styleSheet->setHasSyntacticallyValidCSSHeader(firstRuleValid); |
| 253 | parser.adoptTokenizerEscapedStrings(); |
| 254 | } |
| 255 | |
| 256 | void CSSParserImpl::adoptTokenizerEscapedStrings() |
| 257 | { |
| 258 | if (!m_deferredParser || !m_tokenizer) |
| 259 | return; |
| 260 | m_deferredParser->adoptTokenizerEscapedStrings(m_tokenizer->escapedStringsForAdoption()); |
| 261 | } |
| 262 | |
| 263 | CSSSelectorList CSSParserImpl::parsePageSelector(CSSParserTokenRange range, StyleSheetContents* styleSheet) |
| 264 | { |
| 265 | // We only support a small subset of the css-page spec. |
| 266 | range.consumeWhitespace(); |
| 267 | AtomicString typeSelector; |
| 268 | if (range.peek().type() == IdentToken) |
| 269 | typeSelector = range.consume().value().toAtomicString(); |
| 270 | |
| 271 | AtomicString pseudo; |
| 272 | if (range.peek().type() == ColonToken) { |
| 273 | range.consume(); |
| 274 | if (range.peek().type() != IdentToken) |
| 275 | return CSSSelectorList(); |
| 276 | pseudo = range.consume().value().toAtomicString(); |
| 277 | } |
| 278 | |
| 279 | range.consumeWhitespace(); |
| 280 | if (!range.atEnd()) |
| 281 | return CSSSelectorList(); // Parse error; extra tokens in @page selector |
| 282 | |
| 283 | std::unique_ptr<CSSParserSelector> selector; |
| 284 | if (!typeSelector.isNull() && pseudo.isNull()) |
| 285 | selector = std::make_unique<CSSParserSelector>(QualifiedName(nullAtom(), typeSelector, styleSheet->defaultNamespace())); |
| 286 | else { |
| 287 | selector = std::make_unique<CSSParserSelector>(); |
| 288 | if (!pseudo.isNull()) { |
| 289 | selector = std::unique_ptr<CSSParserSelector>(CSSParserSelector::parsePagePseudoSelector(pseudo)); |
| 290 | if (!selector || selector->match() != CSSSelector::PagePseudoClass) |
| 291 | return CSSSelectorList(); |
| 292 | } |
| 293 | if (!typeSelector.isNull()) |
| 294 | selector->prependTagSelector(QualifiedName(nullAtom(), typeSelector, styleSheet->defaultNamespace())); |
| 295 | } |
| 296 | |
| 297 | selector->setForPage(); |
| 298 | return CSSSelectorList { Vector<std::unique_ptr<CSSParserSelector>>::from(WTFMove(selector)) }; |
| 299 | } |
| 300 | |
| 301 | std::unique_ptr<Vector<double>> CSSParserImpl::parseKeyframeKeyList(const String& keyList) |
| 302 | { |
| 303 | return consumeKeyframeKeyList(CSSTokenizer(keyList).tokenRange()); |
| 304 | } |
| 305 | |
| 306 | bool CSSParserImpl::supportsDeclaration(CSSParserTokenRange& range) |
| 307 | { |
| 308 | ASSERT(m_parsedProperties.isEmpty()); |
| 309 | consumeDeclaration(range, StyleRule::Style); |
| 310 | bool result = !m_parsedProperties.isEmpty(); |
| 311 | m_parsedProperties.clear(); |
| 312 | return result; |
| 313 | } |
| 314 | |
| 315 | void CSSParserImpl::parseDeclarationListForInspector(const String& declaration, const CSSParserContext& context, CSSParserObserver& observer) |
| 316 | { |
| 317 | CSSParserObserverWrapper wrapper(observer); |
| 318 | CSSParserImpl parser(context, declaration, nullptr, &wrapper); |
| 319 | observer.startRuleHeader(StyleRule::Style, 0); |
| 320 | observer.endRuleHeader(1); |
| 321 | parser.consumeDeclarationList(parser.tokenizer()->tokenRange(), StyleRule::Style); |
| 322 | } |
| 323 | |
| 324 | void CSSParserImpl::parseStyleSheetForInspector(const String& string, const CSSParserContext& context, StyleSheetContents* styleSheet, CSSParserObserver& observer) |
| 325 | { |
| 326 | CSSParserObserverWrapper wrapper(observer); |
| 327 | CSSParserImpl parser(context, string, styleSheet, &wrapper); |
| 328 | bool firstRuleValid = parser.consumeRuleList(parser.tokenizer()->tokenRange(), TopLevelRuleList, [&styleSheet](RefPtr<StyleRuleBase> rule) { |
| 329 | if (rule->isCharsetRule()) |
| 330 | return; |
| 331 | styleSheet->parserAppendRule(rule.releaseNonNull()); |
| 332 | }); |
| 333 | styleSheet->setHasSyntacticallyValidCSSHeader(firstRuleValid); |
| 334 | } |
| 335 | |
| 336 | static CSSParserImpl::AllowedRulesType computeNewAllowedRules(CSSParserImpl::AllowedRulesType allowedRules, StyleRuleBase* rule) |
| 337 | { |
| 338 | if (!rule || allowedRules == CSSParserImpl::KeyframeRules || allowedRules == CSSParserImpl::NoRules) |
| 339 | return allowedRules; |
| 340 | ASSERT(allowedRules <= CSSParserImpl::RegularRules); |
| 341 | if (rule->isCharsetRule() || rule->isImportRule()) |
| 342 | return CSSParserImpl::AllowImportRules; |
| 343 | if (rule->isNamespaceRule()) |
| 344 | return CSSParserImpl::AllowNamespaceRules; |
| 345 | return CSSParserImpl::RegularRules; |
| 346 | } |
| 347 | |
| 348 | template<typename T> |
| 349 | bool CSSParserImpl::consumeRuleList(CSSParserTokenRange range, RuleListType ruleListType, const T callback) |
| 350 | { |
| 351 | AllowedRulesType allowedRules = RegularRules; |
| 352 | switch (ruleListType) { |
| 353 | case TopLevelRuleList: |
| 354 | allowedRules = AllowCharsetRules; |
| 355 | break; |
| 356 | case RegularRuleList: |
| 357 | allowedRules = RegularRules; |
| 358 | break; |
| 359 | case KeyframesRuleList: |
| 360 | allowedRules = KeyframeRules; |
| 361 | break; |
| 362 | default: |
| 363 | ASSERT_NOT_REACHED(); |
| 364 | } |
| 365 | |
| 366 | bool seenRule = false; |
| 367 | bool firstRuleValid = false; |
| 368 | while (!range.atEnd()) { |
| 369 | RefPtr<StyleRuleBase> rule; |
| 370 | switch (range.peek().type()) { |
| 371 | case WhitespaceToken: |
| 372 | range.consumeWhitespace(); |
| 373 | continue; |
| 374 | case AtKeywordToken: |
| 375 | rule = consumeAtRule(range, allowedRules); |
| 376 | break; |
| 377 | case CDOToken: |
| 378 | case CDCToken: |
| 379 | if (ruleListType == TopLevelRuleList) { |
| 380 | range.consume(); |
| 381 | continue; |
| 382 | } |
| 383 | FALLTHROUGH; |
| 384 | default: |
| 385 | rule = consumeQualifiedRule(range, allowedRules); |
| 386 | break; |
| 387 | } |
| 388 | if (!seenRule) { |
| 389 | seenRule = true; |
| 390 | firstRuleValid = rule; |
| 391 | } |
| 392 | if (rule) { |
| 393 | allowedRules = computeNewAllowedRules(allowedRules, rule.get()); |
| 394 | callback(rule); |
| 395 | } |
| 396 | } |
| 397 | |
| 398 | return firstRuleValid; |
| 399 | } |
| 400 | |
| 401 | RefPtr<StyleRuleBase> CSSParserImpl::consumeAtRule(CSSParserTokenRange& range, AllowedRulesType allowedRules) |
| 402 | { |
| 403 | ASSERT(range.peek().type() == AtKeywordToken); |
| 404 | const StringView name = range.consumeIncludingWhitespace().value(); |
| 405 | const CSSParserToken* preludeStart = &range.peek(); |
| 406 | while (!range.atEnd() && range.peek().type() != LeftBraceToken && range.peek().type() != SemicolonToken) |
| 407 | range.consumeComponentValue(); |
| 408 | |
| 409 | CSSParserTokenRange prelude = range.makeSubRange(preludeStart, &range.peek()); |
| 410 | CSSAtRuleID id = cssAtRuleID(name); |
| 411 | |
| 412 | if (range.atEnd() || range.peek().type() == SemicolonToken) { |
| 413 | range.consume(); |
| 414 | if (allowedRules == AllowCharsetRules && id == CSSAtRuleCharset) |
| 415 | return consumeCharsetRule(prelude); |
| 416 | if (allowedRules <= AllowImportRules && id == CSSAtRuleImport) |
| 417 | return consumeImportRule(prelude); |
| 418 | if (allowedRules <= AllowNamespaceRules && id == CSSAtRuleNamespace) |
| 419 | return consumeNamespaceRule(prelude); |
| 420 | // FIXME-NEWPARSER: Support "apply" |
| 421 | /*if (allowedRules == ApplyRules && id == CSSAtRuleApply) { |
| 422 | consumeApplyRule(prelude); |
| 423 | return nullptr; // consumeApplyRule just updates m_parsedProperties |
| 424 | }*/ |
| 425 | return nullptr; // Parse error, unrecognised at-rule without block |
| 426 | } |
| 427 | |
| 428 | CSSParserTokenRange block = range.consumeBlock(); |
| 429 | if (allowedRules == KeyframeRules) |
| 430 | return nullptr; // Parse error, no at-rules supported inside @keyframes |
| 431 | if (allowedRules == NoRules || allowedRules == ApplyRules) |
| 432 | return nullptr; // Parse error, no at-rules with blocks supported inside declaration lists |
| 433 | |
| 434 | ASSERT(allowedRules <= RegularRules); |
| 435 | |
| 436 | switch (id) { |
| 437 | case CSSAtRuleMedia: |
| 438 | return consumeMediaRule(prelude, block); |
| 439 | case CSSAtRuleSupports: |
| 440 | return consumeSupportsRule(prelude, block); |
| 441 | #if ENABLE(CSS_DEVICE_ADAPTATION) |
| 442 | case CSSAtRuleViewport: |
| 443 | return consumeViewportRule(prelude, block); |
| 444 | #endif |
| 445 | case CSSAtRuleFontFace: |
| 446 | return consumeFontFaceRule(prelude, block); |
| 447 | case CSSAtRuleWebkitKeyframes: |
| 448 | return consumeKeyframesRule(true, prelude, block); |
| 449 | case CSSAtRuleKeyframes: |
| 450 | return consumeKeyframesRule(false, prelude, block); |
| 451 | case CSSAtRulePage: |
| 452 | return consumePageRule(prelude, block); |
| 453 | default: |
| 454 | return nullptr; // Parse error, unrecognised at-rule with block |
| 455 | } |
| 456 | } |
| 457 | |
| 458 | RefPtr<StyleRuleBase> CSSParserImpl::consumeQualifiedRule(CSSParserTokenRange& range, AllowedRulesType allowedRules) |
| 459 | { |
| 460 | const CSSParserToken* preludeStart = &range.peek(); |
| 461 | while (!range.atEnd() && range.peek().type() != LeftBraceToken) |
| 462 | range.consumeComponentValue(); |
| 463 | |
| 464 | if (range.atEnd()) |
| 465 | return nullptr; // Parse error, EOF instead of qualified rule block |
| 466 | |
| 467 | CSSParserTokenRange prelude = range.makeSubRange(preludeStart, &range.peek()); |
| 468 | CSSParserTokenRange block = range.consumeBlockCheckingForEditability(m_styleSheet.get()); |
| 469 | |
| 470 | if (allowedRules <= RegularRules) |
| 471 | return consumeStyleRule(prelude, block); |
| 472 | if (allowedRules == KeyframeRules) |
| 473 | return consumeKeyframeStyleRule(prelude, block); |
| 474 | |
| 475 | ASSERT_NOT_REACHED(); |
| 476 | return nullptr; |
| 477 | } |
| 478 | |
| 479 | // This may still consume tokens if it fails |
| 480 | static AtomicString consumeStringOrURI(CSSParserTokenRange& range) |
| 481 | { |
| 482 | const CSSParserToken& token = range.peek(); |
| 483 | |
| 484 | if (token.type() == StringToken || token.type() == UrlToken) |
| 485 | return range.consumeIncludingWhitespace().value().toAtomicString(); |
| 486 | |
| 487 | if (token.type() != FunctionToken || !equalIgnoringASCIICase(token.value(), "url" )) |
| 488 | return AtomicString(); |
| 489 | |
| 490 | CSSParserTokenRange contents = range.consumeBlock(); |
| 491 | const CSSParserToken& uri = contents.consumeIncludingWhitespace(); |
| 492 | if (uri.type() == BadStringToken || !contents.atEnd()) |
| 493 | return AtomicString(); |
| 494 | return uri.value().toAtomicString(); |
| 495 | } |
| 496 | |
| 497 | RefPtr<StyleRuleCharset> CSSParserImpl::consumeCharsetRule(CSSParserTokenRange prelude) |
| 498 | { |
| 499 | const CSSParserToken& string = prelude.consumeIncludingWhitespace(); |
| 500 | if (string.type() != StringToken || !prelude.atEnd()) |
| 501 | return nullptr; // Parse error, expected a single string |
| 502 | return StyleRuleCharset::create(); |
| 503 | } |
| 504 | |
| 505 | RefPtr<StyleRuleImport> CSSParserImpl::consumeImportRule(CSSParserTokenRange prelude) |
| 506 | { |
| 507 | AtomicString uri(consumeStringOrURI(prelude)); |
| 508 | if (uri.isNull()) |
| 509 | return nullptr; // Parse error, expected string or URI |
| 510 | |
| 511 | if (m_observerWrapper) { |
| 512 | unsigned endOffset = m_observerWrapper->endOffset(prelude); |
| 513 | m_observerWrapper->observer().startRuleHeader(StyleRule::Import, m_observerWrapper->startOffset(prelude)); |
| 514 | m_observerWrapper->observer().endRuleHeader(endOffset); |
| 515 | m_observerWrapper->observer().startRuleBody(endOffset); |
| 516 | m_observerWrapper->observer().endRuleBody(endOffset); |
| 517 | } |
| 518 | |
| 519 | return StyleRuleImport::create(uri, MediaQueryParser::parseMediaQuerySet(prelude, MediaQueryParserContext(m_context)).releaseNonNull()); |
| 520 | } |
| 521 | |
| 522 | RefPtr<StyleRuleNamespace> CSSParserImpl::consumeNamespaceRule(CSSParserTokenRange prelude) |
| 523 | { |
| 524 | AtomicString namespacePrefix; |
| 525 | if (prelude.peek().type() == IdentToken) |
| 526 | namespacePrefix = prelude.consumeIncludingWhitespace().value().toAtomicString(); |
| 527 | |
| 528 | AtomicString uri(consumeStringOrURI(prelude)); |
| 529 | if (uri.isNull() || !prelude.atEnd()) |
| 530 | return nullptr; // Parse error, expected string or URI |
| 531 | |
| 532 | return StyleRuleNamespace::create(namespacePrefix, uri); |
| 533 | } |
| 534 | |
| 535 | RefPtr<StyleRuleMedia> CSSParserImpl::consumeMediaRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 536 | { |
| 537 | if (m_deferredParser) |
| 538 | return StyleRuleMedia::create(MediaQueryParser::parseMediaQuerySet(prelude, MediaQueryParserContext(m_context)).releaseNonNull(), std::make_unique<DeferredStyleGroupRuleList>(block, *m_deferredParser)); |
| 539 | |
| 540 | Vector<RefPtr<StyleRuleBase>> rules; |
| 541 | |
| 542 | if (m_observerWrapper) { |
| 543 | m_observerWrapper->observer().startRuleHeader(StyleRule::Media, m_observerWrapper->startOffset(prelude)); |
| 544 | m_observerWrapper->observer().endRuleHeader(m_observerWrapper->endOffset(prelude)); |
| 545 | m_observerWrapper->observer().startRuleBody(m_observerWrapper->previousTokenStartOffset(block)); |
| 546 | } |
| 547 | |
| 548 | consumeRuleList(block, RegularRuleList, [&rules](RefPtr<StyleRuleBase> rule) { |
| 549 | rules.append(rule); |
| 550 | }); |
| 551 | rules.shrinkToFit(); |
| 552 | |
| 553 | if (m_observerWrapper) |
| 554 | m_observerWrapper->observer().endRuleBody(m_observerWrapper->endOffset(block)); |
| 555 | |
| 556 | return StyleRuleMedia::create(MediaQueryParser::parseMediaQuerySet(prelude, MediaQueryParserContext(m_context)).releaseNonNull(), rules); |
| 557 | } |
| 558 | |
| 559 | RefPtr<StyleRuleSupports> CSSParserImpl::consumeSupportsRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 560 | { |
| 561 | CSSSupportsParser::SupportsResult supported = CSSSupportsParser::supportsCondition(prelude, *this, CSSSupportsParser::ForAtRule); |
| 562 | if (supported == CSSSupportsParser::Invalid) |
| 563 | return nullptr; // Parse error, invalid @supports condition |
| 564 | |
| 565 | if (m_deferredParser) |
| 566 | return StyleRuleSupports::create(prelude.serialize().stripWhiteSpace(), supported, std::make_unique<DeferredStyleGroupRuleList>(block, *m_deferredParser)); |
| 567 | |
| 568 | if (m_observerWrapper) { |
| 569 | m_observerWrapper->observer().startRuleHeader(StyleRule::Supports, m_observerWrapper->startOffset(prelude)); |
| 570 | m_observerWrapper->observer().endRuleHeader(m_observerWrapper->endOffset(prelude)); |
| 571 | m_observerWrapper->observer().startRuleBody(m_observerWrapper->previousTokenStartOffset(block)); |
| 572 | } |
| 573 | |
| 574 | Vector<RefPtr<StyleRuleBase>> rules; |
| 575 | consumeRuleList(block, RegularRuleList, [&rules](RefPtr<StyleRuleBase> rule) { |
| 576 | rules.append(rule); |
| 577 | }); |
| 578 | rules.shrinkToFit(); |
| 579 | |
| 580 | if (m_observerWrapper) |
| 581 | m_observerWrapper->observer().endRuleBody(m_observerWrapper->endOffset(block)); |
| 582 | |
| 583 | return StyleRuleSupports::create(prelude.serialize().stripWhiteSpace(), supported, rules); |
| 584 | } |
| 585 | |
| 586 | #if ENABLE(CSS_DEVICE_ADAPTATION) |
| 587 | RefPtr<StyleRuleViewport> CSSParserImpl::consumeViewportRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 588 | { |
| 589 | if (!prelude.atEnd()) |
| 590 | return nullptr; // Parser error; @viewport prelude should be empty |
| 591 | |
| 592 | if (m_observerWrapper) { |
| 593 | unsigned endOffset = m_observerWrapper->endOffset(prelude); |
| 594 | m_observerWrapper->observer().startRuleHeader(StyleRule::Viewport, m_observerWrapper->startOffset(prelude)); |
| 595 | m_observerWrapper->observer().endRuleHeader(endOffset); |
| 596 | m_observerWrapper->observer().startRuleBody(endOffset); |
| 597 | m_observerWrapper->observer().endRuleBody(endOffset); |
| 598 | } |
| 599 | |
| 600 | consumeDeclarationList(block, StyleRule::Viewport); |
| 601 | return StyleRuleViewport::create(createStyleProperties(m_parsedProperties, CSSViewportRuleMode)); |
| 602 | } |
| 603 | #endif |
| 604 | |
| 605 | RefPtr<StyleRuleFontFace> CSSParserImpl::consumeFontFaceRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 606 | { |
| 607 | if (!prelude.atEnd()) |
| 608 | return nullptr; // Parse error; @font-face prelude should be empty |
| 609 | |
| 610 | if (m_observerWrapper) { |
| 611 | unsigned endOffset = m_observerWrapper->endOffset(prelude); |
| 612 | m_observerWrapper->observer().startRuleHeader(StyleRule::FontFace, m_observerWrapper->startOffset(prelude)); |
| 613 | m_observerWrapper->observer().endRuleHeader(endOffset); |
| 614 | m_observerWrapper->observer().startRuleBody(endOffset); |
| 615 | m_observerWrapper->observer().endRuleBody(endOffset); |
| 616 | } |
| 617 | |
| 618 | consumeDeclarationList(block, StyleRule::FontFace); |
| 619 | return StyleRuleFontFace::create(createStyleProperties(m_parsedProperties, m_context.mode)); |
| 620 | } |
| 621 | |
| 622 | RefPtr<StyleRuleKeyframes> CSSParserImpl::consumeKeyframesRule(bool webkitPrefixed, CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 623 | { |
| 624 | CSSParserTokenRange rangeCopy = prelude; // For inspector callbacks |
| 625 | const CSSParserToken& nameToken = prelude.consumeIncludingWhitespace(); |
| 626 | if (!prelude.atEnd()) |
| 627 | return nullptr; // Parse error; expected single non-whitespace token in @keyframes header |
| 628 | |
| 629 | String name; |
| 630 | if (nameToken.type() == IdentToken) { |
| 631 | name = nameToken.value().toString(); |
| 632 | } else if (nameToken.type() == StringToken && webkitPrefixed) |
| 633 | name = nameToken.value().toString(); |
| 634 | else |
| 635 | return nullptr; // Parse error; expected ident token in @keyframes header |
| 636 | |
| 637 | if (m_deferredParser) |
| 638 | return StyleRuleKeyframes::create(name, std::make_unique<DeferredStyleGroupRuleList>(block, *m_deferredParser)); |
| 639 | |
| 640 | if (m_observerWrapper) { |
| 641 | m_observerWrapper->observer().startRuleHeader(StyleRule::Keyframes, m_observerWrapper->startOffset(rangeCopy)); |
| 642 | m_observerWrapper->observer().endRuleHeader(m_observerWrapper->endOffset(prelude)); |
| 643 | m_observerWrapper->observer().startRuleBody(m_observerWrapper->previousTokenStartOffset(block)); |
| 644 | m_observerWrapper->observer().endRuleBody(m_observerWrapper->endOffset(block)); |
| 645 | } |
| 646 | |
| 647 | RefPtr<StyleRuleKeyframes> keyframeRule = StyleRuleKeyframes::create(name); |
| 648 | consumeRuleList(block, KeyframesRuleList, [keyframeRule](const RefPtr<StyleRuleBase>& keyframe) { |
| 649 | keyframeRule->parserAppendKeyframe(downcast<const StyleRuleKeyframe>(keyframe.get())); |
| 650 | }); |
| 651 | |
| 652 | // FIXME-NEWPARSER: Find out why this is done. Behavior difference when prefixed? |
| 653 | // keyframeRule->setVendorPrefixed(webkitPrefixed); |
| 654 | return keyframeRule; |
| 655 | } |
| 656 | |
| 657 | RefPtr<StyleRulePage> CSSParserImpl::(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 658 | { |
| 659 | CSSSelectorList selectorList = parsePageSelector(prelude, m_styleSheet.get()); |
| 660 | if (!selectorList.isValid()) |
| 661 | return nullptr; // Parse error, invalid @page selector |
| 662 | |
| 663 | if (m_observerWrapper) { |
| 664 | unsigned endOffset = m_observerWrapper->endOffset(prelude); |
| 665 | m_observerWrapper->observer().startRuleHeader(StyleRule::Page, m_observerWrapper->startOffset(prelude)); |
| 666 | m_observerWrapper->observer().endRuleHeader(endOffset); |
| 667 | } |
| 668 | |
| 669 | consumeDeclarationList(block, StyleRule::Style); |
| 670 | |
| 671 | return StyleRulePage::create(createStyleProperties(m_parsedProperties, m_context.mode), WTFMove(selectorList)); |
| 672 | } |
| 673 | |
| 674 | // FIXME-NEWPARSER: Support "apply" |
| 675 | /*void CSSParserImpl::consumeApplyRule(CSSParserTokenRange prelude) |
| 676 | { |
| 677 | const CSSParserToken& ident = prelude.consumeIncludingWhitespace(); |
| 678 | if (!prelude.atEnd() || !CSSVariableParser::isValidVariableName(ident)) |
| 679 | return; // Parse error, expected a single custom property name |
| 680 | m_parsedProperties.append(CSSProperty( |
| 681 | CSSPropertyApplyAtRule, |
| 682 | *CSSCustomIdentValue::create(ident.value().toString()))); |
| 683 | } |
| 684 | */ |
| 685 | |
| 686 | RefPtr<StyleRuleKeyframe> CSSParserImpl::consumeKeyframeStyleRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 687 | { |
| 688 | std::unique_ptr<Vector<double>> keyList = consumeKeyframeKeyList(prelude); |
| 689 | if (!keyList) |
| 690 | return nullptr; |
| 691 | |
| 692 | if (m_observerWrapper) { |
| 693 | m_observerWrapper->observer().startRuleHeader(StyleRule::Keyframe, m_observerWrapper->startOffset(prelude)); |
| 694 | m_observerWrapper->observer().endRuleHeader(m_observerWrapper->endOffset(prelude)); |
| 695 | } |
| 696 | |
| 697 | consumeDeclarationList(block, StyleRule::Keyframe); |
| 698 | return StyleRuleKeyframe::create(WTFMove(keyList), createStyleProperties(m_parsedProperties, m_context.mode)); |
| 699 | } |
| 700 | |
| 701 | static void observeSelectors(CSSParserObserverWrapper& wrapper, CSSParserTokenRange selectors) |
| 702 | { |
| 703 | // This is easier than hooking into the CSSSelectorParser |
| 704 | selectors.consumeWhitespace(); |
| 705 | CSSParserTokenRange originalRange = selectors; |
| 706 | wrapper.observer().startRuleHeader(StyleRule::Style, wrapper.startOffset(originalRange)); |
| 707 | |
| 708 | while (!selectors.atEnd()) { |
| 709 | const CSSParserToken* selectorStart = &selectors.peek(); |
| 710 | while (!selectors.atEnd() && selectors.peek().type() != CommaToken) |
| 711 | selectors.consumeComponentValue(); |
| 712 | CSSParserTokenRange selector = selectors.makeSubRange(selectorStart, &selectors.peek()); |
| 713 | selectors.consumeIncludingWhitespace(); |
| 714 | |
| 715 | wrapper.observer().observeSelector(wrapper.startOffset(selector), wrapper.endOffset(selector)); |
| 716 | } |
| 717 | |
| 718 | wrapper.observer().endRuleHeader(wrapper.endOffset(originalRange)); |
| 719 | } |
| 720 | |
| 721 | RefPtr<StyleRule> CSSParserImpl::consumeStyleRule(CSSParserTokenRange prelude, CSSParserTokenRange block) |
| 722 | { |
| 723 | CSSSelectorList selectorList = CSSSelectorParser::parseSelector(prelude, m_context, m_styleSheet.get()); |
| 724 | if (!selectorList.isValid()) |
| 725 | return nullptr; // Parse error, invalid selector list |
| 726 | |
| 727 | if (m_observerWrapper) |
| 728 | observeSelectors(*m_observerWrapper, prelude); |
| 729 | |
| 730 | if (m_deferredParser) { |
| 731 | // If a rule is empty (i.e., only whitespace), don't bother using |
| 732 | // deferred parsing. This allows the empty rule optimization in ElementRuleCollector |
| 733 | // to continue to work. Note we don't have to consider CommentTokens, since those |
| 734 | // are stripped out. |
| 735 | CSSParserTokenRange blockCopy = block; |
| 736 | blockCopy.consumeWhitespace(); |
| 737 | if (!blockCopy.atEnd()) { |
| 738 | return StyleRule::create(createDeferredStyleProperties(block), m_context.hasDocumentSecurityOrigin, WTFMove(selectorList)); |
| 739 | } |
| 740 | } |
| 741 | |
| 742 | consumeDeclarationList(block, StyleRule::Style); |
| 743 | return StyleRule::create(createStyleProperties(m_parsedProperties, m_context.mode), m_context.hasDocumentSecurityOrigin, WTFMove(selectorList)); |
| 744 | } |
| 745 | |
| 746 | void CSSParserImpl::consumeDeclarationList(CSSParserTokenRange range, StyleRule::Type ruleType) |
| 747 | { |
| 748 | ASSERT(m_parsedProperties.isEmpty()); |
| 749 | |
| 750 | bool useObserver = m_observerWrapper && (ruleType == StyleRule::Style || ruleType == StyleRule::Keyframe); |
| 751 | if (useObserver) { |
| 752 | m_observerWrapper->observer().startRuleBody(m_observerWrapper->previousTokenStartOffset(range)); |
| 753 | m_observerWrapper->skipCommentsBefore(range, true); |
| 754 | } |
| 755 | |
| 756 | while (!range.atEnd()) { |
| 757 | switch (range.peek().type()) { |
| 758 | case WhitespaceToken: |
| 759 | case SemicolonToken: |
| 760 | range.consume(); |
| 761 | break; |
| 762 | case IdentToken: { |
| 763 | const CSSParserToken* declarationStart = &range.peek(); |
| 764 | |
| 765 | if (useObserver) |
| 766 | m_observerWrapper->yieldCommentsBefore(range); |
| 767 | |
| 768 | while (!range.atEnd() && range.peek().type() != SemicolonToken) |
| 769 | range.consumeComponentValue(); |
| 770 | |
| 771 | consumeDeclaration(range.makeSubRange(declarationStart, &range.peek()), ruleType); |
| 772 | |
| 773 | if (useObserver) |
| 774 | m_observerWrapper->skipCommentsBefore(range, false); |
| 775 | break; |
| 776 | } |
| 777 | case AtKeywordToken: { |
| 778 | // FIXME-NEWPARSER: Support apply |
| 779 | AllowedRulesType allowedRules = /* ruleType == StyleRule::Style && RuntimeEnabledFeatures::cssApplyAtRulesEnabled() ? ApplyRules :*/ NoRules; |
| 780 | RefPtr<StyleRuleBase> rule = consumeAtRule(range, allowedRules); |
| 781 | ASSERT_UNUSED(rule, !rule); |
| 782 | break; |
| 783 | } |
| 784 | default: // Parse error, unexpected token in declaration list |
| 785 | while (!range.atEnd() && range.peek().type() != SemicolonToken) |
| 786 | range.consumeComponentValue(); |
| 787 | break; |
| 788 | } |
| 789 | } |
| 790 | |
| 791 | // Yield remaining comments |
| 792 | if (useObserver) { |
| 793 | m_observerWrapper->yieldCommentsBefore(range); |
| 794 | m_observerWrapper->observer().endRuleBody(m_observerWrapper->endOffset(range)); |
| 795 | } |
| 796 | } |
| 797 | |
| 798 | void CSSParserImpl::consumeDeclaration(CSSParserTokenRange range, StyleRule::Type ruleType) |
| 799 | { |
| 800 | CSSParserTokenRange rangeCopy = range; // For inspector callbacks |
| 801 | |
| 802 | ASSERT(range.peek().type() == IdentToken); |
| 803 | const CSSParserToken& token = range.consumeIncludingWhitespace(); |
| 804 | CSSPropertyID propertyID = token.parseAsCSSPropertyID(); |
| 805 | if (range.consume().type() != ColonToken) |
| 806 | return; // Parse error |
| 807 | |
| 808 | bool important = false; |
| 809 | const CSSParserToken* declarationValueEnd = range.end(); |
| 810 | const CSSParserToken* last = range.end() - 1; |
| 811 | while (last->type() == WhitespaceToken) |
| 812 | --last; |
| 813 | if (last->type() == IdentToken && equalIgnoringASCIICase(last->value(), "important" )) { |
| 814 | --last; |
| 815 | while (last->type() == WhitespaceToken) |
| 816 | --last; |
| 817 | if (last->type() == DelimiterToken && last->delimiter() == '!') { |
| 818 | important = true; |
| 819 | declarationValueEnd = last; |
| 820 | } |
| 821 | } |
| 822 | |
| 823 | size_t propertiesCount = m_parsedProperties.size(); |
| 824 | if (propertyID == CSSPropertyInvalid && CSSVariableParser::isValidVariableName(token)) { |
| 825 | AtomicString variableName = token.value().toAtomicString(); |
| 826 | consumeCustomPropertyValue(range.makeSubRange(&range.peek(), declarationValueEnd), variableName, important); |
| 827 | } |
| 828 | |
| 829 | if (important && (ruleType == StyleRule::FontFace || ruleType == StyleRule::Keyframe)) |
| 830 | return; |
| 831 | |
| 832 | if (propertyID != CSSPropertyInvalid) |
| 833 | consumeDeclarationValue(range.makeSubRange(&range.peek(), declarationValueEnd), propertyID, important, ruleType); |
| 834 | |
| 835 | if (m_observerWrapper && (ruleType == StyleRule::Style || ruleType == StyleRule::Keyframe)) { |
| 836 | m_observerWrapper->observer().observeProperty( |
| 837 | m_observerWrapper->startOffset(rangeCopy), m_observerWrapper->endOffset(rangeCopy), |
| 838 | important, m_parsedProperties.size() != propertiesCount); |
| 839 | } |
| 840 | } |
| 841 | |
| 842 | void CSSParserImpl::consumeCustomPropertyValue(CSSParserTokenRange range, const AtomicString& variableName, bool important) |
| 843 | { |
| 844 | if (RefPtr<CSSCustomPropertyValue> value = CSSVariableParser::parseDeclarationValue(variableName, range, m_context)) |
| 845 | m_parsedProperties.append(CSSProperty(CSSPropertyCustom, WTFMove(value), important)); |
| 846 | } |
| 847 | |
| 848 | void CSSParserImpl::consumeDeclarationValue(CSSParserTokenRange range, CSSPropertyID propertyID, bool important, StyleRule::Type ruleType) |
| 849 | { |
| 850 | CSSPropertyParser::parseValue(propertyID, important, range, m_context, m_parsedProperties, ruleType); |
| 851 | } |
| 852 | |
| 853 | std::unique_ptr<Vector<double>> CSSParserImpl::consumeKeyframeKeyList(CSSParserTokenRange range) |
| 854 | { |
| 855 | std::unique_ptr<Vector<double>> result = std::unique_ptr<Vector<double>>(new Vector<double>); |
| 856 | while (true) { |
| 857 | range.consumeWhitespace(); |
| 858 | const CSSParserToken& token = range.consumeIncludingWhitespace(); |
| 859 | if (token.type() == PercentageToken && token.numericValue() >= 0 && token.numericValue() <= 100) |
| 860 | result->append(token.numericValue() / 100); |
| 861 | else if (token.type() == IdentToken && equalIgnoringASCIICase(token.value(), "from" )) |
| 862 | result->append(0); |
| 863 | else if (token.type() == IdentToken && equalIgnoringASCIICase(token.value(), "to" )) |
| 864 | result->append(1); |
| 865 | else |
| 866 | return nullptr; // Parser error, invalid value in keyframe selector |
| 867 | if (range.atEnd()) |
| 868 | return result; |
| 869 | if (range.consume().type() != CommaToken) |
| 870 | return nullptr; // Parser error |
| 871 | } |
| 872 | } |
| 873 | |
| 874 | } // namespace WebCore |
| 875 | |