/* * Copyright (C) 2011, 2013 Apple Inc. All rights reserved. * Copyright (C) 2014 Samsung Electronics. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "sky/engine/config.h" #include "sky/engine/core/dom/SelectorQuery.h" #include "sky/engine/bindings2/exception_state.h" #include "sky/engine/core/css/SelectorChecker.h" #include "sky/engine/core/css/parser/BisonCSSParser.h" #include "sky/engine/core/dom/Document.h" #include "sky/engine/core/dom/ElementTraversal.h" #include "sky/engine/core/dom/Node.h" #include "sky/engine/core/dom/StaticNodeList.h" namespace blink { PassOwnPtr SelectorQuery::adopt(CSSSelectorList& selectorList) { return adoptPtr(new SelectorQuery(selectorList)); } SelectorQuery::SelectorQuery(CSSSelectorList& selectorList) { m_selectors.adopt(selectorList); } bool SelectorQuery::matches(Element& element) const { return selectorMatches(element, element); } PassRefPtr SelectorQuery::queryAll(ContainerNode& rootNode) const { Vector > result; for (Element* element = ElementTraversal::firstWithin(rootNode); element; element = ElementTraversal::next(*element, &rootNode)) { if (selectorMatches(rootNode, *element)) result.append(element); } return StaticElementList::adopt(result); } PassRefPtr SelectorQuery::queryFirst(ContainerNode& rootNode) const { for (Element* element = ElementTraversal::firstWithin(rootNode); element; element = ElementTraversal::next(*element, &rootNode)) { if (selectorMatches(rootNode, *element)) return element; } return nullptr; } bool SelectorQuery::selectorMatches(ContainerNode& rootNode, Element& element) const { SelectorChecker checker(element); for (const CSSSelector* selector = m_selectors.first(); selector; selector = CSSSelectorList::next(*selector)) { if (checker.match(*selector)) return true; } return false; } SelectorQuery* SelectorQueryCache::add(const AtomicString& selectors, const Document& document, ExceptionState& exceptionState) { HashMap >::iterator it = m_entries.find(selectors); if (it != m_entries.end()) return it->value.get(); CSSParserContext context(document); BisonCSSParser parser(context); CSSSelectorList selectorList; parser.parseSelector(selectors, selectorList); if (!selectorList.first()) { exceptionState.ThrowDOMException(SyntaxError, "'" + selectors + "' is not a valid selector."); return 0; } const unsigned maximumSelectorQueryCacheSize = 256; if (m_entries.size() == maximumSelectorQueryCacheSize) m_entries.remove(m_entries.begin()); return m_entries.add(selectors, SelectorQuery::adopt(selectorList)).storedValue->value.get(); } }