Index: Source/core/css/SelectorCheckerFastPath.cpp |
diff --git a/Source/core/css/SelectorCheckerFastPath.cpp b/Source/core/css/SelectorCheckerFastPath.cpp |
deleted file mode 100644 |
index 943e26239b8220073b34362346f833ce43870cc9..0000000000000000000000000000000000000000 |
--- a/Source/core/css/SelectorCheckerFastPath.cpp |
+++ /dev/null |
@@ -1,224 +0,0 @@ |
-/* |
- * Copyright (C) 1999 Lars Knoll (knoll@kde.org) |
- * (C) 2004-2005 Allan Sandfeld Jensen (kde@carewolf.com) |
- * Copyright (C) 2006, 2007 Nicholas Shanks (webkit@nickshanks.com) |
- * Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013 Apple Inc. All rights reserved. |
- * Copyright (C) 2007 Alexey Proskuryakov <ap@webkit.org> |
- * Copyright (C) 2007, 2008 Eric Seidel <eric@webkit.org> |
- * Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/) |
- * Copyright (c) 2011, Code Aurora Forum. All rights reserved. |
- * Copyright (C) Research In Motion Limited 2011. All rights reserved. |
- * Copyright (C) 2013 Google Inc. All rights reserved. |
- * |
- * This library is free software; you can redistribute it and/or |
- * modify it under the terms of the GNU Library General Public |
- * License as published by the Free Software Foundation; either |
- * version 2 of the License, or (at your option) any later version. |
- * |
- * This library is distributed in the hope that it will be useful, |
- * but WITHOUT ANY WARRANTY; without even the implied warranty of |
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
- * Library General Public License for more details. |
- * |
- * You should have received a copy of the GNU Library General Public License |
- * along with this library; see the file COPYING.LIB. If not, write to |
- * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
- * Boston, MA 02110-1301, USA. |
- */ |
- |
-#include "config.h" |
-#include "core/css/SelectorCheckerFastPath.h" |
- |
-#include "HTMLNames.h" |
-#include "core/dom/Element.h" |
-#include "core/html/HTMLDocument.h" |
- |
-namespace WebCore { |
- |
-using namespace HTMLNames; |
- |
-namespace { |
- |
-template <bool checkValue(const Element&, const CSSSelector&)> |
-inline bool fastCheckSingleSelector(const CSSSelector*& selector, const Element*& element, const CSSSelector*& topChildOrSubselector, const Element*& topChildOrSubselectorMatchElement) |
-{ |
- for (; element; element = element->parentElement()) { |
- if (checkValue(*element, *selector)) { |
- if (selector->relation() == CSSSelector::Descendant) |
- topChildOrSubselector = 0; |
- else if (!topChildOrSubselector) { |
- ASSERT(selector->relation() == CSSSelector::Child || selector->relation() == CSSSelector::SubSelector); |
- topChildOrSubselector = selector; |
- topChildOrSubselectorMatchElement = element; |
- } |
- if (selector->relation() != CSSSelector::SubSelector) |
- element = element->parentElement(); |
- selector = selector->tagHistory(); |
- return true; |
- } |
- if (topChildOrSubselector) { |
- // Child or subselector check failed. |
- // If the match element is null, topChildOrSubselector was also the very topmost selector and had to match |
- // the original element we were checking. |
- if (!topChildOrSubselectorMatchElement) |
- return false; |
- // There may be other matches down the ancestor chain. |
- // Rewind to the topmost child or subselector and the element it matched, continue checking ancestors. |
- selector = topChildOrSubselector; |
- element = topChildOrSubselectorMatchElement->parentElement(); |
- topChildOrSubselector = 0; |
- return true; |
- } |
- } |
- return false; |
-} |
- |
-inline bool checkClassValue(const Element& element, const CSSSelector& selector) |
-{ |
- return element.hasClass() && element.classNames().contains(selector.value()); |
-} |
- |
-inline bool checkIDValue(const Element& element, const CSSSelector& selector) |
-{ |
- return element.hasID() && element.idForStyleResolution() == selector.value(); |
-} |
- |
-inline bool checkExactAttributeValue(const Element& element, const CSSSelector& selector) |
-{ |
- return SelectorChecker::checkExactAttribute(element, selector.attribute(), selector.value().impl()); |
-} |
- |
-inline bool checkTagValue(const Element& element, const CSSSelector& selector) |
-{ |
- return SelectorChecker::tagMatches(element, selector.tagQName()); |
-} |
- |
-} |
- |
-SelectorCheckerFastPath::SelectorCheckerFastPath(const CSSSelector& selector, const Element& element) |
- : m_selector(selector) |
- , m_element(element) |
-{ |
-} |
- |
-bool SelectorCheckerFastPath::matchesRightmostSelector(SelectorChecker::VisitedMatchType visitedMatchType) const |
-{ |
- ASSERT(SelectorCheckerFastPath::canUse(m_selector)); |
- |
- switch (m_selector.m_match) { |
- case CSSSelector::Tag: |
- return checkTagValue(m_element, m_selector); |
- case CSSSelector::Class: |
- return checkClassValue(m_element, m_selector); |
- case CSSSelector::Id: |
- return checkIDValue(m_element, m_selector); |
- case CSSSelector::Exact: |
- case CSSSelector::Set: |
- return checkExactAttributeValue(m_element, m_selector); |
- case CSSSelector::PseudoClass: |
- return commonPseudoClassSelectorMatches(visitedMatchType); |
- default: |
- ASSERT_NOT_REACHED(); |
- } |
- return false; |
-} |
- |
-bool SelectorCheckerFastPath::matches() const |
-{ |
- ASSERT(matchesRightmostSelector(SelectorChecker::VisitedMatchEnabled)); |
- const CSSSelector* selector = &m_selector; |
- const Element* element = &m_element; |
- |
- const CSSSelector* topChildOrSubselector = 0; |
- const Element* topChildOrSubselectorMatchElement = 0; |
- if (selector->relation() == CSSSelector::Child || selector->relation() == CSSSelector::SubSelector) |
- topChildOrSubselector = selector; |
- |
- if (selector->relation() != CSSSelector::SubSelector) |
- element = element->parentElement(); |
- |
- selector = selector->tagHistory(); |
- |
- // We know this compound selector has descendant, child and subselector combinators only and all components are simple. |
- while (selector) { |
- switch (selector->m_match) { |
- case CSSSelector::Class: |
- if (!fastCheckSingleSelector<checkClassValue>(selector, element, topChildOrSubselector, topChildOrSubselectorMatchElement)) |
- return false; |
- break; |
- case CSSSelector::Id: |
- if (!fastCheckSingleSelector<checkIDValue>(selector, element, topChildOrSubselector, topChildOrSubselectorMatchElement)) |
- return false; |
- break; |
- case CSSSelector::Tag: |
- if (!fastCheckSingleSelector<checkTagValue>(selector, element, topChildOrSubselector, topChildOrSubselectorMatchElement)) |
- return false; |
- break; |
- case CSSSelector::Set: |
- case CSSSelector::Exact: |
- if (!fastCheckSingleSelector<checkExactAttributeValue>(selector, element, topChildOrSubselector, topChildOrSubselectorMatchElement)) |
- return false; |
- break; |
- default: |
- ASSERT_NOT_REACHED(); |
- } |
- } |
- return true; |
-} |
- |
-static inline bool isFastCheckableRelation(CSSSelector::Relation relation) |
-{ |
- return relation == CSSSelector::Descendant || relation == CSSSelector::Child || relation == CSSSelector::SubSelector; |
-} |
- |
-static inline bool isFastCheckableMatch(const CSSSelector& selector) |
-{ |
- if (selector.m_match == CSSSelector::Set) { |
- // Style attribute is generated lazily but the fast path doesn't trigger it. |
- // Disallow them here rather than making the fast path more branchy. |
- return selector.attribute() != styleAttr; |
- } |
- if (selector.m_match == CSSSelector::Exact) |
- return selector.attribute() != styleAttr && HTMLDocument::isCaseSensitiveAttribute(selector.attribute()); |
- return selector.m_match == CSSSelector::Tag || selector.m_match == CSSSelector::Id || selector.m_match == CSSSelector::Class; |
-} |
- |
-static inline bool isFastCheckableRightmostSelector(const CSSSelector& selector) |
-{ |
- if (!isFastCheckableRelation(selector.relation())) |
- return false; |
- return isFastCheckableMatch(selector) || SelectorChecker::isCommonPseudoClassSelector(selector); |
-} |
- |
-bool SelectorCheckerFastPath::canUse(const CSSSelector& selector) |
-{ |
- if (!isFastCheckableRightmostSelector(selector)) |
- return false; |
- for (const CSSSelector* current = selector.tagHistory(); current; current = current->tagHistory()) { |
- if (!isFastCheckableRelation(current->relation())) |
- return false; |
- if (!isFastCheckableMatch(*current)) |
- return false; |
- } |
- return true; |
-} |
- |
-bool SelectorCheckerFastPath::commonPseudoClassSelectorMatches(SelectorChecker::VisitedMatchType visitedMatchType) const |
-{ |
- ASSERT(SelectorChecker::isCommonPseudoClassSelector(m_selector)); |
- switch (m_selector.pseudoType()) { |
- case CSSSelector::PseudoLink: |
- case CSSSelector::PseudoAnyLink: |
- return m_element.isLink(); |
- case CSSSelector::PseudoVisited: |
- return m_element.isLink() && visitedMatchType == SelectorChecker::VisitedMatchEnabled; |
- case CSSSelector::PseudoFocus: |
- return SelectorChecker::matchesFocusPseudoClass(m_element); |
- default: |
- ASSERT_NOT_REACHED(); |
- } |
- return true; |
-} |
- |
- |
-} |