/* * Copyright (C) 2015-2016 Apple 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: * 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 INC. ``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 INC. 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 "config.h" #include "ComposedTreeIterator.h" #include "HTMLSlotElement.h" #include "TextStream.h" namespace WebCore { ComposedTreeIterator::ComposedTreeIterator(ContainerNode& root) { ASSERT(!is(root)); #if ENABLE(SHADOW_DOM) || ENABLE(DETAILS_ELEMENT) if (is(root)) { auto& slot = downcast(root); if (auto* assignedNodes = slot.assignedNodes()) { initializeContextStack(root, *assignedNodes->at(0)); return; } } #endif if (auto* shadowRoot = root.shadowRoot()) { auto* firstChild = shadowRoot->firstChild(); initializeContextStack(root, firstChild ? *firstChild : root); return; } m_contextStack.uncheckedAppend(Context(root)); } ComposedTreeIterator::ComposedTreeIterator(ContainerNode& root, Node& current) { ASSERT(!is(root)); ASSERT(!is(current)); bool mayNeedShadowStack = root.shadowRoot() || (¤t != &root && current.parentNode() != &root); if (mayNeedShadowStack) initializeContextStack(root, current); else m_contextStack.uncheckedAppend(Context(root, current)); } void ComposedTreeIterator::initializeContextStack(ContainerNode& root, Node& current) { // This code sets up the iterator for arbitrary node/root pair. It is not needed in common cases // or completes fast because node and root are close (like in composedTreeChildren(*parent).at(node) case). auto* node = ¤t; auto* contextCurrent = node; size_t currentSlotNodeIndex = notFound; while (node != &root) { auto* parent = node->parentNode(); if (!parent) { *this = { }; return; } if (is(*parent)) { auto& shadowRoot = downcast(*parent); m_contextStack.append(Context(shadowRoot, *contextCurrent, currentSlotNodeIndex)); node = shadowRoot.host(); contextCurrent = node; currentSlotNodeIndex = notFound; continue; } if (auto* shadowRoot = parent->shadowRoot()) { #if ENABLE(SHADOW_DOM) || ENABLE(DETAILS_ELEMENT) m_contextStack.append(Context(*parent, *contextCurrent, currentSlotNodeIndex)); auto* assignedSlot = shadowRoot->findAssignedSlot(*node); if (assignedSlot) { currentSlotNodeIndex = assignedSlot->assignedNodes()->find(node); ASSERT(currentSlotNodeIndex != notFound); node = assignedSlot; contextCurrent = assignedSlot; continue; } // The node is not part of the composed tree. #else UNUSED_PARAM(shadowRoot); #endif *this = { }; return; } node = parent; } m_contextStack.append(Context(root, *contextCurrent, currentSlotNodeIndex)); m_contextStack.reverse(); } void ComposedTreeIterator::dropAssertions() { for (auto& context : m_contextStack) context.iterator.dropAssertions(); m_didDropAssertions = true; } void ComposedTreeIterator::traverseShadowRoot(ShadowRoot& shadowRoot) { Context shadowContext(shadowRoot); if (!shadowContext.iterator) { // Empty shadow root. traverseNextSkippingChildren(); return; } if (m_didDropAssertions) shadowContext.iterator.dropAssertions(); m_contextStack.append(WTFMove(shadowContext)); } void ComposedTreeIterator::traverseNextInShadowTree() { ASSERT(m_contextStack.size() > 1); #if ENABLE(SHADOW_DOM) || ENABLE(DETAILS_ELEMENT) if (is(current())) { auto& slot = downcast(current()); if (auto* assignedNodes = slot.assignedNodes()) { context().slotNodeIndex = 0; auto* assignedNode = assignedNodes->at(0); m_contextStack.append(Context(*assignedNode->parentElement(), *assignedNode)); return; } } #endif context().iterator.traverseNext(); if (!context().iterator) traverseNextLeavingContext(); } void ComposedTreeIterator::traverseNextLeavingContext() { ASSERT(m_contextStack.size() > 1); while (!context().iterator && m_contextStack.size() > 1) { m_contextStack.removeLast(); if (!context().iterator) return; #if ENABLE(SHADOW_DOM) || ENABLE(DETAILS_ELEMENT) if (is(current()) && advanceInSlot(1)) return; #endif context().iterator.traverseNextSkippingChildren(); } } #if ENABLE(SHADOW_DOM) || ENABLE(DETAILS_ELEMENT) bool ComposedTreeIterator::advanceInSlot(int direction) { ASSERT(context().slotNodeIndex != notFound); auto& assignedNodes = *downcast(current()).assignedNodes(); // It is fine to underflow this. context().slotNodeIndex += direction; if (context().slotNodeIndex >= assignedNodes.size()) return false; auto* slotNode = assignedNodes.at(context().slotNodeIndex); m_contextStack.append(Context(*slotNode->parentElement(), *slotNode)); return true; } void ComposedTreeIterator::traverseSiblingInSlot(int direction) { ASSERT(m_contextStack.size() > 1); ASSERT(current().parentNode()->shadowRoot()); m_contextStack.removeLast(); if (!advanceInSlot(direction)) *this = { }; } #endif String composedTreeAsText(ContainerNode& root, ComposedTreeAsTextMode mode) { TextStream stream; auto descendants = composedTreeDescendants(root); for (auto it = descendants.begin(), end = descendants.end(); it != end; ++it) { writeIndent(stream, it.depth()); if (is(*it)) { stream << "#text"; if (mode == ComposedTreeAsTextMode::WithPointers) stream << " " << &*it; stream << "\n"; continue; } auto& element = downcast(*it); stream << element.localName(); if (element.shadowRoot()) stream << " (shadow root)"; if (mode == ComposedTreeAsTextMode::WithPointers) stream << " " << &*it; stream << "\n"; } return stream.release(); } }