/* * Copyright (C) 2007, 2008, 2013, 2016 Apple Inc. All rights reserved. * Copyright (C) 2008 Matt Lilek * Copyright (C) 2009 Joseph Pecoraro * * 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. * 3. Neither the name of Apple Inc. ("Apple") 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 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. */ WebInspector.DOMTreeUpdater = function(treeOutline) { WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.NodeInserted, this._nodeInserted, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.NodeRemoved, this._nodeRemoved, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.AttributeModified, this._attributesUpdated, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.AttributeRemoved, this._attributesUpdated, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.CharacterDataModified, this._characterDataModified, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.DocumentUpdated, this._documentUpdated, this); WebInspector.domTreeManager.addEventListener(WebInspector.DOMTreeManager.Event.ChildNodeCountUpdated, this._childNodeCountUpdated, this); this._treeOutline = treeOutline; this._recentlyInsertedNodes = new Map; this._recentlyDeletedNodes = new Map; this._recentlyModifiedNodes = new Set; // Map from attribute names to nodes that had the attributes. this._recentlyModifiedAttributes = new Map; // Dummy "attribute" that is used to track textContent changes. this._textContentAttributeSymbol = Symbol("text-content-attribute"); }; WebInspector.DOMTreeUpdater.prototype = { close: function() { WebInspector.domTreeManager.removeEventListener(null, null, this); }, _documentUpdated: function(event) { this._reset(); }, _attributesUpdated: function(event) { let {node, name} = event.data; this._nodeAttributeModified(node, name); }, _characterDataModified: function(event) { let {node} = event.data; this._nodeAttributeModified(node, this._textContentAttributeSymbol); }, _nodeAttributeModified: function(node, attribute) { if (!this._recentlyModifiedAttributes.has(attribute)) this._recentlyModifiedAttributes.set(attribute, new Set); this._recentlyModifiedAttributes.get(attribute).add(node); this._recentlyModifiedNodes.add(node); if (this._treeOutline._visible) this.onNextFrame._updateModifiedNodes(); }, _nodeInserted: function(event) { this._recentlyInsertedNodes.set(event.data.node, {parent: event.data.parent}); if (this._treeOutline._visible) this.onNextFrame._updateModifiedNodes(); }, _nodeRemoved: function(event) { this._recentlyDeletedNodes.set(event.data.node, {parent: event.data.parent}); if (this._treeOutline._visible) this.onNextFrame._updateModifiedNodes(); }, _childNodeCountUpdated: function(event) { var treeElement = this._treeOutline.findTreeElement(event.data); if (treeElement) treeElement.hasChildren = event.data.hasChildNodes(); }, _updateModifiedNodes: function() { // Update for insertions and deletions before attribute modifications. This ensures // tree elements get created for newly attached children before we try to update them. let parentElementsToUpdate = new Set; let markNodeParentForUpdate = (value, key, map) => { let parentNode = value.parent; let parentTreeElement = this._treeOutline.findTreeElement(parentNode); if (parentTreeElement) parentElementsToUpdate.add(parentTreeElement); }; this._recentlyInsertedNodes.forEach(markNodeParentForUpdate); this._recentlyDeletedNodes.forEach(markNodeParentForUpdate); for (let parentTreeElement of parentElementsToUpdate) { parentTreeElement.updateTitle(); parentTreeElement.updateChildren(); } for (let node of this._recentlyModifiedNodes.values()) { let nodeTreeElement = this._treeOutline.findTreeElement(node); if (!nodeTreeElement) return; for (let [attribute, nodes] of this._recentlyModifiedAttributes.entries()) { // Don't report textContent changes as attribute modifications. if (attribute === this._textContentAttributeSymbol) continue; if (nodes.has(node)) nodeTreeElement.attributeDidChange(attribute); } nodeTreeElement.updateTitle(); } this._recentlyInsertedNodes.clear(); this._recentlyDeletedNodes.clear(); this._recentlyModifiedNodes.clear(); this._recentlyModifiedAttributes.clear(); }, _reset: function() { WebInspector.domTreeManager.hideDOMNodeHighlight(); this._recentlyInsertedNodes.clear(); this._recentlyDeletedNodes.clear(); this._recentlyModifiedNodes.clear(); this._recentlyModifiedAttributes.clear(); } };