/* * Copyright (C) 2009 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 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. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT * OWNER OR 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 "public/web/WebNode.h" #include "bindings/core/v8/ExceptionState.h" #include "core/dom/Document.h" #include "core/dom/Element.h" #include "core/dom/Node.h" #include "core/dom/NodeList.h" #include "core/dom/StaticNodeList.h" #include "core/dom/TagCollection.h" #include "core/editing/serializers/Serialization.h" #include "core/events/Event.h" #include "core/html/HTMLCollection.h" #include "core/html/HTMLElement.h" #include "core/layout/LayoutObject.h" #include "core/layout/LayoutPart.h" #include "modules/accessibility/AXObject.h" #include "modules/accessibility/AXObjectCacheImpl.h" #include "platform/Widget.h" #include "public/platform/WebString.h" #include "public/platform/WebSuspendableTask.h" #include "public/web/WebAXObject.h" #include "public/web/WebDOMEvent.h" #include "public/web/WebDocument.h" #include "public/web/WebElement.h" #include "public/web/WebElementCollection.h" #include "public/web/WebPluginContainer.h" #include "web/FrameLoaderClientImpl.h" #include "web/WebLocalFrameImpl.h" #include "web/WebPluginContainerImpl.h" namespace blink { namespace { class NodeDispatchEventTask: public SuspendableTask { WTF_MAKE_NONCOPYABLE(NodeDispatchEventTask); public: NodeDispatchEventTask(const WebPrivatePtr& node, WebDOMEvent event) : m_event(event) { m_node = node; } ~NodeDispatchEventTask() { m_node.reset(); } void run() override { m_node->dispatchEvent(m_event); } private: WebPrivatePtr m_node; WebDOMEvent m_event; }; class NodeDispatchSimulatedClickTask: public SuspendableTask { WTF_MAKE_NONCOPYABLE(NodeDispatchSimulatedClickTask); public: NodeDispatchSimulatedClickTask(const WebPrivatePtr& node) { m_node = node; } ~NodeDispatchSimulatedClickTask() { m_node.reset(); } void run() override { m_node->dispatchSimulatedClick(nullptr); } private: WebPrivatePtr m_node; }; } // namespace void WebNode::reset() { m_private.reset(); } void WebNode::assign(const WebNode& other) { m_private = other.m_private; } bool WebNode::equals(const WebNode& n) const { return m_private.get() == n.m_private.get(); } bool WebNode::lessThan(const WebNode& n) const { return m_private.get() < n.m_private.get(); } WebNode WebNode::parentNode() const { return WebNode(const_cast(m_private->parentNode())); } WebString WebNode::nodeValue() const { return m_private->nodeValue(); } WebDocument WebNode::document() const { return WebDocument(&m_private->document()); } WebNode WebNode::firstChild() const { return WebNode(m_private->firstChild()); } WebNode WebNode::lastChild() const { return WebNode(m_private->lastChild()); } WebNode WebNode::previousSibling() const { return WebNode(m_private->previousSibling()); } WebNode WebNode::nextSibling() const { return WebNode(m_private->nextSibling()); } bool WebNode::hasChildNodes() const { return m_private->hasChildren(); } bool WebNode::isLink() const { return m_private->isLink(); } bool WebNode::isTextNode() const { return m_private->isTextNode(); } bool WebNode::isCommentNode() const { return m_private->getNodeType() == Node::COMMENT_NODE; } bool WebNode::isFocusable() const { if (!m_private->isElementNode()) return false; m_private->document().updateLayoutIgnorePendingStylesheets(); return toElement(m_private.get())->isFocusable(); } bool WebNode::isContentEditable() const { return m_private->isContentEditable(); } bool WebNode::isInsideFocusableElementOrARIAWidget() const { return AXObject::isInsideFocusableElementOrARIAWidget(*this->constUnwrap()); } bool WebNode::isElementNode() const { return m_private->isElementNode(); } bool WebNode::isDocumentNode() const { return m_private->isDocumentNode(); } bool WebNode::isDocumentTypeNode() const { return m_private->getNodeType() == Node::DOCUMENT_TYPE_NODE; } void WebNode::dispatchEvent(const WebDOMEvent& event) { if (!event.isNull()) m_private->getExecutionContext()->postSuspendableTask(adoptPtr(new NodeDispatchEventTask(m_private, event))); } void WebNode::simulateClick() { m_private->getExecutionContext()->postSuspendableTask(adoptPtr(new NodeDispatchSimulatedClickTask(m_private))); } WebElementCollection WebNode::getElementsByHTMLTagName(const WebString& tag) const { if (m_private->isContainerNode()) return WebElementCollection(toContainerNode(m_private.get())->getElementsByTagNameNS(HTMLNames::xhtmlNamespaceURI, tag)); return WebElementCollection(); } WebElement WebNode::querySelector(const WebString& selector, WebExceptionCode& ec) const { if (!m_private->isContainerNode()) return WebElement(); TrackExceptionState exceptionState; WebElement element = toContainerNode(m_private.get())->querySelector(selector, exceptionState); ec = exceptionState.code(); return element; } WebElement WebNode::querySelector(const WebString& selector) const { WebExceptionCode ec = 0; WebElement element = querySelector(selector, ec); ASSERT(!ec); return element; } void WebNode::querySelectorAll(const WebString& selector, WebVector& results, WebExceptionCode& ec) const { if (!m_private->isContainerNode()) return; TrackExceptionState exceptionState; RefPtrWillBeRawPtr elements = toContainerNode(m_private.get())->querySelectorAll(selector, exceptionState); ec = exceptionState.code(); if (exceptionState.hadException()) return; Vector temp; temp.reserveCapacity(elements->length()); for (unsigned i = 0; i < elements->length(); ++i) temp.append(WebElement(elements->item(i))); results.assign(temp); } void WebNode::querySelectorAll(const WebString& selector, WebVector& results) const { WebExceptionCode ec = 0; querySelectorAll(selector, results, ec); ASSERT(!ec); } bool WebNode::focused() const { return m_private->focused(); } WebPluginContainer* WebNode::pluginContainer() const { if (isNull()) return 0; const Node& coreNode = *constUnwrap(); if (isHTMLObjectElement(coreNode) || isHTMLEmbedElement(coreNode)) { LayoutObject* object = coreNode.layoutObject(); if (object && object->isLayoutPart()) { Widget* widget = toLayoutPart(object)->widget(); if (widget && widget->isPluginContainer()) return toWebPluginContainerImpl(widget); } } return 0; } WebAXObject WebNode::accessibilityObject() { WebDocument webDocument = document(); const Document* doc = document().constUnwrap(); AXObjectCacheImpl* cache = toAXObjectCacheImpl(doc->existingAXObjectCache()); Node* node = unwrap(); return cache ? WebAXObject(cache->get(node)) : WebAXObject(); } WebNode::WebNode(const PassRefPtrWillBeRawPtr& node) : m_private(node) { } WebNode& WebNode::operator=(const PassRefPtrWillBeRawPtr& node) { m_private = node; return *this; } WebNode::operator PassRefPtrWillBeRawPtr() const { return m_private.get(); } } // namespace blink