/* * Copyright 2005 Frerich Raabe * Copyright (C) 2006 Apple Computer, Inc. * * 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 "XPathResult.h" #if ENABLE(XPATH) #include "EventListener.h" #include "EventNames.h" #include "EventTargetNode.h" #include "ExceptionCode.h" #include "XPathEvaluator.h" #include "XPathException.h" namespace WebCore { using namespace XPath; class InvalidatingEventListener : public EventListener { public: InvalidatingEventListener(XPathResult* result) : m_result(result) { } virtual void handleEvent(Event*, bool) { m_result->invalidateIteratorState(); } private: XPathResult* m_result; }; XPathResult::XPathResult(EventTargetNode* eventTarget, const Value& value) : m_value(value) , m_eventTarget(eventTarget) { m_eventListener = new InvalidatingEventListener(this); m_eventTarget->addEventListener(EventNames::DOMSubtreeModifiedEvent, m_eventListener, false); switch (m_value.type()) { case Value::BooleanValue: m_resultType = BOOLEAN_TYPE; return; case Value::NumberValue: m_resultType = NUMBER_TYPE; return; case Value::StringValue: m_resultType = STRING_TYPE; return; case Value::NodeSetValue: m_resultType = UNORDERED_NODE_ITERATOR_TYPE; m_nodeSetPosition = 0; m_nodeSet = m_value.toNodeSet(); m_invalidIteratorState = false; return; } ASSERT_NOT_REACHED(); } XPathResult::~XPathResult() { if (m_eventTarget) m_eventTarget->removeEventListener(EventNames::DOMSubtreeModifiedEvent, m_eventListener.get(), false); } void XPathResult::convertTo(unsigned short type, ExceptionCode& ec) { switch (type) { case ANY_TYPE: break; case NUMBER_TYPE: m_resultType = type; m_value = m_value.toNumber(); break; case STRING_TYPE: m_resultType = type; m_value = m_value.toString(); break; case BOOLEAN_TYPE: m_resultType = type; m_value = m_value.toBoolean(); break; case UNORDERED_NODE_ITERATOR_TYPE: case UNORDERED_NODE_SNAPSHOT_TYPE: case ANY_UNORDERED_NODE_TYPE: case FIRST_ORDERED_NODE_TYPE: // This is correct - singleNodeValue() will take care of ordering. if (!m_value.isNodeSet()) { ec = XPathException::TYPE_ERR; return; } m_resultType = type; break; case ORDERED_NODE_ITERATOR_TYPE: if (!m_value.isNodeSet()) { ec = XPathException::TYPE_ERR; return; } m_nodeSet.sort(); m_resultType = type; break; case ORDERED_NODE_SNAPSHOT_TYPE: if (!m_value.isNodeSet()) { ec = XPathException::TYPE_ERR; return; } m_value.toNodeSet().sort(); m_resultType = type; break; } } unsigned short XPathResult::resultType() const { return m_resultType; } double XPathResult::numberValue(ExceptionCode& ec) const { if (resultType() != NUMBER_TYPE) { ec = XPathException::TYPE_ERR; return 0.0; } return m_value.toNumber(); } String XPathResult::stringValue(ExceptionCode& ec) const { if (resultType() != STRING_TYPE) { ec = XPathException::TYPE_ERR; return String(); } return m_value.toString(); } bool XPathResult::booleanValue(ExceptionCode& ec) const { if (resultType() != BOOLEAN_TYPE) { ec = XPathException::TYPE_ERR; return false; } return m_value.toBoolean(); } Node* XPathResult::singleNodeValue(ExceptionCode& ec) const { if (resultType() != ANY_UNORDERED_NODE_TYPE && resultType() != FIRST_ORDERED_NODE_TYPE) { ec = XPathException::TYPE_ERR; return 0; } const NodeSet& nodes = m_value.toNodeSet(); if (resultType() == FIRST_ORDERED_NODE_TYPE) return nodes.firstNode(); else return nodes.anyNode(); } void XPathResult::invalidateIteratorState() { m_invalidIteratorState = true; ASSERT(m_eventTarget); ASSERT(m_eventListener); m_eventTarget->removeEventListener(EventNames::DOMSubtreeModifiedEvent, m_eventListener.get(), false); m_eventTarget = 0; } bool XPathResult::invalidIteratorState() const { if (resultType() != UNORDERED_NODE_ITERATOR_TYPE && resultType() != ORDERED_NODE_ITERATOR_TYPE) return false; return m_invalidIteratorState; } unsigned long XPathResult::snapshotLength(ExceptionCode& ec) const { if (resultType() != UNORDERED_NODE_SNAPSHOT_TYPE && resultType() != ORDERED_NODE_SNAPSHOT_TYPE) { ec = XPathException::TYPE_ERR; return 0; } return m_value.toNodeSet().size(); } Node* XPathResult::iterateNext(ExceptionCode& ec) { if (resultType() != UNORDERED_NODE_ITERATOR_TYPE && resultType() != ORDERED_NODE_ITERATOR_TYPE) { ec = XPathException::TYPE_ERR; return 0; } if (m_invalidIteratorState) { ec = INVALID_STATE_ERR; return 0; } if (m_nodeSetPosition + 1 > m_nodeSet.size()) return 0; Node* node = m_nodeSet[m_nodeSetPosition]; m_nodeSetPosition++; return node; } Node* XPathResult::snapshotItem(unsigned long index, ExceptionCode& ec) { if (resultType() != UNORDERED_NODE_SNAPSHOT_TYPE && resultType() != ORDERED_NODE_SNAPSHOT_TYPE) { ec = XPathException::TYPE_ERR; return 0; } const NodeSet& nodes = m_value.toNodeSet(); if (index >= nodes.size()) return 0; return nodes[index]; } } #endif // ENABLE(XPATH)