diff options
Diffstat (limited to 'Source/WebInspectorUI/UserInterface/Views/DOMTreeUpdater.js')
-rw-r--r-- | Source/WebInspectorUI/UserInterface/Views/DOMTreeUpdater.js | 159 |
1 files changed, 159 insertions, 0 deletions
diff --git a/Source/WebInspectorUI/UserInterface/Views/DOMTreeUpdater.js b/Source/WebInspectorUI/UserInterface/Views/DOMTreeUpdater.js new file mode 100644 index 000000000..83f7ae77a --- /dev/null +++ b/Source/WebInspectorUI/UserInterface/Views/DOMTreeUpdater.js @@ -0,0 +1,159 @@ +/* + * Copyright (C) 2007, 2008, 2013, 2016 Apple Inc. All rights reserved. + * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com> + * 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(); + } +}; |