| OLD | NEW |
| (Empty) |
| 1 /* | |
| 2 * Copyright (C) 1999 Lars Knoll (knoll@kde.org) | |
| 3 * (C) 2004-2005 Allan Sandfeld Jensen (kde@carewolf.com) | |
| 4 * Copyright (C) 2006, 2007 Nicholas Shanks (webkit@nickshanks.com) | |
| 5 * Copyright (C) 2005, 2006, 2007, 2008 Apple Inc. All rights reserved. | |
| 6 * Copyright (C) 2007 Alexey Proskuryakov <ap@webkit.org> | |
| 7 * Copyright (C) 2007, 2008 Eric Seidel <eric@webkit.org> | |
| 8 * | |
| 9 * This library is free software; you can redistribute it and/or | |
| 10 * modify it under the terms of the GNU Library General Public | |
| 11 * License as published by the Free Software Foundation; either | |
| 12 * version 2 of the License, or (at your option) any later version. | |
| 13 * | |
| 14 * This library is distributed in the hope that it will be useful, | |
| 15 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
| 17 * Library General Public License for more details. | |
| 18 * | |
| 19 * You should have received a copy of the GNU Library General Public License | |
| 20 * along with this library; see the file COPYING.LIB. If not, write to | |
| 21 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, | |
| 22 * Boston, MA 02110-1301, USA. | |
| 23 */ | |
| 24 | |
| 25 #include "config.h" | |
| 26 #include "CSSStyleSelector.h" | |
| 27 | |
| 28 #include "CSSBorderImageValue.h" | |
| 29 #include "CSSCursorImageValue.h" | |
| 30 #include "CSSFontFace.h" | |
| 31 #include "CSSFontFaceRule.h" | |
| 32 #include "CSSFontFaceSource.h" | |
| 33 #include "CSSImportRule.h" | |
| 34 #include "CSSMediaRule.h" | |
| 35 #include "CSSParser.h" | |
| 36 #include "CSSPrimitiveValueMappings.h" | |
| 37 #include "CSSProperty.h" | |
| 38 #include "CSSPropertyNames.h" | |
| 39 #include "CSSReflectValue.h" | |
| 40 #include "CSSRuleList.h" | |
| 41 #include "CSSSelector.h" | |
| 42 #include "CSSStyleRule.h" | |
| 43 #include "CSSStyleSheet.h" | |
| 44 #include "CSSTimingFunctionValue.h" | |
| 45 #include "CSSValueList.h" | |
| 46 #include "CSSVariableDependentValue.h" | |
| 47 #include "CSSVariablesDeclaration.h" | |
| 48 #include "CSSVariablesRule.h" | |
| 49 #include "CachedImage.h" | |
| 50 #include "Counter.h" | |
| 51 #include "FontCache.h" | |
| 52 #include "FontFamilyValue.h" | |
| 53 #include "FontValue.h" | |
| 54 #include "Frame.h" | |
| 55 #include "FrameView.h" | |
| 56 #include "HTMLDocument.h" | |
| 57 #include "HTMLElement.h" | |
| 58 #include "HTMLInputElement.h" | |
| 59 #include "HTMLTextAreaElement.h" | |
| 60 #include "HTMLNames.h" | |
| 61 #include "MediaList.h" | |
| 62 #include "MediaQueryEvaluator.h" | |
| 63 #include "Page.h" | |
| 64 #include "PageGroup.h" | |
| 65 #include "Pair.h" | |
| 66 #include "Rect.h" | |
| 67 #include "RenderTheme.h" | |
| 68 #include "SelectionController.h" | |
| 69 #include "Settings.h" | |
| 70 #include "ShadowValue.h" | |
| 71 #include "StyleCachedImage.h" | |
| 72 #include "StyleGeneratedImage.h" | |
| 73 #include "StyleSheetList.h" | |
| 74 #include "Text.h" | |
| 75 #include "UserAgentStyleSheets.h" | |
| 76 #include "WebKitCSSKeyframeRule.h" | |
| 77 #include "WebKitCSSKeyframesRule.h" | |
| 78 #include "WebKitCSSTransformValue.h" | |
| 79 #include "XMLNames.h" | |
| 80 #include "loader.h" | |
| 81 #include <wtf/Vector.h> | |
| 82 | |
| 83 #if ENABLE(DASHBOARD_SUPPORT) | |
| 84 #include "DashboardRegion.h" | |
| 85 #endif | |
| 86 | |
| 87 #if ENABLE(SVG) | |
| 88 #include "XLinkNames.h" | |
| 89 #include "SVGNames.h" | |
| 90 #endif | |
| 91 | |
| 92 using namespace std; | |
| 93 | |
| 94 namespace WebCore { | |
| 95 | |
| 96 using namespace HTMLNames; | |
| 97 | |
| 98 // #define STYLE_SHARING_STATS 1 | |
| 99 | |
| 100 #define HANDLE_INHERIT(prop, Prop) \ | |
| 101 if (isInherit) { \ | |
| 102 m_style->set##Prop(m_parentStyle->prop()); \ | |
| 103 return; \ | |
| 104 } | |
| 105 | |
| 106 #define HANDLE_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 107 HANDLE_INHERIT(prop, Prop) \ | |
| 108 if (isInitial) { \ | |
| 109 m_style->set##Prop(RenderStyle::initial##Prop()); \ | |
| 110 return; \ | |
| 111 } | |
| 112 | |
| 113 #define HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(prop, Prop, Value) \ | |
| 114 HANDLE_INHERIT(prop, Prop) \ | |
| 115 if (isInitial) { \ | |
| 116 m_style->set##Prop(RenderStyle::initial##Value());\ | |
| 117 return;\ | |
| 118 } | |
| 119 | |
| 120 #define HANDLE_FILL_LAYER_INHERIT_AND_INITIAL(layerType, LayerType, prop, Prop)
\ | |
| 121 if (isInherit) { \ | |
| 122 FillLayer* currChild = m_style->access##LayerType##Layers(); \ | |
| 123 FillLayer* prevChild = 0; \ | |
| 124 const FillLayer* currParent = m_parentStyle->layerType##Layers(); \ | |
| 125 while (currParent && currParent->is##Prop##Set()) { \ | |
| 126 if (!currChild) { \ | |
| 127 /* Need to make a new layer.*/ \ | |
| 128 currChild = new FillLayer(LayerType##FillLayer); \ | |
| 129 prevChild->setNext(currChild); \ | |
| 130 } \ | |
| 131 currChild->set##Prop(currParent->prop()); \ | |
| 132 prevChild = currChild; \ | |
| 133 currChild = prevChild->next(); \ | |
| 134 currParent = currParent->next(); \ | |
| 135 } \ | |
| 136 \ | |
| 137 while (currChild) { \ | |
| 138 /* Reset any remaining layers to not have the property set. */ \ | |
| 139 currChild->clear##Prop(); \ | |
| 140 currChild = currChild->next(); \ | |
| 141 } \ | |
| 142 } else if (isInitial) { \ | |
| 143 FillLayer* currChild = m_style->access##LayerType##Layers(); \ | |
| 144 currChild->set##Prop(FillLayer::initialFill##Prop(LayerType##FillLayer)); \ | |
| 145 for (currChild = currChild->next(); currChild; currChild = currChild->next()
) \ | |
| 146 currChild->clear##Prop(); \ | |
| 147 } | |
| 148 | |
| 149 #define HANDLE_FILL_LAYER_VALUE(layerType, LayerType, prop, Prop, value) { \ | |
| 150 HANDLE_FILL_LAYER_INHERIT_AND_INITIAL(layerType, LayerType, prop, Prop) \ | |
| 151 if (isInherit || isInitial) \ | |
| 152 return; \ | |
| 153 FillLayer* currChild = m_style->access##LayerType##Layers(); \ | |
| 154 FillLayer* prevChild = 0; \ | |
| 155 if (value->isValueList()) { \ | |
| 156 /* Walk each value and put it into a layer, creating new layers as needed. *
/ \ | |
| 157 CSSValueList* valueList = static_cast<CSSValueList*>(value); \ | |
| 158 for (unsigned int i = 0; i < valueList->length(); i++) { \ | |
| 159 if (!currChild) { \ | |
| 160 /* Need to make a new layer to hold this value */ \ | |
| 161 currChild = new FillLayer(LayerType##FillLayer); \ | |
| 162 prevChild->setNext(currChild); \ | |
| 163 } \ | |
| 164 mapFill##Prop(currChild, valueList->itemWithoutBoundsCheck(i)); \ | |
| 165 prevChild = currChild; \ | |
| 166 currChild = currChild->next(); \ | |
| 167 } \ | |
| 168 } else { \ | |
| 169 mapFill##Prop(currChild, value); \ | |
| 170 currChild = currChild->next(); \ | |
| 171 } \ | |
| 172 while (currChild) { \ | |
| 173 /* Reset all remaining layers to not have the property set. */ \ | |
| 174 currChild->clear##Prop(); \ | |
| 175 currChild = currChild->next(); \ | |
| 176 } } | |
| 177 | |
| 178 #define HANDLE_BACKGROUND_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 179 HANDLE_FILL_LAYER_INHERIT_AND_INITIAL(background, Background, prop, Prop) | |
| 180 | |
| 181 #define HANDLE_BACKGROUND_VALUE(prop, Prop, value) \ | |
| 182 HANDLE_FILL_LAYER_VALUE(background, Background, prop, Prop, value) | |
| 183 | |
| 184 #define HANDLE_MASK_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 185 HANDLE_FILL_LAYER_INHERIT_AND_INITIAL(mask, Mask, prop, Prop) | |
| 186 | |
| 187 #define HANDLE_MASK_VALUE(prop, Prop, value) \ | |
| 188 HANDLE_FILL_LAYER_VALUE(mask, Mask, prop, Prop, value) | |
| 189 | |
| 190 #define HANDLE_ANIMATION_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 191 if (isInherit) { \ | |
| 192 AnimationList* list = m_style->accessAnimations(); \ | |
| 193 const AnimationList* parentList = m_parentStyle->animations(); \ | |
| 194 size_t i = 0, parentSize = parentList ? parentList->size() : 0; \ | |
| 195 for ( ; i < parentSize && (*parentList)[i]->is##Prop##Set(); ++i) { \ | |
| 196 if (list->size() <= i) \ | |
| 197 list->append(Animation::create()); \ | |
| 198 (*list)[i]->set##Prop((*parentList)[i]->prop()); \ | |
| 199 } \ | |
| 200 \ | |
| 201 /* Reset any remaining animations to not have the property set. */ \ | |
| 202 for ( ; i < list->size(); ++i) \ | |
| 203 (*list)[i]->clear##Prop(); \ | |
| 204 } else if (isInitial) { \ | |
| 205 AnimationList* list = m_style->accessAnimations(); \ | |
| 206 if (list->isEmpty()) \ | |
| 207 list->append(Animation::create()); \ | |
| 208 (*list)[0]->set##Prop(RenderStyle::initialAnimation##Prop()); \ | |
| 209 for (size_t i = 1; i < list->size(); ++i) \ | |
| 210 (*list)[0]->clear##Prop(); \ | |
| 211 } | |
| 212 | |
| 213 #define HANDLE_ANIMATION_VALUE(prop, Prop, value) { \ | |
| 214 HANDLE_ANIMATION_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 215 if (isInherit || isInitial) \ | |
| 216 return; \ | |
| 217 AnimationList* list = m_style->accessAnimations(); \ | |
| 218 size_t childIndex = 0; \ | |
| 219 if (value->isValueList()) { \ | |
| 220 /* Walk each value and put it into an animation, creating new animations as
needed. */ \ | |
| 221 CSSValueList* valueList = static_cast<CSSValueList*>(value); \ | |
| 222 for (unsigned int i = 0; i < valueList->length(); i++) { \ | |
| 223 if (childIndex <= list->size()) \ | |
| 224 list->append(Animation::create()); \ | |
| 225 mapAnimation##Prop((*list)[childIndex].get(), valueList->itemWithoutBoun
dsCheck(i)); \ | |
| 226 ++childIndex; \ | |
| 227 } \ | |
| 228 } else { \ | |
| 229 if (list->isEmpty()) \ | |
| 230 list->append(Animation::create()); \ | |
| 231 mapAnimation##Prop((*list)[childIndex].get(), value); \ | |
| 232 childIndex = 1; \ | |
| 233 } \ | |
| 234 for ( ; childIndex < list->size(); ++childIndex) { \ | |
| 235 /* Reset all remaining animations to not have the property set. */ \ | |
| 236 (*list)[childIndex]->clear##Prop(); \ | |
| 237 } \ | |
| 238 } | |
| 239 | |
| 240 #define HANDLE_TRANSITION_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 241 if (isInherit) { \ | |
| 242 AnimationList* list = m_style->accessTransitions(); \ | |
| 243 const AnimationList* parentList = m_parentStyle->transitions(); \ | |
| 244 size_t i = 0, parentSize = parentList ? parentList->size() : 0; \ | |
| 245 for ( ; i < parentSize && (*parentList)[i]->is##Prop##Set(); ++i) { \ | |
| 246 if (list->size() <= i) \ | |
| 247 list->append(Animation::create()); \ | |
| 248 (*list)[i]->set##Prop((*parentList)[i]->prop()); \ | |
| 249 } \ | |
| 250 \ | |
| 251 /* Reset any remaining transitions to not have the property set. */ \ | |
| 252 for ( ; i < list->size(); ++i) \ | |
| 253 (*list)[i]->clear##Prop(); \ | |
| 254 } else if (isInitial) { \ | |
| 255 AnimationList* list = m_style->accessTransitions(); \ | |
| 256 if (list->isEmpty()) \ | |
| 257 list->append(Animation::create()); \ | |
| 258 (*list)[0]->set##Prop(RenderStyle::initialAnimation##Prop()); \ | |
| 259 for (size_t i = 1; i < list->size(); ++i) \ | |
| 260 (*list)[0]->clear##Prop(); \ | |
| 261 } | |
| 262 | |
| 263 #define HANDLE_TRANSITION_VALUE(prop, Prop, value) { \ | |
| 264 HANDLE_TRANSITION_INHERIT_AND_INITIAL(prop, Prop) \ | |
| 265 if (isInherit || isInitial) \ | |
| 266 return; \ | |
| 267 AnimationList* list = m_style->accessTransitions(); \ | |
| 268 size_t childIndex = 0; \ | |
| 269 if (value->isValueList()) { \ | |
| 270 /* Walk each value and put it into a transition, creating new animations as
needed. */ \ | |
| 271 CSSValueList* valueList = static_cast<CSSValueList*>(value); \ | |
| 272 for (unsigned int i = 0; i < valueList->length(); i++) { \ | |
| 273 if (childIndex <= list->size()) \ | |
| 274 list->append(Animation::create()); \ | |
| 275 mapAnimation##Prop((*list)[childIndex].get(), valueList->itemWithoutBoun
dsCheck(i)); \ | |
| 276 ++childIndex; \ | |
| 277 } \ | |
| 278 } else { \ | |
| 279 if (list->isEmpty()) \ | |
| 280 list->append(Animation::create()); \ | |
| 281 mapAnimation##Prop((*list)[childIndex].get(), value); \ | |
| 282 childIndex = 1; \ | |
| 283 } \ | |
| 284 for ( ; childIndex < list->size(); ++childIndex) { \ | |
| 285 /* Reset all remaining transitions to not have the property set. */ \ | |
| 286 (*list)[childIndex]->clear##Prop(); \ | |
| 287 } \ | |
| 288 } | |
| 289 | |
| 290 #define HANDLE_INHERIT_COND(propID, prop, Prop) \ | |
| 291 if (id == propID) { \ | |
| 292 m_style->set##Prop(m_parentStyle->prop()); \ | |
| 293 return; \ | |
| 294 } | |
| 295 | |
| 296 #define HANDLE_INHERIT_COND_WITH_BACKUP(propID, prop, propAlt, Prop) \ | |
| 297 if (id == propID) { \ | |
| 298 if (m_parentStyle->prop().isValid()) \ | |
| 299 m_style->set##Prop(m_parentStyle->prop()); \ | |
| 300 else \ | |
| 301 m_style->set##Prop(m_parentStyle->propAlt()); \ | |
| 302 return; \ | |
| 303 } | |
| 304 | |
| 305 #define HANDLE_INITIAL_COND(propID, Prop) \ | |
| 306 if (id == propID) { \ | |
| 307 m_style->set##Prop(RenderStyle::initial##Prop()); \ | |
| 308 return; \ | |
| 309 } | |
| 310 | |
| 311 #define HANDLE_INITIAL_COND_WITH_VALUE(propID, Prop, Value) \ | |
| 312 if (id == propID) { \ | |
| 313 m_style->set##Prop(RenderStyle::initial##Value()); \ | |
| 314 return; \ | |
| 315 } | |
| 316 | |
| 317 class CSSRuleSet { | |
| 318 public: | |
| 319 CSSRuleSet(); | |
| 320 ~CSSRuleSet(); | |
| 321 | |
| 322 typedef HashMap<AtomicStringImpl*, CSSRuleDataList*> AtomRuleMap; | |
| 323 | |
| 324 void addRulesFromSheet(CSSStyleSheet*, const MediaQueryEvaluator&, CSSStyleS
elector* = 0); | |
| 325 | |
| 326 void addRule(CSSStyleRule* rule, CSSSelector* sel); | |
| 327 void addToRuleSet(AtomicStringImpl* key, AtomRuleMap& map, | |
| 328 CSSStyleRule* rule, CSSSelector* sel); | |
| 329 | |
| 330 CSSRuleDataList* getIDRules(AtomicStringImpl* key) { return m_idRules.get(ke
y); } | |
| 331 CSSRuleDataList* getClassRules(AtomicStringImpl* key) { return m_classRules.
get(key); } | |
| 332 CSSRuleDataList* getTagRules(AtomicStringImpl* key) { return m_tagRules.get(
key); } | |
| 333 CSSRuleDataList* getUniversalRules() { return m_universalRules; } | |
| 334 | |
| 335 public: | |
| 336 AtomRuleMap m_idRules; | |
| 337 AtomRuleMap m_classRules; | |
| 338 AtomRuleMap m_tagRules; | |
| 339 CSSRuleDataList* m_universalRules; | |
| 340 unsigned m_ruleCount; | |
| 341 }; | |
| 342 | |
| 343 static CSSRuleSet* defaultStyle; | |
| 344 static CSSRuleSet* defaultQuirksStyle; | |
| 345 static CSSRuleSet* defaultPrintStyle; | |
| 346 static CSSRuleSet* defaultViewSourceStyle; | |
| 347 | |
| 348 RenderStyle* CSSStyleSelector::s_styleNotYetAvailable; | |
| 349 | |
| 350 static PseudoState pseudoState; | |
| 351 | |
| 352 static void loadDefaultStyle(); | |
| 353 | |
| 354 static const MediaQueryEvaluator& screenEval() | |
| 355 { | |
| 356 static const MediaQueryEvaluator staticScreenEval("screen"); | |
| 357 return staticScreenEval; | |
| 358 } | |
| 359 | |
| 360 static const MediaQueryEvaluator& printEval() | |
| 361 { | |
| 362 static const MediaQueryEvaluator staticPrintEval("print"); | |
| 363 return staticPrintEval; | |
| 364 } | |
| 365 | |
| 366 CSSStyleSelector::CSSStyleSelector(Document* doc, const String& userStyleSheet,
StyleSheetList* styleSheets, CSSStyleSheet* mappedElementSheet, bool strictParsi
ng, bool matchAuthorAndUserStyles) | |
| 367 : m_backgroundData(BackgroundFillLayer) | |
| 368 , m_checker(doc, strictParsing, false) | |
| 369 , m_fontSelector(CSSFontSelector::create(doc)) | |
| 370 { | |
| 371 init(); | |
| 372 | |
| 373 m_matchAuthorAndUserStyles = matchAuthorAndUserStyles; | |
| 374 | |
| 375 if (!defaultStyle) | |
| 376 loadDefaultStyle(); | |
| 377 | |
| 378 m_userStyle = 0; | |
| 379 | |
| 380 // construct document root element default style. this is needed | |
| 381 // to evaluate media queries that contain relative constraints, like "screen
and (max-width: 10em)" | |
| 382 // This is here instead of constructor, because when constructor is run, | |
| 383 // document doesn't have documentElement | |
| 384 // NOTE: this assumes that element that gets passed to styleForElement -call | |
| 385 // is always from the document that owns the style selector | |
| 386 FrameView* view = doc->view(); | |
| 387 if (view) | |
| 388 m_medium = new MediaQueryEvaluator(view->mediaType()); | |
| 389 else | |
| 390 m_medium = new MediaQueryEvaluator("all"); | |
| 391 | |
| 392 Element* root = doc->documentElement(); | |
| 393 | |
| 394 if (root) | |
| 395 m_rootDefaultStyle = styleForElement(root, 0, false, true); // dont ref,
because the RenderStyle is allocated from global heap | |
| 396 | |
| 397 if (m_rootDefaultStyle && view) { | |
| 398 delete m_medium; | |
| 399 m_medium = new MediaQueryEvaluator(view->mediaType(), view->frame(), m_r
ootDefaultStyle); | |
| 400 } | |
| 401 | |
| 402 // FIXME: This sucks! The user sheet is reparsed every time! | |
| 403 if (!userStyleSheet.isEmpty()) { | |
| 404 m_userSheet = CSSStyleSheet::create(doc); | |
| 405 m_userSheet->parseString(userStyleSheet, strictParsing); | |
| 406 | |
| 407 m_userStyle = new CSSRuleSet(); | |
| 408 m_userStyle->addRulesFromSheet(m_userSheet.get(), *m_medium, this); | |
| 409 } | |
| 410 | |
| 411 // add stylesheets from document | |
| 412 m_authorStyle = new CSSRuleSet(); | |
| 413 | |
| 414 // Add rules from elments like SVG's <font-face> | |
| 415 if (mappedElementSheet) | |
| 416 m_authorStyle->addRulesFromSheet(mappedElementSheet, *m_medium, this); | |
| 417 | |
| 418 unsigned length = styleSheets->length(); | |
| 419 for (unsigned i = 0; i < length; i++) { | |
| 420 StyleSheet* sheet = styleSheets->item(i); | |
| 421 if (sheet->isCSSStyleSheet() && !sheet->disabled()) | |
| 422 m_authorStyle->addRulesFromSheet(static_cast<CSSStyleSheet*>(sheet),
*m_medium, this); | |
| 423 } | |
| 424 } | |
| 425 | |
| 426 // this is a simplified style setting function for keyframe styles | |
| 427 void CSSStyleSelector::addKeyframeStyle(Document* doc, const WebKitCSSKeyframesR
ule* rule) | |
| 428 { | |
| 429 AtomicString s(rule->name()); | |
| 430 RefPtr<KeyframeList> list; | |
| 431 if (m_keyframeRuleMap.contains(s.impl())) | |
| 432 list = m_keyframeRuleMap.get(s.impl()).get(); | |
| 433 else { | |
| 434 list = KeyframeList::create(s); | |
| 435 m_keyframeRuleMap.add(s.impl(), list); | |
| 436 } | |
| 437 list->clear(); | |
| 438 | |
| 439 for (unsigned i = 0; i < rule->length(); ++i) { | |
| 440 const WebKitCSSKeyframeRule* kf = rule->item(i); | |
| 441 m_style = new (doc->renderArena()) RenderStyle(); | |
| 442 m_style->ref(); | |
| 443 CSSMutableStyleDeclaration* decl = kf->style(); | |
| 444 DeprecatedValueListConstIterator<CSSProperty> end; | |
| 445 for (DeprecatedValueListConstIterator<CSSProperty> it = decl->valuesIter
ator(); it != end; ++it) { | |
| 446 const CSSProperty& current = *it; | |
| 447 applyProperty(current.id(), current.value()); | |
| 448 list->addProperty(current.id()); | |
| 449 } | |
| 450 list->insert(kf->key(), *m_style); | |
| 451 m_style->deref(doc->renderArena()); | |
| 452 m_style = 0; | |
| 453 } | |
| 454 } | |
| 455 | |
| 456 void CSSStyleSelector::init() | |
| 457 { | |
| 458 m_element = 0; | |
| 459 m_matchedDecls.clear(); | |
| 460 m_ruleList = 0; | |
| 461 m_rootDefaultStyle = 0; | |
| 462 m_medium = 0; | |
| 463 } | |
| 464 | |
| 465 CSSStyleSelector::~CSSStyleSelector() | |
| 466 { | |
| 467 m_fontSelector->clearDocument(); | |
| 468 delete m_medium; | |
| 469 ::delete m_rootDefaultStyle; | |
| 470 delete m_authorStyle; | |
| 471 delete m_userStyle; | |
| 472 deleteAllValues(m_viewportDependentMediaQueryResults); | |
| 473 m_keyframeRuleMap.clear(); | |
| 474 } | |
| 475 | |
| 476 static CSSStyleSheet* parseUASheet(const char* characters, unsigned size) | |
| 477 { | |
| 478 CSSStyleSheet* sheet = CSSStyleSheet::create().releaseRef(); // leak the she
et on purpose | |
| 479 sheet->parseString(String(characters, size)); | |
| 480 return sheet; | |
| 481 } | |
| 482 | |
| 483 static void loadDefaultStyle() | |
| 484 { | |
| 485 ASSERT(!defaultStyle); | |
| 486 | |
| 487 defaultStyle = new CSSRuleSet; | |
| 488 defaultPrintStyle = new CSSRuleSet; | |
| 489 defaultQuirksStyle = new CSSRuleSet; | |
| 490 defaultViewSourceStyle = new CSSRuleSet; | |
| 491 | |
| 492 // Strict-mode rules. | |
| 493 CSSStyleSheet* defaultSheet = parseUASheet(html4UserAgentStyleSheet, sizeof(
html4UserAgentStyleSheet)); | |
| 494 RenderTheme::adjustDefaultStyleSheet(defaultSheet); | |
| 495 defaultStyle->addRulesFromSheet(defaultSheet, screenEval()); | |
| 496 defaultPrintStyle->addRulesFromSheet(defaultSheet, printEval()); | |
| 497 | |
| 498 // Quirks-mode rules. | |
| 499 defaultQuirksStyle->addRulesFromSheet(parseUASheet(quirksUserAgentStyleSheet
, sizeof(quirksUserAgentStyleSheet)), screenEval()); | |
| 500 | |
| 501 // View source rules. | |
| 502 defaultViewSourceStyle->addRulesFromSheet(parseUASheet(sourceUserAgentStyleS
heet, sizeof(sourceUserAgentStyleSheet)), screenEval()); | |
| 503 } | |
| 504 | |
| 505 void CSSStyleSelector::addMatchedDeclaration(CSSMutableStyleDeclaration* decl) | |
| 506 { | |
| 507 if (!decl->hasVariableDependentValue()) { | |
| 508 m_matchedDecls.append(decl); | |
| 509 return; | |
| 510 } | |
| 511 | |
| 512 // See if we have already resolved the variables in this declaration. | |
| 513 CSSMutableStyleDeclaration* resolvedDecl = m_resolvedVariablesDeclarations.g
et(decl).get(); | |
| 514 if (resolvedDecl) { | |
| 515 m_matchedDecls.append(resolvedDecl); | |
| 516 return; | |
| 517 } | |
| 518 | |
| 519 // If this declaration has any variables in it, then we need to make a clone
d | |
| 520 // declaration with as many variables resolved as possible for this style se
lector's media. | |
| 521 RefPtr<CSSMutableStyleDeclaration> newDecl = CSSMutableStyleDeclaration::cre
ate(decl->parentRule()); | |
| 522 m_matchedDecls.append(newDecl.get()); | |
| 523 m_resolvedVariablesDeclarations.set(decl, newDecl); | |
| 524 | |
| 525 HashSet<String> usedBlockVariables; | |
| 526 resolveVariablesForDeclaration(decl, newDecl.get(), usedBlockVariables); | |
| 527 } | |
| 528 | |
| 529 void CSSStyleSelector::resolveVariablesForDeclaration(CSSMutableStyleDeclaration
* decl, CSSMutableStyleDeclaration* newDecl, HashSet<String>& usedBlockVariables
) | |
| 530 { | |
| 531 // Now iterate over the properties in the original declaration. As we resol
ve variables we'll end up | |
| 532 // mutating the new declaration (possibly expanding shorthands). The new de
claration has no m_node | |
| 533 // though, so it can't mistakenly call setChanged on anything. | |
| 534 DeprecatedValueListConstIterator<CSSProperty> end; | |
| 535 for (DeprecatedValueListConstIterator<CSSProperty> it = decl->valuesIterator
(); it != end; ++it) { | |
| 536 const CSSProperty& current = *it; | |
| 537 if (!current.value()->isVariableDependentValue()) { | |
| 538 // We can just add the parsed property directly. | |
| 539 newDecl->addParsedProperty(current); | |
| 540 continue; | |
| 541 } | |
| 542 CSSValueList* valueList = static_cast<CSSVariableDependentValue*>(curren
t.value())->valueList(); | |
| 543 if (!valueList) | |
| 544 continue; | |
| 545 CSSParserValueList resolvedValueList; | |
| 546 unsigned s = valueList->length(); | |
| 547 bool fullyResolved = true; | |
| 548 for (unsigned i = 0; i < s; ++i) { | |
| 549 CSSValue* val = valueList->item(i); | |
| 550 CSSPrimitiveValue* primitiveValue = val->isPrimitiveValue() ? static
_cast<CSSPrimitiveValue*>(val) : 0; | |
| 551 if (primitiveValue && primitiveValue->isVariable()) { | |
| 552 CSSVariablesRule* rule = m_variablesMap.get(primitiveValue->getS
tringValue()); | |
| 553 if (!rule || !rule->variables()) { | |
| 554 fullyResolved = false; | |
| 555 break; | |
| 556 } | |
| 557 | |
| 558 if (current.id() == CSSPropertyWebkitVariableDeclarationBlock &&
s == 1) { | |
| 559 fullyResolved = false; | |
| 560 if (!usedBlockVariables.contains(primitiveValue->getStringVa
lue())) { | |
| 561 CSSMutableStyleDeclaration* declBlock = rule->variables(
)->getParsedVariableDeclarationBlock(primitiveValue->getStringValue()); | |
| 562 if (declBlock) { | |
| 563 usedBlockVariables.add(primitiveValue->getStringValu
e()); | |
| 564 resolveVariablesForDeclaration(declBlock, newDecl, u
sedBlockVariables); | |
| 565 } | |
| 566 } | |
| 567 } | |
| 568 | |
| 569 CSSValueList* resolvedVariable = rule->variables()->getParsedVar
iable(primitiveValue->getStringValue()); | |
| 570 if (!resolvedVariable) { | |
| 571 fullyResolved = false; | |
| 572 break; | |
| 573 } | |
| 574 unsigned valueSize = resolvedVariable->length(); | |
| 575 for (unsigned j = 0; j < valueSize; ++j) | |
| 576 resolvedValueList.addValue(resolvedVariable->item(j)->parser
Value()); | |
| 577 } else | |
| 578 resolvedValueList.addValue(val->parserValue()); | |
| 579 } | |
| 580 | |
| 581 if (!fullyResolved) | |
| 582 continue; | |
| 583 | |
| 584 // We now have a fully resolved new value list. We want the parser to u
se this value list | |
| 585 // and parse our new declaration. | |
| 586 CSSParser(m_checker.m_strictParsing).parsePropertyWithResolvedVariables(
current.id(), current.isImportant(), newDecl, &resolvedValueList); | |
| 587 } | |
| 588 } | |
| 589 | |
| 590 void CSSStyleSelector::matchRules(CSSRuleSet* rules, int& firstRuleIndex, int& l
astRuleIndex) | |
| 591 { | |
| 592 m_matchedRules.clear(); | |
| 593 | |
| 594 if (!rules || !m_element) | |
| 595 return; | |
| 596 | |
| 597 // We need to collect the rules for id, class, tag, and everything else into
a buffer and | |
| 598 // then sort the buffer. | |
| 599 if (m_element->hasID()) | |
| 600 matchRulesForList(rules->getIDRules(m_element->getIDAttribute().impl()),
firstRuleIndex, lastRuleIndex); | |
| 601 if (m_element->hasClass()) { | |
| 602 ASSERT(m_styledElement); | |
| 603 const ClassNames& classNames = m_styledElement->classNames(); | |
| 604 size_t size = classNames.size(); | |
| 605 for (size_t i = 0; i < size; ++i) | |
| 606 matchRulesForList(rules->getClassRules(classNames[i].impl()), firstR
uleIndex, lastRuleIndex); | |
| 607 } | |
| 608 matchRulesForList(rules->getTagRules(m_element->localName().impl()), firstRu
leIndex, lastRuleIndex); | |
| 609 matchRulesForList(rules->getUniversalRules(), firstRuleIndex, lastRuleIndex)
; | |
| 610 | |
| 611 // If we didn't match any rules, we're done. | |
| 612 if (m_matchedRules.isEmpty()) | |
| 613 return; | |
| 614 | |
| 615 // Sort the set of matched rules. | |
| 616 sortMatchedRules(0, m_matchedRules.size()); | |
| 617 | |
| 618 // Now transfer the set of matched rules over to our list of decls. | |
| 619 if (!m_checker.m_collectRulesOnly) { | |
| 620 for (unsigned i = 0; i < m_matchedRules.size(); i++) | |
| 621 addMatchedDeclaration(m_matchedRules[i]->rule()->declaration()); | |
| 622 } else { | |
| 623 for (unsigned i = 0; i < m_matchedRules.size(); i++) { | |
| 624 if (!m_ruleList) | |
| 625 m_ruleList = CSSRuleList::create(); | |
| 626 m_ruleList->append(m_matchedRules[i]->rule()); | |
| 627 } | |
| 628 } | |
| 629 } | |
| 630 | |
| 631 void CSSStyleSelector::matchRulesForList(CSSRuleDataList* rules, int& firstRuleI
ndex, int& lastRuleIndex) | |
| 632 { | |
| 633 if (!rules) | |
| 634 return; | |
| 635 | |
| 636 for (CSSRuleData* d = rules->first(); d; d = d->next()) { | |
| 637 CSSStyleRule* rule = d->rule(); | |
| 638 const AtomicString& localName = m_element->localName(); | |
| 639 const AtomicString& selectorLocalName = d->selector()->m_tag.localName()
; | |
| 640 if ((localName == selectorLocalName || selectorLocalName == starAtom) &&
checkSelector(d->selector())) { | |
| 641 // If the rule has no properties to apply, then ignore it. | |
| 642 CSSMutableStyleDeclaration* decl = rule->declaration(); | |
| 643 if (!decl || !decl->length()) | |
| 644 continue; | |
| 645 | |
| 646 // If we're matching normal rules, set a pseudo bit if | |
| 647 // we really just matched a pseudo-element. | |
| 648 if (m_dynamicPseudo != RenderStyle::NOPSEUDO && m_checker.m_pseudoSt
yle == RenderStyle::NOPSEUDO) { | |
| 649 if (m_checker.m_collectRulesOnly) | |
| 650 return; | |
| 651 if (m_dynamicPseudo < RenderStyle::FIRST_INTERNAL_PSEUDOID) | |
| 652 m_style->setHasPseudoStyle(m_dynamicPseudo); | |
| 653 } else { | |
| 654 // Update our first/last rule indices in the matched rules array
. | |
| 655 lastRuleIndex = m_matchedDecls.size() + m_matchedRules.size(); | |
| 656 if (firstRuleIndex == -1) | |
| 657 firstRuleIndex = lastRuleIndex; | |
| 658 | |
| 659 // Add this rule to our list of matched rules. | |
| 660 addMatchedRule(d); | |
| 661 } | |
| 662 } | |
| 663 } | |
| 664 } | |
| 665 | |
| 666 bool operator >(CSSRuleData& r1, CSSRuleData& r2) | |
| 667 { | |
| 668 int spec1 = r1.selector()->specificity(); | |
| 669 int spec2 = r2.selector()->specificity(); | |
| 670 return (spec1 == spec2) ? r1.position() > r2.position() : spec1 > spec2; | |
| 671 } | |
| 672 bool operator <=(CSSRuleData& r1, CSSRuleData& r2) | |
| 673 { | |
| 674 return !(r1 > r2); | |
| 675 } | |
| 676 | |
| 677 void CSSStyleSelector::sortMatchedRules(unsigned start, unsigned end) | |
| 678 { | |
| 679 if (start >= end || (end - start == 1)) | |
| 680 return; // Sanity check. | |
| 681 | |
| 682 if (end - start <= 6) { | |
| 683 // Apply a bubble sort for smaller lists. | |
| 684 for (unsigned i = end - 1; i > start; i--) { | |
| 685 bool sorted = true; | |
| 686 for (unsigned j = start; j < i; j++) { | |
| 687 CSSRuleData* elt = m_matchedRules[j]; | |
| 688 CSSRuleData* elt2 = m_matchedRules[j + 1]; | |
| 689 if (*elt > *elt2) { | |
| 690 sorted = false; | |
| 691 m_matchedRules[j] = elt2; | |
| 692 m_matchedRules[j + 1] = elt; | |
| 693 } | |
| 694 } | |
| 695 if (sorted) | |
| 696 return; | |
| 697 } | |
| 698 return; | |
| 699 } | |
| 700 | |
| 701 // Peform a merge sort for larger lists. | |
| 702 unsigned mid = (start + end) / 2; | |
| 703 sortMatchedRules(start, mid); | |
| 704 sortMatchedRules(mid, end); | |
| 705 | |
| 706 CSSRuleData* elt = m_matchedRules[mid - 1]; | |
| 707 CSSRuleData* elt2 = m_matchedRules[mid]; | |
| 708 | |
| 709 // Handle the fast common case (of equal specificity). The list may already | |
| 710 // be completely sorted. | |
| 711 if (*elt <= *elt2) | |
| 712 return; | |
| 713 | |
| 714 // We have to merge sort. Ensure our merge buffer is big enough to hold | |
| 715 // all the items. | |
| 716 Vector<CSSRuleData*> rulesMergeBuffer; | |
| 717 rulesMergeBuffer.reserveCapacity(end - start); | |
| 718 | |
| 719 unsigned i1 = start; | |
| 720 unsigned i2 = mid; | |
| 721 | |
| 722 elt = m_matchedRules[i1]; | |
| 723 elt2 = m_matchedRules[i2]; | |
| 724 | |
| 725 while (i1 < mid || i2 < end) { | |
| 726 if (i1 < mid && (i2 == end || *elt <= *elt2)) { | |
| 727 rulesMergeBuffer.append(elt); | |
| 728 if (++i1 < mid) | |
| 729 elt = m_matchedRules[i1]; | |
| 730 } else { | |
| 731 rulesMergeBuffer.append(elt2); | |
| 732 if (++i2 < end) | |
| 733 elt2 = m_matchedRules[i2]; | |
| 734 } | |
| 735 } | |
| 736 | |
| 737 for (unsigned i = start; i < end; i++) | |
| 738 m_matchedRules[i] = rulesMergeBuffer[i - start]; | |
| 739 } | |
| 740 | |
| 741 void CSSStyleSelector::initElementAndPseudoState(Element* e) | |
| 742 { | |
| 743 m_element = e; | |
| 744 if (m_element && m_element->isStyledElement()) | |
| 745 m_styledElement = static_cast<StyledElement*>(m_element); | |
| 746 else | |
| 747 m_styledElement = 0; | |
| 748 pseudoState = PseudoUnknown; | |
| 749 } | |
| 750 | |
| 751 void CSSStyleSelector::initForStyleResolve(Element* e, RenderStyle* parentStyle,
RenderStyle::PseudoId pseudoID) | |
| 752 { | |
| 753 m_checker.m_pseudoStyle = pseudoID; | |
| 754 | |
| 755 m_parentNode = e ? e->parentNode() : 0; | |
| 756 | |
| 757 #if ENABLE(SVG) | |
| 758 if (!m_parentNode && e && e->isSVGElement() && e->isShadowNode()) | |
| 759 m_parentNode = e->shadowParentNode(); | |
| 760 #endif | |
| 761 | |
| 762 if (parentStyle) | |
| 763 m_parentStyle = parentStyle; | |
| 764 else | |
| 765 m_parentStyle = m_parentNode ? m_parentNode->renderStyle() : 0; | |
| 766 | |
| 767 m_style = 0; | |
| 768 | |
| 769 m_matchedDecls.clear(); | |
| 770 | |
| 771 m_ruleList = 0; | |
| 772 | |
| 773 m_fontDirty = false; | |
| 774 } | |
| 775 | |
| 776 static inline const AtomicString* linkAttribute(Node* node) | |
| 777 { | |
| 778 if (!node->isLink()) | |
| 779 return 0; | |
| 780 | |
| 781 ASSERT(node->isElementNode()); | |
| 782 Element* element = static_cast<Element*>(node); | |
| 783 if (element->isHTMLElement()) | |
| 784 return &element->getAttribute(hrefAttr); | |
| 785 #if ENABLE(SVG) | |
| 786 if (element->isSVGElement()) | |
| 787 return &element->getAttribute(XLinkNames::hrefAttr); | |
| 788 #endif | |
| 789 return 0; | |
| 790 } | |
| 791 | |
| 792 CSSStyleSelector::SelectorChecker::SelectorChecker(Document* document, bool stri
ctParsing, bool collectRulesOnly) | |
| 793 : m_document(document) | |
| 794 , m_strictParsing(strictParsing) | |
| 795 , m_collectRulesOnly(collectRulesOnly) | |
| 796 , m_pseudoStyle(RenderStyle::NOPSEUDO) | |
| 797 , m_documentIsHTML(document->isHTMLDocument()) | |
| 798 { | |
| 799 } | |
| 800 | |
| 801 PseudoState CSSStyleSelector::SelectorChecker::checkPseudoState(Element* element
, bool checkVisited) const | |
| 802 { | |
| 803 const AtomicString* attr = linkAttribute(element); | |
| 804 if (!attr || attr->isNull()) | |
| 805 return PseudoNone; | |
| 806 | |
| 807 if (!checkVisited) | |
| 808 return PseudoAnyLink; | |
| 809 | |
| 810 unsigned hash = m_document->visitedLinkHash(*attr); | |
| 811 if (!hash) | |
| 812 return PseudoLink; | |
| 813 | |
| 814 Frame* frame = m_document->frame(); | |
| 815 if (!frame) | |
| 816 return PseudoLink; | |
| 817 | |
| 818 Page* page = frame->page(); | |
| 819 if (!page) | |
| 820 return PseudoLink; | |
| 821 | |
| 822 m_linksCheckedForVisitedState.add(hash); | |
| 823 return page->group().isLinkVisited(hash) ? PseudoVisited : PseudoLink; | |
| 824 } | |
| 825 | |
| 826 bool CSSStyleSelector::SelectorChecker::checkSelector(CSSSelector* sel, Element*
element) const | |
| 827 { | |
| 828 pseudoState = PseudoUnknown; | |
| 829 RenderStyle::PseudoId dynamicPseudo = RenderStyle::NOPSEUDO; | |
| 830 | |
| 831 return checkSelector(sel, element, 0, dynamicPseudo, true, false) == Selecto
rMatches; | |
| 832 } | |
| 833 | |
| 834 // a helper function for parsing nth-arguments | |
| 835 static bool parseNth(const String& nth, int &a, int &b) | |
| 836 { | |
| 837 if (nth.isEmpty()) | |
| 838 return false; | |
| 839 a = 0; | |
| 840 b = 0; | |
| 841 if (nth == "odd") { | |
| 842 a = 2; | |
| 843 b = 1; | |
| 844 } else if (nth == "even") { | |
| 845 a = 2; | |
| 846 b = 0; | |
| 847 } else { | |
| 848 int n = nth.find('n'); | |
| 849 if (n != -1) { | |
| 850 if (nth[0] == '-') { | |
| 851 if (n == 1) | |
| 852 a = -1; // -n == -1n | |
| 853 else | |
| 854 a = nth.substring(0, n).toInt(); | |
| 855 } else if (!n) | |
| 856 a = 1; // n == 1n | |
| 857 else | |
| 858 a = nth.substring(0, n).toInt(); | |
| 859 | |
| 860 int p = nth.find('+', n); | |
| 861 if (p != -1) | |
| 862 b = nth.substring(p + 1, nth.length() - p - 1).toInt(); | |
| 863 else { | |
| 864 p = nth.find('-', n); | |
| 865 b = -nth.substring(p + 1, nth.length() - p - 1).toInt(); | |
| 866 } | |
| 867 } else | |
| 868 b = nth.toInt(); | |
| 869 } | |
| 870 return true; | |
| 871 } | |
| 872 | |
| 873 // a helper function for checking nth-arguments | |
| 874 static bool matchNth(int count, int a, int b) | |
| 875 { | |
| 876 if (!a) | |
| 877 return count == b; | |
| 878 else if (a > 0) { | |
| 879 if (count < b) | |
| 880 return false; | |
| 881 return (count - b) % a == 0; | |
| 882 } else { | |
| 883 if (count > b) | |
| 884 return false; | |
| 885 return (b - count) % (-a) == 0; | |
| 886 } | |
| 887 } | |
| 888 | |
| 889 | |
| 890 #ifdef STYLE_SHARING_STATS | |
| 891 static int fraction = 0; | |
| 892 static int total = 0; | |
| 893 #endif | |
| 894 | |
| 895 static const unsigned cStyleSearchThreshold = 10; | |
| 896 | |
| 897 Node* CSSStyleSelector::locateCousinList(Element* parent, unsigned depth) | |
| 898 { | |
| 899 if (parent && parent->isStyledElement()) { | |
| 900 StyledElement* p = static_cast<StyledElement*>(parent); | |
| 901 if (!p->inlineStyleDecl() && !p->hasID()) { | |
| 902 Node* r = p->previousSibling(); | |
| 903 unsigned subcount = 0; | |
| 904 RenderStyle* st = p->renderStyle(); | |
| 905 while (r) { | |
| 906 if (r->renderStyle() == st) | |
| 907 return r->lastChild(); | |
| 908 if (subcount++ == cStyleSearchThreshold) | |
| 909 return 0; | |
| 910 r = r->previousSibling(); | |
| 911 } | |
| 912 if (!r && depth < cStyleSearchThreshold) | |
| 913 r = locateCousinList(static_cast<Element*>(parent->parentNode())
, depth + 1); | |
| 914 while (r) { | |
| 915 if (r->renderStyle() == st) | |
| 916 return r->lastChild(); | |
| 917 if (subcount++ == cStyleSearchThreshold) | |
| 918 return 0; | |
| 919 r = r->previousSibling(); | |
| 920 } | |
| 921 } | |
| 922 } | |
| 923 return 0; | |
| 924 } | |
| 925 | |
| 926 bool CSSStyleSelector::canShareStyleWithElement(Node* n) | |
| 927 { | |
| 928 if (n->isStyledElement()) { | |
| 929 StyledElement* s = static_cast<StyledElement*>(n); | |
| 930 RenderStyle* style = s->renderStyle(); | |
| 931 if (style && !style->unique() && | |
| 932 (s->tagQName() == m_element->tagQName()) && !s->hasID() && | |
| 933 (s->hasClass() == m_element->hasClass()) && !s->inlineStyleDecl() && | |
| 934 (s->hasMappedAttributes() == m_styledElement->hasMappedAttributes())
&& | |
| 935 (s->isLink() == m_element->isLink()) && | |
| 936 !style->affectedByAttributeSelectors() && | |
| 937 (s->hovered() == m_element->hovered()) && | |
| 938 (s->active() == m_element->active()) && | |
| 939 (s->focused() == m_element->focused()) && | |
| 940 (s != s->document()->getCSSTarget() && m_element != m_element->docum
ent()->getCSSTarget()) && | |
| 941 (s->getAttribute(typeAttr) == m_element->getAttribute(typeAttr)) && | |
| 942 (s->getAttribute(XMLNames::langAttr) == m_element->getAttribute(XMLN
ames::langAttr)) && | |
| 943 (s->getAttribute(langAttr) == m_element->getAttribute(langAttr)) && | |
| 944 (s->getAttribute(readonlyAttr) == m_element->getAttribute(readonlyAt
tr)) && | |
| 945 (s->getAttribute(cellpaddingAttr) == m_element->getAttribute(cellpad
dingAttr))) { | |
| 946 bool isControl = s->isControl(); | |
| 947 if (isControl != m_element->isControl()) | |
| 948 return false; | |
| 949 if (isControl && (s->isEnabled() != m_element->isEnabled()) || | |
| 950 (s->isIndeterminate() != m_element->isIndeterminate
()) || | |
| 951 (s->isChecked() != m_element->isChecked())) | |
| 952 return false; | |
| 953 | |
| 954 if (style->transitions() || style->animations()) | |
| 955 return false; | |
| 956 | |
| 957 bool classesMatch = true; | |
| 958 if (s->hasClass()) { | |
| 959 const AtomicString& class1 = m_element->getAttribute(classAttr); | |
| 960 const AtomicString& class2 = s->getAttribute(classAttr); | |
| 961 classesMatch = (class1 == class2); | |
| 962 } | |
| 963 | |
| 964 if (classesMatch) { | |
| 965 bool mappedAttrsMatch = true; | |
| 966 if (s->hasMappedAttributes()) | |
| 967 mappedAttrsMatch = s->mappedAttributes()->mapsEquivalent(m_s
tyledElement->mappedAttributes()); | |
| 968 if (mappedAttrsMatch) { | |
| 969 bool linksMatch = true; | |
| 970 | |
| 971 if (s->isLink()) { | |
| 972 // We need to check to see if the visited state matches. | |
| 973 if (pseudoState == PseudoUnknown) { | |
| 974 const Color& linkColor = m_element->document()->link
Color(); | |
| 975 const Color& visitedColor = m_element->document()->v
isitedLinkColor(); | |
| 976 pseudoState = m_checker.checkPseudoState(m_element,
style->pseudoState() != PseudoAnyLink || linkColor != visitedColor); | |
| 977 } | |
| 978 linksMatch = (pseudoState == style->pseudoState()); | |
| 979 } | |
| 980 | |
| 981 if (linksMatch) | |
| 982 return true; | |
| 983 } | |
| 984 } | |
| 985 } | |
| 986 } | |
| 987 return false; | |
| 988 } | |
| 989 | |
| 990 RenderStyle* CSSStyleSelector::locateSharedStyle() | |
| 991 { | |
| 992 if (m_styledElement && !m_styledElement->inlineStyleDecl() && !m_styledEleme
nt->hasID() && !m_styledElement->document()->usesSiblingRules()) { | |
| 993 // Check previous siblings. | |
| 994 unsigned count = 0; | |
| 995 Node* n; | |
| 996 for (n = m_element->previousSibling(); n && !n->isElementNode(); n = n->
previousSibling()) { } | |
| 997 while (n) { | |
| 998 if (canShareStyleWithElement(n)) | |
| 999 return n->renderStyle(); | |
| 1000 if (count++ == cStyleSearchThreshold) | |
| 1001 return 0; | |
| 1002 for (n = n->previousSibling(); n && !n->isElementNode(); n = n->prev
iousSibling()) { } | |
| 1003 } | |
| 1004 if (!n) | |
| 1005 n = locateCousinList(static_cast<Element*>(m_element->parentNode()))
; | |
| 1006 while (n) { | |
| 1007 if (canShareStyleWithElement(n)) | |
| 1008 return n->renderStyle(); | |
| 1009 if (count++ == cStyleSearchThreshold) | |
| 1010 return 0; | |
| 1011 for (n = n->previousSibling(); n && !n->isElementNode(); n = n->prev
iousSibling()) { } | |
| 1012 } | |
| 1013 } | |
| 1014 return 0; | |
| 1015 } | |
| 1016 | |
| 1017 void CSSStyleSelector::matchUARules(int& firstUARule, int& lastUARule) | |
| 1018 { | |
| 1019 // First we match rules from the user agent sheet. | |
| 1020 CSSRuleSet* userAgentStyleSheet = m_medium->mediaTypeMatchSpecific("print") | |
| 1021 ? defaultPrintStyle : defaultStyle; | |
| 1022 matchRules(userAgentStyleSheet, firstUARule, lastUARule); | |
| 1023 | |
| 1024 // In quirks mode, we match rules from the quirks user agent sheet. | |
| 1025 if (!m_checker.m_strictParsing) | |
| 1026 matchRules(defaultQuirksStyle, firstUARule, lastUARule); | |
| 1027 | |
| 1028 // If we're in view source mode, then we match rules from the view source st
yle sheet. | |
| 1029 if (m_checker.m_document->frame() && m_checker.m_document->frame()->inViewSo
urceMode()) | |
| 1030 matchRules(defaultViewSourceStyle, firstUARule, lastUARule); | |
| 1031 } | |
| 1032 | |
| 1033 // If resolveForRootDefault is true, style based on user agent style sheet only.
This is used in media queries, where | |
| 1034 // relative units are interpreted according to document root element style, styl
ed only with UA stylesheet | |
| 1035 | |
| 1036 RenderStyle* CSSStyleSelector::styleForElement(Element* e, RenderStyle* defaultP
arent, bool allowSharing, bool resolveForRootDefault) | |
| 1037 { | |
| 1038 // Once an element has a renderer, we don't try to destroy it, since otherwi
se the renderer | |
| 1039 // will vanish if a style recalc happens during loading. | |
| 1040 if (allowSharing && !e->document()->haveStylesheetsLoaded() && !e->renderer(
)) { | |
| 1041 if (!s_styleNotYetAvailable) { | |
| 1042 s_styleNotYetAvailable = ::new RenderStyle; | |
| 1043 s_styleNotYetAvailable->ref(); | |
| 1044 s_styleNotYetAvailable->setDisplay(NONE); | |
| 1045 s_styleNotYetAvailable->font().update(m_fontSelector); | |
| 1046 } | |
| 1047 s_styleNotYetAvailable->ref(); | |
| 1048 e->document()->setHasNodesWithPlaceholderStyle(); | |
| 1049 return s_styleNotYetAvailable; | |
| 1050 } | |
| 1051 | |
| 1052 initElementAndPseudoState(e); | |
| 1053 if (allowSharing) { | |
| 1054 m_style = locateSharedStyle(); | |
| 1055 #ifdef STYLE_SHARING_STATS | |
| 1056 fraction += m_style != 0; | |
| 1057 total++; | |
| 1058 printf("Sharing %d out of %d\n", fraction, total); | |
| 1059 #endif | |
| 1060 if (m_style) { | |
| 1061 m_style->ref(); | |
| 1062 return m_style; | |
| 1063 } | |
| 1064 } | |
| 1065 initForStyleResolve(e, defaultParent); | |
| 1066 | |
| 1067 if (resolveForRootDefault) { | |
| 1068 m_style = ::new RenderStyle(); | |
| 1069 // don't ref, because we want to delete this, but we cannot unref it | |
| 1070 } else { | |
| 1071 m_style = new (e->document()->renderArena()) RenderStyle(); | |
| 1072 m_style->ref(); | |
| 1073 } | |
| 1074 if (m_parentStyle) | |
| 1075 m_style->inheritFrom(m_parentStyle); | |
| 1076 else | |
| 1077 m_parentStyle = m_style; | |
| 1078 | |
| 1079 #if ENABLE(SVG) | |
| 1080 static bool loadedSVGUserAgentSheet; | |
| 1081 if (e->isSVGElement() && !loadedSVGUserAgentSheet) { | |
| 1082 // SVG rules. | |
| 1083 loadedSVGUserAgentSheet = true; | |
| 1084 CSSStyleSheet* svgSheet = parseUASheet(svgUserAgentStyleSheet, sizeof(sv
gUserAgentStyleSheet)); | |
| 1085 defaultStyle->addRulesFromSheet(svgSheet, screenEval()); | |
| 1086 defaultPrintStyle->addRulesFromSheet(svgSheet, printEval()); | |
| 1087 } | |
| 1088 #endif | |
| 1089 | |
| 1090 int firstUARule = -1, lastUARule = -1; | |
| 1091 int firstUserRule = -1, lastUserRule = -1; | |
| 1092 int firstAuthorRule = -1, lastAuthorRule = -1; | |
| 1093 matchUARules(firstUARule, lastUARule); | |
| 1094 | |
| 1095 if (!resolveForRootDefault) { | |
| 1096 // 4. Now we check user sheet rules. | |
| 1097 if (m_matchAuthorAndUserStyles) | |
| 1098 matchRules(m_userStyle, firstUserRule, lastUserRule); | |
| 1099 | |
| 1100 // 5. Now check author rules, beginning first with presentational attrib
utes | |
| 1101 // mapped from HTML. | |
| 1102 if (m_styledElement) { | |
| 1103 // Ask if the HTML element has mapped attributes. | |
| 1104 if (m_styledElement->hasMappedAttributes()) { | |
| 1105 // Walk our attribute list and add in each decl. | |
| 1106 const NamedMappedAttrMap* map = m_styledElement->mappedAttribute
s(); | |
| 1107 for (unsigned i = 0; i < map->length(); i++) { | |
| 1108 MappedAttribute* attr = map->attributeItem(i); | |
| 1109 if (attr->decl()) { | |
| 1110 lastAuthorRule = m_matchedDecls.size(); | |
| 1111 if (firstAuthorRule == -1) | |
| 1112 firstAuthorRule = lastAuthorRule; | |
| 1113 addMatchedDeclaration(attr->decl()); | |
| 1114 } | |
| 1115 } | |
| 1116 } | |
| 1117 | |
| 1118 // Now we check additional mapped declarations. | |
| 1119 // Tables and table cells share an additional mapped rule that must
be applied | |
| 1120 // after all attributes, since their mapped style depends on the val
ues of multiple attributes. | |
| 1121 if (m_styledElement->canHaveAdditionalAttributeStyleDecls()) { | |
| 1122 m_additionalAttributeStyleDecls.clear(); | |
| 1123 m_styledElement->additionalAttributeStyleDecls(m_additionalAttri
buteStyleDecls); | |
| 1124 if (!m_additionalAttributeStyleDecls.isEmpty()) { | |
| 1125 unsigned additionalDeclsSize = m_additionalAttributeStyleDec
ls.size(); | |
| 1126 if (firstAuthorRule == -1) | |
| 1127 firstAuthorRule = m_matchedDecls.size(); | |
| 1128 lastAuthorRule = m_matchedDecls.size() + additionalDeclsSize
- 1; | |
| 1129 for (unsigned i = 0; i < additionalDeclsSize; i++) | |
| 1130 addMatchedDeclaration(m_additionalAttributeStyleDecls[i]
); | |
| 1131 } | |
| 1132 } | |
| 1133 } | |
| 1134 | |
| 1135 // 6. Check the rules in author sheets next. | |
| 1136 if (m_matchAuthorAndUserStyles) | |
| 1137 matchRules(m_authorStyle, firstAuthorRule, lastAuthorRule); | |
| 1138 | |
| 1139 // 7. Now check our inline style attribute. | |
| 1140 if (m_matchAuthorAndUserStyles && m_styledElement) { | |
| 1141 CSSMutableStyleDeclaration* inlineDecl = m_styledElement->inlineStyl
eDecl(); | |
| 1142 if (inlineDecl) { | |
| 1143 lastAuthorRule = m_matchedDecls.size(); | |
| 1144 if (firstAuthorRule == -1) | |
| 1145 firstAuthorRule = lastAuthorRule; | |
| 1146 addMatchedDeclaration(inlineDecl); | |
| 1147 } | |
| 1148 } | |
| 1149 } | |
| 1150 | |
| 1151 // Now we have all of the matched rules in the appropriate order. Walk the
rules and apply | |
| 1152 // high-priority properties first, i.e., those properties that other propert
ies depend on. | |
| 1153 // The order is (1) high-priority not important, (2) high-priority important
, (3) normal not important | |
| 1154 // and (4) normal important. | |
| 1155 m_lineHeightValue = 0; | |
| 1156 applyDeclarations(true, false, 0, m_matchedDecls.size() - 1); | |
| 1157 if (!resolveForRootDefault) { | |
| 1158 applyDeclarations(true, true, firstAuthorRule, lastAuthorRule); | |
| 1159 applyDeclarations(true, true, firstUserRule, lastUserRule); | |
| 1160 } | |
| 1161 applyDeclarations(true, true, firstUARule, lastUARule); | |
| 1162 | |
| 1163 // If our font got dirtied, go ahead and update it now. | |
| 1164 if (m_fontDirty) | |
| 1165 updateFont(); | |
| 1166 | |
| 1167 // Line-height is set when we are sure we decided on the font-size | |
| 1168 if (m_lineHeightValue) | |
| 1169 applyProperty(CSSPropertyLineHeight, m_lineHeightValue); | |
| 1170 | |
| 1171 // Now do the normal priority UA properties. | |
| 1172 applyDeclarations(false, false, firstUARule, lastUARule); | |
| 1173 | |
| 1174 // Cache our border and background so that we can examine them later. | |
| 1175 cacheBorderAndBackground(); | |
| 1176 | |
| 1177 // Now do the author and user normal priority properties and all the !import
ant properties. | |
| 1178 if (!resolveForRootDefault) { | |
| 1179 applyDeclarations(false, false, lastUARule + 1, m_matchedDecls.size() -
1); | |
| 1180 applyDeclarations(false, true, firstAuthorRule, lastAuthorRule); | |
| 1181 applyDeclarations(false, true, firstUserRule, lastUserRule); | |
| 1182 } | |
| 1183 applyDeclarations(false, true, firstUARule, lastUARule); | |
| 1184 | |
| 1185 // If our font got dirtied by one of the non-essential font props, | |
| 1186 // go ahead and update it a second time. | |
| 1187 if (m_fontDirty) | |
| 1188 updateFont(); | |
| 1189 | |
| 1190 // Clean up our style object's display and text decorations (among other fix
ups). | |
| 1191 adjustRenderStyle(m_style, e); | |
| 1192 | |
| 1193 // If we are a link, cache the determined pseudo-state. | |
| 1194 if (e->isLink()) | |
| 1195 m_style->setPseudoState(pseudoState); | |
| 1196 | |
| 1197 // If we have first-letter pseudo style, do not share this style | |
| 1198 if (m_style->hasPseudoStyle(RenderStyle::FIRST_LETTER)) | |
| 1199 m_style->setUnique(); | |
| 1200 | |
| 1201 // Now return the style. | |
| 1202 return m_style; | |
| 1203 } | |
| 1204 | |
| 1205 RenderStyle* CSSStyleSelector::pseudoStyleForElement(RenderStyle::PseudoId pseud
o, Element* e, RenderStyle* parentStyle) | |
| 1206 { | |
| 1207 if (!e) | |
| 1208 return 0; | |
| 1209 | |
| 1210 initElementAndPseudoState(e); | |
| 1211 initForStyleResolve(e, parentStyle, pseudo); | |
| 1212 m_style = parentStyle; | |
| 1213 | |
| 1214 // Since we don't use pseudo-elements in any of our quirk/print user agent r
ules, don't waste time walking | |
| 1215 // those rules. | |
| 1216 | |
| 1217 // Check UA, user and author rules. | |
| 1218 int firstUARule = -1, lastUARule = -1, firstUserRule = -1, lastUserRule = -1
, firstAuthorRule = -1, lastAuthorRule = -1; | |
| 1219 matchUARules(firstUARule, lastUARule); | |
| 1220 | |
| 1221 if (m_matchAuthorAndUserStyles) { | |
| 1222 matchRules(m_userStyle, firstUserRule, lastUserRule); | |
| 1223 matchRules(m_authorStyle, firstAuthorRule, lastAuthorRule); | |
| 1224 } | |
| 1225 | |
| 1226 if (m_matchedDecls.isEmpty()) | |
| 1227 return 0; | |
| 1228 | |
| 1229 m_style = new (e->document()->renderArena()) RenderStyle(); | |
| 1230 m_style->ref(); | |
| 1231 if (parentStyle) | |
| 1232 m_style->inheritFrom(parentStyle); | |
| 1233 | |
| 1234 m_style->noninherited_flags._styleType = pseudo; | |
| 1235 | |
| 1236 m_lineHeightValue = 0; | |
| 1237 // High-priority properties. | |
| 1238 applyDeclarations(true, false, 0, m_matchedDecls.size() - 1); | |
| 1239 applyDeclarations(true, true, firstAuthorRule, lastAuthorRule); | |
| 1240 applyDeclarations(true, true, firstUserRule, lastUserRule); | |
| 1241 applyDeclarations(true, true, firstUARule, lastUARule); | |
| 1242 | |
| 1243 // If our font got dirtied, go ahead and update it now. | |
| 1244 if (m_fontDirty) | |
| 1245 updateFont(); | |
| 1246 | |
| 1247 // Line-height is set when we are sure we decided on the font-size | |
| 1248 if (m_lineHeightValue) | |
| 1249 applyProperty(CSSPropertyLineHeight, m_lineHeightValue); | |
| 1250 | |
| 1251 // Now do the normal priority properties. | |
| 1252 applyDeclarations(false, false, firstUARule, lastUARule); | |
| 1253 | |
| 1254 // Cache our border and background so that we can examine them later. | |
| 1255 cacheBorderAndBackground(); | |
| 1256 | |
| 1257 applyDeclarations(false, false, lastUARule + 1, m_matchedDecls.size() - 1); | |
| 1258 applyDeclarations(false, true, firstAuthorRule, lastAuthorRule); | |
| 1259 applyDeclarations(false, true, firstUserRule, lastUserRule); | |
| 1260 applyDeclarations(false, true, firstUARule, lastUARule); | |
| 1261 | |
| 1262 // If our font got dirtied by one of the non-essential font props, | |
| 1263 // go ahead and update it a second time. | |
| 1264 if (m_fontDirty) | |
| 1265 updateFont(); | |
| 1266 // Clean up our style object's display and text decorations (among other fix
ups). | |
| 1267 adjustRenderStyle(m_style, 0); | |
| 1268 | |
| 1269 // Now return the style. | |
| 1270 return m_style; | |
| 1271 } | |
| 1272 | |
| 1273 static void addIntrinsicMargins(RenderStyle* style) | |
| 1274 { | |
| 1275 // Intrinsic margin value. | |
| 1276 const int intrinsicMargin = 2 * style->effectiveZoom(); | |
| 1277 | |
| 1278 // FIXME: Using width/height alone and not also dealing with min-width/max-w
idth is flawed. | |
| 1279 // FIXME: Using "quirk" to decide the margin wasn't set is kind of lame. | |
| 1280 if (style->width().isIntrinsicOrAuto()) { | |
| 1281 if (style->marginLeft().quirk()) | |
| 1282 style->setMarginLeft(Length(intrinsicMargin, Fixed)); | |
| 1283 if (style->marginRight().quirk()) | |
| 1284 style->setMarginRight(Length(intrinsicMargin, Fixed)); | |
| 1285 } | |
| 1286 | |
| 1287 if (style->height().isAuto()) { | |
| 1288 if (style->marginTop().quirk()) | |
| 1289 style->setMarginTop(Length(intrinsicMargin, Fixed)); | |
| 1290 if (style->marginBottom().quirk()) | |
| 1291 style->setMarginBottom(Length(intrinsicMargin, Fixed)); | |
| 1292 } | |
| 1293 } | |
| 1294 | |
| 1295 void CSSStyleSelector::adjustRenderStyle(RenderStyle* style, Element *e) | |
| 1296 { | |
| 1297 // Cache our original display. | |
| 1298 style->setOriginalDisplay(style->display()); | |
| 1299 | |
| 1300 if (style->display() != NONE) { | |
| 1301 // If we have a <td> that specifies a float property, in quirks mode we
just drop the float | |
| 1302 // property. | |
| 1303 // Sites also commonly use display:inline/block on <td>s and <table>s.
In quirks mode we force | |
| 1304 // these tags to retain their display types. | |
| 1305 if (!m_checker.m_strictParsing && e) { | |
| 1306 if (e->hasTagName(tdTag)) { | |
| 1307 style->setDisplay(TABLE_CELL); | |
| 1308 style->setFloating(FNONE); | |
| 1309 } | |
| 1310 else if (e->hasTagName(tableTag)) | |
| 1311 style->setDisplay(style->isDisplayInlineType() ? INLINE_TABLE :
TABLE); | |
| 1312 } | |
| 1313 | |
| 1314 // Tables never support the -webkit-* values for text-align and will res
et back to the default. | |
| 1315 if (e && e->hasTagName(tableTag) && (style->textAlign() == WEBKIT_LEFT |
| style->textAlign() == WEBKIT_CENTER || style->textAlign() == WEBKIT_RIGHT)) | |
| 1316 style->setTextAlign(TAAUTO); | |
| 1317 | |
| 1318 // Frames and framesets never honor position:relative or position:absolu
te. This is necessary to | |
| 1319 // fix a crash where a site tries to position these objects. They also
never honor display. | |
| 1320 if (e && (e->hasTagName(frameTag) || e->hasTagName(framesetTag))) { | |
| 1321 style->setPosition(StaticPosition); | |
| 1322 style->setDisplay(BLOCK); | |
| 1323 } | |
| 1324 | |
| 1325 // Table headers with a text-align of auto will change the text-align to
center. | |
| 1326 if (e && e->hasTagName(thTag) && style->textAlign() == TAAUTO) | |
| 1327 style->setTextAlign(CENTER); | |
| 1328 | |
| 1329 // Mutate the display to BLOCK or TABLE for certain cases, e.g., if some
one attempts to | |
| 1330 // position or float an inline, compact, or run-in. Cache the original
display, since it | |
| 1331 // may be needed for positioned elements that have to compute their stat
ic normal flow | |
| 1332 // positions. We also force inline-level roots to be block-level. | |
| 1333 if (style->display() != BLOCK && style->display() != TABLE && style->dis
play() != BOX && | |
| 1334 (style->position() == AbsolutePosition || style->position() == Fixed
Position || style->floating() != FNONE || | |
| 1335 (e && e->document()->documentElement() == e))) { | |
| 1336 if (style->display() == INLINE_TABLE) | |
| 1337 style->setDisplay(TABLE); | |
| 1338 else if (style->display() == INLINE_BOX) | |
| 1339 style->setDisplay(BOX); | |
| 1340 else if (style->display() == LIST_ITEM) { | |
| 1341 // It is a WinIE bug that floated list items lose their bullets,
so we'll emulate the quirk, | |
| 1342 // but only in quirks mode. | |
| 1343 if (!m_checker.m_strictParsing && style->floating() != FNONE) | |
| 1344 style->setDisplay(BLOCK); | |
| 1345 } | |
| 1346 else | |
| 1347 style->setDisplay(BLOCK); | |
| 1348 } | |
| 1349 | |
| 1350 // After performing the display mutation, check table rows. We do not h
onor position:relative on | |
| 1351 // table rows or cells. This has been established in CSS2.1 (and caused
a crash in containingBlock() | |
| 1352 // on some sites). | |
| 1353 if ((style->display() == TABLE_HEADER_GROUP || style->display() == TABLE
_ROW_GROUP || | |
| 1354 style->display() == TABLE_FOOTER_GROUP || style->display() == TABLE
_ROW || style->display() == TABLE_CELL) && | |
| 1355 style->position() == RelativePosition) | |
| 1356 style->setPosition(StaticPosition); | |
| 1357 } | |
| 1358 | |
| 1359 // Make sure our z-index value is only applied if the object is positioned. | |
| 1360 if (style->position() == StaticPosition) | |
| 1361 style->setHasAutoZIndex(); | |
| 1362 | |
| 1363 // Auto z-index becomes 0 for the root element and transparent objects. Thi
s prevents | |
| 1364 // cases where objects that should be blended as a single unit end up with a
non-transparent | |
| 1365 // object wedged in between them. Auto z-index also becomes 0 for objects t
hat specify transforms/masks/reflections. | |
| 1366 if (style->hasAutoZIndex() && ((e && e->document()->documentElement() == e)
|| style->opacity() < 1.0f || | |
| 1367 style->hasTransform() || style->hasMask() || style->boxReflect())) | |
| 1368 style->setZIndex(0); | |
| 1369 | |
| 1370 // Button, legend, input, select and textarea all consider width values of '
auto' to be 'intrinsic'. | |
| 1371 // This will be important when we use block flows for all form controls. | |
| 1372 if (e && (e->hasTagName(legendTag) || e->hasTagName(buttonTag) || e->hasTagN
ame(inputTag) || | |
| 1373 e->hasTagName(selectTag) || e->hasTagName(textareaTag))) { | |
| 1374 if (style->width().isAuto()) | |
| 1375 style->setWidth(Length(Intrinsic)); | |
| 1376 } | |
| 1377 | |
| 1378 // Textarea considers overflow visible as auto. | |
| 1379 if (e && e->hasTagName(textareaTag)) { | |
| 1380 style->setOverflowX(style->overflowX() == OVISIBLE ? OAUTO : style->over
flowX()); | |
| 1381 style->setOverflowY(style->overflowY() == OVISIBLE ? OAUTO : style->over
flowY()); | |
| 1382 } | |
| 1383 | |
| 1384 // Finally update our text decorations in effect, but don't allow text-decor
ation to percolate through | |
| 1385 // tables, inline blocks, inline tables, or run-ins. | |
| 1386 if (style->display() == TABLE || style->display() == INLINE_TABLE || style->
display() == RUN_IN | |
| 1387 || style->display() == INLINE_BLOCK || style->display() == INLINE_BOX) | |
| 1388 style->setTextDecorationsInEffect(style->textDecoration()); | |
| 1389 else | |
| 1390 style->addToTextDecorationsInEffect(style->textDecoration()); | |
| 1391 | |
| 1392 // If either overflow value is not visible, change to auto. | |
| 1393 if (style->overflowX() == OMARQUEE && style->overflowY() != OMARQUEE) | |
| 1394 style->setOverflowY(OMARQUEE); | |
| 1395 else if (style->overflowY() == OMARQUEE && style->overflowX() != OMARQUEE) | |
| 1396 style->setOverflowX(OMARQUEE); | |
| 1397 else if (style->overflowX() == OVISIBLE && style->overflowY() != OVISIBLE) | |
| 1398 style->setOverflowX(OAUTO); | |
| 1399 else if (style->overflowY() == OVISIBLE && style->overflowX() != OVISIBLE) | |
| 1400 style->setOverflowY(OAUTO); | |
| 1401 | |
| 1402 // Table rows, sections and the table itself will support overflow:hidden an
d will ignore scroll/auto. | |
| 1403 // FIXME: Eventually table sections will support auto and scroll. | |
| 1404 if (style->display() == TABLE || style->display() == INLINE_TABLE || | |
| 1405 style->display() == TABLE_ROW_GROUP || style->display() == TABLE_ROW) { | |
| 1406 if (style->overflowX() != OVISIBLE && style->overflowX() != OHIDDEN) | |
| 1407 style->setOverflowX(OVISIBLE); | |
| 1408 if (style->overflowY() != OVISIBLE && style->overflowY() != OHIDDEN) | |
| 1409 style->setOverflowY(OVISIBLE); | |
| 1410 } | |
| 1411 | |
| 1412 // Cull out any useless layers and also repeat patterns into additional laye
rs. | |
| 1413 style->adjustBackgroundLayers(); | |
| 1414 style->adjustMaskLayers(); | |
| 1415 | |
| 1416 // Do the same for animations and transitions. | |
| 1417 style->adjustAnimations(); | |
| 1418 style->adjustTransitions(); | |
| 1419 | |
| 1420 // Important: Intrinsic margins get added to controls before the theme has a
djusted the style, since the theme will | |
| 1421 // alter fonts and heights/widths. | |
| 1422 if (e && e->isControl() && style->fontSize() >= 11) { | |
| 1423 // Don't apply intrinsic margins to image buttons. The designer knows h
ow big the images are, | |
| 1424 // so we have to treat all image buttons as though they were explicitly
sized. | |
| 1425 if (!e->hasTagName(inputTag) || static_cast<HTMLInputElement*>(e)->input
Type() != HTMLInputElement::IMAGE) | |
| 1426 addIntrinsicMargins(style); | |
| 1427 } | |
| 1428 | |
| 1429 // Let the theme also have a crack at adjusting the style. | |
| 1430 if (style->hasAppearance()) | |
| 1431 theme()->adjustStyle(this, style, e, m_hasUAAppearance, m_borderData, m_
backgroundData, m_backgroundColor); | |
| 1432 | |
| 1433 #if ENABLE(SVG) | |
| 1434 if (e && e->isSVGElement()) { | |
| 1435 // Spec: http://www.w3.org/TR/SVG/masking.html#OverflowProperty | |
| 1436 if (style->overflowY() == OSCROLL) | |
| 1437 style->setOverflowY(OHIDDEN); | |
| 1438 else if (style->overflowY() == OAUTO) | |
| 1439 style->setOverflowY(OVISIBLE); | |
| 1440 | |
| 1441 if (style->overflowX() == OSCROLL) | |
| 1442 style->setOverflowX(OHIDDEN); | |
| 1443 else if (style->overflowX() == OAUTO) | |
| 1444 style->setOverflowX(OVISIBLE); | |
| 1445 | |
| 1446 // Only the root <svg> element in an SVG document fragment tree honors c
ss position | |
| 1447 if (!(e->hasTagName(SVGNames::svgTag) && e->parentNode() && !e->parentNo
de()->isSVGElement())) | |
| 1448 style->setPosition(RenderStyle::initialPosition()); | |
| 1449 } | |
| 1450 #endif | |
| 1451 } | |
| 1452 | |
| 1453 void CSSStyleSelector::updateFont() | |
| 1454 { | |
| 1455 checkForTextSizeAdjust(); | |
| 1456 checkForGenericFamilyChange(m_style, m_parentStyle); | |
| 1457 checkForZoomChange(m_style, m_parentStyle); | |
| 1458 m_style->font().update(m_fontSelector); | |
| 1459 m_fontDirty = false; | |
| 1460 } | |
| 1461 | |
| 1462 void CSSStyleSelector::cacheBorderAndBackground() | |
| 1463 { | |
| 1464 m_hasUAAppearance = m_style->hasAppearance(); | |
| 1465 if (m_hasUAAppearance) { | |
| 1466 m_borderData = m_style->border(); | |
| 1467 m_backgroundData = *m_style->backgroundLayers(); | |
| 1468 m_backgroundColor = m_style->backgroundColor(); | |
| 1469 } | |
| 1470 } | |
| 1471 | |
| 1472 PassRefPtr<CSSRuleList> CSSStyleSelector::styleRulesForElement(Element* e, bool
authorOnly) | |
| 1473 { | |
| 1474 if (!e || !e->document()->haveStylesheetsLoaded()) | |
| 1475 return 0; | |
| 1476 | |
| 1477 m_checker.m_collectRulesOnly = true; | |
| 1478 | |
| 1479 initElementAndPseudoState(e); | |
| 1480 initForStyleResolve(e); | |
| 1481 | |
| 1482 if (!authorOnly) { | |
| 1483 int firstUARule = -1, lastUARule = -1; | |
| 1484 // First we match rules from the user agent sheet. | |
| 1485 matchUARules(firstUARule, lastUARule); | |
| 1486 | |
| 1487 // Now we check user sheet rules. | |
| 1488 if (m_matchAuthorAndUserStyles) { | |
| 1489 int firstUserRule = -1, lastUserRule = -1; | |
| 1490 matchRules(m_userStyle, firstUserRule, lastUserRule); | |
| 1491 } | |
| 1492 } | |
| 1493 | |
| 1494 if (m_matchAuthorAndUserStyles) { | |
| 1495 // Check the rules in author sheets. | |
| 1496 int firstAuthorRule = -1, lastAuthorRule = -1; | |
| 1497 matchRules(m_authorStyle, firstAuthorRule, lastAuthorRule); | |
| 1498 } | |
| 1499 | |
| 1500 m_checker.m_collectRulesOnly = false; | |
| 1501 | |
| 1502 return m_ruleList.release(); | |
| 1503 } | |
| 1504 | |
| 1505 PassRefPtr<CSSRuleList> CSSStyleSelector::pseudoStyleRulesForElement(Element*, c
onst String& pseudoStyle, bool authorOnly) | |
| 1506 { | |
| 1507 // FIXME: Implement this. | |
| 1508 return 0; | |
| 1509 } | |
| 1510 | |
| 1511 bool CSSStyleSelector::checkSelector(CSSSelector* sel) | |
| 1512 { | |
| 1513 m_dynamicPseudo = RenderStyle::NOPSEUDO; | |
| 1514 | |
| 1515 // Check the selector | |
| 1516 SelectorMatch match = m_checker.checkSelector(sel, m_element, &m_selectorAtt
rs, m_dynamicPseudo, true, false, m_style, m_parentStyle); | |
| 1517 if (match != SelectorMatches) | |
| 1518 return false; | |
| 1519 | |
| 1520 if (m_checker.m_pseudoStyle != RenderStyle::NOPSEUDO && m_checker.m_pseudoSt
yle != m_dynamicPseudo) | |
| 1521 return false; | |
| 1522 | |
| 1523 return true; | |
| 1524 } | |
| 1525 | |
| 1526 // Recursive check of selectors and combinators | |
| 1527 // It can return 3 different values: | |
| 1528 // * SelectorMatches - the selector matches the element e | |
| 1529 // * SelectorFailsLocally - the selector fails for the element e | |
| 1530 // * SelectorFailsCompletely - the selector fails for e and any sibling or ances
tor of e | |
| 1531 CSSStyleSelector::SelectorMatch CSSStyleSelector::SelectorChecker::checkSelector
(CSSSelector* sel, Element* e, HashSet<AtomicStringImpl*>* selectorAttrs, Render
Style::PseudoId& dynamicPseudo, bool isAncestor, bool isSubSelector, RenderStyle
* elementStyle, RenderStyle* elementParentStyle) const | |
| 1532 { | |
| 1533 #if ENABLE(SVG) | |
| 1534 // Spec: CSS2 selectors cannot be applied to the (conceptually) cloned DOM t
ree | |
| 1535 // because its contents are not part of the formal document structure. | |
| 1536 if (e->isSVGElement() && e->isShadowNode()) | |
| 1537 return SelectorFailsCompletely; | |
| 1538 #endif | |
| 1539 | |
| 1540 // first selector has to match | |
| 1541 if (!checkOneSelector(sel, e, selectorAttrs, dynamicPseudo, isAncestor, isSu
bSelector, elementStyle, elementParentStyle)) | |
| 1542 return SelectorFailsLocally; | |
| 1543 | |
| 1544 // The rest of the selectors has to match | |
| 1545 CSSSelector::Relation relation = sel->relation(); | |
| 1546 | |
| 1547 // Prepare next sel | |
| 1548 sel = sel->m_tagHistory; | |
| 1549 if (!sel) | |
| 1550 return SelectorMatches; | |
| 1551 | |
| 1552 if (relation != CSSSelector::SubSelector) | |
| 1553 // Bail-out if this selector is irrelevant for the pseudoStyle | |
| 1554 if (m_pseudoStyle != RenderStyle::NOPSEUDO && m_pseudoStyle != dynamicPs
eudo) | |
| 1555 return SelectorFailsCompletely; | |
| 1556 | |
| 1557 switch (relation) { | |
| 1558 case CSSSelector::Descendant: | |
| 1559 while (true) { | |
| 1560 Node* n = e->parentNode(); | |
| 1561 if (!n || !n->isElementNode()) | |
| 1562 return SelectorFailsCompletely; | |
| 1563 e = static_cast<Element*>(n); | |
| 1564 SelectorMatch match = checkSelector(sel, e, selectorAttrs, dynam
icPseudo, true, false); | |
| 1565 if (match != SelectorFailsLocally) | |
| 1566 return match; | |
| 1567 } | |
| 1568 break; | |
| 1569 case CSSSelector::Child: | |
| 1570 { | |
| 1571 Node* n = e->parentNode(); | |
| 1572 if (!n || !n->isElementNode()) | |
| 1573 return SelectorFailsCompletely; | |
| 1574 e = static_cast<Element*>(n); | |
| 1575 return checkSelector(sel, e, selectorAttrs, dynamicPseudo, true, fal
se); | |
| 1576 } | |
| 1577 case CSSSelector::DirectAdjacent: | |
| 1578 { | |
| 1579 if (!m_collectRulesOnly && e->parentNode() && e->parentNode()->isEle
mentNode()) { | |
| 1580 RenderStyle* parentStyle = elementStyle ? elementParentStyle : e
->parentNode()->renderStyle(); | |
| 1581 if (parentStyle) | |
| 1582 parentStyle->setChildrenAffectedByDirectAdjacentRules(); | |
| 1583 } | |
| 1584 Node* n = e->previousSibling(); | |
| 1585 while (n && !n->isElementNode()) | |
| 1586 n = n->previousSibling(); | |
| 1587 if (!n) | |
| 1588 return SelectorFailsLocally; | |
| 1589 e = static_cast<Element*>(n); | |
| 1590 return checkSelector(sel, e, selectorAttrs, dynamicPseudo, false, fa
lse); | |
| 1591 } | |
| 1592 case CSSSelector::IndirectAdjacent: | |
| 1593 if (!m_collectRulesOnly && e->parentNode() && e->parentNode()->isEle
mentNode()) { | |
| 1594 RenderStyle* parentStyle = elementStyle ? elementParentStyle : e
->parentNode()->renderStyle(); | |
| 1595 if (parentStyle) | |
| 1596 parentStyle->setChildrenAffectedByForwardPositionalRules(); | |
| 1597 } | |
| 1598 while (true) { | |
| 1599 Node* n = e->previousSibling(); | |
| 1600 while (n && !n->isElementNode()) | |
| 1601 n = n->previousSibling(); | |
| 1602 if (!n) | |
| 1603 return SelectorFailsLocally; | |
| 1604 e = static_cast<Element*>(n); | |
| 1605 SelectorMatch match = checkSelector(sel, e, selectorAttrs, dynam
icPseudo, false, false); | |
| 1606 if (match != SelectorFailsLocally) | |
| 1607 return match; | |
| 1608 }; | |
| 1609 break; | |
| 1610 case CSSSelector::SubSelector: | |
| 1611 // a selector is invalid if something follows a pseudo-element | |
| 1612 if (elementStyle && dynamicPseudo != RenderStyle::NOPSEUDO) | |
| 1613 return SelectorFailsCompletely; | |
| 1614 return checkSelector(sel, e, selectorAttrs, dynamicPseudo, isAncesto
r, true, elementStyle, elementParentStyle); | |
| 1615 } | |
| 1616 | |
| 1617 return SelectorFailsCompletely; | |
| 1618 } | |
| 1619 | |
| 1620 static void addLocalNameToSet(HashSet<AtomicStringImpl*>* set, const QualifiedNa
me& qName) | |
| 1621 { | |
| 1622 set->add(qName.localName().impl()); | |
| 1623 } | |
| 1624 | |
| 1625 static HashSet<AtomicStringImpl*>* createHtmlCaseInsensitiveAttributesSet() | |
| 1626 { | |
| 1627 // This is the list of attributes in HTML 4.01 with values marked as "[CI]"
or case-insensitive | |
| 1628 // Mozilla treats all other values as case-sensitive, thus so do we. | |
| 1629 HashSet<AtomicStringImpl*>* attrSet = new HashSet<AtomicStringImpl*>; | |
| 1630 | |
| 1631 addLocalNameToSet(attrSet, accept_charsetAttr); | |
| 1632 addLocalNameToSet(attrSet, acceptAttr); | |
| 1633 addLocalNameToSet(attrSet, alignAttr); | |
| 1634 addLocalNameToSet(attrSet, alinkAttr); | |
| 1635 addLocalNameToSet(attrSet, axisAttr); | |
| 1636 addLocalNameToSet(attrSet, bgcolorAttr); | |
| 1637 addLocalNameToSet(attrSet, charsetAttr); | |
| 1638 addLocalNameToSet(attrSet, checkedAttr); | |
| 1639 addLocalNameToSet(attrSet, clearAttr); | |
| 1640 addLocalNameToSet(attrSet, codetypeAttr); | |
| 1641 addLocalNameToSet(attrSet, colorAttr); | |
| 1642 addLocalNameToSet(attrSet, compactAttr); | |
| 1643 addLocalNameToSet(attrSet, declareAttr); | |
| 1644 addLocalNameToSet(attrSet, deferAttr); | |
| 1645 addLocalNameToSet(attrSet, dirAttr); | |
| 1646 addLocalNameToSet(attrSet, disabledAttr); | |
| 1647 addLocalNameToSet(attrSet, enctypeAttr); | |
| 1648 addLocalNameToSet(attrSet, faceAttr); | |
| 1649 addLocalNameToSet(attrSet, frameAttr); | |
| 1650 addLocalNameToSet(attrSet, hreflangAttr); | |
| 1651 addLocalNameToSet(attrSet, http_equivAttr); | |
| 1652 addLocalNameToSet(attrSet, langAttr); | |
| 1653 addLocalNameToSet(attrSet, languageAttr); | |
| 1654 addLocalNameToSet(attrSet, linkAttr); | |
| 1655 addLocalNameToSet(attrSet, mediaAttr); | |
| 1656 addLocalNameToSet(attrSet, methodAttr); | |
| 1657 addLocalNameToSet(attrSet, multipleAttr); | |
| 1658 addLocalNameToSet(attrSet, nohrefAttr); | |
| 1659 addLocalNameToSet(attrSet, noresizeAttr); | |
| 1660 addLocalNameToSet(attrSet, noshadeAttr); | |
| 1661 addLocalNameToSet(attrSet, nowrapAttr); | |
| 1662 addLocalNameToSet(attrSet, readonlyAttr); | |
| 1663 addLocalNameToSet(attrSet, relAttr); | |
| 1664 addLocalNameToSet(attrSet, revAttr); | |
| 1665 addLocalNameToSet(attrSet, rulesAttr); | |
| 1666 addLocalNameToSet(attrSet, scopeAttr); | |
| 1667 addLocalNameToSet(attrSet, scrollingAttr); | |
| 1668 addLocalNameToSet(attrSet, selectedAttr); | |
| 1669 addLocalNameToSet(attrSet, shapeAttr); | |
| 1670 addLocalNameToSet(attrSet, targetAttr); | |
| 1671 addLocalNameToSet(attrSet, textAttr); | |
| 1672 addLocalNameToSet(attrSet, typeAttr); | |
| 1673 addLocalNameToSet(attrSet, valignAttr); | |
| 1674 addLocalNameToSet(attrSet, valuetypeAttr); | |
| 1675 addLocalNameToSet(attrSet, vlinkAttr); | |
| 1676 | |
| 1677 return attrSet; | |
| 1678 } | |
| 1679 | |
| 1680 static bool htmlAttributeHasCaseInsensitiveValue(const QualifiedName& attr) | |
| 1681 { | |
| 1682 static HashSet<AtomicStringImpl*>* htmlCaseInsensitiveAttributesSet = create
HtmlCaseInsensitiveAttributesSet(); | |
| 1683 bool isPossibleHTMLAttr = !attr.hasPrefix() && (attr.namespaceURI() == nullA
tom); | |
| 1684 return isPossibleHTMLAttr && htmlCaseInsensitiveAttributesSet->contains(attr
.localName().impl()); | |
| 1685 } | |
| 1686 | |
| 1687 bool CSSStyleSelector::SelectorChecker::checkOneSelector(CSSSelector* sel, Eleme
nt* e, HashSet<AtomicStringImpl*>* selectorAttrs, RenderStyle::PseudoId& dynamic
Pseudo, bool isAncestor, bool isSubSelector, RenderStyle* elementStyle, RenderSt
yle* elementParentStyle) const | |
| 1688 { | |
| 1689 if (!e) | |
| 1690 return false; | |
| 1691 | |
| 1692 if (sel->hasTag()) { | |
| 1693 const AtomicString& selLocalName = sel->m_tag.localName(); | |
| 1694 if (selLocalName != starAtom && selLocalName != e->localName()) | |
| 1695 return false; | |
| 1696 const AtomicString& selNS = sel->m_tag.namespaceURI(); | |
| 1697 if (selNS != starAtom && selNS != e->namespaceURI()) | |
| 1698 return false; | |
| 1699 } | |
| 1700 | |
| 1701 if (sel->hasAttribute()) { | |
| 1702 if (sel->m_match == CSSSelector::Class) | |
| 1703 return e->hasClass() && static_cast<StyledElement*>(e)->classNames()
.contains(sel->m_value); | |
| 1704 | |
| 1705 if (sel->m_match == CSSSelector::Id) | |
| 1706 return e->hasID() && e->getIDAttribute() == sel->m_value; | |
| 1707 | |
| 1708 // FIXME: Handle the case were elementStyle is 0. | |
| 1709 if (elementStyle && (!e->isStyledElement() || (!static_cast<StyledElemen
t*>(e)->isMappedAttribute(sel->m_attr) && sel->m_attr != typeAttr && sel->m_attr
!= readonlyAttr))) { | |
| 1710 elementStyle->setAffectedByAttributeSelectors(); // Special-case the
"type" and "readonly" attributes so input form controls can share style. | |
| 1711 if (selectorAttrs) | |
| 1712 selectorAttrs->add(sel->m_attr.localName().impl()); | |
| 1713 } | |
| 1714 | |
| 1715 const AtomicString& value = e->getAttribute(sel->m_attr); | |
| 1716 if (value.isNull()) | |
| 1717 return false; // attribute is not set | |
| 1718 | |
| 1719 bool caseSensitive = !m_documentIsHTML || !htmlAttributeHasCaseInsensiti
veValue(sel->m_attr); | |
| 1720 | |
| 1721 switch (sel->m_match) { | |
| 1722 case CSSSelector::Exact: | |
| 1723 if (caseSensitive ? sel->m_value != value : !equalIgnoringCase(sel->
m_value, value)) | |
| 1724 return false; | |
| 1725 break; | |
| 1726 case CSSSelector::List: | |
| 1727 { | |
| 1728 // Ignore empty selectors or selectors containing spaces | |
| 1729 if (sel->m_value.contains(' ') || sel->m_value.isEmpty()) | |
| 1730 return false; | |
| 1731 | |
| 1732 int startSearchAt = 0; | |
| 1733 while (true) { | |
| 1734 int foundPos = value.find(sel->m_value, startSearchAt, caseSensi
tive); | |
| 1735 if (foundPos == -1) | |
| 1736 return false; | |
| 1737 if (foundPos == 0 || value[foundPos-1] == ' ') { | |
| 1738 unsigned endStr = foundPos + sel->m_value.length(); | |
| 1739 if (endStr == value.length() || value[endStr] == ' ') | |
| 1740 break; // We found a match. | |
| 1741 } | |
| 1742 | |
| 1743 // No match. Keep looking. | |
| 1744 startSearchAt = foundPos + 1; | |
| 1745 } | |
| 1746 break; | |
| 1747 } | |
| 1748 case CSSSelector::Contain: | |
| 1749 if (!value.contains(sel->m_value, caseSensitive) || sel->m_value.isE
mpty()) | |
| 1750 return false; | |
| 1751 break; | |
| 1752 case CSSSelector::Begin: | |
| 1753 if (!value.startsWith(sel->m_value, caseSensitive) || sel->m_value.i
sEmpty()) | |
| 1754 return false; | |
| 1755 break; | |
| 1756 case CSSSelector::End: | |
| 1757 if (!value.endsWith(sel->m_value, caseSensitive) || sel->m_value.isE
mpty()) | |
| 1758 return false; | |
| 1759 break; | |
| 1760 case CSSSelector::Hyphen: | |
| 1761 if (value.length() < sel->m_value.length()) | |
| 1762 return false; | |
| 1763 if (!value.startsWith(sel->m_value, caseSensitive)) | |
| 1764 return false; | |
| 1765 // It they start the same, check for exact match or following '-': | |
| 1766 if (value.length() != sel->m_value.length() && value[sel->m_value.le
ngth()] != '-') | |
| 1767 return false; | |
| 1768 break; | |
| 1769 case CSSSelector::PseudoClass: | |
| 1770 case CSSSelector::PseudoElement: | |
| 1771 default: | |
| 1772 break; | |
| 1773 } | |
| 1774 } | |
| 1775 if (sel->m_match == CSSSelector::PseudoClass) { | |
| 1776 switch (sel->pseudoType()) { | |
| 1777 // Pseudo classes: | |
| 1778 case CSSSelector::PseudoEmpty: { | |
| 1779 bool result = true; | |
| 1780 for (Node* n = e->firstChild(); n; n = n->nextSibling()) { | |
| 1781 if (n->isElementNode()) { | |
| 1782 result = false; | |
| 1783 break; | |
| 1784 } else if (n->isTextNode()) { | |
| 1785 Text* textNode = static_cast<Text*>(n); | |
| 1786 if (!textNode->data().isEmpty()) { | |
| 1787 result = false; | |
| 1788 break; | |
| 1789 } | |
| 1790 } | |
| 1791 } | |
| 1792 if (!m_collectRulesOnly) { | |
| 1793 if (elementStyle) | |
| 1794 elementStyle->setEmptyState(result); | |
| 1795 else if (e->renderStyle() && (e->document()->usesSiblingRule
s() || e->renderStyle()->unique())) | |
| 1796 e->renderStyle()->setEmptyState(result); | |
| 1797 } | |
| 1798 return result; | |
| 1799 } | |
| 1800 case CSSSelector::PseudoFirstChild: { | |
| 1801 // first-child matches the first child that is an element | |
| 1802 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1803 bool result = false; | |
| 1804 Node* n = e->previousSibling(); | |
| 1805 while (n && !n->isElementNode()) | |
| 1806 n = n->previousSibling(); | |
| 1807 if (!n) | |
| 1808 result = true; | |
| 1809 if (!m_collectRulesOnly) { | |
| 1810 RenderStyle* childStyle = elementStyle ? elementStyle :
e->renderStyle(); | |
| 1811 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : e->parentNode()->renderStyle(); | |
| 1812 if (parentStyle) | |
| 1813 parentStyle->setChildrenAffectedByFirstChildRules(); | |
| 1814 if (result && childStyle) | |
| 1815 childStyle->setFirstChildState(); | |
| 1816 } | |
| 1817 return result; | |
| 1818 } | |
| 1819 break; | |
| 1820 } | |
| 1821 case CSSSelector::PseudoFirstOfType: { | |
| 1822 // first-of-type matches the first element of its type | |
| 1823 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1824 bool result = false; | |
| 1825 const QualifiedName& type = e->tagQName(); | |
| 1826 Node* n = e->previousSibling(); | |
| 1827 while (n) { | |
| 1828 if (n->isElementNode() && static_cast<Element*>(n)->hasT
agName(type)) | |
| 1829 break; | |
| 1830 n = n->previousSibling(); | |
| 1831 } | |
| 1832 if (!n) | |
| 1833 result = true; | |
| 1834 if (!m_collectRulesOnly) { | |
| 1835 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : e->parentNode()->renderStyle(); | |
| 1836 if (parentStyle) | |
| 1837 parentStyle->setChildrenAffectedByForwardPositionalR
ules(); | |
| 1838 } | |
| 1839 return result; | |
| 1840 } | |
| 1841 break; | |
| 1842 } | |
| 1843 case CSSSelector::PseudoLastChild: { | |
| 1844 // last-child matches the last child that is an element | |
| 1845 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1846 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 1847 bool result = false; | |
| 1848 if (parentNode->isFinishedParsingChildren()) { | |
| 1849 Node* n = e->nextSibling(); | |
| 1850 while (n && !n->isElementNode()) | |
| 1851 n = n->nextSibling(); | |
| 1852 if (!n) | |
| 1853 result = true; | |
| 1854 } | |
| 1855 if (!m_collectRulesOnly) { | |
| 1856 RenderStyle* childStyle = elementStyle ? elementStyle :
e->renderStyle(); | |
| 1857 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 1858 if (parentStyle) | |
| 1859 parentStyle->setChildrenAffectedByLastChildRules(); | |
| 1860 if (result && childStyle) | |
| 1861 childStyle->setLastChildState(); | |
| 1862 } | |
| 1863 return result; | |
| 1864 } | |
| 1865 break; | |
| 1866 } | |
| 1867 case CSSSelector::PseudoLastOfType: { | |
| 1868 // last-of-type matches the last element of its type | |
| 1869 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1870 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 1871 if (!m_collectRulesOnly) { | |
| 1872 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 1873 if (parentStyle) | |
| 1874 parentStyle->setChildrenAffectedByBackwardPositional
Rules(); | |
| 1875 } | |
| 1876 if (!parentNode->isFinishedParsingChildren()) | |
| 1877 return false; | |
| 1878 bool result = false; | |
| 1879 const QualifiedName& type = e->tagQName(); | |
| 1880 Node* n = e->nextSibling(); | |
| 1881 while (n) { | |
| 1882 if (n->isElementNode() && static_cast<Element*>(n)->hasT
agName(type)) | |
| 1883 break; | |
| 1884 n = n->nextSibling(); | |
| 1885 } | |
| 1886 if (!n) | |
| 1887 result = true; | |
| 1888 return result; | |
| 1889 } | |
| 1890 break; | |
| 1891 } | |
| 1892 case CSSSelector::PseudoOnlyChild: { | |
| 1893 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1894 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 1895 bool firstChild = false; | |
| 1896 bool lastChild = false; | |
| 1897 | |
| 1898 Node* n = e->previousSibling(); | |
| 1899 while (n && !n->isElementNode()) | |
| 1900 n = n->previousSibling(); | |
| 1901 if (!n) | |
| 1902 firstChild = true; | |
| 1903 if (firstChild && parentNode->isFinishedParsingChildren()) { | |
| 1904 n = e->nextSibling(); | |
| 1905 while (n && !n->isElementNode()) | |
| 1906 n = n->nextSibling(); | |
| 1907 if (!n) | |
| 1908 lastChild = true; | |
| 1909 } | |
| 1910 if (!m_collectRulesOnly) { | |
| 1911 RenderStyle* childStyle = elementStyle ? elementStyle :
e->renderStyle(); | |
| 1912 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 1913 if (parentStyle) { | |
| 1914 parentStyle->setChildrenAffectedByFirstChildRules(); | |
| 1915 parentStyle->setChildrenAffectedByLastChildRules(); | |
| 1916 } | |
| 1917 if (firstChild && childStyle) | |
| 1918 childStyle->setFirstChildState(); | |
| 1919 if (lastChild && childStyle) | |
| 1920 childStyle->setLastChildState(); | |
| 1921 } | |
| 1922 return firstChild && lastChild; | |
| 1923 } | |
| 1924 break; | |
| 1925 } | |
| 1926 case CSSSelector::PseudoOnlyOfType: { | |
| 1927 // FIXME: This selector is very slow. | |
| 1928 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1929 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 1930 if (!m_collectRulesOnly) { | |
| 1931 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 1932 if (parentStyle) { | |
| 1933 parentStyle->setChildrenAffectedByForwardPositionalR
ules(); | |
| 1934 parentStyle->setChildrenAffectedByBackwardPositional
Rules(); | |
| 1935 } | |
| 1936 } | |
| 1937 if (!parentNode->isFinishedParsingChildren()) | |
| 1938 return false; | |
| 1939 bool firstChild = false; | |
| 1940 bool lastChild = false; | |
| 1941 const QualifiedName& type = e->tagQName(); | |
| 1942 Node* n = e->previousSibling(); | |
| 1943 while (n) { | |
| 1944 if (n->isElementNode() && static_cast<Element*>(n)->hasT
agName(type)) | |
| 1945 break; | |
| 1946 n = n->previousSibling(); | |
| 1947 } | |
| 1948 if (!n) | |
| 1949 firstChild = true; | |
| 1950 if (firstChild) { | |
| 1951 n = e->nextSibling(); | |
| 1952 while (n) { | |
| 1953 if (n->isElementNode() && static_cast<Element*>(n)->
hasTagName(type)) | |
| 1954 break; | |
| 1955 n = n->nextSibling(); | |
| 1956 } | |
| 1957 if (!n) | |
| 1958 lastChild = true; | |
| 1959 } | |
| 1960 return firstChild && lastChild; | |
| 1961 } | |
| 1962 break; | |
| 1963 } | |
| 1964 case CSSSelector::PseudoNthChild: { | |
| 1965 int a, b; | |
| 1966 // calculate a and b every time we run through checkOneSelector | |
| 1967 // this should probably be saved after we calculate it once, but
currently | |
| 1968 // would require increasing the size of CSSSelector | |
| 1969 if (!parseNth(sel->m_argument, a, b)) | |
| 1970 break; | |
| 1971 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 1972 int count = 1; | |
| 1973 Node* n = e->previousSibling(); | |
| 1974 while (n) { | |
| 1975 if (n->isElementNode()) { | |
| 1976 RenderStyle* s = n->renderStyle(); | |
| 1977 unsigned index = s ? s->childIndex() : 0; | |
| 1978 if (index) { | |
| 1979 count += index; | |
| 1980 break; | |
| 1981 } | |
| 1982 count++; | |
| 1983 } | |
| 1984 n = n->previousSibling(); | |
| 1985 } | |
| 1986 | |
| 1987 if (!m_collectRulesOnly) { | |
| 1988 RenderStyle* childStyle = elementStyle ? elementStyle :
e->renderStyle(); | |
| 1989 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : e->parentNode()->renderStyle(); | |
| 1990 if (childStyle) | |
| 1991 childStyle->setChildIndex(count); | |
| 1992 if (parentStyle) | |
| 1993 parentStyle->setChildrenAffectedByForwardPositionalR
ules(); | |
| 1994 } | |
| 1995 | |
| 1996 if (matchNth(count, a, b)) | |
| 1997 return true; | |
| 1998 } | |
| 1999 break; | |
| 2000 } | |
| 2001 case CSSSelector::PseudoNthOfType: { | |
| 2002 // FIXME: This selector is very slow. | |
| 2003 int a, b; | |
| 2004 // calculate a and b every time we run through checkOneSelector
(see above) | |
| 2005 if (!parseNth(sel->m_argument, a, b)) | |
| 2006 break; | |
| 2007 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 2008 int count = 1; | |
| 2009 const QualifiedName& type = e->tagQName(); | |
| 2010 Node* n = e->previousSibling(); | |
| 2011 while (n) { | |
| 2012 if (n->isElementNode() && static_cast<Element*>(n)->hasT
agName(type)) | |
| 2013 count++; | |
| 2014 n = n->previousSibling(); | |
| 2015 } | |
| 2016 | |
| 2017 if (!m_collectRulesOnly) { | |
| 2018 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : e->parentNode()->renderStyle(); | |
| 2019 if (parentStyle) | |
| 2020 parentStyle->setChildrenAffectedByForwardPositionalR
ules(); | |
| 2021 } | |
| 2022 | |
| 2023 if (matchNth(count, a, b)) | |
| 2024 return true; | |
| 2025 } | |
| 2026 break; | |
| 2027 } | |
| 2028 case CSSSelector::PseudoNthLastChild: { | |
| 2029 int a, b; | |
| 2030 // calculate a and b every time we run through checkOneSelector | |
| 2031 // this should probably be saved after we calculate it once, but
currently | |
| 2032 // would require increasing the size of CSSSelector | |
| 2033 if (!parseNth(sel->m_argument, a, b)) | |
| 2034 break; | |
| 2035 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 2036 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 2037 if (!m_collectRulesOnly) { | |
| 2038 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 2039 if (parentStyle) | |
| 2040 parentStyle->setChildrenAffectedByBackwardPositional
Rules(); | |
| 2041 } | |
| 2042 if (!parentNode->isFinishedParsingChildren()) | |
| 2043 return false; | |
| 2044 int count = 1; | |
| 2045 Node* n = e->nextSibling(); | |
| 2046 while (n) { | |
| 2047 if (n->isElementNode()) | |
| 2048 count++; | |
| 2049 n = n->nextSibling(); | |
| 2050 } | |
| 2051 if (matchNth(count, a, b)) | |
| 2052 return true; | |
| 2053 } | |
| 2054 break; | |
| 2055 } | |
| 2056 case CSSSelector::PseudoNthLastOfType: { | |
| 2057 // FIXME: This selector is very slow. | |
| 2058 int a, b; | |
| 2059 // calculate a and b every time we run through checkOneSelector
(see above) | |
| 2060 if (!parseNth(sel->m_argument, a, b)) | |
| 2061 break; | |
| 2062 if (e->parentNode() && e->parentNode()->isElementNode()) { | |
| 2063 Element* parentNode = static_cast<Element*>(e->parentNode())
; | |
| 2064 if (!m_collectRulesOnly) { | |
| 2065 RenderStyle* parentStyle = elementStyle ? elementParentS
tyle : parentNode->renderStyle(); | |
| 2066 if (parentStyle) | |
| 2067 parentStyle->setChildrenAffectedByBackwardPositional
Rules(); | |
| 2068 } | |
| 2069 if (!parentNode->isFinishedParsingChildren()) | |
| 2070 return false; | |
| 2071 int count = 1; | |
| 2072 const QualifiedName& type = e->tagQName(); | |
| 2073 Node* n = e->nextSibling(); | |
| 2074 while (n) { | |
| 2075 if (n->isElementNode() && static_cast<Element*>(n)->hasT
agName(type)) | |
| 2076 count++; | |
| 2077 n = n->nextSibling(); | |
| 2078 } | |
| 2079 if (matchNth(count, a, b)) | |
| 2080 return true; | |
| 2081 } | |
| 2082 break; | |
| 2083 } | |
| 2084 case CSSSelector::PseudoTarget: | |
| 2085 if (e == e->document()->getCSSTarget()) | |
| 2086 return true; | |
| 2087 break; | |
| 2088 case CSSSelector::PseudoAnyLink: | |
| 2089 if (pseudoState == PseudoUnknown) | |
| 2090 pseudoState = checkPseudoState(e, false); | |
| 2091 if (pseudoState == PseudoAnyLink || pseudoState == PseudoLink ||
pseudoState == PseudoVisited) | |
| 2092 return true; | |
| 2093 break; | |
| 2094 case CSSSelector::PseudoAutofill: | |
| 2095 if (e && e->hasTagName(inputTag)) | |
| 2096 return static_cast<HTMLInputElement*>(e)->autofilled(); | |
| 2097 break; | |
| 2098 case CSSSelector::PseudoLink: | |
| 2099 if (pseudoState == PseudoUnknown || pseudoState == PseudoAnyLink
) | |
| 2100 pseudoState = checkPseudoState(e); | |
| 2101 if (pseudoState == PseudoLink) | |
| 2102 return true; | |
| 2103 break; | |
| 2104 case CSSSelector::PseudoVisited: | |
| 2105 if (pseudoState == PseudoUnknown || pseudoState == PseudoAnyLink
) | |
| 2106 pseudoState = checkPseudoState(e); | |
| 2107 if (pseudoState == PseudoVisited) | |
| 2108 return true; | |
| 2109 break; | |
| 2110 case CSSSelector::PseudoDrag: { | |
| 2111 if (elementStyle) | |
| 2112 elementStyle->setAffectedByDragRules(true); | |
| 2113 else if (e->renderStyle()) | |
| 2114 e->renderStyle()->setAffectedByDragRules(true); | |
| 2115 if (e->renderer() && e->renderer()->isDragging()) | |
| 2116 return true; | |
| 2117 break; | |
| 2118 } | |
| 2119 case CSSSelector::PseudoFocus: | |
| 2120 if (e && e->focused() && e->document()->frame()->selection()->is
FocusedAndActive()) | |
| 2121 return true; | |
| 2122 break; | |
| 2123 case CSSSelector::PseudoHover: { | |
| 2124 // If we're in quirks mode, then hover should never match anchor
s with no | |
| 2125 // href and *:hover should not match anything. This is importan
t for sites like wsj.com. | |
| 2126 if (m_strictParsing || isSubSelector || (sel->hasTag() && !e->ha
sTagName(aTag)) || e->isLink()) { | |
| 2127 if (elementStyle) | |
| 2128 elementStyle->setAffectedByHoverRules(true); | |
| 2129 else if (e->renderStyle()) | |
| 2130 e->renderStyle()->setAffectedByHoverRules(true); | |
| 2131 if (e->hovered()) | |
| 2132 return true; | |
| 2133 } | |
| 2134 break; | |
| 2135 } | |
| 2136 case CSSSelector::PseudoActive: | |
| 2137 // If we're in quirks mode, then :active should never match anch
ors with no | |
| 2138 // href and *:active should not match anything. | |
| 2139 if (m_strictParsing || isSubSelector || (sel->hasTag() && !e->ha
sTagName(aTag)) || e->isLink()) { | |
| 2140 if (elementStyle) | |
| 2141 elementStyle->setAffectedByActiveRules(true); | |
| 2142 else if (e->renderStyle()) | |
| 2143 e->renderStyle()->setAffectedByActiveRules(true); | |
| 2144 if (e->active()) | |
| 2145 return true; | |
| 2146 } | |
| 2147 break; | |
| 2148 case CSSSelector::PseudoEnabled: | |
| 2149 if (e && e->isControl() && !e->isInputTypeHidden()) | |
| 2150 // The UI spec states that you can't match :enabled unless y
ou are an object that can | |
| 2151 // "receive focus and be activated." We will limit matching
of this pseudo-class to elements | |
| 2152 // that are non-"hidden" controls. | |
| 2153 return e->isEnabled(); | |
| 2154 break; | |
| 2155 case CSSSelector::PseudoFullPageMedia: | |
| 2156 return e && e->document() && e->document()->isMediaDocument(); | |
| 2157 break; | |
| 2158 case CSSSelector::PseudoDisabled: | |
| 2159 if (e && e->isControl() && !e->isInputTypeHidden()) | |
| 2160 // The UI spec states that you can't match :enabled unless y
ou are an object that can | |
| 2161 // "receive focus and be activated." We will limit matching
of this pseudo-class to elements | |
| 2162 // that are non-"hidden" controls. | |
| 2163 return !e->isEnabled(); | |
| 2164 break; | |
| 2165 case CSSSelector::PseudoReadOnly: | |
| 2166 return e && e->isTextControl() && e->isReadOnlyControl(); | |
| 2167 case CSSSelector::PseudoReadWrite: | |
| 2168 return e && e->isTextControl() && !e->isReadOnlyControl(); | |
| 2169 case CSSSelector::PseudoChecked: | |
| 2170 // Even though WinIE allows checked and indeterminate to co-exis
t, the CSS selector spec says that | |
| 2171 // you can't be both checked and indeterminate. We will behave
like WinIE behind the scenes and just | |
| 2172 // obey the CSS spec here in the test for matching the pseudo. | |
| 2173 if (e && e->isChecked() && !e->isIndeterminate()) | |
| 2174 return true; | |
| 2175 break; | |
| 2176 case CSSSelector::PseudoIndeterminate: | |
| 2177 if (e && e->isIndeterminate()) | |
| 2178 return true; | |
| 2179 break; | |
| 2180 case CSSSelector::PseudoRoot: | |
| 2181 if (e == e->document()->documentElement()) | |
| 2182 return true; | |
| 2183 break; | |
| 2184 case CSSSelector::PseudoLang: { | |
| 2185 Node* n = e; | |
| 2186 AtomicString value; | |
| 2187 // The language property is inherited, so we iterate over the pa
rents | |
| 2188 // to find the first language. | |
| 2189 while (n && value.isEmpty()) { | |
| 2190 if (n->isElementNode()) { | |
| 2191 // Spec: xml:lang takes precedence -- http://www.w3.org/
TR/xhtml1/#C_7 | |
| 2192 value = static_cast<Element*>(n)->getAttribute(XMLNames:
:langAttr); | |
| 2193 if (value.isEmpty()) | |
| 2194 value = static_cast<Element*>(n)->getAttribute(langA
ttr); | |
| 2195 } else if (n->isDocumentNode()) | |
| 2196 // checking the MIME content-language | |
| 2197 value = static_cast<Document*>(n)->contentLanguage(); | |
| 2198 | |
| 2199 n = n->parent(); | |
| 2200 } | |
| 2201 if (value.isEmpty() || !value.startsWith(sel->m_argument, false)
) | |
| 2202 break; | |
| 2203 if (value.length() != sel->m_argument.length() && value[sel->m_a
rgument.length()] != '-') | |
| 2204 break; | |
| 2205 return true; | |
| 2206 } | |
| 2207 case CSSSelector::PseudoNot: { | |
| 2208 // check the simple selector | |
| 2209 for (CSSSelector* subSel = sel->m_simpleSelector; subSel; subSel
= subSel->m_tagHistory) { | |
| 2210 // :not cannot nest. I don't really know why this is a | |
| 2211 // restriction in CSS3, but it is, so let's honour it. | |
| 2212 if (subSel->m_simpleSelector) | |
| 2213 break; | |
| 2214 if (!checkOneSelector(subSel, e, selectorAttrs, dynamicPseud
o, isAncestor, true, elementStyle, elementParentStyle)) | |
| 2215 return true; | |
| 2216 } | |
| 2217 break; | |
| 2218 } | |
| 2219 case CSSSelector::PseudoUnknown: | |
| 2220 case CSSSelector::PseudoNotParsed: | |
| 2221 default: | |
| 2222 ASSERT_NOT_REACHED(); | |
| 2223 break; | |
| 2224 } | |
| 2225 return false; | |
| 2226 } | |
| 2227 if (sel->m_match == CSSSelector::PseudoElement) { | |
| 2228 if (!elementStyle) | |
| 2229 return false; | |
| 2230 | |
| 2231 switch (sel->pseudoType()) { | |
| 2232 // Pseudo-elements: | |
| 2233 case CSSSelector::PseudoFirstLine: | |
| 2234 dynamicPseudo = RenderStyle::FIRST_LINE; | |
| 2235 return true; | |
| 2236 case CSSSelector::PseudoFirstLetter: | |
| 2237 dynamicPseudo = RenderStyle::FIRST_LETTER; | |
| 2238 if (Document* doc = e->document()) | |
| 2239 doc->setUsesFirstLetterRules(true); | |
| 2240 return true; | |
| 2241 case CSSSelector::PseudoSelection: | |
| 2242 dynamicPseudo = RenderStyle::SELECTION; | |
| 2243 return true; | |
| 2244 case CSSSelector::PseudoBefore: | |
| 2245 dynamicPseudo = RenderStyle::BEFORE; | |
| 2246 return true; | |
| 2247 case CSSSelector::PseudoAfter: | |
| 2248 dynamicPseudo = RenderStyle::AFTER; | |
| 2249 return true; | |
| 2250 case CSSSelector::PseudoFileUploadButton: | |
| 2251 dynamicPseudo = RenderStyle::FILE_UPLOAD_BUTTON; | |
| 2252 return true; | |
| 2253 case CSSSelector::PseudoSliderThumb: | |
| 2254 dynamicPseudo = RenderStyle::SLIDER_THUMB; | |
| 2255 return true; | |
| 2256 case CSSSelector::PseudoSearchCancelButton: | |
| 2257 dynamicPseudo = RenderStyle::SEARCH_CANCEL_BUTTON; | |
| 2258 return true; | |
| 2259 case CSSSelector::PseudoSearchDecoration: | |
| 2260 dynamicPseudo = RenderStyle::SEARCH_DECORATION; | |
| 2261 return true; | |
| 2262 case CSSSelector::PseudoSearchResultsDecoration: | |
| 2263 dynamicPseudo = RenderStyle::SEARCH_RESULTS_DECORATION; | |
| 2264 return true; | |
| 2265 case CSSSelector::PseudoSearchResultsButton: | |
| 2266 dynamicPseudo = RenderStyle::SEARCH_RESULTS_BUTTON; | |
| 2267 return true; | |
| 2268 case CSSSelector::PseudoMediaControlsPanel: | |
| 2269 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_PANEL; | |
| 2270 return true; | |
| 2271 case CSSSelector::PseudoMediaControlsMuteButton: | |
| 2272 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_MUTE_BUTTON; | |
| 2273 return true; | |
| 2274 case CSSSelector::PseudoMediaControlsPlayButton: | |
| 2275 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_PLAY_BUTTON; | |
| 2276 return true; | |
| 2277 case CSSSelector::PseudoMediaControlsTimeDisplay: | |
| 2278 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_TIME_DISPLAY; | |
| 2279 return true; | |
| 2280 case CSSSelector::PseudoMediaControlsTimeline: | |
| 2281 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_TIMELINE; | |
| 2282 return true; | |
| 2283 case CSSSelector::PseudoMediaControlsSeekBackButton: | |
| 2284 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_SEEK_BACK_BUTTON; | |
| 2285 return true; | |
| 2286 case CSSSelector::PseudoMediaControlsSeekForwardButton: | |
| 2287 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_SEEK_FORWARD_BUTTON; | |
| 2288 return true; | |
| 2289 case CSSSelector::PseudoMediaControlsFullscreenButton: | |
| 2290 dynamicPseudo = RenderStyle::MEDIA_CONTROLS_FULLSCREEN_BUTTON; | |
| 2291 return true; | |
| 2292 case CSSSelector::PseudoUnknown: | |
| 2293 case CSSSelector::PseudoNotParsed: | |
| 2294 default: | |
| 2295 ASSERT_NOT_REACHED(); | |
| 2296 break; | |
| 2297 } | |
| 2298 return false; | |
| 2299 } | |
| 2300 // ### add the rest of the checks... | |
| 2301 return true; | |
| 2302 } | |
| 2303 | |
| 2304 void CSSStyleSelector::addVariables(CSSVariablesRule* variables) | |
| 2305 { | |
| 2306 CSSVariablesDeclaration* decl = variables->variables(); | |
| 2307 if (!decl) | |
| 2308 return; | |
| 2309 unsigned size = decl->length(); | |
| 2310 for (unsigned i = 0; i < size; ++i) { | |
| 2311 String name = decl->item(i); | |
| 2312 m_variablesMap.set(name, variables); | |
| 2313 } | |
| 2314 } | |
| 2315 | |
| 2316 CSSValue* CSSStyleSelector::resolveVariableDependentValue(CSSVariableDependentVa
lue* val) | |
| 2317 { | |
| 2318 return 0; | |
| 2319 } | |
| 2320 | |
| 2321 // ----------------------------------------------------------------- | |
| 2322 | |
| 2323 CSSRuleSet::CSSRuleSet() | |
| 2324 { | |
| 2325 m_universalRules = 0; | |
| 2326 m_ruleCount = 0; | |
| 2327 } | |
| 2328 | |
| 2329 CSSRuleSet::~CSSRuleSet() | |
| 2330 { | |
| 2331 deleteAllValues(m_idRules); | |
| 2332 deleteAllValues(m_classRules); | |
| 2333 deleteAllValues(m_tagRules); | |
| 2334 | |
| 2335 delete m_universalRules; | |
| 2336 } | |
| 2337 | |
| 2338 | |
| 2339 void CSSRuleSet::addToRuleSet(AtomicStringImpl* key, AtomRuleMap& map, | |
| 2340 CSSStyleRule* rule, CSSSelector* sel) | |
| 2341 { | |
| 2342 if (!key) return; | |
| 2343 CSSRuleDataList* rules = map.get(key); | |
| 2344 if (!rules) { | |
| 2345 rules = new CSSRuleDataList(m_ruleCount++, rule, sel); | |
| 2346 map.set(key, rules); | |
| 2347 } else | |
| 2348 rules->append(m_ruleCount++, rule, sel); | |
| 2349 } | |
| 2350 | |
| 2351 void CSSRuleSet::addRule(CSSStyleRule* rule, CSSSelector* sel) | |
| 2352 { | |
| 2353 if (sel->m_match == CSSSelector::Id) { | |
| 2354 addToRuleSet(sel->m_value.impl(), m_idRules, rule, sel); | |
| 2355 return; | |
| 2356 } | |
| 2357 if (sel->m_match == CSSSelector::Class) { | |
| 2358 addToRuleSet(sel->m_value.impl(), m_classRules, rule, sel); | |
| 2359 return; | |
| 2360 } | |
| 2361 | |
| 2362 const AtomicString& localName = sel->m_tag.localName(); | |
| 2363 if (localName != starAtom) { | |
| 2364 addToRuleSet(localName.impl(), m_tagRules, rule, sel); | |
| 2365 return; | |
| 2366 } | |
| 2367 | |
| 2368 // Just put it in the universal rule set. | |
| 2369 if (!m_universalRules) | |
| 2370 m_universalRules = new CSSRuleDataList(m_ruleCount++, rule, sel); | |
| 2371 else | |
| 2372 m_universalRules->append(m_ruleCount++, rule, sel); | |
| 2373 } | |
| 2374 | |
| 2375 void CSSRuleSet::addRulesFromSheet(CSSStyleSheet* sheet, const MediaQueryEvaluat
or& medium, CSSStyleSelector* styleSelector) | |
| 2376 { | |
| 2377 if (!sheet) | |
| 2378 return; | |
| 2379 | |
| 2380 // No media implies "all", but if a media list exists it must | |
| 2381 // contain our current medium | |
| 2382 if (sheet->media() && !medium.eval(sheet->media(), styleSelector)) | |
| 2383 return; // the style sheet doesn't apply | |
| 2384 | |
| 2385 int len = sheet->length(); | |
| 2386 | |
| 2387 for (int i = 0; i < len; i++) { | |
| 2388 StyleBase* item = sheet->item(i); | |
| 2389 if (item->isStyleRule()) { | |
| 2390 CSSStyleRule* rule = static_cast<CSSStyleRule*>(item); | |
| 2391 for (CSSSelector* s = rule->selector(); s; s = s->next()) | |
| 2392 addRule(rule, s); | |
| 2393 } | |
| 2394 else if (item->isImportRule()) { | |
| 2395 CSSImportRule* import = static_cast<CSSImportRule*>(item); | |
| 2396 if (!import->media() || medium.eval(import->media(), styleSelector)) | |
| 2397 addRulesFromSheet(import->styleSheet(), medium, styleSelector); | |
| 2398 } | |
| 2399 else if (item->isMediaRule()) { | |
| 2400 CSSMediaRule* r = static_cast<CSSMediaRule*>(item); | |
| 2401 CSSRuleList* rules = r->cssRules(); | |
| 2402 | |
| 2403 if ((!r->media() || medium.eval(r->media(), styleSelector)) && rules
) { | |
| 2404 // Traverse child elements of the @media rule. | |
| 2405 for (unsigned j = 0; j < rules->length(); j++) { | |
| 2406 CSSRule *childItem = rules->item(j); | |
| 2407 if (childItem->isStyleRule()) { | |
| 2408 // It is a StyleRule, so append it to our list | |
| 2409 CSSStyleRule* rule = static_cast<CSSStyleRule*>(childIte
m); | |
| 2410 for (CSSSelector* s = rule->selector(); s; s = s->next()
) | |
| 2411 addRule(rule, s); | |
| 2412 } else if (childItem->isFontFaceRule() && styleSelector) { | |
| 2413 // Add this font face to our set. | |
| 2414 const CSSFontFaceRule* fontFaceRule = static_cast<CSSFon
tFaceRule*>(childItem); | |
| 2415 styleSelector->fontSelector()->addFontFaceRule(fontFaceR
ule); | |
| 2416 } else if (childItem->isKeyframesRule() && styleSelector) { | |
| 2417 // Add this keyframe rule to our set. | |
| 2418 const WebKitCSSKeyframesRule* keyframesRule = static_cas
t<WebKitCSSKeyframesRule*>(childItem); | |
| 2419 styleSelector->addKeyframeStyle(sheet->doc(), keyframesR
ule); | |
| 2420 } | |
| 2421 } // for rules | |
| 2422 } // if rules | |
| 2423 } else if (item->isFontFaceRule() && styleSelector) { | |
| 2424 // Add this font face to our set. | |
| 2425 const CSSFontFaceRule* fontFaceRule = static_cast<CSSFontFaceRule*>(
item); | |
| 2426 styleSelector->fontSelector()->addFontFaceRule(fontFaceRule); | |
| 2427 } else if (item->isVariablesRule()) { | |
| 2428 // Evaluate the media query and make sure it matches. | |
| 2429 CSSVariablesRule* variables = static_cast<CSSVariablesRule*>(item); | |
| 2430 if (!variables->media() || medium.eval(variables->media(), styleSele
ctor)) | |
| 2431 styleSelector->addVariables(variables); | |
| 2432 } else if (item->isKeyframesRule()) { | |
| 2433 WebKitCSSKeyframesRule* r = static_cast<WebKitCSSKeyframesRule*>(ite
m); | |
| 2434 styleSelector->addKeyframeStyle(sheet->doc(), r); | |
| 2435 } | |
| 2436 } | |
| 2437 } | |
| 2438 | |
| 2439 // -----------------------------------------------------------------------------
-------- | |
| 2440 // this is mostly boring stuff on how to apply a certain rule to the renderstyle
... | |
| 2441 | |
| 2442 static Length convertToLength(CSSPrimitiveValue *primitiveValue, RenderStyle *st
yle, bool *ok = 0) | |
| 2443 { | |
| 2444 Length l; | |
| 2445 if (!primitiveValue) { | |
| 2446 if (ok) | |
| 2447 *ok = false; | |
| 2448 } else { | |
| 2449 int type = primitiveValue->primitiveType(); | |
| 2450 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveValue
::CSS_DEG) | |
| 2451 l = Length(primitiveValue->computeLengthIntForLength(style), Fixed); | |
| 2452 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 2453 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 2454 else if (type == CSSPrimitiveValue::CSS_NUMBER) | |
| 2455 l = Length(primitiveValue->getDoubleValue() * 100.0, Percent); | |
| 2456 else if (ok) | |
| 2457 *ok = false; | |
| 2458 } | |
| 2459 return l; | |
| 2460 } | |
| 2461 | |
| 2462 void CSSStyleSelector::applyDeclarations(bool applyFirst, bool isImportant, | |
| 2463 int startIndex, int endIndex) | |
| 2464 { | |
| 2465 if (startIndex == -1) | |
| 2466 return; | |
| 2467 | |
| 2468 for (int i = startIndex; i <= endIndex; i++) { | |
| 2469 CSSMutableStyleDeclaration* decl = m_matchedDecls[i]; | |
| 2470 DeprecatedValueListConstIterator<CSSProperty> end; | |
| 2471 for (DeprecatedValueListConstIterator<CSSProperty> it = decl->valuesIter
ator(); it != end; ++it) { | |
| 2472 const CSSProperty& current = *it; | |
| 2473 // give special priority to font-xxx, color properties | |
| 2474 if (isImportant == current.isImportant()) { | |
| 2475 bool first; | |
| 2476 switch (current.id()) { | |
| 2477 case CSSPropertyLineHeight: | |
| 2478 m_lineHeightValue = current.value(); | |
| 2479 first = !applyFirst; // we apply line-height later | |
| 2480 break; | |
| 2481 case CSSPropertyColor: | |
| 2482 case CSSPropertyDirection: | |
| 2483 case CSSPropertyDisplay: | |
| 2484 case CSSPropertyFont: | |
| 2485 case CSSPropertyFontSize: | |
| 2486 case CSSPropertyFontStyle: | |
| 2487 case CSSPropertyFontFamily: | |
| 2488 case CSSPropertyFontWeight: | |
| 2489 case CSSPropertyWebkitTextSizeAdjust: | |
| 2490 case CSSPropertyFontVariant: | |
| 2491 case CSSPropertyZoom: | |
| 2492 // these have to be applied first, because other propert
ies use the computed | |
| 2493 // values of these porperties. | |
| 2494 first = true; | |
| 2495 break; | |
| 2496 default: | |
| 2497 first = false; | |
| 2498 break; | |
| 2499 } | |
| 2500 if (first == applyFirst) | |
| 2501 applyProperty(current.id(), current.value()); | |
| 2502 } | |
| 2503 } | |
| 2504 } | |
| 2505 } | |
| 2506 | |
| 2507 static void applyCounterList(RenderStyle* style, CSSValueList* list, bool isRese
t) | |
| 2508 { | |
| 2509 CounterDirectiveMap& map = style->accessCounterDirectives(); | |
| 2510 typedef CounterDirectiveMap::iterator Iterator; | |
| 2511 | |
| 2512 Iterator end = map.end(); | |
| 2513 for (Iterator it = map.begin(); it != end; ++it) | |
| 2514 if (isReset) | |
| 2515 it->second.m_reset = false; | |
| 2516 else | |
| 2517 it->second.m_increment = false; | |
| 2518 | |
| 2519 int length = list ? list->length() : 0; | |
| 2520 for (int i = 0; i < length; ++i) { | |
| 2521 Pair* pair = static_cast<CSSPrimitiveValue*>(list->itemWithoutBoundsChec
k(i))->getPairValue(); | |
| 2522 AtomicString identifier = static_cast<CSSPrimitiveValue*>(pair->first())
->getStringValue(); | |
| 2523 // FIXME: What about overflow? | |
| 2524 int value = static_cast<CSSPrimitiveValue*>(pair->second())->getIntValue
(); | |
| 2525 CounterDirectives& directives = map.add(identifier.impl(), CounterDirect
ives()).first->second; | |
| 2526 if (isReset) { | |
| 2527 directives.m_reset = true; | |
| 2528 directives.m_resetValue = value; | |
| 2529 } else { | |
| 2530 if (directives.m_increment) | |
| 2531 directives.m_incrementValue += value; | |
| 2532 else { | |
| 2533 directives.m_increment = true; | |
| 2534 directives.m_incrementValue = value; | |
| 2535 } | |
| 2536 } | |
| 2537 } | |
| 2538 } | |
| 2539 | |
| 2540 struct ScriptFamilyState { | |
| 2541 bool isGenericAdded; | |
| 2542 bool isPerScriptGenericChecked; | |
| 2543 ScriptFamilyState() : isGenericAdded(false), isPerScriptGenericChecked(false) | |
| 2544 {} | |
| 2545 }; | |
| 2546 | |
| 2547 inline static void handleScriptFamily(const char* webkitFamily, UScriptCode scri
pt, | |
| 2548 FontDescription::GenericFamilyType generic
, | |
| 2549 AtomicString& face, ScriptFamilyState& sta
te, FontDescription& fontDescription, | |
| 2550 int& familyIndex) | |
| 2551 { | |
| 2552 if (!state.isGenericAdded) { | |
| 2553 face = webkitFamily; | |
| 2554 state.isGenericAdded = true; | |
| 2555 fontDescription.setGenericFamily(generic); | |
| 2556 // go through this once more to add per-script generic family. | |
| 2557 --familyIndex; | |
| 2558 } else if (!state.isPerScriptGenericChecked) { | |
| 2559 face = FontCache::getGenericFontForScript(script, fontDescription); | |
| 2560 state.isPerScriptGenericChecked = true; | |
| 2561 } | |
| 2562 } | |
| 2563 | |
| 2564 void CSSStyleSelector::applyPropertyToStyle(int id, CSSValue *value, RenderStyle
* style) | |
| 2565 { | |
| 2566 initElementAndPseudoState(0); | |
| 2567 initForStyleResolve(0, style); | |
| 2568 m_style = style; | |
| 2569 applyProperty(id, value); | |
| 2570 } | |
| 2571 | |
| 2572 void CSSStyleSelector::applyProperty(int id, CSSValue *value) | |
| 2573 { | |
| 2574 CSSPrimitiveValue* primitiveValue = 0; | |
| 2575 if (value->isPrimitiveValue()) | |
| 2576 primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 2577 | |
| 2578 float zoomFactor = m_style->effectiveZoom(); | |
| 2579 | |
| 2580 Length l; | |
| 2581 bool apply = false; | |
| 2582 | |
| 2583 unsigned short valueType = value->cssValueType(); | |
| 2584 | |
| 2585 bool isInherit = m_parentNode && valueType == CSSValue::CSS_INHERIT; | |
| 2586 bool isInitial = valueType == CSSValue::CSS_INITIAL || (!m_parentNode && val
ueType == CSSValue::CSS_INHERIT); | |
| 2587 | |
| 2588 // These properties are used to set the correct margins/padding on RTL lists
. | |
| 2589 if (id == CSSPropertyWebkitMarginStart) | |
| 2590 id = m_style->direction() == LTR ? CSSPropertyMarginLeft : CSSPropertyMa
rginRight; | |
| 2591 else if (id == CSSPropertyWebkitPaddingStart) | |
| 2592 id = m_style->direction() == LTR ? CSSPropertyPaddingLeft : CSSPropertyP
addingRight; | |
| 2593 | |
| 2594 // What follows is a list that maps the CSS properties into their correspond
ing front-end | |
| 2595 // RenderStyle values. Shorthands (e.g. border, background) occur in this l
ist as well and | |
| 2596 // are only hit when mapping "inherit" or "initial" into front-end values. | |
| 2597 switch (static_cast<CSSPropertyID>(id)) { | |
| 2598 // ident only properties | |
| 2599 case CSSPropertyBackgroundAttachment: | |
| 2600 HANDLE_BACKGROUND_VALUE(attachment, Attachment, value) | |
| 2601 return; | |
| 2602 case CSSPropertyWebkitBackgroundClip: | |
| 2603 HANDLE_BACKGROUND_VALUE(clip, Clip, value) | |
| 2604 return; | |
| 2605 case CSSPropertyWebkitBackgroundComposite: | |
| 2606 HANDLE_BACKGROUND_VALUE(composite, Composite, value) | |
| 2607 return; | |
| 2608 case CSSPropertyWebkitBackgroundOrigin: | |
| 2609 HANDLE_BACKGROUND_VALUE(origin, Origin, value) | |
| 2610 return; | |
| 2611 case CSSPropertyBackgroundRepeat: | |
| 2612 HANDLE_BACKGROUND_VALUE(repeat, Repeat, value) | |
| 2613 return; | |
| 2614 case CSSPropertyWebkitBackgroundSize: | |
| 2615 HANDLE_BACKGROUND_VALUE(size, Size, value) | |
| 2616 return; | |
| 2617 case CSSPropertyWebkitMaskAttachment: | |
| 2618 HANDLE_MASK_VALUE(attachment, Attachment, value) | |
| 2619 return; | |
| 2620 case CSSPropertyWebkitMaskClip: | |
| 2621 HANDLE_MASK_VALUE(clip, Clip, value) | |
| 2622 return; | |
| 2623 case CSSPropertyWebkitMaskComposite: | |
| 2624 HANDLE_MASK_VALUE(composite, Composite, value) | |
| 2625 return; | |
| 2626 case CSSPropertyWebkitMaskOrigin: | |
| 2627 HANDLE_MASK_VALUE(origin, Origin, value) | |
| 2628 return; | |
| 2629 case CSSPropertyWebkitMaskRepeat: | |
| 2630 HANDLE_MASK_VALUE(repeat, Repeat, value) | |
| 2631 return; | |
| 2632 case CSSPropertyWebkitMaskSize: | |
| 2633 HANDLE_MASK_VALUE(size, Size, value) | |
| 2634 return; | |
| 2635 case CSSPropertyBorderCollapse: | |
| 2636 HANDLE_INHERIT_AND_INITIAL(borderCollapse, BorderCollapse) | |
| 2637 if (!primitiveValue) | |
| 2638 return; | |
| 2639 switch (primitiveValue->getIdent()) { | |
| 2640 case CSSValueCollapse: | |
| 2641 m_style->setBorderCollapse(true); | |
| 2642 break; | |
| 2643 case CSSValueSeparate: | |
| 2644 m_style->setBorderCollapse(false); | |
| 2645 break; | |
| 2646 default: | |
| 2647 return; | |
| 2648 } | |
| 2649 return; | |
| 2650 | |
| 2651 case CSSPropertyBorderTopStyle: | |
| 2652 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(borderTopStyle, BorderTopStyle, Bo
rderStyle) | |
| 2653 if (primitiveValue) | |
| 2654 m_style->setBorderTopStyle(*primitiveValue); | |
| 2655 return; | |
| 2656 case CSSPropertyBorderRightStyle: | |
| 2657 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(borderRightStyle, BorderRightStyle
, BorderStyle) | |
| 2658 if (primitiveValue) | |
| 2659 m_style->setBorderRightStyle(*primitiveValue); | |
| 2660 return; | |
| 2661 case CSSPropertyBorderBottomStyle: | |
| 2662 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(borderBottomStyle, BorderBottomSty
le, BorderStyle) | |
| 2663 if (primitiveValue) | |
| 2664 m_style->setBorderBottomStyle(*primitiveValue); | |
| 2665 return; | |
| 2666 case CSSPropertyBorderLeftStyle: | |
| 2667 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(borderLeftStyle, BorderLeftStyle,
BorderStyle) | |
| 2668 if (primitiveValue) | |
| 2669 m_style->setBorderLeftStyle(*primitiveValue); | |
| 2670 return; | |
| 2671 case CSSPropertyOutlineStyle: | |
| 2672 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(outlineStyle, OutlineStyle, Border
Style) | |
| 2673 if (primitiveValue) { | |
| 2674 if (primitiveValue->getIdent() == CSSValueAuto) | |
| 2675 m_style->setOutlineStyle(DOTTED, true); | |
| 2676 else | |
| 2677 m_style->setOutlineStyle(*primitiveValue); | |
| 2678 } | |
| 2679 return; | |
| 2680 case CSSPropertyCaptionSide: | |
| 2681 { | |
| 2682 HANDLE_INHERIT_AND_INITIAL(captionSide, CaptionSide) | |
| 2683 if (primitiveValue) | |
| 2684 m_style->setCaptionSide(*primitiveValue); | |
| 2685 return; | |
| 2686 } | |
| 2687 case CSSPropertyClear: | |
| 2688 { | |
| 2689 HANDLE_INHERIT_AND_INITIAL(clear, Clear) | |
| 2690 if (primitiveValue) | |
| 2691 m_style->setClear(*primitiveValue); | |
| 2692 return; | |
| 2693 } | |
| 2694 case CSSPropertyDirection: | |
| 2695 { | |
| 2696 HANDLE_INHERIT_AND_INITIAL(direction, Direction) | |
| 2697 if (primitiveValue) | |
| 2698 m_style->setDirection(*primitiveValue); | |
| 2699 return; | |
| 2700 } | |
| 2701 case CSSPropertyDisplay: | |
| 2702 { | |
| 2703 HANDLE_INHERIT_AND_INITIAL(display, Display) | |
| 2704 if (primitiveValue) | |
| 2705 m_style->setDisplay(*primitiveValue); | |
| 2706 return; | |
| 2707 } | |
| 2708 | |
| 2709 case CSSPropertyEmptyCells: | |
| 2710 { | |
| 2711 HANDLE_INHERIT_AND_INITIAL(emptyCells, EmptyCells) | |
| 2712 if (primitiveValue) | |
| 2713 m_style->setEmptyCells(*primitiveValue); | |
| 2714 return; | |
| 2715 } | |
| 2716 case CSSPropertyFloat: | |
| 2717 { | |
| 2718 HANDLE_INHERIT_AND_INITIAL(floating, Floating) | |
| 2719 if (primitiveValue) | |
| 2720 m_style->setFloating(*primitiveValue); | |
| 2721 return; | |
| 2722 } | |
| 2723 | |
| 2724 case CSSPropertyFontStyle: | |
| 2725 { | |
| 2726 FontDescription fontDescription = m_style->fontDescription(); | |
| 2727 if (isInherit) | |
| 2728 fontDescription.setItalic(m_parentStyle->fontDescription().italic())
; | |
| 2729 else if (isInitial) | |
| 2730 fontDescription.setItalic(false); | |
| 2731 else { | |
| 2732 if (!primitiveValue) | |
| 2733 return; | |
| 2734 switch (primitiveValue->getIdent()) { | |
| 2735 case CSSValueOblique: | |
| 2736 // FIXME: oblique is the same as italic for the moment... | |
| 2737 case CSSValueItalic: | |
| 2738 fontDescription.setItalic(true); | |
| 2739 break; | |
| 2740 case CSSValueNormal: | |
| 2741 fontDescription.setItalic(false); | |
| 2742 break; | |
| 2743 default: | |
| 2744 return; | |
| 2745 } | |
| 2746 } | |
| 2747 if (m_style->setFontDescription(fontDescription)) | |
| 2748 m_fontDirty = true; | |
| 2749 return; | |
| 2750 } | |
| 2751 | |
| 2752 case CSSPropertyFontVariant: | |
| 2753 { | |
| 2754 FontDescription fontDescription = m_style->fontDescription(); | |
| 2755 if (isInherit) | |
| 2756 fontDescription.setSmallCaps(m_parentStyle->fontDescription().smallC
aps()); | |
| 2757 else if (isInitial) | |
| 2758 fontDescription.setSmallCaps(false); | |
| 2759 else { | |
| 2760 if (!primitiveValue) | |
| 2761 return; | |
| 2762 int id = primitiveValue->getIdent(); | |
| 2763 if (id == CSSValueNormal) | |
| 2764 fontDescription.setSmallCaps(false); | |
| 2765 else if (id == CSSValueSmallCaps) | |
| 2766 fontDescription.setSmallCaps(true); | |
| 2767 else | |
| 2768 return; | |
| 2769 } | |
| 2770 if (m_style->setFontDescription(fontDescription)) | |
| 2771 m_fontDirty = true; | |
| 2772 return; | |
| 2773 } | |
| 2774 | |
| 2775 case CSSPropertyFontWeight: | |
| 2776 { | |
| 2777 FontDescription fontDescription = m_style->fontDescription(); | |
| 2778 if (isInherit) | |
| 2779 fontDescription.setWeight(m_parentStyle->fontDescription().weight())
; | |
| 2780 else if (isInitial) | |
| 2781 fontDescription.setWeight(FontWeightNormal); | |
| 2782 else { | |
| 2783 if (!primitiveValue) | |
| 2784 return; | |
| 2785 if (primitiveValue->getIdent()) { | |
| 2786 switch (primitiveValue->getIdent()) { | |
| 2787 case CSSValueBolder: | |
| 2788 fontDescription.setWeight(fontDescription.bolderWeight()
); | |
| 2789 break; | |
| 2790 case CSSValueLighter: | |
| 2791 fontDescription.setWeight(fontDescription.lighterWeight(
)); | |
| 2792 break; | |
| 2793 case CSSValueBold: | |
| 2794 case CSSValue700: | |
| 2795 fontDescription.setWeight(FontWeightBold); | |
| 2796 break; | |
| 2797 case CSSValueNormal: | |
| 2798 case CSSValue400: | |
| 2799 fontDescription.setWeight(FontWeightNormal); | |
| 2800 break; | |
| 2801 case CSSValue900: | |
| 2802 fontDescription.setWeight(FontWeight900); | |
| 2803 break; | |
| 2804 case CSSValue800: | |
| 2805 fontDescription.setWeight(FontWeight800); | |
| 2806 break; | |
| 2807 case CSSValue600: | |
| 2808 fontDescription.setWeight(FontWeight600); | |
| 2809 break; | |
| 2810 case CSSValue500: | |
| 2811 fontDescription.setWeight(FontWeight500); | |
| 2812 break; | |
| 2813 case CSSValue300: | |
| 2814 fontDescription.setWeight(FontWeight300); | |
| 2815 break; | |
| 2816 case CSSValue200: | |
| 2817 fontDescription.setWeight(FontWeight200); | |
| 2818 break; | |
| 2819 case CSSValue100: | |
| 2820 fontDescription.setWeight(FontWeight100); | |
| 2821 break; | |
| 2822 default: | |
| 2823 return; | |
| 2824 } | |
| 2825 } else | |
| 2826 ASSERT_NOT_REACHED(); | |
| 2827 } | |
| 2828 if (m_style->setFontDescription(fontDescription)) | |
| 2829 m_fontDirty = true; | |
| 2830 return; | |
| 2831 } | |
| 2832 | |
| 2833 case CSSPropertyListStylePosition: | |
| 2834 { | |
| 2835 HANDLE_INHERIT_AND_INITIAL(listStylePosition, ListStylePosition) | |
| 2836 if (primitiveValue) | |
| 2837 m_style->setListStylePosition(*primitiveValue); | |
| 2838 return; | |
| 2839 } | |
| 2840 | |
| 2841 case CSSPropertyListStyleType: | |
| 2842 { | |
| 2843 HANDLE_INHERIT_AND_INITIAL(listStyleType, ListStyleType) | |
| 2844 if (primitiveValue) | |
| 2845 m_style->setListStyleType(*primitiveValue); | |
| 2846 return; | |
| 2847 } | |
| 2848 | |
| 2849 case CSSPropertyOverflow: | |
| 2850 { | |
| 2851 if (isInherit) { | |
| 2852 m_style->setOverflowX(m_parentStyle->overflowX()); | |
| 2853 m_style->setOverflowY(m_parentStyle->overflowY()); | |
| 2854 return; | |
| 2855 } | |
| 2856 | |
| 2857 if (isInitial) { | |
| 2858 m_style->setOverflowX(RenderStyle::initialOverflowX()); | |
| 2859 m_style->setOverflowY(RenderStyle::initialOverflowY()); | |
| 2860 return; | |
| 2861 } | |
| 2862 | |
| 2863 EOverflow o = *primitiveValue; | |
| 2864 | |
| 2865 m_style->setOverflowX(o); | |
| 2866 m_style->setOverflowY(o); | |
| 2867 return; | |
| 2868 } | |
| 2869 | |
| 2870 case CSSPropertyOverflowX: | |
| 2871 { | |
| 2872 HANDLE_INHERIT_AND_INITIAL(overflowX, OverflowX) | |
| 2873 m_style->setOverflowX(*primitiveValue); | |
| 2874 return; | |
| 2875 } | |
| 2876 | |
| 2877 case CSSPropertyOverflowY: | |
| 2878 { | |
| 2879 HANDLE_INHERIT_AND_INITIAL(overflowY, OverflowY) | |
| 2880 m_style->setOverflowY(*primitiveValue); | |
| 2881 return; | |
| 2882 } | |
| 2883 | |
| 2884 case CSSPropertyPageBreakBefore: | |
| 2885 { | |
| 2886 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(pageBreakBefore, PageBreakBefore,
PageBreak) | |
| 2887 if (primitiveValue) | |
| 2888 m_style->setPageBreakBefore(*primitiveValue); | |
| 2889 return; | |
| 2890 } | |
| 2891 | |
| 2892 case CSSPropertyPageBreakAfter: | |
| 2893 { | |
| 2894 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(pageBreakAfter, PageBreakAfter, Pa
geBreak) | |
| 2895 if (primitiveValue) | |
| 2896 m_style->setPageBreakAfter(*primitiveValue); | |
| 2897 return; | |
| 2898 } | |
| 2899 | |
| 2900 case CSSPropertyPageBreakInside: { | |
| 2901 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(pageBreakInside, PageBreakInside,
PageBreak) | |
| 2902 if (!primitiveValue) | |
| 2903 return; | |
| 2904 EPageBreak pageBreak = *primitiveValue; | |
| 2905 if (pageBreak != PBALWAYS) | |
| 2906 m_style->setPageBreakInside(pageBreak); | |
| 2907 return; | |
| 2908 } | |
| 2909 | |
| 2910 case CSSPropertyPosition: | |
| 2911 { | |
| 2912 HANDLE_INHERIT_AND_INITIAL(position, Position) | |
| 2913 if (primitiveValue) | |
| 2914 m_style->setPosition(*primitiveValue); | |
| 2915 return; | |
| 2916 } | |
| 2917 | |
| 2918 case CSSPropertyTableLayout: { | |
| 2919 HANDLE_INHERIT_AND_INITIAL(tableLayout, TableLayout) | |
| 2920 | |
| 2921 ETableLayout l = *primitiveValue; | |
| 2922 if (l == TAUTO) | |
| 2923 l = RenderStyle::initialTableLayout(); | |
| 2924 | |
| 2925 m_style->setTableLayout(l); | |
| 2926 return; | |
| 2927 } | |
| 2928 | |
| 2929 case CSSPropertyUnicodeBidi: { | |
| 2930 HANDLE_INHERIT_AND_INITIAL(unicodeBidi, UnicodeBidi) | |
| 2931 m_style->setUnicodeBidi(*primitiveValue); | |
| 2932 return; | |
| 2933 } | |
| 2934 case CSSPropertyTextTransform: { | |
| 2935 HANDLE_INHERIT_AND_INITIAL(textTransform, TextTransform) | |
| 2936 m_style->setTextTransform(*primitiveValue); | |
| 2937 return; | |
| 2938 } | |
| 2939 | |
| 2940 case CSSPropertyVisibility: | |
| 2941 { | |
| 2942 HANDLE_INHERIT_AND_INITIAL(visibility, Visibility) | |
| 2943 m_style->setVisibility(*primitiveValue); | |
| 2944 return; | |
| 2945 } | |
| 2946 case CSSPropertyWhiteSpace: | |
| 2947 HANDLE_INHERIT_AND_INITIAL(whiteSpace, WhiteSpace) | |
| 2948 m_style->setWhiteSpace(*primitiveValue); | |
| 2949 return; | |
| 2950 | |
| 2951 case CSSPropertyBackgroundPosition: | |
| 2952 HANDLE_BACKGROUND_INHERIT_AND_INITIAL(xPosition, XPosition); | |
| 2953 HANDLE_BACKGROUND_INHERIT_AND_INITIAL(yPosition, YPosition); | |
| 2954 return; | |
| 2955 case CSSPropertyBackgroundPositionX: { | |
| 2956 HANDLE_BACKGROUND_VALUE(xPosition, XPosition, value) | |
| 2957 return; | |
| 2958 } | |
| 2959 case CSSPropertyBackgroundPositionY: { | |
| 2960 HANDLE_BACKGROUND_VALUE(yPosition, YPosition, value) | |
| 2961 return; | |
| 2962 } | |
| 2963 case CSSPropertyWebkitMaskPosition: | |
| 2964 HANDLE_MASK_INHERIT_AND_INITIAL(xPosition, XPosition); | |
| 2965 HANDLE_MASK_INHERIT_AND_INITIAL(yPosition, YPosition); | |
| 2966 return; | |
| 2967 case CSSPropertyWebkitMaskPositionX: { | |
| 2968 HANDLE_MASK_VALUE(xPosition, XPosition, value) | |
| 2969 return; | |
| 2970 } | |
| 2971 case CSSPropertyWebkitMaskPositionY: { | |
| 2972 HANDLE_MASK_VALUE(yPosition, YPosition, value) | |
| 2973 return; | |
| 2974 } | |
| 2975 case CSSPropertyBorderSpacing: { | |
| 2976 if (isInherit) { | |
| 2977 m_style->setHorizontalBorderSpacing(m_parentStyle->horizontalBorderS
pacing()); | |
| 2978 m_style->setVerticalBorderSpacing(m_parentStyle->verticalBorderSpaci
ng()); | |
| 2979 } | |
| 2980 else if (isInitial) { | |
| 2981 m_style->setHorizontalBorderSpacing(0); | |
| 2982 m_style->setVerticalBorderSpacing(0); | |
| 2983 } | |
| 2984 return; | |
| 2985 } | |
| 2986 case CSSPropertyWebkitBorderHorizontalSpacing: { | |
| 2987 HANDLE_INHERIT_AND_INITIAL(horizontalBorderSpacing, HorizontalBorderSpac
ing) | |
| 2988 if (!primitiveValue) | |
| 2989 return; | |
| 2990 short spacing = primitiveValue->computeLengthShort(m_style, zoomFactor)
; | |
| 2991 m_style->setHorizontalBorderSpacing(spacing); | |
| 2992 return; | |
| 2993 } | |
| 2994 case CSSPropertyWebkitBorderVerticalSpacing: { | |
| 2995 HANDLE_INHERIT_AND_INITIAL(verticalBorderSpacing, VerticalBorderSpacing) | |
| 2996 if (!primitiveValue) | |
| 2997 return; | |
| 2998 short spacing = primitiveValue->computeLengthShort(m_style, zoomFactor)
; | |
| 2999 m_style->setVerticalBorderSpacing(spacing); | |
| 3000 return; | |
| 3001 } | |
| 3002 case CSSPropertyCursor: | |
| 3003 if (isInherit) { | |
| 3004 m_style->setCursor(m_parentStyle->cursor()); | |
| 3005 m_style->setCursorList(m_parentStyle->cursors()); | |
| 3006 return; | |
| 3007 } | |
| 3008 m_style->clearCursorList(); | |
| 3009 if (isInitial) { | |
| 3010 m_style->setCursor(RenderStyle::initialCursor()); | |
| 3011 return; | |
| 3012 } | |
| 3013 if (value->isValueList()) { | |
| 3014 CSSValueList* list = static_cast<CSSValueList*>(value); | |
| 3015 int len = list->length(); | |
| 3016 m_style->setCursor(CURSOR_AUTO); | |
| 3017 for (int i = 0; i < len; i++) { | |
| 3018 CSSValue* item = list->itemWithoutBoundsCheck(i); | |
| 3019 if (!item->isPrimitiveValue()) | |
| 3020 continue; | |
| 3021 primitiveValue = static_cast<CSSPrimitiveValue*>(item); | |
| 3022 int type = primitiveValue->primitiveType(); | |
| 3023 if (type == CSSPrimitiveValue::CSS_URI) { | |
| 3024 CSSCursorImageValue* image = static_cast<CSSCursorImageValue
*>(primitiveValue); | |
| 3025 if (image->updateIfSVGCursorIsUsed(m_element)) // Elements w
ith SVG cursors are not allowed to share style. | |
| 3026 m_style->setUnique(); | |
| 3027 // FIXME: Temporary clumsiness to pass off a CachedImage to
an API that will eventually convert to using | |
| 3028 // StyleImage. | |
| 3029 RefPtr<StyleCachedImage> styleCachedImage(image->cachedImage
(m_element->document()->docLoader())); | |
| 3030 if (styleCachedImage) | |
| 3031 m_style->addCursor(styleCachedImage->cachedImage(), imag
e->hotspot()); | |
| 3032 } else if (type == CSSPrimitiveValue::CSS_IDENT) | |
| 3033 m_style->setCursor(*primitiveValue); | |
| 3034 } | |
| 3035 } else if (primitiveValue) { | |
| 3036 int type = primitiveValue->primitiveType(); | |
| 3037 if (type == CSSPrimitiveValue::CSS_IDENT) | |
| 3038 m_style->setCursor(*primitiveValue); | |
| 3039 } | |
| 3040 return; | |
| 3041 // colors || inherit | |
| 3042 case CSSPropertyColor: | |
| 3043 // If the 'currentColor' keyword is set on the 'color' property itself, | |
| 3044 // it is treated as 'color:inherit' at parse time | |
| 3045 if (primitiveValue && primitiveValue->getIdent() == CSSValueCurrentcolor
) | |
| 3046 isInherit = true; | |
| 3047 case CSSPropertyBackgroundColor: | |
| 3048 case CSSPropertyBorderTopColor: | |
| 3049 case CSSPropertyBorderRightColor: | |
| 3050 case CSSPropertyBorderBottomColor: | |
| 3051 case CSSPropertyBorderLeftColor: | |
| 3052 case CSSPropertyOutlineColor: | |
| 3053 case CSSPropertyWebkitColumnRuleColor: | |
| 3054 case CSSPropertyWebkitTextStrokeColor: | |
| 3055 case CSSPropertyWebkitTextFillColor: { | |
| 3056 Color col; | |
| 3057 if (isInherit) { | |
| 3058 HANDLE_INHERIT_COND(CSSPropertyBackgroundColor, backgroundColor, Bac
kgroundColor) | |
| 3059 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyBorderTopColor, borderTop
Color, color, BorderTopColor) | |
| 3060 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyBorderBottomColor, border
BottomColor, color, BorderBottomColor) | |
| 3061 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyBorderRightColor, borderR
ightColor, color, BorderRightColor) | |
| 3062 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyBorderLeftColor, borderLe
ftColor, color, BorderLeftColor) | |
| 3063 HANDLE_INHERIT_COND(CSSPropertyColor, color, Color) | |
| 3064 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyOutlineColor, outlineColo
r, color, OutlineColor) | |
| 3065 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyWebkitColumnRuleColor, co
lumnRuleColor, color, ColumnRuleColor) | |
| 3066 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyWebkitTextStrokeColor, te
xtStrokeColor, color, TextStrokeColor) | |
| 3067 HANDLE_INHERIT_COND_WITH_BACKUP(CSSPropertyWebkitTextFillColor, text
FillColor, color, TextFillColor) | |
| 3068 return; | |
| 3069 } | |
| 3070 if (isInitial) { | |
| 3071 // The border/outline colors will just map to the invalid color |col
| above. This will have the | |
| 3072 // effect of forcing the use of the currentColor when it comes time
to draw the borders (and of | |
| 3073 // not painting the background since the color won't be valid). | |
| 3074 if (id == CSSPropertyColor) | |
| 3075 col = RenderStyle::initialColor(); | |
| 3076 } else { | |
| 3077 if (!primitiveValue) | |
| 3078 return; | |
| 3079 col = getColorFromPrimitiveValue(primitiveValue); | |
| 3080 } | |
| 3081 | |
| 3082 switch (id) { | |
| 3083 case CSSPropertyBackgroundColor: | |
| 3084 m_style->setBackgroundColor(col); | |
| 3085 break; | |
| 3086 case CSSPropertyBorderTopColor: | |
| 3087 m_style->setBorderTopColor(col); | |
| 3088 break; | |
| 3089 case CSSPropertyBorderRightColor: | |
| 3090 m_style->setBorderRightColor(col); | |
| 3091 break; | |
| 3092 case CSSPropertyBorderBottomColor: | |
| 3093 m_style->setBorderBottomColor(col); | |
| 3094 break; | |
| 3095 case CSSPropertyBorderLeftColor: | |
| 3096 m_style->setBorderLeftColor(col); | |
| 3097 break; | |
| 3098 case CSSPropertyColor: | |
| 3099 m_style->setColor(col); | |
| 3100 break; | |
| 3101 case CSSPropertyOutlineColor: | |
| 3102 m_style->setOutlineColor(col); | |
| 3103 break; | |
| 3104 case CSSPropertyWebkitColumnRuleColor: | |
| 3105 m_style->setColumnRuleColor(col); | |
| 3106 break; | |
| 3107 case CSSPropertyWebkitTextStrokeColor: | |
| 3108 m_style->setTextStrokeColor(col); | |
| 3109 break; | |
| 3110 case CSSPropertyWebkitTextFillColor: | |
| 3111 m_style->setTextFillColor(col); | |
| 3112 break; | |
| 3113 } | |
| 3114 | |
| 3115 return; | |
| 3116 } | |
| 3117 | |
| 3118 // uri || inherit | |
| 3119 case CSSPropertyBackgroundImage: | |
| 3120 HANDLE_BACKGROUND_VALUE(image, Image, value) | |
| 3121 return; | |
| 3122 case CSSPropertyWebkitMaskImage: | |
| 3123 HANDLE_MASK_VALUE(image, Image, value) | |
| 3124 return; | |
| 3125 case CSSPropertyListStyleImage: | |
| 3126 { | |
| 3127 HANDLE_INHERIT_AND_INITIAL(listStyleImage, ListStyleImage) | |
| 3128 m_style->setListStyleImage(styleImage(value)); | |
| 3129 return; | |
| 3130 } | |
| 3131 | |
| 3132 // length | |
| 3133 case CSSPropertyBorderTopWidth: | |
| 3134 case CSSPropertyBorderRightWidth: | |
| 3135 case CSSPropertyBorderBottomWidth: | |
| 3136 case CSSPropertyBorderLeftWidth: | |
| 3137 case CSSPropertyOutlineWidth: | |
| 3138 case CSSPropertyWebkitColumnRuleWidth: | |
| 3139 { | |
| 3140 if (isInherit) { | |
| 3141 HANDLE_INHERIT_COND(CSSPropertyBorderTopWidth, borderTopWidth, Borde
rTopWidth) | |
| 3142 HANDLE_INHERIT_COND(CSSPropertyBorderRightWidth, borderRightWidth, B
orderRightWidth) | |
| 3143 HANDLE_INHERIT_COND(CSSPropertyBorderBottomWidth, borderBottomWidth,
BorderBottomWidth) | |
| 3144 HANDLE_INHERIT_COND(CSSPropertyBorderLeftWidth, borderLeftWidth, Bor
derLeftWidth) | |
| 3145 HANDLE_INHERIT_COND(CSSPropertyOutlineWidth, outlineWidth, OutlineWi
dth) | |
| 3146 HANDLE_INHERIT_COND(CSSPropertyWebkitColumnRuleWidth, columnRuleWidt
h, ColumnRuleWidth) | |
| 3147 return; | |
| 3148 } | |
| 3149 else if (isInitial) { | |
| 3150 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyBorderTopWidth, BorderTopW
idth, BorderWidth) | |
| 3151 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyBorderRightWidth, BorderRi
ghtWidth, BorderWidth) | |
| 3152 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyBorderBottomWidth, BorderB
ottomWidth, BorderWidth) | |
| 3153 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyBorderLeftWidth, BorderLef
tWidth, BorderWidth) | |
| 3154 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyOutlineWidth, OutlineWidth
, BorderWidth) | |
| 3155 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitColumnRuleWidth, Col
umnRuleWidth, BorderWidth) | |
| 3156 return; | |
| 3157 } | |
| 3158 | |
| 3159 if (!primitiveValue) | |
| 3160 return; | |
| 3161 short width = 3; | |
| 3162 switch (primitiveValue->getIdent()) { | |
| 3163 case CSSValueThin: | |
| 3164 width = 1; | |
| 3165 break; | |
| 3166 case CSSValueMedium: | |
| 3167 width = 3; | |
| 3168 break; | |
| 3169 case CSSValueThick: | |
| 3170 width = 5; | |
| 3171 break; | |
| 3172 case CSSValueInvalid: | |
| 3173 width = primitiveValue->computeLengthShort(m_style, zoomFactor); | |
| 3174 break; | |
| 3175 default: | |
| 3176 return; | |
| 3177 } | |
| 3178 | |
| 3179 if (width < 0) return; | |
| 3180 switch (id) { | |
| 3181 case CSSPropertyBorderTopWidth: | |
| 3182 m_style->setBorderTopWidth(width); | |
| 3183 break; | |
| 3184 case CSSPropertyBorderRightWidth: | |
| 3185 m_style->setBorderRightWidth(width); | |
| 3186 break; | |
| 3187 case CSSPropertyBorderBottomWidth: | |
| 3188 m_style->setBorderBottomWidth(width); | |
| 3189 break; | |
| 3190 case CSSPropertyBorderLeftWidth: | |
| 3191 m_style->setBorderLeftWidth(width); | |
| 3192 break; | |
| 3193 case CSSPropertyOutlineWidth: | |
| 3194 m_style->setOutlineWidth(width); | |
| 3195 break; | |
| 3196 case CSSPropertyWebkitColumnRuleWidth: | |
| 3197 m_style->setColumnRuleWidth(width); | |
| 3198 break; | |
| 3199 default: | |
| 3200 return; | |
| 3201 } | |
| 3202 return; | |
| 3203 } | |
| 3204 | |
| 3205 case CSSPropertyLetterSpacing: | |
| 3206 case CSSPropertyWordSpacing: | |
| 3207 { | |
| 3208 | |
| 3209 if (isInherit) { | |
| 3210 HANDLE_INHERIT_COND(CSSPropertyLetterSpacing, letterSpacing, LetterS
pacing) | |
| 3211 HANDLE_INHERIT_COND(CSSPropertyWordSpacing, wordSpacing, WordSpacing
) | |
| 3212 return; | |
| 3213 } | |
| 3214 else if (isInitial) { | |
| 3215 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyLetterSpacing, LetterSpaci
ng, LetterWordSpacing) | |
| 3216 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWordSpacing, WordSpacing,
LetterWordSpacing) | |
| 3217 return; | |
| 3218 } | |
| 3219 | |
| 3220 int width = 0; | |
| 3221 if (primitiveValue && primitiveValue->getIdent() == CSSValueNormal){ | |
| 3222 width = 0; | |
| 3223 } else { | |
| 3224 if (!primitiveValue) | |
| 3225 return; | |
| 3226 width = primitiveValue->computeLengthInt(m_style, zoomFactor); | |
| 3227 } | |
| 3228 switch (id) { | |
| 3229 case CSSPropertyLetterSpacing: | |
| 3230 m_style->setLetterSpacing(width); | |
| 3231 break; | |
| 3232 case CSSPropertyWordSpacing: | |
| 3233 m_style->setWordSpacing(width); | |
| 3234 break; | |
| 3235 // ### needs the definitions in renderstyle | |
| 3236 default: break; | |
| 3237 } | |
| 3238 return; | |
| 3239 } | |
| 3240 | |
| 3241 case CSSPropertyWordBreak: { | |
| 3242 HANDLE_INHERIT_AND_INITIAL(wordBreak, WordBreak) | |
| 3243 m_style->setWordBreak(*primitiveValue); | |
| 3244 return; | |
| 3245 } | |
| 3246 | |
| 3247 case CSSPropertyWordWrap: { | |
| 3248 HANDLE_INHERIT_AND_INITIAL(wordWrap, WordWrap) | |
| 3249 m_style->setWordWrap(*primitiveValue); | |
| 3250 return; | |
| 3251 } | |
| 3252 | |
| 3253 case CSSPropertyWebkitNbspMode: | |
| 3254 { | |
| 3255 HANDLE_INHERIT_AND_INITIAL(nbspMode, NBSPMode) | |
| 3256 m_style->setNBSPMode(*primitiveValue); | |
| 3257 return; | |
| 3258 } | |
| 3259 | |
| 3260 case CSSPropertyWebkitLineBreak: | |
| 3261 { | |
| 3262 HANDLE_INHERIT_AND_INITIAL(khtmlLineBreak, KHTMLLineBreak) | |
| 3263 m_style->setKHTMLLineBreak(*primitiveValue); | |
| 3264 return; | |
| 3265 } | |
| 3266 | |
| 3267 case CSSPropertyWebkitMatchNearestMailBlockquoteColor: | |
| 3268 { | |
| 3269 HANDLE_INHERIT_AND_INITIAL(matchNearestMailBlockquoteColor, MatchNearest
MailBlockquoteColor) | |
| 3270 m_style->setMatchNearestMailBlockquoteColor(*primitiveValue); | |
| 3271 return; | |
| 3272 } | |
| 3273 | |
| 3274 case CSSPropertyResize: | |
| 3275 { | |
| 3276 HANDLE_INHERIT_AND_INITIAL(resize, Resize) | |
| 3277 | |
| 3278 if (!primitiveValue->getIdent()) | |
| 3279 return; | |
| 3280 | |
| 3281 EResize r = RESIZE_NONE; | |
| 3282 if (primitiveValue->getIdent() == CSSValueAuto) { | |
| 3283 if (Settings* settings = m_checker.m_document->settings()) | |
| 3284 r = settings->textAreasAreResizable() ? RESIZE_BOTH : RESIZE_NON
E; | |
| 3285 } else | |
| 3286 r = *primitiveValue; | |
| 3287 | |
| 3288 m_style->setResize(r); | |
| 3289 return; | |
| 3290 } | |
| 3291 | |
| 3292 // length, percent | |
| 3293 case CSSPropertyMaxWidth: | |
| 3294 // +none +inherit | |
| 3295 if (primitiveValue && primitiveValue->getIdent() == CSSValueNone) | |
| 3296 apply = true; | |
| 3297 case CSSPropertyTop: | |
| 3298 case CSSPropertyLeft: | |
| 3299 case CSSPropertyRight: | |
| 3300 case CSSPropertyBottom: | |
| 3301 case CSSPropertyWidth: | |
| 3302 case CSSPropertyMinWidth: | |
| 3303 case CSSPropertyMarginTop: | |
| 3304 case CSSPropertyMarginRight: | |
| 3305 case CSSPropertyMarginBottom: | |
| 3306 case CSSPropertyMarginLeft: | |
| 3307 // +inherit +auto | |
| 3308 if (id == CSSPropertyWidth || id == CSSPropertyMinWidth || id == CSSProp
ertyMaxWidth) { | |
| 3309 if (primitiveValue && primitiveValue->getIdent() == CSSValueIntrinsi
c) { | |
| 3310 l = Length(Intrinsic); | |
| 3311 apply = true; | |
| 3312 } | |
| 3313 else if (primitiveValue && primitiveValue->getIdent() == CSSValueMin
Intrinsic) { | |
| 3314 l = Length(MinIntrinsic); | |
| 3315 apply = true; | |
| 3316 } | |
| 3317 } | |
| 3318 if (id != CSSPropertyMaxWidth && primitiveValue && primitiveValue->getId
ent() == CSSValueAuto) | |
| 3319 apply = true; | |
| 3320 case CSSPropertyPaddingTop: | |
| 3321 case CSSPropertyPaddingRight: | |
| 3322 case CSSPropertyPaddingBottom: | |
| 3323 case CSSPropertyPaddingLeft: | |
| 3324 case CSSPropertyTextIndent: | |
| 3325 // +inherit | |
| 3326 { | |
| 3327 if (isInherit) { | |
| 3328 HANDLE_INHERIT_COND(CSSPropertyMaxWidth, maxWidth, MaxWidth) | |
| 3329 HANDLE_INHERIT_COND(CSSPropertyBottom, bottom, Bottom) | |
| 3330 HANDLE_INHERIT_COND(CSSPropertyTop, top, Top) | |
| 3331 HANDLE_INHERIT_COND(CSSPropertyLeft, left, Left) | |
| 3332 HANDLE_INHERIT_COND(CSSPropertyRight, right, Right) | |
| 3333 HANDLE_INHERIT_COND(CSSPropertyWidth, width, Width) | |
| 3334 HANDLE_INHERIT_COND(CSSPropertyMinWidth, minWidth, MinWidth) | |
| 3335 HANDLE_INHERIT_COND(CSSPropertyPaddingTop, paddingTop, PaddingTop) | |
| 3336 HANDLE_INHERIT_COND(CSSPropertyPaddingRight, paddingRight, PaddingRi
ght) | |
| 3337 HANDLE_INHERIT_COND(CSSPropertyPaddingBottom, paddingBottom, Padding
Bottom) | |
| 3338 HANDLE_INHERIT_COND(CSSPropertyPaddingLeft, paddingLeft, PaddingLeft
) | |
| 3339 HANDLE_INHERIT_COND(CSSPropertyMarginTop, marginTop, MarginTop) | |
| 3340 HANDLE_INHERIT_COND(CSSPropertyMarginRight, marginRight, MarginRight
) | |
| 3341 HANDLE_INHERIT_COND(CSSPropertyMarginBottom, marginBottom, MarginBot
tom) | |
| 3342 HANDLE_INHERIT_COND(CSSPropertyMarginLeft, marginLeft, MarginLeft) | |
| 3343 HANDLE_INHERIT_COND(CSSPropertyTextIndent, textIndent, TextIndent) | |
| 3344 return; | |
| 3345 } | |
| 3346 else if (isInitial) { | |
| 3347 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMaxWidth, MaxWidth, MaxSiz
e) | |
| 3348 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyBottom, Bottom, Offset) | |
| 3349 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyTop, Top, Offset) | |
| 3350 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyLeft, Left, Offset) | |
| 3351 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyRight, Right, Offset) | |
| 3352 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWidth, Width, Size) | |
| 3353 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMinWidth, MinWidth, MinSiz
e) | |
| 3354 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyPaddingTop, PaddingTop, Pa
dding) | |
| 3355 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyPaddingRight, PaddingRight
, Padding) | |
| 3356 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyPaddingBottom, PaddingBott
om, Padding) | |
| 3357 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyPaddingLeft, PaddingLeft,
Padding) | |
| 3358 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMarginTop, MarginTop, Marg
in) | |
| 3359 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMarginRight, MarginRight,
Margin) | |
| 3360 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMarginBottom, MarginBottom
, Margin) | |
| 3361 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMarginLeft, MarginLeft, Ma
rgin) | |
| 3362 HANDLE_INITIAL_COND(CSSPropertyTextIndent, TextIndent) | |
| 3363 return; | |
| 3364 } | |
| 3365 | |
| 3366 if (primitiveValue && !apply) { | |
| 3367 int type = primitiveValue->primitiveType(); | |
| 3368 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveV
alue::CSS_DEG) | |
| 3369 // Handle our quirky margin units if we have them. | |
| 3370 l = Length(primitiveValue->computeLengthIntForLength(m_style, zo
omFactor), Fixed, | |
| 3371 primitiveValue->isQuirkValue()); | |
| 3372 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 3373 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 3374 else | |
| 3375 return; | |
| 3376 if (id == CSSPropertyPaddingLeft || id == CSSPropertyPaddingRight || | |
| 3377 id == CSSPropertyPaddingTop || id == CSSPropertyPaddingBottom) | |
| 3378 // Padding can't be negative | |
| 3379 apply = !((l.isFixed() || l.isPercent()) && l.calcValue(100) < 0
); | |
| 3380 else | |
| 3381 apply = true; | |
| 3382 } | |
| 3383 if (!apply) return; | |
| 3384 switch (id) { | |
| 3385 case CSSPropertyMaxWidth: | |
| 3386 m_style->setMaxWidth(l); | |
| 3387 break; | |
| 3388 case CSSPropertyBottom: | |
| 3389 m_style->setBottom(l); | |
| 3390 break; | |
| 3391 case CSSPropertyTop: | |
| 3392 m_style->setTop(l); | |
| 3393 break; | |
| 3394 case CSSPropertyLeft: | |
| 3395 m_style->setLeft(l); | |
| 3396 break; | |
| 3397 case CSSPropertyRight: | |
| 3398 m_style->setRight(l); | |
| 3399 break; | |
| 3400 case CSSPropertyWidth: | |
| 3401 m_style->setWidth(l); | |
| 3402 break; | |
| 3403 case CSSPropertyMinWidth: | |
| 3404 m_style->setMinWidth(l); | |
| 3405 break; | |
| 3406 case CSSPropertyPaddingTop: | |
| 3407 m_style->setPaddingTop(l); | |
| 3408 break; | |
| 3409 case CSSPropertyPaddingRight: | |
| 3410 m_style->setPaddingRight(l); | |
| 3411 break; | |
| 3412 case CSSPropertyPaddingBottom: | |
| 3413 m_style->setPaddingBottom(l); | |
| 3414 break; | |
| 3415 case CSSPropertyPaddingLeft: | |
| 3416 m_style->setPaddingLeft(l); | |
| 3417 break; | |
| 3418 case CSSPropertyMarginTop: | |
| 3419 m_style->setMarginTop(l); | |
| 3420 break; | |
| 3421 case CSSPropertyMarginRight: | |
| 3422 m_style->setMarginRight(l); | |
| 3423 break; | |
| 3424 case CSSPropertyMarginBottom: | |
| 3425 m_style->setMarginBottom(l); | |
| 3426 break; | |
| 3427 case CSSPropertyMarginLeft: | |
| 3428 m_style->setMarginLeft(l); | |
| 3429 break; | |
| 3430 case CSSPropertyTextIndent: | |
| 3431 m_style->setTextIndent(l); | |
| 3432 break; | |
| 3433 default: | |
| 3434 break; | |
| 3435 } | |
| 3436 return; | |
| 3437 } | |
| 3438 | |
| 3439 case CSSPropertyMaxHeight: | |
| 3440 if (primitiveValue && primitiveValue->getIdent() == CSSValueNone) { | |
| 3441 l = Length(undefinedLength, Fixed); | |
| 3442 apply = true; | |
| 3443 } | |
| 3444 case CSSPropertyHeight: | |
| 3445 case CSSPropertyMinHeight: | |
| 3446 if (primitiveValue && primitiveValue->getIdent() == CSSValueIntrinsic) { | |
| 3447 l = Length(Intrinsic); | |
| 3448 apply = true; | |
| 3449 } else if (primitiveValue && primitiveValue->getIdent() == CSSValueMinIn
trinsic) { | |
| 3450 l = Length(MinIntrinsic); | |
| 3451 apply = true; | |
| 3452 } else if (id != CSSPropertyMaxHeight && primitiveValue && primitiveValu
e->getIdent() == CSSValueAuto) | |
| 3453 apply = true; | |
| 3454 if (isInherit) { | |
| 3455 HANDLE_INHERIT_COND(CSSPropertyMaxHeight, maxHeight, MaxHeight) | |
| 3456 HANDLE_INHERIT_COND(CSSPropertyHeight, height, Height) | |
| 3457 HANDLE_INHERIT_COND(CSSPropertyMinHeight, minHeight, MinHeight) | |
| 3458 return; | |
| 3459 } | |
| 3460 if (isInitial) { | |
| 3461 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMaxHeight, MaxHeight, MaxS
ize) | |
| 3462 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyHeight, Height, Size) | |
| 3463 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyMinHeight, MinHeight, MinS
ize) | |
| 3464 return; | |
| 3465 } | |
| 3466 | |
| 3467 if (primitiveValue && !apply) { | |
| 3468 unsigned short type = primitiveValue->primitiveType(); | |
| 3469 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveV
alue::CSS_DEG) | |
| 3470 l = Length(primitiveValue->computeLengthIntForLength(m_style, zo
omFactor), Fixed); | |
| 3471 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 3472 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 3473 else | |
| 3474 return; | |
| 3475 apply = true; | |
| 3476 } | |
| 3477 if (apply) | |
| 3478 switch (id) { | |
| 3479 case CSSPropertyMaxHeight: | |
| 3480 m_style->setMaxHeight(l); | |
| 3481 break; | |
| 3482 case CSSPropertyHeight: | |
| 3483 m_style->setHeight(l); | |
| 3484 break; | |
| 3485 case CSSPropertyMinHeight: | |
| 3486 m_style->setMinHeight(l); | |
| 3487 break; | |
| 3488 } | |
| 3489 return; | |
| 3490 | |
| 3491 case CSSPropertyVerticalAlign: | |
| 3492 HANDLE_INHERIT_AND_INITIAL(verticalAlign, VerticalAlign) | |
| 3493 if (!primitiveValue) | |
| 3494 return; | |
| 3495 if (primitiveValue->getIdent()) { | |
| 3496 EVerticalAlign align; | |
| 3497 | |
| 3498 switch (primitiveValue->getIdent()) { | |
| 3499 case CSSValueTop: | |
| 3500 align = TOP; break; | |
| 3501 case CSSValueBottom: | |
| 3502 align = BOTTOM; break; | |
| 3503 case CSSValueMiddle: | |
| 3504 align = MIDDLE; break; | |
| 3505 case CSSValueBaseline: | |
| 3506 align = BASELINE; break; | |
| 3507 case CSSValueTextBottom: | |
| 3508 align = TEXT_BOTTOM; break; | |
| 3509 case CSSValueTextTop: | |
| 3510 align = TEXT_TOP; break; | |
| 3511 case CSSValueSub: | |
| 3512 align = SUB; break; | |
| 3513 case CSSValueSuper: | |
| 3514 align = SUPER; break; | |
| 3515 case CSSValueWebkitBaselineMiddle: | |
| 3516 align = BASELINE_MIDDLE; break; | |
| 3517 default: | |
| 3518 return; | |
| 3519 } | |
| 3520 m_style->setVerticalAlign(align); | |
| 3521 return; | |
| 3522 } else { | |
| 3523 int type = primitiveValue->primitiveType(); | |
| 3524 Length l; | |
| 3525 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveVal
ue::CSS_DEG) | |
| 3526 l = Length(primitiveValue->computeLengthIntForLength(m_style, zoomFa
ctor), Fixed); | |
| 3527 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 3528 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 3529 | |
| 3530 m_style->setVerticalAlign(LENGTH); | |
| 3531 m_style->setVerticalAlignLength(l); | |
| 3532 } | |
| 3533 return; | |
| 3534 | |
| 3535 case CSSPropertyFontSize: | |
| 3536 { | |
| 3537 FontDescription fontDescription = m_style->fontDescription(); | |
| 3538 fontDescription.setKeywordSize(0); | |
| 3539 bool familyIsFixed = fontDescription.genericFamily() == FontDescription:
:MonospaceFamily; | |
| 3540 float oldSize = 0; | |
| 3541 float size = 0; | |
| 3542 | |
| 3543 bool parentIsAbsoluteSize = false; | |
| 3544 if (m_parentNode) { | |
| 3545 oldSize = m_parentStyle->fontDescription().specifiedSize(); | |
| 3546 parentIsAbsoluteSize = m_parentStyle->fontDescription().isAbsoluteSi
ze(); | |
| 3547 } | |
| 3548 | |
| 3549 if (isInherit) { | |
| 3550 size = oldSize; | |
| 3551 if (m_parentNode) | |
| 3552 fontDescription.setKeywordSize(m_parentStyle->fontDescription().
keywordSize()); | |
| 3553 } else if (isInitial) { | |
| 3554 size = fontSizeForKeyword(CSSValueMedium, m_style->htmlHacks(), fami
lyIsFixed); | |
| 3555 fontDescription.setKeywordSize(CSSValueMedium - CSSValueXxSmall + 1)
; | |
| 3556 } else if (primitiveValue->getIdent()) { | |
| 3557 // Keywords are being used. | |
| 3558 switch (primitiveValue->getIdent()) { | |
| 3559 case CSSValueXxSmall: | |
| 3560 case CSSValueXSmall: | |
| 3561 case CSSValueSmall: | |
| 3562 case CSSValueMedium: | |
| 3563 case CSSValueLarge: | |
| 3564 case CSSValueXLarge: | |
| 3565 case CSSValueXxLarge: | |
| 3566 case CSSValueWebkitXxxLarge: | |
| 3567 size = fontSizeForKeyword(primitiveValue->getIdent(), m_styl
e->htmlHacks(), familyIsFixed); | |
| 3568 fontDescription.setKeywordSize(primitiveValue->getIdent() -
CSSValueXxSmall + 1); | |
| 3569 break; | |
| 3570 case CSSValueLarger: | |
| 3571 size = largerFontSize(oldSize, m_style->htmlHacks()); | |
| 3572 break; | |
| 3573 case CSSValueSmaller: | |
| 3574 size = smallerFontSize(oldSize, m_style->htmlHacks()); | |
| 3575 break; | |
| 3576 default: | |
| 3577 return; | |
| 3578 } | |
| 3579 | |
| 3580 fontDescription.setIsAbsoluteSize(parentIsAbsoluteSize && | |
| 3581 (primitiveValue->getIdent() == CSS
ValueLarger || | |
| 3582 primitiveValue->getIdent() == CSS
ValueSmaller)); | |
| 3583 } else { | |
| 3584 int type = primitiveValue->primitiveType(); | |
| 3585 fontDescription.setIsAbsoluteSize(parentIsAbsoluteSize || | |
| 3586 (type != CSSPrimitiveValue::CSS_PE
RCENTAGE && | |
| 3587 type != CSSPrimitiveValue::CSS_EM
S && | |
| 3588 type != CSSPrimitiveValue::CSS_EX
S)); | |
| 3589 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveV
alue::CSS_DEG) | |
| 3590 size = primitiveValue->computeLengthFloat(m_parentStyle, true); | |
| 3591 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 3592 size = (primitiveValue->getFloatValue() * oldSize) / 100.0f; | |
| 3593 else | |
| 3594 return; | |
| 3595 } | |
| 3596 | |
| 3597 if (size < 0) | |
| 3598 return; | |
| 3599 | |
| 3600 setFontSize(fontDescription, size); | |
| 3601 if (m_style->setFontDescription(fontDescription)) | |
| 3602 m_fontDirty = true; | |
| 3603 return; | |
| 3604 } | |
| 3605 | |
| 3606 case CSSPropertyZIndex: { | |
| 3607 if (isInherit) { | |
| 3608 if (m_parentStyle->hasAutoZIndex()) | |
| 3609 m_style->setHasAutoZIndex(); | |
| 3610 else | |
| 3611 m_style->setZIndex(m_parentStyle->zIndex()); | |
| 3612 return; | |
| 3613 } else if (isInitial || primitiveValue->getIdent() == CSSValueAuto) { | |
| 3614 m_style->setHasAutoZIndex(); | |
| 3615 return; | |
| 3616 } | |
| 3617 | |
| 3618 // FIXME: Should clamp all sorts of other integer properties too. | |
| 3619 const double minIntAsDouble = INT_MIN; | |
| 3620 const double maxIntAsDouble = INT_MAX; | |
| 3621 m_style->setZIndex(static_cast<int>(max(minIntAsDouble, min(primitiveVal
ue->getDoubleValue(), maxIntAsDouble)))); | |
| 3622 return; | |
| 3623 } | |
| 3624 case CSSPropertyWidows: | |
| 3625 { | |
| 3626 HANDLE_INHERIT_AND_INITIAL(widows, Widows) | |
| 3627 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 3628 return; | |
| 3629 m_style->setWidows(primitiveValue->getIntValue()); | |
| 3630 return; | |
| 3631 } | |
| 3632 | |
| 3633 case CSSPropertyOrphans: | |
| 3634 { | |
| 3635 HANDLE_INHERIT_AND_INITIAL(orphans, Orphans) | |
| 3636 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 3637 return; | |
| 3638 m_style->setOrphans(primitiveValue->getIntValue()); | |
| 3639 return; | |
| 3640 } | |
| 3641 | |
| 3642 // length, percent, number | |
| 3643 case CSSPropertyLineHeight: | |
| 3644 { | |
| 3645 HANDLE_INHERIT_AND_INITIAL(lineHeight, LineHeight) | |
| 3646 if (!primitiveValue) | |
| 3647 return; | |
| 3648 Length lineHeight; | |
| 3649 int type = primitiveValue->primitiveType(); | |
| 3650 if (primitiveValue->getIdent() == CSSValueNormal) | |
| 3651 lineHeight = Length(-100.0, Percent); | |
| 3652 else if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitive
Value::CSS_DEG) { | |
| 3653 double multiplier = m_style->effectiveZoom(); | |
| 3654 if (m_style->textSizeAdjust() && m_checker.m_document->frame() && m_
checker.m_document->frame()->shouldApplyTextZoom()) | |
| 3655 multiplier *= m_checker.m_document->frame()->textZoomFactor(); | |
| 3656 lineHeight = Length(primitiveValue->computeLengthIntForLength(m_styl
e, multiplier), Fixed); | |
| 3657 } else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 3658 lineHeight = Length((m_style->fontSize() * primitiveValue->getIntVal
ue()) / 100, Fixed); | |
| 3659 else if (type == CSSPrimitiveValue::CSS_NUMBER) | |
| 3660 lineHeight = Length(primitiveValue->getDoubleValue() * 100.0, Percen
t); | |
| 3661 else | |
| 3662 return; | |
| 3663 m_style->setLineHeight(lineHeight); | |
| 3664 return; | |
| 3665 } | |
| 3666 | |
| 3667 // string | |
| 3668 case CSSPropertyTextAlign: | |
| 3669 { | |
| 3670 HANDLE_INHERIT_AND_INITIAL(textAlign, TextAlign) | |
| 3671 if (!primitiveValue) | |
| 3672 return; | |
| 3673 int id = primitiveValue->getIdent(); | |
| 3674 if (id == CSSValueStart) | |
| 3675 m_style->setTextAlign(m_style->direction() == LTR ? LEFT : RIGHT); | |
| 3676 else if (id == CSSValueEnd) | |
| 3677 m_style->setTextAlign(m_style->direction() == LTR ? RIGHT : LEFT); | |
| 3678 else | |
| 3679 m_style->setTextAlign(*primitiveValue); | |
| 3680 return; | |
| 3681 } | |
| 3682 | |
| 3683 // rect | |
| 3684 case CSSPropertyClip: | |
| 3685 { | |
| 3686 Length top; | |
| 3687 Length right; | |
| 3688 Length bottom; | |
| 3689 Length left; | |
| 3690 bool hasClip = true; | |
| 3691 if (isInherit) { | |
| 3692 if (m_parentStyle->hasClip()) { | |
| 3693 top = m_parentStyle->clipTop(); | |
| 3694 right = m_parentStyle->clipRight(); | |
| 3695 bottom = m_parentStyle->clipBottom(); | |
| 3696 left = m_parentStyle->clipLeft(); | |
| 3697 } | |
| 3698 else { | |
| 3699 hasClip = false; | |
| 3700 top = right = bottom = left = Length(); | |
| 3701 } | |
| 3702 } else if (isInitial) { | |
| 3703 hasClip = false; | |
| 3704 top = right = bottom = left = Length(); | |
| 3705 } else if (!primitiveValue) { | |
| 3706 return; | |
| 3707 } else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_REC
T) { | |
| 3708 Rect* rect = primitiveValue->getRectValue(); | |
| 3709 if (!rect) | |
| 3710 return; | |
| 3711 top = convertToLength(rect->top(), m_style); | |
| 3712 right = convertToLength(rect->right(), m_style); | |
| 3713 bottom = convertToLength(rect->bottom(), m_style); | |
| 3714 left = convertToLength(rect->left(), m_style); | |
| 3715 | |
| 3716 } else if (primitiveValue->getIdent() != CSSValueAuto) { | |
| 3717 return; | |
| 3718 } | |
| 3719 m_style->setClip(top, right, bottom, left); | |
| 3720 m_style->setHasClip(hasClip); | |
| 3721 | |
| 3722 // rect, ident | |
| 3723 return; | |
| 3724 } | |
| 3725 | |
| 3726 // lists | |
| 3727 case CSSPropertyContent: | |
| 3728 // list of string, uri, counter, attr, i | |
| 3729 { | |
| 3730 // FIXME: In CSS3, it will be possible to inherit content. In CSS2 it i
s not. This | |
| 3731 // note is a reminder that eventually "inherit" needs to be supported. | |
| 3732 | |
| 3733 if (isInitial) { | |
| 3734 m_style->clearContent(); | |
| 3735 return; | |
| 3736 } | |
| 3737 | |
| 3738 if (!value->isValueList()) | |
| 3739 return; | |
| 3740 | |
| 3741 CSSValueList* list = static_cast<CSSValueList*>(value); | |
| 3742 int len = list->length(); | |
| 3743 | |
| 3744 bool didSet = false; | |
| 3745 for (int i = 0; i < len; i++) { | |
| 3746 CSSValue* item = list->itemWithoutBoundsCheck(i); | |
| 3747 if (item->isImageGeneratorValue()) { | |
| 3748 m_style->setContent(static_cast<CSSImageGeneratorValue*>(item)->
generatedImage(), didSet); | |
| 3749 didSet = true; | |
| 3750 } | |
| 3751 | |
| 3752 if (!item->isPrimitiveValue()) | |
| 3753 continue; | |
| 3754 | |
| 3755 CSSPrimitiveValue* val = static_cast<CSSPrimitiveValue*>(item); | |
| 3756 switch (val->primitiveType()) { | |
| 3757 case CSSPrimitiveValue::CSS_STRING: | |
| 3758 m_style->setContent(val->getStringValue().impl(), didSet); | |
| 3759 didSet = true; | |
| 3760 break; | |
| 3761 case CSSPrimitiveValue::CSS_ATTR: { | |
| 3762 // FIXME: Can a namespace be specified for an attr(foo)? | |
| 3763 if (m_style->styleType() == RenderStyle::NOPSEUDO) | |
| 3764 m_style->setUnique(); | |
| 3765 else | |
| 3766 m_parentStyle->setUnique(); | |
| 3767 QualifiedName attr(nullAtom, val->getStringValue().impl(), n
ullAtom); | |
| 3768 m_style->setContent(m_element->getAttribute(attr).impl(), di
dSet); | |
| 3769 didSet = true; | |
| 3770 // register the fact that the attribute value affects the st
yle | |
| 3771 m_selectorAttrs.add(attr.localName().impl()); | |
| 3772 break; | |
| 3773 } | |
| 3774 case CSSPrimitiveValue::CSS_URI: { | |
| 3775 CSSImageValue* image = static_cast<CSSImageValue*>(val); | |
| 3776 m_style->setContent(image->cachedImage(m_element->document()
->docLoader()), didSet); | |
| 3777 didSet = true; | |
| 3778 break; | |
| 3779 } | |
| 3780 case CSSPrimitiveValue::CSS_COUNTER: { | |
| 3781 Counter* counterValue = val->getCounterValue(); | |
| 3782 CounterContent* counter = new CounterContent(counterValue->i
dentifier(), | |
| 3783 (EListStyleType)counterValue->listStyleNumber(), counter
Value->separator()); | |
| 3784 m_style->setContent(counter, didSet); | |
| 3785 didSet = true; | |
| 3786 } | |
| 3787 } | |
| 3788 } | |
| 3789 if (!didSet) | |
| 3790 m_style->clearContent(); | |
| 3791 return; | |
| 3792 } | |
| 3793 | |
| 3794 case CSSPropertyCounterIncrement: | |
| 3795 applyCounterList(m_style, value->isValueList() ? static_cast<CSSValueLis
t*>(value) : 0, false); | |
| 3796 return; | |
| 3797 case CSSPropertyCounterReset: | |
| 3798 applyCounterList(m_style, value->isValueList() ? static_cast<CSSValueLis
t*>(value) : 0, true); | |
| 3799 return; | |
| 3800 | |
| 3801 case CSSPropertyFontFamily: { | |
| 3802 // list of strings and ids | |
| 3803 if (isInherit) { | |
| 3804 FontDescription parentFontDescription = m_parentStyle->fontDescripti
on(); | |
| 3805 FontDescription fontDescription = m_style->fontDescription(); | |
| 3806 fontDescription.setGenericFamily(parentFontDescription.genericFamily
()); | |
| 3807 fontDescription.setFamily(parentFontDescription.firstFamily()); | |
| 3808 if (m_style->setFontDescription(fontDescription)) | |
| 3809 m_fontDirty = true; | |
| 3810 return; | |
| 3811 } | |
| 3812 else if (isInitial) { | |
| 3813 FontDescription initialDesc = FontDescription(); | |
| 3814 FontDescription fontDescription = m_style->fontDescription(); | |
| 3815 // We need to adjust the size to account for the generic family chan
ge from monospace | |
| 3816 // to non-monospace. | |
| 3817 if (fontDescription.keywordSize() && fontDescription.genericFamily()
== FontDescription::MonospaceFamily) | |
| 3818 setFontSize(fontDescription, fontSizeForKeyword(CSSValueXxSmall
+ fontDescription.keywordSize() - 1, m_style->htmlHacks(), false)); | |
| 3819 fontDescription.setGenericFamily(initialDesc.genericFamily()); | |
| 3820 if (!initialDesc.firstFamily().familyIsEmpty()) | |
| 3821 fontDescription.setFamily(initialDesc.firstFamily()); | |
| 3822 if (m_style->setFontDescription(fontDescription)) | |
| 3823 m_fontDirty = true; | |
| 3824 return; | |
| 3825 } | |
| 3826 | |
| 3827 if (!value->isValueList()) return; | |
| 3828 FontDescription fontDescription = m_style->fontDescription(); | |
| 3829 CSSValueList *list = static_cast<CSSValueList*>(value); | |
| 3830 int len = list->length(); | |
| 3831 FontFamily& firstFamily = fontDescription.firstFamily(); | |
| 3832 FontFamily *currFamily = 0; | |
| 3833 | |
| 3834 // Before mapping in a new font-family property, we should reset the gen
eric family. | |
| 3835 bool oldFamilyIsMonospace = fontDescription.genericFamily() == FontDescr
iption::MonospaceFamily; | |
| 3836 fontDescription.setGenericFamily(FontDescription::NoFamily); | |
| 3837 | |
| 3838 // |script| is used to add a font per script and per CSS generic family.
| |
| 3839 // Adding it here is not very efficient because we may never use it | |
| 3840 // if all the characters are covered by fonts specified for this element
. | |
| 3841 // TODO(jungshik): Currently, it's document-wide constant inferred from | |
| 3842 // the document charset, but we should infer it from the value of | |
| 3843 // xml:lang or lang for |m_element|. | |
| 3844 UScriptCode script = m_checker.m_document->dominantScript(); | |
| 3845 // serif, sans-serif, cursive, fantasy, monospace | |
| 3846 ScriptFamilyState scriptFamilyStates[5]; | |
| 3847 Settings* settings = m_checker.m_document->settings(); | |
| 3848 for (int i = 0; i < len; i++) { | |
| 3849 CSSValue *item = list->itemWithoutBoundsCheck(i); | |
| 3850 if (!item->isPrimitiveValue()) continue; | |
| 3851 CSSPrimitiveValue *val = static_cast<CSSPrimitiveValue*>(item); | |
| 3852 AtomicString face; | |
| 3853 if (val->primitiveType() == CSSPrimitiveValue::CSS_STRING) | |
| 3854 face = static_cast<FontFamilyValue*>(val)->familyName(); | |
| 3855 else if (val->primitiveType() == CSSPrimitiveValue::CSS_IDENT && set
tings) { | |
| 3856 switch (val->getIdent()) { | |
| 3857 case CSSValueWebkitBody: | |
| 3858 face = settings->standardFontFamily(); | |
| 3859 break; | |
| 3860 // For each of 5 CSS generic families, | |
| 3861 // we add '-webkit-FOO' and a per-script generic family. | |
| 3862 // When |Settings| becomes expressive enough to support | |
| 3863 // per-script&per-generic family and we have a UI for | |
| 3864 // that, we'd just add the latter. Even without that, | |
| 3865 // I'm tempted to add per-script generic first, but I can't. | |
| 3866 // If I did, our font-selection UI would be all but | |
| 3867 // non-functional. Another issue is that we're adding | |
| 3868 // these fonts without regard for actual need in page | |
| 3869 // rendering. That is, it's not done in a lazy manner. | |
| 3870 // Somewhere in getGlyphDataForCharacter() could be | |
| 3871 // a better place in terms of performance. | |
| 3872 // See https://bugs.webkit.org/show_bug.cgi?id=18085 | |
| 3873 // and http://bugs.webkit.org/show_bug.cgi?id=10874 | |
| 3874 case CSSValueSerif: | |
| 3875 handleScriptFamily("-webkit-serif", script, | |
| 3876 FontDescription::SerifFamily, face, | |
| 3877 scriptFamilyStates[0], fontDescription, i); | |
| 3878 break; | |
| 3879 case CSSValueSansSerif: | |
| 3880 handleScriptFamily("-webkit-sans-serif", script, | |
| 3881 FontDescription::SansSerifFamily, face, | |
| 3882 scriptFamilyStates[1], fontDescription, i); | |
| 3883 break; | |
| 3884 case CSSValueCursive: | |
| 3885 handleScriptFamily("-webkit-cursive", script, | |
| 3886 FontDescription::CursiveFamily, face, | |
| 3887 scriptFamilyStates[2], fontDescription, i); | |
| 3888 break; | |
| 3889 case CSSValueFantasy: | |
| 3890 handleScriptFamily("-webkit-fantasy", script, | |
| 3891 FontDescription::FantasyFamily, face, | |
| 3892 scriptFamilyStates[3], fontDescription, i); | |
| 3893 break; | |
| 3894 case CSSValueMonospace: | |
| 3895 handleScriptFamily("-webkit-monospace", script, | |
| 3896 FontDescription::MonospaceFamily, face, | |
| 3897 scriptFamilyStates[4], fontDescription, i); | |
| 3898 break; | |
| 3899 } | |
| 3900 } | |
| 3901 | |
| 3902 if (!face.isEmpty()) { | |
| 3903 if (!currFamily) { | |
| 3904 // Filling in the first family. | |
| 3905 firstFamily.setFamily(face); | |
| 3906 currFamily = &firstFamily; | |
| 3907 } | |
| 3908 else { | |
| 3909 RefPtr<SharedFontFamily> newFamily = SharedFontFamily::creat
e(); | |
| 3910 newFamily->setFamily(face); | |
| 3911 currFamily->appendFamily(newFamily); | |
| 3912 currFamily = newFamily.get(); | |
| 3913 } | |
| 3914 | |
| 3915 if (fontDescription.keywordSize() && (fontDescription.genericFam
ily() == FontDescription::MonospaceFamily) != oldFamilyIsMonospace) | |
| 3916 setFontSize(fontDescription, fontSizeForKeyword(CSSValueXxSm
all + fontDescription.keywordSize() - 1, m_style->htmlHacks(), !oldFamilyIsMonos
pace)); | |
| 3917 | |
| 3918 if (m_style->setFontDescription(fontDescription)) | |
| 3919 m_fontDirty = true; | |
| 3920 } | |
| 3921 } | |
| 3922 | |
| 3923 if (fontDescription.genericFamily() == FontDescription::NoFamily && curr
Family) { | |
| 3924 FontDescription::GenericFamilyType generic; | |
| 3925 // TODO(jungshik) : Perhaps, we'd better add isStandardSerif() | |
| 3926 // method to |Settings| which will be set via WebPreference. | |
| 3927 if (settings) { | |
| 3928 if (settings->serifFontFamily() == settings->standardFontFamily(
)) | |
| 3929 generic = FontDescription::SerifFamily ; | |
| 3930 else | |
| 3931 generic = FontDescription::SansSerifFamily; | |
| 3932 } else | |
| 3933 generic = FontDescription::StandardFamily; | |
| 3934 fontDescription.setGenericFamily(generic); | |
| 3935 AtomicString face = FontCache::getGenericFontForScript(script, fontD
escription); | |
| 3936 if (!face.isEmpty()) { | |
| 3937 RefPtr<SharedFontFamily> newFamily = SharedFontFamily::create(); | |
| 3938 newFamily->setFamily(face); | |
| 3939 currFamily->appendFamily(newFamily); | |
| 3940 currFamily = newFamily.get(); | |
| 3941 if (m_style->setFontDescription(fontDescription)) | |
| 3942 m_fontDirty = true; | |
| 3943 } | |
| 3944 } | |
| 3945 return; | |
| 3946 } | |
| 3947 case CSSPropertyTextDecoration: { | |
| 3948 // list of ident | |
| 3949 HANDLE_INHERIT_AND_INITIAL(textDecoration, TextDecoration) | |
| 3950 int t = RenderStyle::initialTextDecoration(); | |
| 3951 if (primitiveValue && primitiveValue->getIdent() == CSSValueNone) { | |
| 3952 // do nothing | |
| 3953 } else { | |
| 3954 if (!value->isValueList()) return; | |
| 3955 CSSValueList *list = static_cast<CSSValueList*>(value); | |
| 3956 int len = list->length(); | |
| 3957 for (int i = 0; i < len; i++) | |
| 3958 { | |
| 3959 CSSValue *item = list->itemWithoutBoundsCheck(i); | |
| 3960 if (!item->isPrimitiveValue()) continue; | |
| 3961 primitiveValue = static_cast<CSSPrimitiveValue*>(item); | |
| 3962 switch (primitiveValue->getIdent()) { | |
| 3963 case CSSValueNone: | |
| 3964 t = TDNONE; break; | |
| 3965 case CSSValueUnderline: | |
| 3966 t |= UNDERLINE; break; | |
| 3967 case CSSValueOverline: | |
| 3968 t |= OVERLINE; break; | |
| 3969 case CSSValueLineThrough: | |
| 3970 t |= LINE_THROUGH; break; | |
| 3971 case CSSValueBlink: | |
| 3972 t |= BLINK; break; | |
| 3973 default: | |
| 3974 return; | |
| 3975 } | |
| 3976 } | |
| 3977 } | |
| 3978 | |
| 3979 m_style->setTextDecoration(t); | |
| 3980 return; | |
| 3981 } | |
| 3982 | |
| 3983 case CSSPropertyZoom: | |
| 3984 { | |
| 3985 // Reset the zoom in effect before we do anything. This allows the setZ
oom method to accurately compute a new | |
| 3986 // zoom in effect. | |
| 3987 m_style->setEffectiveZoom(m_parentStyle ? m_parentStyle->effectiveZoom()
: RenderStyle::initialZoom()); | |
| 3988 | |
| 3989 // Now we can handle inherit and initial. | |
| 3990 HANDLE_INHERIT_AND_INITIAL(zoom, Zoom) | |
| 3991 | |
| 3992 // Handle normal/reset, numbers and percentages. | |
| 3993 int type = primitiveValue->primitiveType(); | |
| 3994 if (primitiveValue->getIdent() == CSSValueNormal) | |
| 3995 m_style->setZoom(RenderStyle::initialZoom()); | |
| 3996 else if (primitiveValue->getIdent() == CSSValueReset) { | |
| 3997 m_style->setEffectiveZoom(RenderStyle::initialZoom()); | |
| 3998 m_style->setZoom(RenderStyle::initialZoom()); | |
| 3999 } else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) { | |
| 4000 if (primitiveValue->getFloatValue()) | |
| 4001 m_style->setZoom(primitiveValue->getFloatValue() / 100.0f); | |
| 4002 } else if (type == CSSPrimitiveValue::CSS_NUMBER) { | |
| 4003 if (primitiveValue->getFloatValue()) | |
| 4004 m_style->setZoom(primitiveValue->getFloatValue()); | |
| 4005 } | |
| 4006 | |
| 4007 m_fontDirty = true; | |
| 4008 return; | |
| 4009 } | |
| 4010 // shorthand properties | |
| 4011 case CSSPropertyBackground: | |
| 4012 if (isInitial) { | |
| 4013 m_style->clearBackgroundLayers(); | |
| 4014 m_style->setBackgroundColor(Color()); | |
| 4015 } | |
| 4016 else if (isInherit) { | |
| 4017 m_style->inheritBackgroundLayers(*m_parentStyle->backgroundLayers())
; | |
| 4018 m_style->setBackgroundColor(m_parentStyle->backgroundColor()); | |
| 4019 } | |
| 4020 return; | |
| 4021 case CSSPropertyWebkitMask: | |
| 4022 if (isInitial) | |
| 4023 m_style->clearMaskLayers(); | |
| 4024 else if (isInherit) | |
| 4025 m_style->inheritMaskLayers(*m_parentStyle->maskLayers()); | |
| 4026 return; | |
| 4027 | |
| 4028 case CSSPropertyBorder: | |
| 4029 case CSSPropertyBorderStyle: | |
| 4030 case CSSPropertyBorderWidth: | |
| 4031 case CSSPropertyBorderColor: | |
| 4032 if (id == CSSPropertyBorder || id == CSSPropertyBorderColor) | |
| 4033 { | |
| 4034 if (isInherit) { | |
| 4035 m_style->setBorderTopColor(m_parentStyle->borderTopColor().isVal
id() ? m_parentStyle->borderTopColor() : m_parentStyle->color()); | |
| 4036 m_style->setBorderBottomColor(m_parentStyle->borderBottomColor()
.isValid() ? m_parentStyle->borderBottomColor() : m_parentStyle->color()); | |
| 4037 m_style->setBorderLeftColor(m_parentStyle->borderLeftColor().isV
alid() ? m_parentStyle->borderLeftColor() : m_parentStyle->color()); | |
| 4038 m_style->setBorderRightColor(m_parentStyle->borderRightColor().i
sValid() ? m_parentStyle->borderRightColor(): m_parentStyle->color()); | |
| 4039 } | |
| 4040 else if (isInitial) { | |
| 4041 m_style->setBorderTopColor(Color()); // Reset to invalid color s
o currentColor is used instead. | |
| 4042 m_style->setBorderBottomColor(Color()); | |
| 4043 m_style->setBorderLeftColor(Color()); | |
| 4044 m_style->setBorderRightColor(Color()); | |
| 4045 } | |
| 4046 } | |
| 4047 if (id == CSSPropertyBorder || id == CSSPropertyBorderStyle) | |
| 4048 { | |
| 4049 if (isInherit) { | |
| 4050 m_style->setBorderTopStyle(m_parentStyle->borderTopStyle()); | |
| 4051 m_style->setBorderBottomStyle(m_parentStyle->borderBottomStyle()
); | |
| 4052 m_style->setBorderLeftStyle(m_parentStyle->borderLeftStyle()); | |
| 4053 m_style->setBorderRightStyle(m_parentStyle->borderRightStyle()); | |
| 4054 } | |
| 4055 else if (isInitial) { | |
| 4056 m_style->setBorderTopStyle(RenderStyle::initialBorderStyle()); | |
| 4057 m_style->setBorderBottomStyle(RenderStyle::initialBorderStyle())
; | |
| 4058 m_style->setBorderLeftStyle(RenderStyle::initialBorderStyle()); | |
| 4059 m_style->setBorderRightStyle(RenderStyle::initialBorderStyle()); | |
| 4060 } | |
| 4061 } | |
| 4062 if (id == CSSPropertyBorder || id == CSSPropertyBorderWidth) | |
| 4063 { | |
| 4064 if (isInherit) { | |
| 4065 m_style->setBorderTopWidth(m_parentStyle->borderTopWidth()); | |
| 4066 m_style->setBorderBottomWidth(m_parentStyle->borderBottomWidth()
); | |
| 4067 m_style->setBorderLeftWidth(m_parentStyle->borderLeftWidth()); | |
| 4068 m_style->setBorderRightWidth(m_parentStyle->borderRightWidth()); | |
| 4069 } | |
| 4070 else if (isInitial) { | |
| 4071 m_style->setBorderTopWidth(RenderStyle::initialBorderWidth()); | |
| 4072 m_style->setBorderBottomWidth(RenderStyle::initialBorderWidth())
; | |
| 4073 m_style->setBorderLeftWidth(RenderStyle::initialBorderWidth()); | |
| 4074 m_style->setBorderRightWidth(RenderStyle::initialBorderWidth()); | |
| 4075 } | |
| 4076 } | |
| 4077 return; | |
| 4078 case CSSPropertyBorderTop: | |
| 4079 if (isInherit) { | |
| 4080 m_style->setBorderTopColor(m_parentStyle->borderTopColor().isValid()
? m_parentStyle->borderTopColor() : m_parentStyle->color()); | |
| 4081 m_style->setBorderTopStyle(m_parentStyle->borderTopStyle()); | |
| 4082 m_style->setBorderTopWidth(m_parentStyle->borderTopWidth()); | |
| 4083 } | |
| 4084 else if (isInitial) | |
| 4085 m_style->resetBorderTop(); | |
| 4086 return; | |
| 4087 case CSSPropertyBorderRight: | |
| 4088 if (isInherit) { | |
| 4089 m_style->setBorderRightColor(m_parentStyle->borderRightColor().isVal
id() ? m_parentStyle->borderRightColor() : m_parentStyle->color()); | |
| 4090 m_style->setBorderRightStyle(m_parentStyle->borderRightStyle()); | |
| 4091 m_style->setBorderRightWidth(m_parentStyle->borderRightWidth()); | |
| 4092 } | |
| 4093 else if (isInitial) | |
| 4094 m_style->resetBorderRight(); | |
| 4095 return; | |
| 4096 case CSSPropertyBorderBottom: | |
| 4097 if (isInherit) { | |
| 4098 m_style->setBorderBottomColor(m_parentStyle->borderBottomColor().isV
alid() ? m_parentStyle->borderBottomColor() : m_parentStyle->color()); | |
| 4099 m_style->setBorderBottomStyle(m_parentStyle->borderBottomStyle()); | |
| 4100 m_style->setBorderBottomWidth(m_parentStyle->borderBottomWidth()); | |
| 4101 } | |
| 4102 else if (isInitial) | |
| 4103 m_style->resetBorderBottom(); | |
| 4104 return; | |
| 4105 case CSSPropertyBorderLeft: | |
| 4106 if (isInherit) { | |
| 4107 m_style->setBorderLeftColor(m_parentStyle->borderLeftColor().isValid
() ? m_parentStyle->borderLeftColor() : m_parentStyle->color()); | |
| 4108 m_style->setBorderLeftStyle(m_parentStyle->borderLeftStyle()); | |
| 4109 m_style->setBorderLeftWidth(m_parentStyle->borderLeftWidth()); | |
| 4110 } | |
| 4111 else if (isInitial) | |
| 4112 m_style->resetBorderLeft(); | |
| 4113 return; | |
| 4114 case CSSPropertyMargin: | |
| 4115 if (isInherit) { | |
| 4116 m_style->setMarginTop(m_parentStyle->marginTop()); | |
| 4117 m_style->setMarginBottom(m_parentStyle->marginBottom()); | |
| 4118 m_style->setMarginLeft(m_parentStyle->marginLeft()); | |
| 4119 m_style->setMarginRight(m_parentStyle->marginRight()); | |
| 4120 } | |
| 4121 else if (isInitial) | |
| 4122 m_style->resetMargin(); | |
| 4123 return; | |
| 4124 case CSSPropertyPadding: | |
| 4125 if (isInherit) { | |
| 4126 m_style->setPaddingTop(m_parentStyle->paddingTop()); | |
| 4127 m_style->setPaddingBottom(m_parentStyle->paddingBottom()); | |
| 4128 m_style->setPaddingLeft(m_parentStyle->paddingLeft()); | |
| 4129 m_style->setPaddingRight(m_parentStyle->paddingRight()); | |
| 4130 } | |
| 4131 else if (isInitial) | |
| 4132 m_style->resetPadding(); | |
| 4133 return; | |
| 4134 case CSSPropertyFont: | |
| 4135 if (isInherit) { | |
| 4136 FontDescription fontDescription = m_parentStyle->fontDescription(); | |
| 4137 m_style->setLineHeight(m_parentStyle->lineHeight()); | |
| 4138 m_lineHeightValue = 0; | |
| 4139 if (m_style->setFontDescription(fontDescription)) | |
| 4140 m_fontDirty = true; | |
| 4141 } else if (isInitial) { | |
| 4142 Settings* settings = m_checker.m_document->settings(); | |
| 4143 FontDescription fontDescription; | |
| 4144 fontDescription.setGenericFamily(FontDescription::StandardFamily); | |
| 4145 fontDescription.setRenderingMode(settings->fontRenderingMode()); | |
| 4146 fontDescription.setUsePrinterFont(m_checker.m_document->printing()); | |
| 4147 const AtomicString& standardFontFamily = m_checker.m_document->setti
ngs()->standardFontFamily(); | |
| 4148 if (!standardFontFamily.isEmpty()) { | |
| 4149 fontDescription.firstFamily().setFamily(standardFontFamily); | |
| 4150 fontDescription.firstFamily().appendFamily(0); | |
| 4151 } | |
| 4152 fontDescription.setKeywordSize(CSSValueMedium - CSSValueXxSmall + 1)
; | |
| 4153 setFontSize(fontDescription, fontSizeForKeyword(CSSValueMedium, m_st
yle->htmlHacks(), false)); | |
| 4154 m_style->setLineHeight(RenderStyle::initialLineHeight()); | |
| 4155 m_lineHeightValue = 0; | |
| 4156 if (m_style->setFontDescription(fontDescription)) | |
| 4157 m_fontDirty = true; | |
| 4158 } else if (primitiveValue) { | |
| 4159 m_style->setLineHeight(RenderStyle::initialLineHeight()); | |
| 4160 m_lineHeightValue = 0; | |
| 4161 FontDescription fontDescription; | |
| 4162 theme()->systemFont(primitiveValue->getIdent(), m_checker.m_document
, fontDescription); | |
| 4163 // Double-check and see if the theme did anything. If not, don't bo
ther updating the font. | |
| 4164 if (fontDescription.isAbsoluteSize()) { | |
| 4165 // Handle the zoom factor. | |
| 4166 fontDescription.setComputedSize(getComputedSizeFromSpecifiedSize
(fontDescription.isAbsoluteSize(), fontDescription.specifiedSize())); | |
| 4167 if (m_style->setFontDescription(fontDescription)) | |
| 4168 m_fontDirty = true; | |
| 4169 } | |
| 4170 } else if (value->isFontValue()) { | |
| 4171 FontValue *font = static_cast<FontValue*>(value); | |
| 4172 if (!font->style || !font->variant || !font->weight || | |
| 4173 !font->size || !font->lineHeight || !font->family) | |
| 4174 return; | |
| 4175 applyProperty(CSSPropertyFontStyle, font->style.get()); | |
| 4176 applyProperty(CSSPropertyFontVariant, font->variant.get()); | |
| 4177 applyProperty(CSSPropertyFontWeight, font->weight.get()); | |
| 4178 applyProperty(CSSPropertyFontSize, font->size.get()); | |
| 4179 | |
| 4180 m_lineHeightValue = font->lineHeight.get(); | |
| 4181 | |
| 4182 applyProperty(CSSPropertyFontFamily, font->family.get()); | |
| 4183 } | |
| 4184 return; | |
| 4185 | |
| 4186 case CSSPropertyListStyle: | |
| 4187 if (isInherit) { | |
| 4188 m_style->setListStyleType(m_parentStyle->listStyleType()); | |
| 4189 m_style->setListStyleImage(m_parentStyle->listStyleImage()); | |
| 4190 m_style->setListStylePosition(m_parentStyle->listStylePosition()); | |
| 4191 } | |
| 4192 else if (isInitial) { | |
| 4193 m_style->setListStyleType(RenderStyle::initialListStyleType()); | |
| 4194 m_style->setListStyleImage(RenderStyle::initialListStyleImage()); | |
| 4195 m_style->setListStylePosition(RenderStyle::initialListStylePosition(
)); | |
| 4196 } | |
| 4197 return; | |
| 4198 case CSSPropertyOutline: | |
| 4199 if (isInherit) { | |
| 4200 m_style->setOutlineWidth(m_parentStyle->outlineWidth()); | |
| 4201 m_style->setOutlineColor(m_parentStyle->outlineColor().isValid() ? m
_parentStyle->outlineColor() : m_parentStyle->color()); | |
| 4202 m_style->setOutlineStyle(m_parentStyle->outlineStyle()); | |
| 4203 } | |
| 4204 else if (isInitial) | |
| 4205 m_style->resetOutline(); | |
| 4206 return; | |
| 4207 | |
| 4208 // CSS3 Properties | |
| 4209 case CSSPropertyWebkitAppearance: { | |
| 4210 HANDLE_INHERIT_AND_INITIAL(appearance, Appearance) | |
| 4211 if (!primitiveValue) | |
| 4212 return; | |
| 4213 m_style->setAppearance(*primitiveValue); | |
| 4214 return; | |
| 4215 } | |
| 4216 case CSSPropertyWebkitBinding: { | |
| 4217 #if ENABLE(XBL) | |
| 4218 if (isInitial || (primitiveValue && primitiveValue->getIdent() == CSSVal
ueNone)) { | |
| 4219 m_style->deleteBindingURIs(); | |
| 4220 return; | |
| 4221 } | |
| 4222 else if (isInherit) { | |
| 4223 if (m_parentStyle->bindingURIs()) | |
| 4224 m_style->inheritBindingURIs(m_parentStyle->bindingURIs()); | |
| 4225 else | |
| 4226 m_style->deleteBindingURIs(); | |
| 4227 return; | |
| 4228 } | |
| 4229 | |
| 4230 if (!value->isValueList()) return; | |
| 4231 CSSValueList* list = static_cast<CSSValueList*>(value); | |
| 4232 bool firstBinding = true; | |
| 4233 for (unsigned int i = 0; i < list->length(); i++) { | |
| 4234 CSSValue *item = list->itemWithoutBoundsCheck(i); | |
| 4235 CSSPrimitiveValue *val = static_cast<CSSPrimitiveValue*>(item); | |
| 4236 if (val->primitiveType() == CSSPrimitiveValue::CSS_URI) { | |
| 4237 if (firstBinding) { | |
| 4238 firstBinding = false; | |
| 4239 m_style->deleteBindingURIs(); | |
| 4240 } | |
| 4241 m_style->addBindingURI(val->getStringValue()); | |
| 4242 } | |
| 4243 } | |
| 4244 #endif | |
| 4245 return; | |
| 4246 } | |
| 4247 | |
| 4248 case CSSPropertyWebkitBorderImage: | |
| 4249 case CSSPropertyWebkitMaskBoxImage: { | |
| 4250 if (isInherit) { | |
| 4251 HANDLE_INHERIT_COND(CSSPropertyWebkitBorderImage, borderImage, Borde
rImage) | |
| 4252 HANDLE_INHERIT_COND(CSSPropertyWebkitMaskBoxImage, maskBoxImage, Mas
kBoxImage) | |
| 4253 return; | |
| 4254 } else if (isInitial) { | |
| 4255 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitBorderImage, BorderI
mage, NinePieceImage) | |
| 4256 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitMaskBoxImage, MaskBo
xImage, NinePieceImage) | |
| 4257 return; | |
| 4258 } | |
| 4259 | |
| 4260 NinePieceImage image; | |
| 4261 mapNinePieceImage(value, image); | |
| 4262 | |
| 4263 if (id == CSSPropertyWebkitBorderImage) | |
| 4264 m_style->setBorderImage(image); | |
| 4265 else | |
| 4266 m_style->setMaskBoxImage(image); | |
| 4267 return; | |
| 4268 } | |
| 4269 | |
| 4270 case CSSPropertyWebkitBorderRadius: | |
| 4271 if (isInherit) { | |
| 4272 m_style->setBorderTopLeftRadius(m_parentStyle->borderTopLeftRadius()
); | |
| 4273 m_style->setBorderTopRightRadius(m_parentStyle->borderTopRightRadius
()); | |
| 4274 m_style->setBorderBottomLeftRadius(m_parentStyle->borderBottomLeftRa
dius()); | |
| 4275 m_style->setBorderBottomRightRadius(m_parentStyle->borderBottomRight
Radius()); | |
| 4276 return; | |
| 4277 } | |
| 4278 if (isInitial) { | |
| 4279 m_style->resetBorderRadius(); | |
| 4280 return; | |
| 4281 } | |
| 4282 // Fall through | |
| 4283 case CSSPropertyWebkitBorderTopLeftRadius: | |
| 4284 case CSSPropertyWebkitBorderTopRightRadius: | |
| 4285 case CSSPropertyWebkitBorderBottomLeftRadius: | |
| 4286 case CSSPropertyWebkitBorderBottomRightRadius: { | |
| 4287 if (isInherit) { | |
| 4288 HANDLE_INHERIT_COND(CSSPropertyWebkitBorderTopLeftRadius, borderTopL
eftRadius, BorderTopLeftRadius) | |
| 4289 HANDLE_INHERIT_COND(CSSPropertyWebkitBorderTopRightRadius, borderTop
RightRadius, BorderTopRightRadius) | |
| 4290 HANDLE_INHERIT_COND(CSSPropertyWebkitBorderBottomLeftRadius, borderB
ottomLeftRadius, BorderBottomLeftRadius) | |
| 4291 HANDLE_INHERIT_COND(CSSPropertyWebkitBorderBottomRightRadius, border
BottomRightRadius, BorderBottomRightRadius) | |
| 4292 return; | |
| 4293 } | |
| 4294 | |
| 4295 if (isInitial) { | |
| 4296 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitBorderTopLeftRadius,
BorderTopLeftRadius, BorderRadius) | |
| 4297 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitBorderTopRightRadius
, BorderTopRightRadius, BorderRadius) | |
| 4298 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitBorderBottomLeftRadi
us, BorderBottomLeftRadius, BorderRadius) | |
| 4299 HANDLE_INITIAL_COND_WITH_VALUE(CSSPropertyWebkitBorderBottomRightRad
ius, BorderBottomRightRadius, BorderRadius) | |
| 4300 return; | |
| 4301 } | |
| 4302 | |
| 4303 if (!primitiveValue) | |
| 4304 return; | |
| 4305 | |
| 4306 Pair* pair = primitiveValue->getPairValue(); | |
| 4307 if (!pair) | |
| 4308 return; | |
| 4309 | |
| 4310 int width = pair->first()->computeLengthInt(m_style, zoomFactor); | |
| 4311 int height = pair->second()->computeLengthInt(m_style, zoomFactor); | |
| 4312 if (width < 0 || height < 0) | |
| 4313 return; | |
| 4314 | |
| 4315 if (width == 0) | |
| 4316 height = 0; // Null out the other value. | |
| 4317 else if (height == 0) | |
| 4318 width = 0; // Null out the other value. | |
| 4319 | |
| 4320 IntSize size(width, height); | |
| 4321 switch (id) { | |
| 4322 case CSSPropertyWebkitBorderTopLeftRadius: | |
| 4323 m_style->setBorderTopLeftRadius(size); | |
| 4324 break; | |
| 4325 case CSSPropertyWebkitBorderTopRightRadius: | |
| 4326 m_style->setBorderTopRightRadius(size); | |
| 4327 break; | |
| 4328 case CSSPropertyWebkitBorderBottomLeftRadius: | |
| 4329 m_style->setBorderBottomLeftRadius(size); | |
| 4330 break; | |
| 4331 case CSSPropertyWebkitBorderBottomRightRadius: | |
| 4332 m_style->setBorderBottomRightRadius(size); | |
| 4333 break; | |
| 4334 default: | |
| 4335 m_style->setBorderRadius(size); | |
| 4336 break; | |
| 4337 } | |
| 4338 return; | |
| 4339 } | |
| 4340 | |
| 4341 case CSSPropertyOutlineOffset: | |
| 4342 HANDLE_INHERIT_AND_INITIAL(outlineOffset, OutlineOffset) | |
| 4343 m_style->setOutlineOffset(primitiveValue->computeLengthInt(m_style, zoom
Factor)); | |
| 4344 return; | |
| 4345 | |
| 4346 case CSSPropertyTextShadow: | |
| 4347 case CSSPropertyWebkitBoxShadow: { | |
| 4348 if (isInherit) { | |
| 4349 if (id == CSSPropertyTextShadow) | |
| 4350 return m_style->setTextShadow(m_parentStyle->textShadow() ? new
ShadowData(*m_parentStyle->textShadow()) : 0); | |
| 4351 return m_style->setBoxShadow(m_parentStyle->boxShadow() ? new Shadow
Data(*m_parentStyle->boxShadow()) : 0); | |
| 4352 } | |
| 4353 if (isInitial || primitiveValue) // initial | none | |
| 4354 return id == CSSPropertyTextShadow ? m_style->setTextShadow(0) : m_s
tyle->setBoxShadow(0); | |
| 4355 | |
| 4356 if (!value->isValueList()) | |
| 4357 return; | |
| 4358 | |
| 4359 CSSValueList *list = static_cast<CSSValueList*>(value); | |
| 4360 int len = list->length(); | |
| 4361 for (int i = 0; i < len; i++) { | |
| 4362 ShadowValue* item = static_cast<ShadowValue*>(list->itemWithoutBound
sCheck(i)); | |
| 4363 int x = item->x->computeLengthInt(m_style, zoomFactor); | |
| 4364 int y = item->y->computeLengthInt(m_style, zoomFactor); | |
| 4365 int blur = item->blur ? item->blur->computeLengthInt(m_style, zoomFa
ctor) : 0; | |
| 4366 Color color; | |
| 4367 if (item->color) | |
| 4368 color = getColorFromPrimitiveValue(item->color.get()); | |
| 4369 ShadowData* shadowData = new ShadowData(x, y, blur, color.isValid()
? color : Color::transparent); | |
| 4370 if (id == CSSPropertyTextShadow) | |
| 4371 m_style->setTextShadow(shadowData, i != 0); | |
| 4372 else | |
| 4373 m_style->setBoxShadow(shadowData, i != 0); | |
| 4374 } | |
| 4375 return; | |
| 4376 } | |
| 4377 case CSSPropertyWebkitBoxReflect: { | |
| 4378 HANDLE_INHERIT_AND_INITIAL(boxReflect, BoxReflect) | |
| 4379 if (primitiveValue) { | |
| 4380 m_style->setBoxReflect(RenderStyle::initialBoxReflect()); | |
| 4381 return; | |
| 4382 } | |
| 4383 CSSReflectValue* reflectValue = static_cast<CSSReflectValue*>(value); | |
| 4384 RefPtr<StyleReflection> reflection = StyleReflection::create(); | |
| 4385 reflection->setDirection(reflectValue->direction()); | |
| 4386 if (reflectValue->offset()) { | |
| 4387 int type = reflectValue->offset()->primitiveType(); | |
| 4388 if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 4389 reflection->setOffset(Length(reflectValue->offset()->getDoubleVa
lue(), Percent)); | |
| 4390 else | |
| 4391 reflection->setOffset(Length(reflectValue->offset()->computeLeng
thIntForLength(m_style, zoomFactor), Fixed)); | |
| 4392 } | |
| 4393 NinePieceImage mask; | |
| 4394 mapNinePieceImage(reflectValue->mask(), mask); | |
| 4395 reflection->setMask(mask); | |
| 4396 | |
| 4397 m_style->setBoxReflect(reflection.release()); | |
| 4398 return; | |
| 4399 } | |
| 4400 case CSSPropertyOpacity: | |
| 4401 HANDLE_INHERIT_AND_INITIAL(opacity, Opacity) | |
| 4402 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 4403 return; // Error case. | |
| 4404 // Clamp opacity to the range 0-1 | |
| 4405 m_style->setOpacity(min(1.0f, max(0.0f, primitiveValue->getFloatValue())
)); | |
| 4406 return; | |
| 4407 case CSSPropertyWebkitBoxAlign: | |
| 4408 { | |
| 4409 HANDLE_INHERIT_AND_INITIAL(boxAlign, BoxAlign) | |
| 4410 if (!primitiveValue) | |
| 4411 return; | |
| 4412 EBoxAlignment boxAlignment = *primitiveValue; | |
| 4413 if (boxAlignment != BJUSTIFY) | |
| 4414 m_style->setBoxAlign(boxAlignment); | |
| 4415 return; | |
| 4416 } | |
| 4417 case CSSPropertySrc: // Only used in @font-face rules. | |
| 4418 return; | |
| 4419 case CSSPropertyUnicodeRange: // Only used in @font-face rules. | |
| 4420 return; | |
| 4421 case CSSPropertyWebkitBoxDirection: | |
| 4422 HANDLE_INHERIT_AND_INITIAL(boxDirection, BoxDirection) | |
| 4423 if (primitiveValue) | |
| 4424 m_style->setBoxDirection(*primitiveValue); | |
| 4425 return; | |
| 4426 case CSSPropertyWebkitBoxLines: | |
| 4427 HANDLE_INHERIT_AND_INITIAL(boxLines, BoxLines) | |
| 4428 if (primitiveValue) | |
| 4429 m_style->setBoxLines(*primitiveValue); | |
| 4430 return; | |
| 4431 case CSSPropertyWebkitBoxOrient: | |
| 4432 HANDLE_INHERIT_AND_INITIAL(boxOrient, BoxOrient) | |
| 4433 if (primitiveValue) | |
| 4434 m_style->setBoxOrient(*primitiveValue); | |
| 4435 return; | |
| 4436 case CSSPropertyWebkitBoxPack: | |
| 4437 { | |
| 4438 HANDLE_INHERIT_AND_INITIAL(boxPack, BoxPack) | |
| 4439 if (!primitiveValue) | |
| 4440 return; | |
| 4441 EBoxAlignment boxPack = *primitiveValue; | |
| 4442 if (boxPack != BSTRETCH && boxPack != BBASELINE) | |
| 4443 m_style->setBoxPack(boxPack); | |
| 4444 return; | |
| 4445 } | |
| 4446 case CSSPropertyWebkitBoxFlex: | |
| 4447 HANDLE_INHERIT_AND_INITIAL(boxFlex, BoxFlex) | |
| 4448 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 4449 return; // Error case. | |
| 4450 m_style->setBoxFlex(primitiveValue->getFloatValue()); | |
| 4451 return; | |
| 4452 case CSSPropertyWebkitBoxFlexGroup: | |
| 4453 HANDLE_INHERIT_AND_INITIAL(boxFlexGroup, BoxFlexGroup) | |
| 4454 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 4455 return; // Error case. | |
| 4456 m_style->setBoxFlexGroup((unsigned int)(primitiveValue->getDoubleValue()
)); | |
| 4457 return; | |
| 4458 case CSSPropertyWebkitBoxOrdinalGroup: | |
| 4459 HANDLE_INHERIT_AND_INITIAL(boxOrdinalGroup, BoxOrdinalGroup) | |
| 4460 if (!primitiveValue || primitiveValue->primitiveType() != CSSPrimitiveVa
lue::CSS_NUMBER) | |
| 4461 return; // Error case. | |
| 4462 m_style->setBoxOrdinalGroup((unsigned int)(primitiveValue->getDoubleValu
e())); | |
| 4463 return; | |
| 4464 case CSSPropertyWebkitBoxSizing: | |
| 4465 HANDLE_INHERIT_AND_INITIAL(boxSizing, BoxSizing) | |
| 4466 if (!primitiveValue) | |
| 4467 return; | |
| 4468 if (primitiveValue->getIdent() == CSSValueContentBox) | |
| 4469 m_style->setBoxSizing(CONTENT_BOX); | |
| 4470 else | |
| 4471 m_style->setBoxSizing(BORDER_BOX); | |
| 4472 return; | |
| 4473 case CSSPropertyWebkitColumnCount: { | |
| 4474 if (isInherit) { | |
| 4475 if (m_parentStyle->hasAutoColumnCount()) | |
| 4476 m_style->setHasAutoColumnCount(); | |
| 4477 else | |
| 4478 m_style->setColumnCount(m_parentStyle->columnCount()); | |
| 4479 return; | |
| 4480 } else if (isInitial || primitiveValue->getIdent() == CSSValueAuto) { | |
| 4481 m_style->setHasAutoColumnCount(); | |
| 4482 return; | |
| 4483 } | |
| 4484 m_style->setColumnCount(static_cast<unsigned short>(primitiveValue->getD
oubleValue())); | |
| 4485 return; | |
| 4486 } | |
| 4487 case CSSPropertyWebkitColumnGap: { | |
| 4488 if (isInherit) { | |
| 4489 if (m_parentStyle->hasNormalColumnGap()) | |
| 4490 m_style->setHasNormalColumnGap(); | |
| 4491 else | |
| 4492 m_style->setColumnGap(m_parentStyle->columnGap()); | |
| 4493 return; | |
| 4494 } else if (isInitial || primitiveValue->getIdent() == CSSValueNormal) { | |
| 4495 m_style->setHasNormalColumnGap(); | |
| 4496 return; | |
| 4497 } | |
| 4498 m_style->setColumnGap(primitiveValue->computeLengthFloat(m_style, zoomFa
ctor)); | |
| 4499 return; | |
| 4500 } | |
| 4501 case CSSPropertyWebkitColumnWidth: { | |
| 4502 if (isInherit) { | |
| 4503 if (m_parentStyle->hasAutoColumnWidth()) | |
| 4504 m_style->setHasAutoColumnWidth(); | |
| 4505 else | |
| 4506 m_style->setColumnWidth(m_parentStyle->columnWidth()); | |
| 4507 return; | |
| 4508 } else if (isInitial || primitiveValue->getIdent() == CSSValueAuto) { | |
| 4509 m_style->setHasAutoColumnWidth(); | |
| 4510 return; | |
| 4511 } | |
| 4512 m_style->setColumnWidth(primitiveValue->computeLengthFloat(m_style, zoom
Factor)); | |
| 4513 return; | |
| 4514 } | |
| 4515 case CSSPropertyWebkitColumnRuleStyle: | |
| 4516 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(columnRuleStyle, ColumnRuleStyle,
BorderStyle) | |
| 4517 m_style->setColumnRuleStyle(*primitiveValue); | |
| 4518 return; | |
| 4519 case CSSPropertyWebkitColumnBreakBefore: { | |
| 4520 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(columnBreakBefore, ColumnBreakBefo
re, PageBreak) | |
| 4521 m_style->setColumnBreakBefore(*primitiveValue); | |
| 4522 return; | |
| 4523 } | |
| 4524 case CSSPropertyWebkitColumnBreakAfter: { | |
| 4525 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(columnBreakAfter, ColumnBreakAfter
, PageBreak) | |
| 4526 m_style->setColumnBreakAfter(*primitiveValue); | |
| 4527 return; | |
| 4528 } | |
| 4529 case CSSPropertyWebkitColumnBreakInside: { | |
| 4530 HANDLE_INHERIT_AND_INITIAL_WITH_VALUE(columnBreakInside, ColumnBreakInsi
de, PageBreak) | |
| 4531 EPageBreak pb = *primitiveValue; | |
| 4532 if (pb != PBALWAYS) | |
| 4533 m_style->setColumnBreakInside(pb); | |
| 4534 return; | |
| 4535 } | |
| 4536 case CSSPropertyWebkitColumnRule: | |
| 4537 if (isInherit) { | |
| 4538 m_style->setColumnRuleColor(m_parentStyle->columnRuleColor().isValid
() ? m_parentStyle->columnRuleColor() : m_parentStyle->color()); | |
| 4539 m_style->setColumnRuleStyle(m_parentStyle->columnRuleStyle()); | |
| 4540 m_style->setColumnRuleWidth(m_parentStyle->columnRuleWidth()); | |
| 4541 } | |
| 4542 else if (isInitial) | |
| 4543 m_style->resetColumnRule(); | |
| 4544 return; | |
| 4545 case CSSPropertyWebkitColumns: | |
| 4546 if (isInherit) { | |
| 4547 if (m_parentStyle->hasAutoColumnWidth()) | |
| 4548 m_style->setHasAutoColumnWidth(); | |
| 4549 else | |
| 4550 m_style->setColumnWidth(m_parentStyle->columnWidth()); | |
| 4551 m_style->setColumnCount(m_parentStyle->columnCount()); | |
| 4552 } else if (isInitial) { | |
| 4553 m_style->setHasAutoColumnWidth(); | |
| 4554 m_style->setColumnCount(RenderStyle::initialColumnCount()); | |
| 4555 } | |
| 4556 return; | |
| 4557 case CSSPropertyWebkitMarquee: | |
| 4558 if (valueType != CSSValue::CSS_INHERIT || !m_parentNode) return; | |
| 4559 m_style->setMarqueeDirection(m_parentStyle->marqueeDirection()); | |
| 4560 m_style->setMarqueeIncrement(m_parentStyle->marqueeIncrement()); | |
| 4561 m_style->setMarqueeSpeed(m_parentStyle->marqueeSpeed()); | |
| 4562 m_style->setMarqueeLoopCount(m_parentStyle->marqueeLoopCount()); | |
| 4563 m_style->setMarqueeBehavior(m_parentStyle->marqueeBehavior()); | |
| 4564 return; | |
| 4565 case CSSPropertyWebkitMarqueeRepetition: { | |
| 4566 HANDLE_INHERIT_AND_INITIAL(marqueeLoopCount, MarqueeLoopCount) | |
| 4567 if (!primitiveValue) | |
| 4568 return; | |
| 4569 if (primitiveValue->getIdent() == CSSValueInfinite) | |
| 4570 m_style->setMarqueeLoopCount(-1); // -1 means repeat forever. | |
| 4571 else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_NUMBE
R) | |
| 4572 m_style->setMarqueeLoopCount(primitiveValue->getIntValue()); | |
| 4573 return; | |
| 4574 } | |
| 4575 case CSSPropertyWebkitMarqueeSpeed: { | |
| 4576 HANDLE_INHERIT_AND_INITIAL(marqueeSpeed, MarqueeSpeed) | |
| 4577 if (!primitiveValue) | |
| 4578 return; | |
| 4579 if (primitiveValue->getIdent()) { | |
| 4580 switch (primitiveValue->getIdent()) { | |
| 4581 case CSSValueSlow: | |
| 4582 m_style->setMarqueeSpeed(500); // 500 msec. | |
| 4583 break; | |
| 4584 case CSSValueNormal: | |
| 4585 m_style->setMarqueeSpeed(85); // 85msec. The WinIE default. | |
| 4586 break; | |
| 4587 case CSSValueFast: | |
| 4588 m_style->setMarqueeSpeed(10); // 10msec. Super fast. | |
| 4589 break; | |
| 4590 } | |
| 4591 } | |
| 4592 else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_S) | |
| 4593 m_style->setMarqueeSpeed(1000 * primitiveValue->getIntValue()); | |
| 4594 else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_MS) | |
| 4595 m_style->setMarqueeSpeed(primitiveValue->getIntValue()); | |
| 4596 else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_NUMBE
R) // For scrollamount support. | |
| 4597 m_style->setMarqueeSpeed(primitiveValue->getIntValue()); | |
| 4598 return; | |
| 4599 } | |
| 4600 case CSSPropertyWebkitMarqueeIncrement: { | |
| 4601 HANDLE_INHERIT_AND_INITIAL(marqueeIncrement, MarqueeIncrement) | |
| 4602 if (!primitiveValue) | |
| 4603 return; | |
| 4604 if (primitiveValue->getIdent()) { | |
| 4605 switch (primitiveValue->getIdent()) { | |
| 4606 case CSSValueSmall: | |
| 4607 m_style->setMarqueeIncrement(Length(1, Fixed)); // 1px. | |
| 4608 break; | |
| 4609 case CSSValueNormal: | |
| 4610 m_style->setMarqueeIncrement(Length(6, Fixed)); // 6px. The
WinIE default. | |
| 4611 break; | |
| 4612 case CSSValueLarge: | |
| 4613 m_style->setMarqueeIncrement(Length(36, Fixed)); // 36px. | |
| 4614 break; | |
| 4615 } | |
| 4616 } | |
| 4617 else { | |
| 4618 bool ok = true; | |
| 4619 Length l = convertToLength(primitiveValue, m_style, &ok); | |
| 4620 if (ok) | |
| 4621 m_style->setMarqueeIncrement(l); | |
| 4622 } | |
| 4623 return; | |
| 4624 } | |
| 4625 case CSSPropertyWebkitMarqueeStyle: { | |
| 4626 HANDLE_INHERIT_AND_INITIAL(marqueeBehavior, MarqueeBehavior) | |
| 4627 if (primitiveValue) | |
| 4628 m_style->setMarqueeBehavior(*primitiveValue); | |
| 4629 return; | |
| 4630 } | |
| 4631 case CSSPropertyWebkitMarqueeDirection: { | |
| 4632 HANDLE_INHERIT_AND_INITIAL(marqueeDirection, MarqueeDirection) | |
| 4633 if (primitiveValue) | |
| 4634 m_style->setMarqueeDirection(*primitiveValue); | |
| 4635 return; | |
| 4636 } | |
| 4637 case CSSPropertyWebkitUserDrag: { | |
| 4638 HANDLE_INHERIT_AND_INITIAL(userDrag, UserDrag) | |
| 4639 if (primitiveValue) | |
| 4640 m_style->setUserDrag(*primitiveValue); | |
| 4641 return; | |
| 4642 } | |
| 4643 case CSSPropertyWebkitUserModify: { | |
| 4644 HANDLE_INHERIT_AND_INITIAL(userModify, UserModify) | |
| 4645 if (primitiveValue) | |
| 4646 m_style->setUserModify(*primitiveValue); | |
| 4647 return; | |
| 4648 } | |
| 4649 case CSSPropertyWebkitUserSelect: { | |
| 4650 HANDLE_INHERIT_AND_INITIAL(userSelect, UserSelect) | |
| 4651 if (primitiveValue) | |
| 4652 m_style->setUserSelect(*primitiveValue); | |
| 4653 return; | |
| 4654 } | |
| 4655 case CSSPropertyTextOverflow: { | |
| 4656 // This property is supported by WinIE, and so we leave off the "-webkit
-" in order to | |
| 4657 // work with WinIE-specific pages that use the property. | |
| 4658 HANDLE_INHERIT_AND_INITIAL(textOverflow, TextOverflow) | |
| 4659 if (!primitiveValue || !primitiveValue->getIdent()) | |
| 4660 return; | |
| 4661 m_style->setTextOverflow(primitiveValue->getIdent() == CSSValueEllipsis)
; | |
| 4662 return; | |
| 4663 } | |
| 4664 case CSSPropertyWebkitMarginCollapse: { | |
| 4665 if (isInherit) { | |
| 4666 m_style->setMarginTopCollapse(m_parentStyle->marginTopCollapse()); | |
| 4667 m_style->setMarginBottomCollapse(m_parentStyle->marginBottomCollapse
()); | |
| 4668 } | |
| 4669 else if (isInitial) { | |
| 4670 m_style->setMarginTopCollapse(MCOLLAPSE); | |
| 4671 m_style->setMarginBottomCollapse(MCOLLAPSE); | |
| 4672 } | |
| 4673 return; | |
| 4674 } | |
| 4675 case CSSPropertyWebkitMarginTopCollapse: { | |
| 4676 HANDLE_INHERIT_AND_INITIAL(marginTopCollapse, MarginTopCollapse) | |
| 4677 if (primitiveValue) | |
| 4678 m_style->setMarginTopCollapse(*primitiveValue); | |
| 4679 return; | |
| 4680 } | |
| 4681 case CSSPropertyWebkitMarginBottomCollapse: { | |
| 4682 HANDLE_INHERIT_AND_INITIAL(marginBottomCollapse, MarginBottomCollapse) | |
| 4683 if (primitiveValue) | |
| 4684 m_style->setMarginBottomCollapse(*primitiveValue); | |
| 4685 return; | |
| 4686 } | |
| 4687 | |
| 4688 // Apple-specific changes. Do not merge these properties into KHTML. | |
| 4689 case CSSPropertyWebkitLineClamp: { | |
| 4690 HANDLE_INHERIT_AND_INITIAL(lineClamp, LineClamp) | |
| 4691 if (!primitiveValue) | |
| 4692 return; | |
| 4693 m_style->setLineClamp(primitiveValue->getIntValue(CSSPrimitiveValue::CSS
_PERCENTAGE)); | |
| 4694 return; | |
| 4695 } | |
| 4696 case CSSPropertyWebkitHighlight: { | |
| 4697 HANDLE_INHERIT_AND_INITIAL(highlight, Highlight); | |
| 4698 if (primitiveValue->getIdent() == CSSValueNone) | |
| 4699 m_style->setHighlight(nullAtom); | |
| 4700 else | |
| 4701 m_style->setHighlight(primitiveValue->getStringValue()); | |
| 4702 return; | |
| 4703 } | |
| 4704 case CSSPropertyWebkitBorderFit: { | |
| 4705 HANDLE_INHERIT_AND_INITIAL(borderFit, BorderFit); | |
| 4706 if (primitiveValue->getIdent() == CSSValueBorder) | |
| 4707 m_style->setBorderFit(BorderFitBorder); | |
| 4708 else | |
| 4709 m_style->setBorderFit(BorderFitLines); | |
| 4710 return; | |
| 4711 } | |
| 4712 case CSSPropertyWebkitTextSizeAdjust: { | |
| 4713 HANDLE_INHERIT_AND_INITIAL(textSizeAdjust, TextSizeAdjust) | |
| 4714 if (!primitiveValue || !primitiveValue->getIdent()) return; | |
| 4715 m_style->setTextSizeAdjust(primitiveValue->getIdent() == CSSValueAuto); | |
| 4716 m_fontDirty = true; | |
| 4717 return; | |
| 4718 } | |
| 4719 case CSSPropertyWebkitTextSecurity: { | |
| 4720 HANDLE_INHERIT_AND_INITIAL(textSecurity, TextSecurity) | |
| 4721 if (primitiveValue) | |
| 4722 m_style->setTextSecurity(*primitiveValue); | |
| 4723 return; | |
| 4724 } | |
| 4725 #if ENABLE(DASHBOARD_SUPPORT) | |
| 4726 case CSSPropertyWebkitDashboardRegion: { | |
| 4727 HANDLE_INHERIT_AND_INITIAL(dashboardRegions, DashboardRegions) | |
| 4728 if (!primitiveValue) | |
| 4729 return; | |
| 4730 | |
| 4731 if (primitiveValue->getIdent() == CSSValueNone) { | |
| 4732 m_style->setDashboardRegions(RenderStyle::noneDashboardRegions()); | |
| 4733 return; | |
| 4734 } | |
| 4735 | |
| 4736 DashboardRegion *region = primitiveValue->getDashboardRegionValue(); | |
| 4737 if (!region) | |
| 4738 return; | |
| 4739 | |
| 4740 DashboardRegion *first = region; | |
| 4741 while (region) { | |
| 4742 Length top = convertToLength (region->top(), m_style); | |
| 4743 Length right = convertToLength (region->right(), m_style); | |
| 4744 Length bottom = convertToLength (region->bottom(), m_style); | |
| 4745 Length left = convertToLength (region->left(), m_style); | |
| 4746 if (region->m_isCircle) | |
| 4747 m_style->setDashboardRegion(StyleDashboardRegion::Circle, region
->m_label, top, right, bottom, left, region == first ? false : true); | |
| 4748 else if (region->m_isRectangle) | |
| 4749 m_style->setDashboardRegion(StyleDashboardRegion::Rectangle, reg
ion->m_label, top, right, bottom, left, region == first ? false : true); | |
| 4750 region = region->m_next.get(); | |
| 4751 } | |
| 4752 | |
| 4753 m_element->document()->setHasDashboardRegions(true); | |
| 4754 | |
| 4755 return; | |
| 4756 } | |
| 4757 #endif | |
| 4758 case CSSPropertyWebkitRtlOrdering: | |
| 4759 HANDLE_INHERIT_AND_INITIAL(visuallyOrdered, VisuallyOrdered) | |
| 4760 if (!primitiveValue || !primitiveValue->getIdent()) | |
| 4761 return; | |
| 4762 m_style->setVisuallyOrdered(primitiveValue->getIdent() == CSSValueVisual
); | |
| 4763 return; | |
| 4764 case CSSPropertyWebkitTextStrokeWidth: { | |
| 4765 HANDLE_INHERIT_AND_INITIAL(textStrokeWidth, TextStrokeWidth) | |
| 4766 float width = 0; | |
| 4767 switch (primitiveValue->getIdent()) { | |
| 4768 case CSSValueThin: | |
| 4769 case CSSValueMedium: | |
| 4770 case CSSValueThick: { | |
| 4771 double result = 1.0 / 48; | |
| 4772 if (primitiveValue->getIdent() == CSSValueMedium) | |
| 4773 result *= 3; | |
| 4774 else if (primitiveValue->getIdent() == CSSValueThick) | |
| 4775 result *= 5; | |
| 4776 width = CSSPrimitiveValue::create(result, CSSPrimitiveValue::CSS
_EMS)->computeLengthFloat(m_style, zoomFactor); | |
| 4777 break; | |
| 4778 } | |
| 4779 default: | |
| 4780 width = primitiveValue->computeLengthFloat(m_style, zoomFactor); | |
| 4781 break; | |
| 4782 } | |
| 4783 m_style->setTextStrokeWidth(width); | |
| 4784 return; | |
| 4785 } | |
| 4786 case CSSPropertyWebkitTransform: { | |
| 4787 HANDLE_INHERIT_AND_INITIAL(transform, Transform); | |
| 4788 TransformOperations operations; | |
| 4789 if (!value->isPrimitiveValue()) { | |
| 4790 CSSValueList* list = static_cast<CSSValueList*>(value); | |
| 4791 unsigned size = list->length(); | |
| 4792 for (unsigned i = 0; i < size; i++) { | |
| 4793 WebKitCSSTransformValue* val = static_cast<WebKitCSSTransformVal
ue*>(list->itemWithoutBoundsCheck(i)); | |
| 4794 | |
| 4795 CSSPrimitiveValue* firstValue = static_cast<CSSPrimitiveValue*>(
val->itemWithoutBoundsCheck(0)); | |
| 4796 | |
| 4797 switch (val->operationType()) { | |
| 4798 case WebKitCSSTransformValue::ScaleTransformOperation: | |
| 4799 case WebKitCSSTransformValue::ScaleXTransformOperation: | |
| 4800 case WebKitCSSTransformValue::ScaleYTransformOperation: { | |
| 4801 double sx = 1.0; | |
| 4802 double sy = 1.0; | |
| 4803 if (val->operationType() == WebKitCSSTransformValue::Sca
leYTransformOperation) | |
| 4804 sy = firstValue->getDoubleValue(); | |
| 4805 else { | |
| 4806 sx = firstValue->getDoubleValue(); | |
| 4807 if (val->operationType() == WebKitCSSTransformValue:
:ScaleTransformOperation) { | |
| 4808 if (val->length() > 1) { | |
| 4809 CSSPrimitiveValue* secondValue = static_cast
<CSSPrimitiveValue*>(val->itemWithoutBoundsCheck(1)); | |
| 4810 sy = secondValue->getDoubleValue(); | |
| 4811 } else | |
| 4812 sy = sx; | |
| 4813 } | |
| 4814 } | |
| 4815 operations.append(ScaleTransformOperation::create(sx, sy
)); | |
| 4816 break; | |
| 4817 } | |
| 4818 case WebKitCSSTransformValue::TranslateTransformOperation: | |
| 4819 case WebKitCSSTransformValue::TranslateXTransformOperation: | |
| 4820 case WebKitCSSTransformValue::TranslateYTransformOperation:
{ | |
| 4821 bool ok = true; | |
| 4822 Length tx = Length(0, Fixed); | |
| 4823 Length ty = Length(0, Fixed); | |
| 4824 if (val->operationType() == WebKitCSSTransformValue::Tra
nslateYTransformOperation) | |
| 4825 ty = convertToLength(firstValue, m_style, &ok); | |
| 4826 else { | |
| 4827 tx = convertToLength(firstValue, m_style, &ok); | |
| 4828 if (val->operationType() == WebKitCSSTransformValue:
:TranslateTransformOperation) { | |
| 4829 if (val->length() > 1) { | |
| 4830 CSSPrimitiveValue* secondValue = static_cast
<CSSPrimitiveValue*>(val->itemWithoutBoundsCheck(1)); | |
| 4831 ty = convertToLength(secondValue, m_style, &
ok); | |
| 4832 } | |
| 4833 } | |
| 4834 } | |
| 4835 | |
| 4836 if (!ok) | |
| 4837 return; | |
| 4838 | |
| 4839 operations.append(TranslateTransformOperation::create(tx
, ty)); | |
| 4840 break; | |
| 4841 } | |
| 4842 case WebKitCSSTransformValue::RotateTransformOperation: { | |
| 4843 double angle = firstValue->getDoubleValue(); | |
| 4844 if (firstValue->primitiveType() == CSSPrimitiveValue::CS
S_RAD) | |
| 4845 angle = rad2deg(angle); | |
| 4846 else if (firstValue->primitiveType() == CSSPrimitiveValu
e::CSS_GRAD) | |
| 4847 angle = grad2deg(angle); | |
| 4848 operations.append(RotateTransformOperation::create(angle
)); | |
| 4849 break; | |
| 4850 } | |
| 4851 case WebKitCSSTransformValue::SkewTransformOperation: | |
| 4852 case WebKitCSSTransformValue::SkewXTransformOperation: | |
| 4853 case WebKitCSSTransformValue::SkewYTransformOperation: { | |
| 4854 double angleX = 0; | |
| 4855 double angleY = 0; | |
| 4856 double angle = firstValue->getDoubleValue(); | |
| 4857 if (firstValue->primitiveType() == CSSPrimitiveValue::CS
S_RAD) | |
| 4858 angle = rad2deg(angle); | |
| 4859 else if (firstValue->primitiveType() == CSSPrimitiveValu
e::CSS_GRAD) | |
| 4860 angle = grad2deg(angle); | |
| 4861 if (val->operationType() == WebKitCSSTransformValue::Ske
wYTransformOperation) | |
| 4862 angleY = angle; | |
| 4863 else { | |
| 4864 angleX = angle; | |
| 4865 if (val->operationType() == WebKitCSSTransformValue:
:SkewTransformOperation) { | |
| 4866 if (val->length() > 1) { | |
| 4867 CSSPrimitiveValue* secondValue = static_cast
<CSSPrimitiveValue*>(val->itemWithoutBoundsCheck(1)); | |
| 4868 angleY = secondValue->getDoubleValue(); | |
| 4869 if (secondValue->primitiveType() == CSSPrimi
tiveValue::CSS_RAD) | |
| 4870 angleY = rad2deg(angle); | |
| 4871 else if (secondValue->primitiveType() == CSS
PrimitiveValue::CSS_GRAD) | |
| 4872 angleY = grad2deg(angle); | |
| 4873 } | |
| 4874 } | |
| 4875 } | |
| 4876 operations.append(SkewTransformOperation::create(angleX,
angleY)); | |
| 4877 break; | |
| 4878 } | |
| 4879 case WebKitCSSTransformValue::MatrixTransformOperation: { | |
| 4880 double a = firstValue->getDoubleValue(); | |
| 4881 double b = static_cast<CSSPrimitiveValue*>(val->itemWith
outBoundsCheck(1))->getDoubleValue(); | |
| 4882 double c = static_cast<CSSPrimitiveValue*>(val->itemWith
outBoundsCheck(2))->getDoubleValue(); | |
| 4883 double d = static_cast<CSSPrimitiveValue*>(val->itemWith
outBoundsCheck(3))->getDoubleValue(); | |
| 4884 double e = static_cast<CSSPrimitiveValue*>(val->itemWith
outBoundsCheck(4))->getDoubleValue(); | |
| 4885 double f = static_cast<CSSPrimitiveValue*>(val->itemWith
outBoundsCheck(5))->getDoubleValue(); | |
| 4886 operations.append(MatrixTransformOperation::create(a, b,
c, d, e, f)); | |
| 4887 break; | |
| 4888 } | |
| 4889 case WebKitCSSTransformValue::UnknownTransformOperation: | |
| 4890 ASSERT_NOT_REACHED(); | |
| 4891 break; | |
| 4892 } | |
| 4893 } | |
| 4894 } | |
| 4895 m_style->setTransform(operations); | |
| 4896 return; | |
| 4897 } | |
| 4898 case CSSPropertyWebkitTransformOrigin: | |
| 4899 HANDLE_INHERIT_AND_INITIAL(transformOriginX, TransformOriginX) | |
| 4900 HANDLE_INHERIT_AND_INITIAL(transformOriginY, TransformOriginY) | |
| 4901 return; | |
| 4902 case CSSPropertyWebkitTransformOriginX: { | |
| 4903 HANDLE_INHERIT_AND_INITIAL(transformOriginX, TransformOriginX) | |
| 4904 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(valu
e); | |
| 4905 Length l; | |
| 4906 int type = primitiveValue->primitiveType(); | |
| 4907 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveValue
::CSS_DEG) | |
| 4908 l = Length(primitiveValue->computeLengthIntForLength(m_style, zoomFa
ctor), Fixed); | |
| 4909 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 4910 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 4911 else | |
| 4912 return; | |
| 4913 m_style->setTransformOriginX(l); | |
| 4914 break; | |
| 4915 } | |
| 4916 case CSSPropertyWebkitTransformOriginY: { | |
| 4917 HANDLE_INHERIT_AND_INITIAL(transformOriginY, TransformOriginY) | |
| 4918 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(valu
e); | |
| 4919 Length l; | |
| 4920 int type = primitiveValue->primitiveType(); | |
| 4921 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveValue
::CSS_DEG) | |
| 4922 l = Length(primitiveValue->computeLengthIntForLength(m_style, zoomFa
ctor), Fixed); | |
| 4923 else if(type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 4924 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 4925 else | |
| 4926 return; | |
| 4927 m_style->setTransformOriginY(l); | |
| 4928 break; | |
| 4929 } | |
| 4930 case CSSPropertyWebkitAnimation: | |
| 4931 if (isInitial) | |
| 4932 m_style->clearAnimations(); | |
| 4933 else if (isInherit) | |
| 4934 m_style->inheritAnimations(m_parentStyle->animations()); | |
| 4935 return; | |
| 4936 case CSSPropertyWebkitAnimationDelay: | |
| 4937 HANDLE_ANIMATION_VALUE(delay, Delay, value) | |
| 4938 return; | |
| 4939 case CSSPropertyWebkitAnimationDirection: | |
| 4940 HANDLE_ANIMATION_VALUE(direction, Direction, value) | |
| 4941 return; | |
| 4942 case CSSPropertyWebkitAnimationDuration: | |
| 4943 HANDLE_ANIMATION_VALUE(duration, Duration, value) | |
| 4944 return; | |
| 4945 case CSSPropertyWebkitAnimationIterationCount: | |
| 4946 HANDLE_ANIMATION_VALUE(iterationCount, IterationCount, value) | |
| 4947 return; | |
| 4948 case CSSPropertyWebkitAnimationName: | |
| 4949 HANDLE_ANIMATION_VALUE(name, Name, value) | |
| 4950 return; | |
| 4951 case CSSPropertyWebkitAnimationPlayState: | |
| 4952 HANDLE_ANIMATION_VALUE(playState, PlayState, value) | |
| 4953 return; | |
| 4954 case CSSPropertyWebkitAnimationTimingFunction: | |
| 4955 HANDLE_ANIMATION_VALUE(timingFunction, TimingFunction, value) | |
| 4956 return; | |
| 4957 case CSSPropertyWebkitTransition: | |
| 4958 if (isInitial) | |
| 4959 m_style->clearTransitions(); | |
| 4960 else if (isInherit) | |
| 4961 m_style->inheritTransitions(m_parentStyle->transitions()); | |
| 4962 return; | |
| 4963 case CSSPropertyWebkitTransitionDelay: | |
| 4964 HANDLE_TRANSITION_VALUE(delay, Delay, value) | |
| 4965 return; | |
| 4966 case CSSPropertyWebkitTransitionDuration: | |
| 4967 HANDLE_TRANSITION_VALUE(duration, Duration, value) | |
| 4968 return; | |
| 4969 case CSSPropertyWebkitTransitionProperty: | |
| 4970 HANDLE_TRANSITION_VALUE(property, Property, value) | |
| 4971 return; | |
| 4972 case CSSPropertyWebkitTransitionTimingFunction: | |
| 4973 HANDLE_TRANSITION_VALUE(timingFunction, TimingFunction, value) | |
| 4974 return; | |
| 4975 case CSSPropertyInvalid: | |
| 4976 return; | |
| 4977 case CSSPropertyFontStretch: | |
| 4978 case CSSPropertyPage: | |
| 4979 case CSSPropertyQuotes: | |
| 4980 case CSSPropertyScrollbar3dlightColor: | |
| 4981 case CSSPropertyScrollbarArrowColor: | |
| 4982 case CSSPropertyScrollbarDarkshadowColor: | |
| 4983 case CSSPropertyScrollbarFaceColor: | |
| 4984 case CSSPropertyScrollbarHighlightColor: | |
| 4985 case CSSPropertyScrollbarShadowColor: | |
| 4986 case CSSPropertyScrollbarTrackColor: | |
| 4987 case CSSPropertySize: | |
| 4988 case CSSPropertyTextLineThrough: | |
| 4989 case CSSPropertyTextLineThroughColor: | |
| 4990 case CSSPropertyTextLineThroughMode: | |
| 4991 case CSSPropertyTextLineThroughStyle: | |
| 4992 case CSSPropertyTextLineThroughWidth: | |
| 4993 case CSSPropertyTextOverline: | |
| 4994 case CSSPropertyTextOverlineColor: | |
| 4995 case CSSPropertyTextOverlineMode: | |
| 4996 case CSSPropertyTextOverlineStyle: | |
| 4997 case CSSPropertyTextOverlineWidth: | |
| 4998 case CSSPropertyTextUnderline: | |
| 4999 case CSSPropertyTextUnderlineColor: | |
| 5000 case CSSPropertyTextUnderlineMode: | |
| 5001 case CSSPropertyTextUnderlineStyle: | |
| 5002 case CSSPropertyTextUnderlineWidth: | |
| 5003 case CSSPropertyWebkitFontSizeDelta: | |
| 5004 case CSSPropertyWebkitMarginStart: | |
| 5005 case CSSPropertyWebkitPaddingStart: | |
| 5006 case CSSPropertyWebkitTextDecorationsInEffect: | |
| 5007 case CSSPropertyWebkitTextStroke: | |
| 5008 return; | |
| 5009 #if ENABLE(SVG) | |
| 5010 default: | |
| 5011 // Try the SVG properties | |
| 5012 applySVGProperty(id, value); | |
| 5013 #endif | |
| 5014 } | |
| 5015 } | |
| 5016 | |
| 5017 void CSSStyleSelector::mapFillAttachment(FillLayer* layer, CSSValue* value) | |
| 5018 { | |
| 5019 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5020 layer->setAttachment(FillLayer::initialFillAttachment(layer->type())); | |
| 5021 return; | |
| 5022 } | |
| 5023 | |
| 5024 if (!value->isPrimitiveValue()) | |
| 5025 return; | |
| 5026 | |
| 5027 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5028 switch (primitiveValue->getIdent()) { | |
| 5029 case CSSValueFixed: | |
| 5030 layer->setAttachment(false); | |
| 5031 break; | |
| 5032 case CSSValueScroll: | |
| 5033 layer->setAttachment(true); | |
| 5034 break; | |
| 5035 default: | |
| 5036 return; | |
| 5037 } | |
| 5038 } | |
| 5039 | |
| 5040 void CSSStyleSelector::mapFillClip(FillLayer* layer, CSSValue* value) | |
| 5041 { | |
| 5042 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5043 layer->setClip(FillLayer::initialFillClip(layer->type())); | |
| 5044 return; | |
| 5045 } | |
| 5046 | |
| 5047 if (!value->isPrimitiveValue()) | |
| 5048 return; | |
| 5049 | |
| 5050 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5051 layer->setClip(*primitiveValue); | |
| 5052 } | |
| 5053 | |
| 5054 void CSSStyleSelector::mapFillComposite(FillLayer* layer, CSSValue* value) | |
| 5055 { | |
| 5056 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5057 layer->setComposite(FillLayer::initialFillComposite(layer->type())); | |
| 5058 return; | |
| 5059 } | |
| 5060 | |
| 5061 if (!value->isPrimitiveValue()) | |
| 5062 return; | |
| 5063 | |
| 5064 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5065 layer->setComposite(*primitiveValue); | |
| 5066 } | |
| 5067 | |
| 5068 void CSSStyleSelector::mapFillOrigin(FillLayer* layer, CSSValue* value) | |
| 5069 { | |
| 5070 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5071 layer->setOrigin(FillLayer::initialFillOrigin(layer->type())); | |
| 5072 return; | |
| 5073 } | |
| 5074 | |
| 5075 if (!value->isPrimitiveValue()) | |
| 5076 return; | |
| 5077 | |
| 5078 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5079 layer->setOrigin(*primitiveValue); | |
| 5080 } | |
| 5081 | |
| 5082 StyleImage* CSSStyleSelector::styleImage(CSSValue* value) | |
| 5083 { | |
| 5084 if (value->isImageValue()) | |
| 5085 return static_cast<CSSImageValue*>(value)->cachedImage(m_element->docume
nt()->docLoader()); | |
| 5086 if (value->isImageGeneratorValue()) | |
| 5087 return static_cast<CSSImageGeneratorValue*>(value)->generatedImage(); | |
| 5088 return 0; | |
| 5089 } | |
| 5090 | |
| 5091 void CSSStyleSelector::mapFillImage(FillLayer* layer, CSSValue* value) | |
| 5092 { | |
| 5093 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5094 layer->setImage(FillLayer::initialFillImage(layer->type())); | |
| 5095 return; | |
| 5096 } | |
| 5097 | |
| 5098 layer->setImage(styleImage(value)); | |
| 5099 } | |
| 5100 | |
| 5101 void CSSStyleSelector::mapFillRepeat(FillLayer* layer, CSSValue* value) | |
| 5102 { | |
| 5103 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5104 layer->setRepeat(FillLayer::initialFillRepeat(layer->type())); | |
| 5105 return; | |
| 5106 } | |
| 5107 | |
| 5108 if (!value->isPrimitiveValue()) | |
| 5109 return; | |
| 5110 | |
| 5111 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5112 layer->setRepeat(*primitiveValue); | |
| 5113 } | |
| 5114 | |
| 5115 void CSSStyleSelector::mapFillSize(FillLayer* layer, CSSValue* value) | |
| 5116 { | |
| 5117 LengthSize b = FillLayer::initialFillSize(layer->type()); | |
| 5118 | |
| 5119 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5120 layer->setSize(b); | |
| 5121 return; | |
| 5122 } | |
| 5123 | |
| 5124 if (!value->isPrimitiveValue()) | |
| 5125 return; | |
| 5126 | |
| 5127 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5128 Pair* pair = primitiveValue->getPairValue(); | |
| 5129 if (!pair) | |
| 5130 return; | |
| 5131 | |
| 5132 CSSPrimitiveValue* first = static_cast<CSSPrimitiveValue*>(pair->first()); | |
| 5133 CSSPrimitiveValue* second = static_cast<CSSPrimitiveValue*>(pair->second()); | |
| 5134 | |
| 5135 if (!first || !second) | |
| 5136 return; | |
| 5137 | |
| 5138 Length firstLength, secondLength; | |
| 5139 int firstType = first->primitiveType(); | |
| 5140 int secondType = second->primitiveType(); | |
| 5141 | |
| 5142 float zoomFactor = m_style->effectiveZoom(); | |
| 5143 | |
| 5144 if (firstType == CSSPrimitiveValue::CSS_UNKNOWN) | |
| 5145 firstLength = Length(Auto); | |
| 5146 else if (firstType > CSSPrimitiveValue::CSS_PERCENTAGE && firstType < CSSPri
mitiveValue::CSS_DEG) | |
| 5147 firstLength = Length(first->computeLengthIntForLength(m_style, zoomFacto
r), Fixed); | |
| 5148 else if (firstType == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5149 firstLength = Length(first->getDoubleValue(), Percent); | |
| 5150 else | |
| 5151 return; | |
| 5152 | |
| 5153 if (secondType == CSSPrimitiveValue::CSS_UNKNOWN) | |
| 5154 secondLength = Length(Auto); | |
| 5155 else if (secondType > CSSPrimitiveValue::CSS_PERCENTAGE && secondType < CSSP
rimitiveValue::CSS_DEG) | |
| 5156 secondLength = Length(second->computeLengthIntForLength(m_style, zoomFac
tor), Fixed); | |
| 5157 else if (secondType == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5158 secondLength = Length(second->getDoubleValue(), Percent); | |
| 5159 else | |
| 5160 return; | |
| 5161 | |
| 5162 b.width = firstLength; | |
| 5163 b.height = secondLength; | |
| 5164 layer->setSize(b); | |
| 5165 } | |
| 5166 | |
| 5167 void CSSStyleSelector::mapFillXPosition(FillLayer* layer, CSSValue* value) | |
| 5168 { | |
| 5169 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5170 layer->setXPosition(FillLayer::initialFillXPosition(layer->type())); | |
| 5171 return; | |
| 5172 } | |
| 5173 | |
| 5174 if (!value->isPrimitiveValue()) | |
| 5175 return; | |
| 5176 | |
| 5177 float zoomFactor = m_style->effectiveZoom(); | |
| 5178 | |
| 5179 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5180 Length l; | |
| 5181 int type = primitiveValue->primitiveType(); | |
| 5182 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveValue::CS
S_DEG) | |
| 5183 l = Length(primitiveValue->computeLengthIntForLength(m_style, zoomFactor
), Fixed); | |
| 5184 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5185 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 5186 else | |
| 5187 return; | |
| 5188 layer->setXPosition(l); | |
| 5189 } | |
| 5190 | |
| 5191 void CSSStyleSelector::mapFillYPosition(FillLayer* layer, CSSValue* value) | |
| 5192 { | |
| 5193 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5194 layer->setYPosition(FillLayer::initialFillYPosition(layer->type())); | |
| 5195 return; | |
| 5196 } | |
| 5197 | |
| 5198 if (!value->isPrimitiveValue()) | |
| 5199 return; | |
| 5200 | |
| 5201 float zoomFactor = m_style->effectiveZoom(); | |
| 5202 | |
| 5203 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5204 Length l; | |
| 5205 int type = primitiveValue->primitiveType(); | |
| 5206 if (type > CSSPrimitiveValue::CSS_PERCENTAGE && type < CSSPrimitiveValue::CS
S_DEG) | |
| 5207 l = Length(primitiveValue->computeLengthIntForLength(m_style, zoomFactor
), Fixed); | |
| 5208 else if (type == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5209 l = Length(primitiveValue->getDoubleValue(), Percent); | |
| 5210 else | |
| 5211 return; | |
| 5212 layer->setYPosition(l); | |
| 5213 } | |
| 5214 | |
| 5215 void CSSStyleSelector::mapAnimationDelay(Animation* animation, CSSValue* value) | |
| 5216 { | |
| 5217 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5218 animation->setDelay(RenderStyle::initialAnimationDelay()); | |
| 5219 return; | |
| 5220 } | |
| 5221 | |
| 5222 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5223 if (primitiveValue->getIdent() == CSSValueNow) | |
| 5224 animation->setDelay(0); | |
| 5225 else { | |
| 5226 if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_S) | |
| 5227 animation->setDelay(primitiveValue->getFloatValue()); | |
| 5228 else | |
| 5229 animation->setDelay(primitiveValue->getFloatValue()/1000.0f); | |
| 5230 } | |
| 5231 } | |
| 5232 | |
| 5233 void CSSStyleSelector::mapAnimationDirection(Animation* layer, CSSValue* value) | |
| 5234 { | |
| 5235 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5236 layer->setDirection(RenderStyle::initialAnimationDirection()); | |
| 5237 return; | |
| 5238 } | |
| 5239 | |
| 5240 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5241 layer->setDirection(primitiveValue->getIdent() == CSSValueAlternate); | |
| 5242 } | |
| 5243 | |
| 5244 void CSSStyleSelector::mapAnimationDuration(Animation* animation, CSSValue* valu
e) | |
| 5245 { | |
| 5246 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5247 animation->setDuration(RenderStyle::initialAnimationDuration()); | |
| 5248 return; | |
| 5249 } | |
| 5250 | |
| 5251 if (!value->isPrimitiveValue()) | |
| 5252 return; | |
| 5253 | |
| 5254 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5255 if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_S) | |
| 5256 animation->setDuration(primitiveValue->getFloatValue()); | |
| 5257 else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_MS) | |
| 5258 animation->setDuration(primitiveValue->getFloatValue()/1000.0f); | |
| 5259 } | |
| 5260 | |
| 5261 void CSSStyleSelector::mapAnimationIterationCount(Animation* animation, CSSValue
* value) | |
| 5262 { | |
| 5263 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5264 animation->setIterationCount(RenderStyle::initialAnimationIterationCount
()); | |
| 5265 return; | |
| 5266 } | |
| 5267 | |
| 5268 if (!value->isPrimitiveValue()) | |
| 5269 return; | |
| 5270 | |
| 5271 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5272 if (primitiveValue->getIdent() == CSSValueInfinite) | |
| 5273 animation->setIterationCount(-1); | |
| 5274 else | |
| 5275 animation->setIterationCount(int(primitiveValue->getFloatValue())); | |
| 5276 } | |
| 5277 | |
| 5278 void CSSStyleSelector::mapAnimationName(Animation* layer, CSSValue* value) | |
| 5279 { | |
| 5280 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5281 layer->setName(RenderStyle::initialAnimationName()); | |
| 5282 return; | |
| 5283 } | |
| 5284 | |
| 5285 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5286 | |
| 5287 if (primitiveValue->getIdent() == CSSValueNone) { | |
| 5288 layer->setIsNoneAnimation(true); | |
| 5289 } else { | |
| 5290 layer->setName(primitiveValue->getStringValue()); | |
| 5291 | |
| 5292 // resolve to the keyframes | |
| 5293 RefPtr<KeyframeList> keyframe = findKeyframeRule(primitiveValue->getStri
ngValue()); | |
| 5294 layer->setAnimationKeyframe(keyframe); | |
| 5295 } | |
| 5296 } | |
| 5297 | |
| 5298 void CSSStyleSelector::mapAnimationPlayState(Animation* layer, CSSValue* value) | |
| 5299 { | |
| 5300 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5301 layer->setPlayState(RenderStyle::initialAnimationPlayState()); | |
| 5302 return; | |
| 5303 } | |
| 5304 | |
| 5305 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5306 layer->setPlayState((primitiveValue->getIdent() == CSSValuePaused) ? AnimPla
yStatePaused : AnimPlayStatePlaying); | |
| 5307 } | |
| 5308 | |
| 5309 void CSSStyleSelector::mapAnimationProperty(Animation* animation, CSSValue* valu
e) | |
| 5310 { | |
| 5311 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5312 animation->setProperty(RenderStyle::initialAnimationProperty()); | |
| 5313 return; | |
| 5314 } | |
| 5315 | |
| 5316 if (!value->isPrimitiveValue()) | |
| 5317 return; | |
| 5318 | |
| 5319 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(value); | |
| 5320 animation->setProperty(static_cast<CSSPropertyID>(primitiveValue->getIdent()
)); | |
| 5321 } | |
| 5322 | |
| 5323 void CSSStyleSelector::mapAnimationTimingFunction(Animation* animation, CSSValue
* value) | |
| 5324 { | |
| 5325 if (value->cssValueType() == CSSValue::CSS_INITIAL) { | |
| 5326 animation->setTimingFunction(RenderStyle::initialAnimationTimingFunction
()); | |
| 5327 return; | |
| 5328 } | |
| 5329 | |
| 5330 if (value->isPrimitiveValue()) { | |
| 5331 CSSPrimitiveValue* primitiveValue = static_cast<CSSPrimitiveValue*>(valu
e); | |
| 5332 switch (primitiveValue->getIdent()) { | |
| 5333 case CSSValueLinear: | |
| 5334 animation->setTimingFunction(TimingFunction(LinearTimingFunction
, 0.0, 0.0, 1.0, 1.0)); | |
| 5335 break; | |
| 5336 case CSSValueEase: | |
| 5337 animation->setTimingFunction(TimingFunction()); | |
| 5338 break; | |
| 5339 case CSSValueEaseIn: | |
| 5340 animation->setTimingFunction(TimingFunction(CubicBezierTimingFun
ction, 0.42, 0.0, 1.0, 1.0)); | |
| 5341 break; | |
| 5342 case CSSValueEaseOut: | |
| 5343 animation->setTimingFunction(TimingFunction(CubicBezierTimingFun
ction, 0.0, 0.0, 0.58, 1.0)); | |
| 5344 break; | |
| 5345 case CSSValueEaseInOut: | |
| 5346 animation->setTimingFunction(TimingFunction(CubicBezierTimingFun
ction, 0.42, 0.0, 0.58, 1.0)); | |
| 5347 break; | |
| 5348 } | |
| 5349 return; | |
| 5350 } | |
| 5351 | |
| 5352 if (value->isTimingFunctionValue()) { | |
| 5353 CSSTimingFunctionValue* timingFunction = static_cast<CSSTimingFunctionVa
lue*>(value); | |
| 5354 animation->setTimingFunction(TimingFunction(CubicBezierTimingFunction, t
imingFunction->x1(), timingFunction->y1(), timingFunction->x2(), timingFunction-
>y2())); | |
| 5355 } | |
| 5356 } | |
| 5357 | |
| 5358 void CSSStyleSelector::mapNinePieceImage(CSSValue* value, NinePieceImage& image) | |
| 5359 { | |
| 5360 // If we're a primitive value, then we are "none" and don't need to alter th
e empty image at all. | |
| 5361 if (!value || value->isPrimitiveValue()) | |
| 5362 return; | |
| 5363 | |
| 5364 // Retrieve the border image value. | |
| 5365 CSSBorderImageValue* borderImage = static_cast<CSSBorderImageValue*>(value); | |
| 5366 | |
| 5367 // Set the image (this kicks off the load). | |
| 5368 image.m_image = styleImage(borderImage->imageValue()); | |
| 5369 | |
| 5370 // Set up a length box to represent our image slices. | |
| 5371 LengthBox& l = image.m_slices; | |
| 5372 Rect* r = borderImage->m_imageSliceRect.get(); | |
| 5373 if (r->top()->primitiveType() == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5374 l.top = Length(r->top()->getDoubleValue(), Percent); | |
| 5375 else | |
| 5376 l.top = Length(r->top()->getIntValue(CSSPrimitiveValue::CSS_NUMBER), Fix
ed); | |
| 5377 if (r->bottom()->primitiveType() == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5378 l.bottom = Length(r->bottom()->getDoubleValue(), Percent); | |
| 5379 else | |
| 5380 l.bottom = Length((int)r->bottom()->getFloatValue(CSSPrimitiveValue::CSS
_NUMBER), Fixed); | |
| 5381 if (r->left()->primitiveType() == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5382 l.left = Length(r->left()->getDoubleValue(), Percent); | |
| 5383 else | |
| 5384 l.left = Length(r->left()->getIntValue(CSSPrimitiveValue::CSS_NUMBER), F
ixed); | |
| 5385 if (r->right()->primitiveType() == CSSPrimitiveValue::CSS_PERCENTAGE) | |
| 5386 l.right = Length(r->right()->getDoubleValue(), Percent); | |
| 5387 else | |
| 5388 l.right = Length(r->right()->getIntValue(CSSPrimitiveValue::CSS_NUMBER),
Fixed); | |
| 5389 | |
| 5390 // Set the appropriate rules for stretch/round/repeat of the slices | |
| 5391 switch (borderImage->m_horizontalSizeRule) { | |
| 5392 case CSSValueStretch: | |
| 5393 image.m_horizontalRule = StretchImageRule; | |
| 5394 break; | |
| 5395 case CSSValueRound: | |
| 5396 image.m_horizontalRule = RoundImageRule; | |
| 5397 break; | |
| 5398 default: // CSSValueRepeat | |
| 5399 image.m_horizontalRule = RepeatImageRule; | |
| 5400 break; | |
| 5401 } | |
| 5402 | |
| 5403 switch (borderImage->m_verticalSizeRule) { | |
| 5404 case CSSValueStretch: | |
| 5405 image.m_verticalRule = StretchImageRule; | |
| 5406 break; | |
| 5407 case CSSValueRound: | |
| 5408 image.m_verticalRule = RoundImageRule; | |
| 5409 break; | |
| 5410 default: // CSSValueRepeat | |
| 5411 image.m_verticalRule = RepeatImageRule; | |
| 5412 break; | |
| 5413 } | |
| 5414 } | |
| 5415 | |
| 5416 void CSSStyleSelector::checkForTextSizeAdjust() | |
| 5417 { | |
| 5418 if (m_style->textSizeAdjust()) | |
| 5419 return; | |
| 5420 | |
| 5421 FontDescription newFontDescription(m_style->fontDescription()); | |
| 5422 newFontDescription.setComputedSize(newFontDescription.specifiedSize()); | |
| 5423 m_style->setFontDescription(newFontDescription); | |
| 5424 } | |
| 5425 | |
| 5426 void CSSStyleSelector::checkForZoomChange(RenderStyle* style, RenderStyle* paren
tStyle) | |
| 5427 { | |
| 5428 if (style->effectiveZoom() == parentStyle->effectiveZoom()) | |
| 5429 return; | |
| 5430 | |
| 5431 const FontDescription& childFont = style->fontDescription(); | |
| 5432 FontDescription newFontDescription(childFont); | |
| 5433 setFontSize(newFontDescription, childFont.specifiedSize()); | |
| 5434 style->setFontDescription(newFontDescription); | |
| 5435 } | |
| 5436 | |
| 5437 void CSSStyleSelector::checkForGenericFamilyChange(RenderStyle* style, RenderSty
le* parentStyle) | |
| 5438 { | |
| 5439 const FontDescription& childFont = style->fontDescription(); | |
| 5440 | |
| 5441 if (childFont.isAbsoluteSize() || !parentStyle) | |
| 5442 return; | |
| 5443 | |
| 5444 const FontDescription& parentFont = parentStyle->fontDescription(); | |
| 5445 | |
| 5446 if (childFont.genericFamily() == parentFont.genericFamily()) | |
| 5447 return; | |
| 5448 | |
| 5449 // For now, lump all families but monospace together. | |
| 5450 if (childFont.genericFamily() != FontDescription::MonospaceFamily && | |
| 5451 parentFont.genericFamily() != FontDescription::MonospaceFamily) | |
| 5452 return; | |
| 5453 | |
| 5454 // We know the parent is monospace or the child is monospace, and that font | |
| 5455 // size was unspecified. We want to scale our font size as appropriate. | |
| 5456 // If the font uses a keyword size, then we refetch from the table rather th
an | |
| 5457 // multiplying by our scale factor. | |
| 5458 float size; | |
| 5459 if (childFont.keywordSize()) { | |
| 5460 size = fontSizeForKeyword(CSSValueXxSmall + childFont.keywordSize() - 1,
style->htmlHacks(), | |
| 5461 childFont.genericFamily() == FontDescription::
MonospaceFamily); | |
| 5462 } else { | |
| 5463 Settings* settings = m_checker.m_document->settings(); | |
| 5464 float fixedScaleFactor = settings | |
| 5465 ? static_cast<float>(settings->defaultFixedFontSize()) / settings->d
efaultFontSize() | |
| 5466 : 1; | |
| 5467 size = (parentFont.genericFamily() == FontDescription::MonospaceFamily)
? | |
| 5468 childFont.specifiedSize()/fixedScaleFactor : | |
| 5469 childFont.specifiedSize()*fixedScaleFactor; | |
| 5470 } | |
| 5471 | |
| 5472 FontDescription newFontDescription(childFont); | |
| 5473 setFontSize(newFontDescription, size); | |
| 5474 style->setFontDescription(newFontDescription); | |
| 5475 } | |
| 5476 | |
| 5477 void CSSStyleSelector::setFontSize(FontDescription& fontDescription, float size) | |
| 5478 { | |
| 5479 fontDescription.setSpecifiedSize(size); | |
| 5480 fontDescription.setComputedSize(getComputedSizeFromSpecifiedSize(fontDescrip
tion.isAbsoluteSize(), size)); | |
| 5481 } | |
| 5482 | |
| 5483 float CSSStyleSelector::getComputedSizeFromSpecifiedSize(bool isAbsoluteSize, fl
oat specifiedSize) | |
| 5484 { | |
| 5485 // We support two types of minimum font size. The first is a hard override
that applies to | |
| 5486 // all fonts. This is "minSize." The second type of minimum font size is a
"smart minimum" | |
| 5487 // that is applied only when the Web page can't know what size it really ask
ed for, e.g., | |
| 5488 // when it uses logical sizes like "small" or expresses the font-size as a p
ercentage of | |
| 5489 // the user's default font setting. | |
| 5490 | |
| 5491 // With the smart minimum, we never want to get smaller than the minimum fon
t size to keep fonts readable. | |
| 5492 // However we always allow the page to set an explicit pixel size that is sm
aller, | |
| 5493 // since sites will mis-render otherwise (e.g., http://www.gamespot.com with
a 9px minimum). | |
| 5494 | |
| 5495 Settings* settings = m_checker.m_document->settings(); | |
| 5496 if (!settings) | |
| 5497 return 1.0f; | |
| 5498 | |
| 5499 int minSize = settings->minimumFontSize(); | |
| 5500 int minLogicalSize = settings->minimumLogicalFontSize(); | |
| 5501 | |
| 5502 float zoomFactor = m_style->effectiveZoom(); | |
| 5503 if (m_checker.m_document->frame() && m_checker.m_document->frame()->shouldAp
plyTextZoom()) | |
| 5504 zoomFactor *= m_checker.m_document->frame()->textZoomFactor(); | |
| 5505 | |
| 5506 float zoomedSize = specifiedSize * zoomFactor; | |
| 5507 | |
| 5508 // Apply the hard minimum first. We only apply the hard minimum if after zo
oming we're still too small. | |
| 5509 if (zoomedSize < minSize) | |
| 5510 zoomedSize = minSize; | |
| 5511 | |
| 5512 // Now apply the "smart minimum." This minimum is also only applied if we'r
e still too small | |
| 5513 // after zooming. The font size must either be relative to the user default
or the original size | |
| 5514 // must have been acceptable. In other words, we only apply the smart minim
um whenever we're positive | |
| 5515 // doing so won't disrupt the layout. | |
| 5516 if (zoomedSize < minLogicalSize && (specifiedSize >= minLogicalSize || !isAb
soluteSize)) | |
| 5517 zoomedSize = minLogicalSize; | |
| 5518 | |
| 5519 // Also clamp to a reasonable maximum to prevent insane font sizes from caus
ing crashes on various | |
| 5520 // platforms (I'm looking at you, Windows.) | |
| 5521 return min(1000000.0f, max(zoomedSize, 1.0f)); | |
| 5522 } | |
| 5523 | |
| 5524 const int fontSizeTableMax = 16; | |
| 5525 const int fontSizeTableMin = 9; | |
| 5526 const int totalKeywords = 8; | |
| 5527 | |
| 5528 // WinIE/Nav4 table for font sizes. Designed to match the legacy font mapping s
ystem of HTML. | |
| 5529 static const int quirksFontSizeTable[fontSizeTableMax - fontSizeTableMin + 1][to
talKeywords] = | |
| 5530 { | |
| 5531 { 9, 9, 9, 9, 11, 14, 18, 28 }, | |
| 5532 { 9, 9, 9, 10, 12, 15, 20, 31 }, | |
| 5533 { 9, 9, 9, 11, 13, 17, 22, 34 }, | |
| 5534 { 9, 9, 10, 12, 14, 18, 24, 37 }, | |
| 5535 { 9, 9, 10, 13, 16, 20, 26, 40 }, // fixed font defau
lt (13) | |
| 5536 { 9, 9, 11, 14, 17, 21, 28, 42 }, | |
| 5537 { 9, 10, 12, 15, 17, 23, 30, 45 }, | |
| 5538 { 9, 10, 13, 16, 18, 24, 32, 48 } // proportional fon
t default (16) | |
| 5539 }; | |
| 5540 // HTML 1 2 3 4 5 6 7 | |
| 5541 // CSS xxs xs s m l xl xxl | |
| 5542 // | | |
| 5543 // user pref | |
| 5544 | |
| 5545 // Strict mode table matches MacIE and Mozilla's settings exactly. | |
| 5546 static const int strictFontSizeTable[fontSizeTableMax - fontSizeTableMin + 1][to
talKeywords] = | |
| 5547 { | |
| 5548 { 9, 9, 9, 9, 11, 14, 18, 27 }, | |
| 5549 { 9, 9, 9, 10, 12, 15, 20, 30 }, | |
| 5550 { 9, 9, 10, 11, 13, 17, 22, 33 }, | |
| 5551 { 9, 9, 10, 12, 14, 18, 24, 36 }, | |
| 5552 { 9, 10, 12, 13, 16, 20, 26, 39 }, // fixed font defau
lt (13) | |
| 5553 { 9, 10, 12, 14, 17, 21, 28, 42 }, | |
| 5554 { 9, 10, 13, 15, 18, 23, 30, 45 }, | |
| 5555 { 9, 10, 13, 16, 18, 24, 32, 48 } // proportional fon
t default (16) | |
| 5556 }; | |
| 5557 // HTML 1 2 3 4 5 6 7 | |
| 5558 // CSS xxs xs s m l xl xxl | |
| 5559 // | | |
| 5560 // user pref | |
| 5561 | |
| 5562 // For values outside the range of the table, we use Todd Fahrner's suggested sc
ale | |
| 5563 // factors for each keyword value. | |
| 5564 static const float fontSizeFactors[totalKeywords] = { 0.60f, 0.75f, 0.89f, 1.0f,
1.2f, 1.5f, 2.0f, 3.0f }; | |
| 5565 | |
| 5566 float CSSStyleSelector::fontSizeForKeyword(int keyword, bool quirksMode, bool fi
xed) const | |
| 5567 { | |
| 5568 Settings* settings = m_checker.m_document->settings(); | |
| 5569 if (!settings) | |
| 5570 return 1.0f; | |
| 5571 | |
| 5572 int mediumSize = fixed ? settings->defaultFixedFontSize() : settings->defaul
tFontSize(); | |
| 5573 if (mediumSize >= fontSizeTableMin && mediumSize <= fontSizeTableMax) { | |
| 5574 // Look up the entry in the table. | |
| 5575 int row = mediumSize - fontSizeTableMin; | |
| 5576 int col = (keyword - CSSValueXxSmall); | |
| 5577 return quirksMode ? quirksFontSizeTable[row][col] : strictFontSizeTable[
row][col]; | |
| 5578 } | |
| 5579 | |
| 5580 // Value is outside the range of the table. Apply the scale factor instead. | |
| 5581 float minLogicalSize = max(settings->minimumLogicalFontSize(), 1); | |
| 5582 return max(fontSizeFactors[keyword - CSSValueXxSmall]*mediumSize, minLogical
Size); | |
| 5583 } | |
| 5584 | |
| 5585 float CSSStyleSelector::largerFontSize(float size, bool quirksMode) const | |
| 5586 { | |
| 5587 // FIXME: Figure out where we fall in the size ranges (xx-small to xxx-large
) and scale up to | |
| 5588 // the next size level. | |
| 5589 return size * 1.2f; | |
| 5590 } | |
| 5591 | |
| 5592 float CSSStyleSelector::smallerFontSize(float size, bool quirksMode) const | |
| 5593 { | |
| 5594 // FIXME: Figure out where we fall in the size ranges (xx-small to xxx-large
) and scale down to | |
| 5595 // the next size level. | |
| 5596 return size / 1.2f; | |
| 5597 } | |
| 5598 | |
| 5599 static Color colorForCSSValue(int cssValueId) | |
| 5600 { | |
| 5601 struct ColorValue { | |
| 5602 int cssValueId; | |
| 5603 RGBA32 color; | |
| 5604 }; | |
| 5605 | |
| 5606 static const ColorValue colorValues[] = { | |
| 5607 { CSSValueAqua, 0xFF00FFFF }, | |
| 5608 { CSSValueBlack, 0xFF000000 }, | |
| 5609 { CSSValueBlue, 0xFF0000FF }, | |
| 5610 { CSSValueFuchsia, 0xFFFF00FF }, | |
| 5611 { CSSValueGray, 0xFF808080 }, | |
| 5612 { CSSValueGreen, 0xFF008000 }, | |
| 5613 { CSSValueGrey, 0xFF808080 }, | |
| 5614 { CSSValueLime, 0xFF00FF00 }, | |
| 5615 { CSSValueMaroon, 0xFF800000 }, | |
| 5616 { CSSValueNavy, 0xFF000080 }, | |
| 5617 { CSSValueOlive, 0xFF808000 }, | |
| 5618 { CSSValueOrange, 0xFFFFA500 }, | |
| 5619 { CSSValuePurple, 0xFF800080 }, | |
| 5620 { CSSValueRed, 0xFFFF0000 }, | |
| 5621 { CSSValueSilver, 0xFFC0C0C0 }, | |
| 5622 { CSSValueTeal, 0xFF008080 }, | |
| 5623 { CSSValueTransparent, 0x00000000 }, | |
| 5624 { CSSValueWhite, 0xFFFFFFFF }, | |
| 5625 { CSSValueYellow, 0xFFFFFF00 }, | |
| 5626 { 0, 0 } | |
| 5627 }; | |
| 5628 | |
| 5629 for (const ColorValue* col = colorValues; col->cssValueId; ++col) { | |
| 5630 if (col->cssValueId == cssValueId) | |
| 5631 return col->color; | |
| 5632 } | |
| 5633 return theme()->systemColor(cssValueId); | |
| 5634 } | |
| 5635 | |
| 5636 Color CSSStyleSelector::getColorFromPrimitiveValue(CSSPrimitiveValue* primitiveV
alue) | |
| 5637 { | |
| 5638 Color col; | |
| 5639 int ident = primitiveValue->getIdent(); | |
| 5640 if (ident) { | |
| 5641 if (ident == CSSValueWebkitText) | |
| 5642 col = m_element->document()->textColor(); | |
| 5643 else if (ident == CSSValueWebkitLink) { | |
| 5644 const Color& linkColor = m_element->document()->linkColor(); | |
| 5645 const Color& visitedColor = m_element->document()->visitedLinkColor(
); | |
| 5646 if (linkColor == visitedColor) | |
| 5647 col = linkColor; | |
| 5648 else { | |
| 5649 if (pseudoState == PseudoUnknown || pseudoState == PseudoAnyLink
) | |
| 5650 pseudoState = m_checker.checkPseudoState(m_element); | |
| 5651 col = (pseudoState == PseudoLink) ? linkColor : visitedColor; | |
| 5652 } | |
| 5653 } else if (ident == CSSValueWebkitActivelink) | |
| 5654 col = m_element->document()->activeLinkColor(); | |
| 5655 else if (ident == CSSValueWebkitFocusRingColor) | |
| 5656 col = focusRingColor(); | |
| 5657 else if (ident == CSSValueCurrentcolor) | |
| 5658 col = m_style->color(); | |
| 5659 else | |
| 5660 col = colorForCSSValue(ident); | |
| 5661 } else if (primitiveValue->primitiveType() == CSSPrimitiveValue::CSS_RGBCOLO
R) | |
| 5662 col.setRGB(primitiveValue->getRGBColorValue()); | |
| 5663 return col; | |
| 5664 } | |
| 5665 | |
| 5666 bool CSSStyleSelector::hasSelectorForAttribute(const AtomicString &attrname) | |
| 5667 { | |
| 5668 return m_selectorAttrs.contains(attrname.impl()); | |
| 5669 } | |
| 5670 | |
| 5671 void CSSStyleSelector::addViewportDependentMediaQueryResult(const MediaQueryExp*
expr, bool result) | |
| 5672 { | |
| 5673 m_viewportDependentMediaQueryResults.append(new MediaQueryResult(*expr, resu
lt)); | |
| 5674 } | |
| 5675 | |
| 5676 bool CSSStyleSelector::affectedByViewportChange() const | |
| 5677 { | |
| 5678 unsigned s = m_viewportDependentMediaQueryResults.size(); | |
| 5679 for (unsigned i = 0; i < s; i++) { | |
| 5680 if (m_medium->eval(&m_viewportDependentMediaQueryResults[i]->m_expressio
n) != m_viewportDependentMediaQueryResults[i]->m_result) | |
| 5681 return true; | |
| 5682 } | |
| 5683 return false; | |
| 5684 } | |
| 5685 | |
| 5686 void CSSStyleSelector::SelectorChecker::allVisitedStateChanged() | |
| 5687 { | |
| 5688 if (m_linksCheckedForVisitedState.isEmpty()) | |
| 5689 return; | |
| 5690 for (Node* node = m_document; node; node = node->traverseNextNode()) { | |
| 5691 if (node->isLink()) | |
| 5692 node->setChanged(); | |
| 5693 } | |
| 5694 } | |
| 5695 | |
| 5696 void CSSStyleSelector::SelectorChecker::visitedStateChanged(unsigned visitedHash
) | |
| 5697 { | |
| 5698 if (!m_linksCheckedForVisitedState.contains(visitedHash)) | |
| 5699 return; | |
| 5700 for (Node* node = m_document; node; node = node->traverseNextNode()) { | |
| 5701 const AtomicString* attr = linkAttribute(node); | |
| 5702 if (attr && m_document->visitedLinkHash(*attr) == visitedHash) | |
| 5703 node->setChanged(); | |
| 5704 } | |
| 5705 } | |
| 5706 | |
| 5707 } // namespace WebCore | |
| OLD | NEW |