⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xpathpath.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
字号:
/* * Copyright 2005 Frerich Raabe <raabe@kde.org> * Copyright (C) 2006 Apple Computer, Inc. * Copyright (C) 2007 Alexey Proskuryakov <ap@webkit.org> * * 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 THE AUTHOR ``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 THE AUTHOR 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 "config.h"#include "XPathPath.h"#if ENABLE(XPATH)#include "Document.h"#include "XPathPredicate.h"#include "XPathStep.h"#include "XPathValue.h"namespace WebCore {namespace XPath {        Filter::Filter(Expression* expr, const Vector<Predicate*>& predicates)    : m_expr(expr), m_predicates(predicates){}Filter::~Filter(){    delete m_expr;    deleteAllValues(m_predicates);}Value Filter::evaluate() const{    Value v = m_expr->evaluate();        if (!v.isNodeSet())         return v;    NodeSet& nodes = v.modifiableNodeSet();    nodes.sort();    EvaluationContext& evaluationContext = Expression::evaluationContext();    for (unsigned i = 0; i < m_predicates.size(); i++) {        NodeSet newNodes;        evaluationContext.size = nodes.size();        evaluationContext.position = 0;                for (unsigned j = 0; j < nodes.size(); j++) {            Node* node = nodes[j];                        evaluationContext.node = node;            ++evaluationContext.position;                        if (m_predicates[i]->evaluate())                newNodes.append(node);        }        nodes.swap(newNodes);    }    return v;}LocationPath::LocationPath()    : m_absolute(false){}LocationPath::~LocationPath(){    deleteAllValues(m_steps);}Value LocationPath::evaluate() const{    EvaluationContext& evaluationContext = Expression::evaluationContext();    EvaluationContext backupContext = evaluationContext;    /* For absolute location paths, the context node is ignored - the     * document's root node is used instead.     */    Node* context = evaluationContext.node.get();    if (m_absolute && context->nodeType() != Node::DOCUMENT_NODE)         context = context->ownerDocument();    NodeSet nodes;    nodes.append(context);    evaluate(nodes);        evaluationContext = backupContext;    return Value(nodes, Value::adopt);}void LocationPath::evaluate(NodeSet& nodes) const{    for (unsigned i = 0; i < m_steps.size(); i++) {        Step* step = m_steps[i];        NodeSet newNodes;        HashSet<Node*> newNodesSet;        for (unsigned j = 0; j < nodes.size(); j++) {            NodeSet matches;            step->evaluate(nodes[j], matches);                        for (size_t nodeIndex = 0; nodeIndex < matches.size(); ++nodeIndex) {                Node* node = matches[nodeIndex];                if (newNodesSet.add(node).second)                    newNodes.append(node);            }        }                nodes.swap(newNodes);    }    nodes.markSorted(false);}void LocationPath::optimizeStepPair(unsigned index){    Step* first = m_steps[index];        if (first->axis() == Step::DescendantOrSelfAxis        && first->nodeTest().kind() == Step::NodeTest::AnyNodeTest        && first->predicates().size() == 0) {        Step* second = m_steps[index + 1];        if (second->axis() == Step::ChildAxis            && second->nodeTest().namespaceURI().isEmpty()            && second->nodeTest().kind() == Step::NodeTest::NameTest            && second->nodeTest().data() == "*") {            // Optimize the common case of "//*" AKA descendant-or-self::node()/child::*.            first->setAxis(Step::DescendantAxis);            second->setAxis(Step::SelfAxis);            second->setNodeTest(Step::NodeTest::ElementNodeTest);            ASSERT(second->nodeTest().data().isEmpty());        }    }}void LocationPath::appendStep(Step* step){    m_steps.append(step);        unsigned stepCount = m_steps.size();    if (stepCount > 1)        optimizeStepPair(stepCount - 2);}void LocationPath::insertFirstStep(Step* step){    m_steps.insert(0, step);    if (m_steps.size() > 1)        optimizeStepPair(0);}Path::Path(Filter* filter, LocationPath* path)    : m_filter(filter),    m_path(path){}Path::~Path(){    delete m_filter;    delete m_path;}Value Path::evaluate() const{    Value v = m_filter->evaluate();    NodeSet& nodes = v.modifiableNodeSet();    m_path->evaluate(nodes);        return v;}}}#endif // ENABLE(XPATH)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -