| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved. | 2 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved. |
| 3 * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com> | 3 * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com> |
| 4 * Copyright (C) 2009 Joseph Pecoraro | 4 * Copyright (C) 2009 Joseph Pecoraro |
| 5 * | 5 * |
| 6 * Redistribution and use in source and binary forms, with or without | 6 * Redistribution and use in source and binary forms, with or without |
| 7 * modification, are permitted provided that the following conditions | 7 * modification, are permitted provided that the following conditions |
| 8 * are met: | 8 * are met: |
| 9 * | 9 * |
| 10 * 1. Redistributions of source code must retain the above copyright | 10 * 1. Redistributions of source code must retain the above copyright |
| 11 * notice, this list of conditions and the following disclaimer. | 11 * notice, this list of conditions and the following disclaimer. |
| 12 * 2. Redistributions in binary form must reproduce the above copyright | 12 * 2. Redistributions in binary form must reproduce the above copyright |
| 13 * notice, this list of conditions and the following disclaimer in the | 13 * notice, this list of conditions and the following disclaimer in the |
| 14 * documentation and/or other materials provided with the distribution. | 14 * documentation and/or other materials provided with the distribution. |
| 15 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of | 15 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
| 16 * its contributors may be used to endorse or promote products derived | 16 * its contributors may be used to endorse or promote products derived |
| 17 * from this software without specific prior written permission. | 17 * from this software without specific prior written permission. |
| 18 * | 18 * |
| 19 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY | 19 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| 20 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | 20 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| 21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| 22 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY | 22 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| 23 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 23 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| 24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| 25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | 25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| 26 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 26 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| 28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 29 */ | 29 */ |
| 30 | |
| 31 /** | 30 /** |
| 32 * @constructor | 31 * @unrestricted |
| 33 * @extends {TreeOutline} | |
| 34 * @param {!WebInspector.DOMModel} domModel | |
| 35 * @param {boolean=} omitRootDOMNode | |
| 36 * @param {boolean=} selectEnabled | |
| 37 */ | 32 */ |
| 38 WebInspector.ElementsTreeOutline = function(domModel, omitRootDOMNode, selectEna
bled) | 33 WebInspector.ElementsTreeOutline = class extends TreeOutline { |
| 39 { | 34 /** |
| 40 TreeOutline.call(this); | 35 * @param {!WebInspector.DOMModel} domModel |
| 36 * @param {boolean=} omitRootDOMNode |
| 37 * @param {boolean=} selectEnabled |
| 38 */ |
| 39 constructor(domModel, omitRootDOMNode, selectEnabled) { |
| 40 super(); |
| 41 | 41 |
| 42 this._domModel = domModel; | 42 this._domModel = domModel; |
| 43 this._treeElementSymbol = Symbol("treeElement"); | 43 this._treeElementSymbol = Symbol('treeElement'); |
| 44 var shadowContainer = createElement("div"); | 44 var shadowContainer = createElement('div'); |
| 45 this._shadowRoot = WebInspector.createShadowRootWithCoreStyles(shadowContain
er, "elements/elementsTreeOutline.css"); | 45 this._shadowRoot = WebInspector.createShadowRootWithCoreStyles(shadowContain
er, 'elements/elementsTreeOutline.css'); |
| 46 var outlineDisclosureElement = this._shadowRoot.createChild("div", "elements
-disclosure"); | 46 var outlineDisclosureElement = this._shadowRoot.createChild('div', 'elements
-disclosure'); |
| 47 | 47 |
| 48 this._element = this.element; | 48 this._element = this.element; |
| 49 this._element.classList.add("elements-tree-outline", "source-code"); | 49 this._element.classList.add('elements-tree-outline', 'source-code'); |
| 50 this._element.addEventListener("mousedown", this._onmousedown.bind(this), fa
lse); | 50 this._element.addEventListener('mousedown', this._onmousedown.bind(this), fa
lse); |
| 51 this._element.addEventListener("mousemove", this._onmousemove.bind(this), fa
lse); | 51 this._element.addEventListener('mousemove', this._onmousemove.bind(this), fa
lse); |
| 52 this._element.addEventListener("mouseleave", this._onmouseleave.bind(this),
false); | 52 this._element.addEventListener('mouseleave', this._onmouseleave.bind(this),
false); |
| 53 this._element.addEventListener("dragstart", this._ondragstart.bind(this), fa
lse); | 53 this._element.addEventListener('dragstart', this._ondragstart.bind(this), fa
lse); |
| 54 this._element.addEventListener("dragover", this._ondragover.bind(this), fals
e); | 54 this._element.addEventListener('dragover', this._ondragover.bind(this), fals
e); |
| 55 this._element.addEventListener("dragleave", this._ondragleave.bind(this), fa
lse); | 55 this._element.addEventListener('dragleave', this._ondragleave.bind(this), fa
lse); |
| 56 this._element.addEventListener("drop", this._ondrop.bind(this), false); | 56 this._element.addEventListener('drop', this._ondrop.bind(this), false); |
| 57 this._element.addEventListener("dragend", this._ondragend.bind(this), false)
; | 57 this._element.addEventListener('dragend', this._ondragend.bind(this), false)
; |
| 58 this._element.addEventListener("contextmenu", this._contextMenuEventFired.bi
nd(this), false); | 58 this._element.addEventListener('contextmenu', this._contextMenuEventFired.bi
nd(this), false); |
| 59 this._element.addEventListener("clipboard-beforecopy", this._onBeforeCopy.bi
nd(this), false); | 59 this._element.addEventListener('clipboard-beforecopy', this._onBeforeCopy.bi
nd(this), false); |
| 60 this._element.addEventListener("clipboard-copy", this._onCopyOrCut.bind(this
, false), false); | 60 this._element.addEventListener('clipboard-copy', this._onCopyOrCut.bind(this
, false), false); |
| 61 this._element.addEventListener("clipboard-cut", this._onCopyOrCut.bind(this,
true), false); | 61 this._element.addEventListener('clipboard-cut', this._onCopyOrCut.bind(this,
true), false); |
| 62 this._element.addEventListener("clipboard-paste", this._onPaste.bind(this),
false); | 62 this._element.addEventListener('clipboard-paste', this._onPaste.bind(this),
false); |
| 63 | 63 |
| 64 outlineDisclosureElement.appendChild(this._element); | 64 outlineDisclosureElement.appendChild(this._element); |
| 65 this.element = shadowContainer; | 65 this.element = shadowContainer; |
| 66 | 66 |
| 67 this._includeRootDOMNode = !omitRootDOMNode; | 67 this._includeRootDOMNode = !omitRootDOMNode; |
| 68 this._selectEnabled = selectEnabled; | 68 this._selectEnabled = selectEnabled; |
| 69 /** @type {?WebInspector.DOMNode} */ | 69 /** @type {?WebInspector.DOMNode} */ |
| 70 this._rootDOMNode = null; | 70 this._rootDOMNode = null; |
| 71 /** @type {?WebInspector.DOMNode} */ | 71 /** @type {?WebInspector.DOMNode} */ |
| 72 this._selectedDOMNode = null; | 72 this._selectedDOMNode = null; |
| 73 | 73 |
| 74 this._visible = false; | 74 this._visible = false; |
| 75 | 75 |
| 76 this._popoverHelper = new WebInspector.PopoverHelper(this._element); | 76 this._popoverHelper = new WebInspector.PopoverHelper(this._element); |
| 77 this._popoverHelper.initializeCallbacks(this._getPopoverAnchor.bind(this), t
his._showPopover.bind(this)); | 77 this._popoverHelper.initializeCallbacks(this._getPopoverAnchor.bind(this), t
his._showPopover.bind(this)); |
| 78 this._popoverHelper.setTimeout(0, 100); | 78 this._popoverHelper.setTimeout(0, 100); |
| 79 | 79 |
| 80 /** @type {!Map<!WebInspector.DOMNode, !WebInspector.ElementsTreeOutline.Upd
ateRecord>} */ | 80 /** @type {!Map<!WebInspector.DOMNode, !WebInspector.ElementsTreeOutline.Upd
ateRecord>} */ |
| 81 this._updateRecords = new Map(); | 81 this._updateRecords = new Map(); |
| 82 /** @type {!Set<!WebInspector.ElementsTreeElement>} */ | 82 /** @type {!Set<!WebInspector.ElementsTreeElement>} */ |
| 83 this._treeElementsBeingUpdated = new Set(); | 83 this._treeElementsBeingUpdated = new Set(); |
| 84 | 84 |
| 85 this._domModel.addEventListener(WebInspector.DOMModel.Events.MarkersChanged,
this._markersChanged, this); | 85 this._domModel.addEventListener(WebInspector.DOMModel.Events.MarkersChanged,
this._markersChanged, this); |
| 86 this._showHTMLCommentsSetting = WebInspector.moduleSetting("showHTMLComments
"); | 86 this._showHTMLCommentsSetting = WebInspector.moduleSetting('showHTMLComments
'); |
| 87 this._showHTMLCommentsSetting.addChangeListener(this._onShowHTMLCommentsChan
ge.bind(this)); | 87 this._showHTMLCommentsSetting.addChangeListener(this._onShowHTMLCommentsChan
ge.bind(this)); |
| 88 } |
| 89 |
| 90 /** |
| 91 * @param {!WebInspector.DOMModel} domModel |
| 92 * @return {?WebInspector.ElementsTreeOutline} |
| 93 */ |
| 94 static forDOMModel(domModel) { |
| 95 return domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbol] || null
; |
| 96 } |
| 97 |
| 98 _onShowHTMLCommentsChange() { |
| 99 var selectedNode = this.selectedDOMNode(); |
| 100 if (selectedNode && selectedNode.nodeType() === Node.COMMENT_NODE && !this._
showHTMLCommentsSetting.get()) |
| 101 this.selectDOMNode(selectedNode.parentNode); |
| 102 this.update(); |
| 103 } |
| 104 |
| 105 /** |
| 106 * @return {symbol} |
| 107 */ |
| 108 treeElementSymbol() { |
| 109 return this._treeElementSymbol; |
| 110 } |
| 111 |
| 112 /** |
| 113 * @override |
| 114 */ |
| 115 focus() { |
| 116 this._element.focus(); |
| 117 } |
| 118 |
| 119 /** |
| 120 * @param {boolean} wrap |
| 121 */ |
| 122 setWordWrap(wrap) { |
| 123 this._element.classList.toggle('elements-tree-nowrap', !wrap); |
| 124 } |
| 125 |
| 126 /** |
| 127 * @return {!WebInspector.DOMModel} |
| 128 */ |
| 129 domModel() { |
| 130 return this._domModel; |
| 131 } |
| 132 |
| 133 /** |
| 134 * @param {?WebInspector.InplaceEditor.Controller} multilineEditing |
| 135 */ |
| 136 setMultilineEditing(multilineEditing) { |
| 137 this._multilineEditing = multilineEditing; |
| 138 } |
| 139 |
| 140 /** |
| 141 * @return {number} |
| 142 */ |
| 143 visibleWidth() { |
| 144 return this._visibleWidth; |
| 145 } |
| 146 |
| 147 /** |
| 148 * @param {number} width |
| 149 */ |
| 150 setVisibleWidth(width) { |
| 151 this._visibleWidth = width; |
| 152 if (this._multilineEditing) |
| 153 this._multilineEditing.setWidth(this._visibleWidth); |
| 154 } |
| 155 |
| 156 /** |
| 157 * @param {?WebInspector.ElementsTreeOutline.ClipboardData} data |
| 158 */ |
| 159 _setClipboardData(data) { |
| 160 if (this._clipboardNodeData) { |
| 161 var treeElement = this.findTreeElement(this._clipboardNodeData.node); |
| 162 if (treeElement) |
| 163 treeElement.setInClipboard(false); |
| 164 delete this._clipboardNodeData; |
| 165 } |
| 166 |
| 167 if (data) { |
| 168 var treeElement = this.findTreeElement(data.node); |
| 169 if (treeElement) |
| 170 treeElement.setInClipboard(true); |
| 171 this._clipboardNodeData = data; |
| 172 } |
| 173 } |
| 174 |
| 175 /** |
| 176 * @param {!WebInspector.DOMNode} removedNode |
| 177 */ |
| 178 resetClipboardIfNeeded(removedNode) { |
| 179 if (this._clipboardNodeData && this._clipboardNodeData.node === removedNode) |
| 180 this._setClipboardData(null); |
| 181 } |
| 182 |
| 183 /** |
| 184 * @param {!Event} event |
| 185 */ |
| 186 _onBeforeCopy(event) { |
| 187 event.handled = true; |
| 188 } |
| 189 |
| 190 /** |
| 191 * @param {boolean} isCut |
| 192 * @param {!Event} event |
| 193 */ |
| 194 _onCopyOrCut(isCut, event) { |
| 195 this._setClipboardData(null); |
| 196 var originalEvent = event['original']; |
| 197 |
| 198 // Don't prevent the normal copy if the user has a selection. |
| 199 if (!originalEvent.target.isComponentSelectionCollapsed()) |
| 200 return; |
| 201 |
| 202 // Do not interfere with text editing. |
| 203 if (WebInspector.isEditing()) |
| 204 return; |
| 205 |
| 206 var targetNode = this.selectedDOMNode(); |
| 207 if (!targetNode) |
| 208 return; |
| 209 |
| 210 originalEvent.clipboardData.clearData(); |
| 211 event.handled = true; |
| 212 |
| 213 this.performCopyOrCut(isCut, targetNode); |
| 214 } |
| 215 |
| 216 /** |
| 217 * @param {boolean} isCut |
| 218 * @param {?WebInspector.DOMNode} node |
| 219 */ |
| 220 performCopyOrCut(isCut, node) { |
| 221 if (isCut && (node.isShadowRoot() || node.ancestorUserAgentShadowRoot())) |
| 222 return; |
| 223 |
| 224 node.copyNode(); |
| 225 this._setClipboardData({node: node, isCut: isCut}); |
| 226 } |
| 227 |
| 228 /** |
| 229 * @param {!WebInspector.DOMNode} targetNode |
| 230 * @return {boolean} |
| 231 */ |
| 232 canPaste(targetNode) { |
| 233 if (targetNode.isShadowRoot() || targetNode.ancestorUserAgentShadowRoot()) |
| 234 return false; |
| 235 |
| 236 if (!this._clipboardNodeData) |
| 237 return false; |
| 238 |
| 239 var node = this._clipboardNodeData.node; |
| 240 if (this._clipboardNodeData.isCut && (node === targetNode || node.isAncestor
(targetNode))) |
| 241 return false; |
| 242 |
| 243 if (targetNode.target() !== node.target()) |
| 244 return false; |
| 245 return true; |
| 246 } |
| 247 |
| 248 /** |
| 249 * @param {!WebInspector.DOMNode} targetNode |
| 250 */ |
| 251 pasteNode(targetNode) { |
| 252 if (this.canPaste(targetNode)) |
| 253 this._performPaste(targetNode); |
| 254 } |
| 255 |
| 256 /** |
| 257 * @param {!Event} event |
| 258 */ |
| 259 _onPaste(event) { |
| 260 // Do not interfere with text editing. |
| 261 if (WebInspector.isEditing()) |
| 262 return; |
| 263 |
| 264 var targetNode = this.selectedDOMNode(); |
| 265 if (!targetNode || !this.canPaste(targetNode)) |
| 266 return; |
| 267 |
| 268 event.handled = true; |
| 269 this._performPaste(targetNode); |
| 270 } |
| 271 |
| 272 /** |
| 273 * @param {!WebInspector.DOMNode} targetNode |
| 274 */ |
| 275 _performPaste(targetNode) { |
| 276 if (this._clipboardNodeData.isCut) { |
| 277 this._clipboardNodeData.node.moveTo(targetNode, null, expandCallback.bind(
this)); |
| 278 this._setClipboardData(null); |
| 279 } else { |
| 280 this._clipboardNodeData.node.copyTo(targetNode, null, expandCallback.bind(
this)); |
| 281 } |
| 282 |
| 283 /** |
| 284 * @param {?Protocol.Error} error |
| 285 * @param {!DOMAgent.NodeId} nodeId |
| 286 * @this {WebInspector.ElementsTreeOutline} |
| 287 */ |
| 288 function expandCallback(error, nodeId) { |
| 289 if (error) |
| 290 return; |
| 291 var pastedNode = this._domModel.nodeForId(nodeId); |
| 292 if (!pastedNode) |
| 293 return; |
| 294 this.selectDOMNode(pastedNode); |
| 295 } |
| 296 } |
| 297 |
| 298 /** |
| 299 * @param {boolean} visible |
| 300 */ |
| 301 setVisible(visible) { |
| 302 this._visible = visible; |
| 303 if (!this._visible) { |
| 304 this._popoverHelper.hidePopover(); |
| 305 if (this._multilineEditing) |
| 306 this._multilineEditing.cancel(); |
| 307 return; |
| 308 } |
| 309 |
| 310 this.runPendingUpdates(); |
| 311 if (this._selectedDOMNode) |
| 312 this._revealAndSelectNode(this._selectedDOMNode, false); |
| 313 } |
| 314 |
| 315 get rootDOMNode() { |
| 316 return this._rootDOMNode; |
| 317 } |
| 318 |
| 319 set rootDOMNode(x) { |
| 320 if (this._rootDOMNode === x) |
| 321 return; |
| 322 |
| 323 this._rootDOMNode = x; |
| 324 |
| 325 this._isXMLMimeType = x && x.isXMLNode(); |
| 326 |
| 327 this.update(); |
| 328 } |
| 329 |
| 330 get isXMLMimeType() { |
| 331 return this._isXMLMimeType; |
| 332 } |
| 333 |
| 334 /** |
| 335 * @return {?WebInspector.DOMNode} |
| 336 */ |
| 337 selectedDOMNode() { |
| 338 return this._selectedDOMNode; |
| 339 } |
| 340 |
| 341 /** |
| 342 * @param {?WebInspector.DOMNode} node |
| 343 * @param {boolean=} focus |
| 344 */ |
| 345 selectDOMNode(node, focus) { |
| 346 if (this._selectedDOMNode === node) { |
| 347 this._revealAndSelectNode(node, !focus); |
| 348 return; |
| 349 } |
| 350 |
| 351 this._selectedDOMNode = node; |
| 352 this._revealAndSelectNode(node, !focus); |
| 353 |
| 354 // The _revealAndSelectNode() method might find a different element if there
is inlined text, |
| 355 // and the select() call would change the selectedDOMNode and reenter this s
etter. So to |
| 356 // avoid calling _selectedNodeChanged() twice, first check if _selectedDOMNo
de is the same |
| 357 // node as the one passed in. |
| 358 if (this._selectedDOMNode === node) |
| 359 this._selectedNodeChanged(!!focus); |
| 360 } |
| 361 |
| 362 /** |
| 363 * @return {boolean} |
| 364 */ |
| 365 editing() { |
| 366 var node = this.selectedDOMNode(); |
| 367 if (!node) |
| 368 return false; |
| 369 var treeElement = this.findTreeElement(node); |
| 370 if (!treeElement) |
| 371 return false; |
| 372 return treeElement.isEditing() || false; |
| 373 } |
| 374 |
| 375 update() { |
| 376 var selectedNode = this.selectedDOMNode(); |
| 377 this.removeChildren(); |
| 378 if (!this.rootDOMNode) |
| 379 return; |
| 380 |
| 381 if (this._includeRootDOMNode) { |
| 382 var treeElement = this._createElementTreeElement(this.rootDOMNode); |
| 383 this.appendChild(treeElement); |
| 384 } else { |
| 385 // FIXME: this could use findTreeElement to reuse a tree element if it alr
eady exists |
| 386 var children = this._visibleChildren(this.rootDOMNode); |
| 387 for (var child of children) { |
| 388 var treeElement = this._createElementTreeElement(child); |
| 389 this.appendChild(treeElement); |
| 390 } |
| 391 } |
| 392 |
| 393 if (selectedNode) |
| 394 this._revealAndSelectNode(selectedNode, true); |
| 395 } |
| 396 |
| 397 /** |
| 398 * @param {boolean} focus |
| 399 */ |
| 400 _selectedNodeChanged(focus) { |
| 401 this.dispatchEventToListeners( |
| 402 WebInspector.ElementsTreeOutline.Events.SelectedNodeChanged, {node: this
._selectedDOMNode, focus: focus}); |
| 403 } |
| 404 |
| 405 /** |
| 406 * @param {!Array.<!WebInspector.DOMNode>} nodes |
| 407 */ |
| 408 _fireElementsTreeUpdated(nodes) { |
| 409 this.dispatchEventToListeners(WebInspector.ElementsTreeOutline.Events.Elemen
tsTreeUpdated, nodes); |
| 410 } |
| 411 |
| 412 /** |
| 413 * @param {!WebInspector.DOMNode} node |
| 414 * @return {?WebInspector.ElementsTreeElement} |
| 415 */ |
| 416 findTreeElement(node) { |
| 417 var treeElement = this._lookUpTreeElement(node); |
| 418 if (!treeElement && node.nodeType() === Node.TEXT_NODE) { |
| 419 // The text node might have been inlined if it was short, so try to find t
he parent element. |
| 420 treeElement = this._lookUpTreeElement(node.parentNode); |
| 421 } |
| 422 |
| 423 return /** @type {?WebInspector.ElementsTreeElement} */ (treeElement); |
| 424 } |
| 425 |
| 426 /** |
| 427 * @param {?WebInspector.DOMNode} node |
| 428 * @return {?TreeElement} |
| 429 */ |
| 430 _lookUpTreeElement(node) { |
| 431 if (!node) |
| 432 return null; |
| 433 |
| 434 var cachedElement = node[this._treeElementSymbol]; |
| 435 if (cachedElement) |
| 436 return cachedElement; |
| 437 |
| 438 // Walk up the parent pointers from the desired node |
| 439 var ancestors = []; |
| 440 for (var currentNode = node.parentNode; currentNode; currentNode = currentNo
de.parentNode) { |
| 441 ancestors.push(currentNode); |
| 442 if (currentNode[this._treeElementSymbol]) // stop climbing as soon as we
hit |
| 443 break; |
| 444 } |
| 445 |
| 446 if (!currentNode) |
| 447 return null; |
| 448 |
| 449 // Walk down to populate each ancestor's children, to fill in the tree and t
he cache. |
| 450 for (var i = ancestors.length - 1; i >= 0; --i) { |
| 451 var treeElement = ancestors[i][this._treeElementSymbol]; |
| 452 if (treeElement) |
| 453 treeElement.onpopulate(); // fill the cache with the children of treeEl
ement |
| 454 } |
| 455 |
| 456 return node[this._treeElementSymbol]; |
| 457 } |
| 458 |
| 459 /** |
| 460 * @param {!WebInspector.DOMNode} node |
| 461 * @return {?WebInspector.ElementsTreeElement} |
| 462 */ |
| 463 createTreeElementFor(node) { |
| 464 var treeElement = this.findTreeElement(node); |
| 465 if (treeElement) |
| 466 return treeElement; |
| 467 if (!node.parentNode) |
| 468 return null; |
| 469 |
| 470 treeElement = this.createTreeElementFor(node.parentNode); |
| 471 return treeElement ? this._showChild(treeElement, node) : null; |
| 472 } |
| 473 |
| 474 set suppressRevealAndSelect(x) { |
| 475 if (this._suppressRevealAndSelect === x) |
| 476 return; |
| 477 this._suppressRevealAndSelect = x; |
| 478 } |
| 479 |
| 480 /** |
| 481 * @param {?WebInspector.DOMNode} node |
| 482 * @param {boolean} omitFocus |
| 483 */ |
| 484 _revealAndSelectNode(node, omitFocus) { |
| 485 if (this._suppressRevealAndSelect) |
| 486 return; |
| 487 |
| 488 if (!this._includeRootDOMNode && node === this.rootDOMNode && this.rootDOMNo
de) |
| 489 node = this.rootDOMNode.firstChild; |
| 490 if (!node) |
| 491 return; |
| 492 var treeElement = this.createTreeElementFor(node); |
| 493 if (!treeElement) |
| 494 return; |
| 495 |
| 496 treeElement.revealAndSelect(omitFocus); |
| 497 } |
| 498 |
| 499 /** |
| 500 * @return {?TreeElement} |
| 501 */ |
| 502 _treeElementFromEvent(event) { |
| 503 var scrollContainer = this.element.parentElement; |
| 504 |
| 505 // We choose this X coordinate based on the knowledge that our list |
| 506 // items extend at least to the right edge of the outer <ol> container. |
| 507 // In the no-word-wrap mode the outer <ol> may be wider than the tree contai
ner |
| 508 // (and partially hidden), in which case we are left to use only its right b
oundary. |
| 509 var x = scrollContainer.totalOffsetLeft() + scrollContainer.offsetWidth - 36
; |
| 510 |
| 511 var y = event.pageY; |
| 512 |
| 513 // Our list items have 1-pixel cracks between them vertically. We avoid |
| 514 // the cracks by checking slightly above and slightly below the mouse |
| 515 // and seeing if we hit the same element each time. |
| 516 var elementUnderMouse = this.treeElementFromPoint(x, y); |
| 517 var elementAboveMouse = this.treeElementFromPoint(x, y - 2); |
| 518 var element; |
| 519 if (elementUnderMouse === elementAboveMouse) |
| 520 element = elementUnderMouse; |
| 521 else |
| 522 element = this.treeElementFromPoint(x, y + 2); |
| 523 |
| 524 return element; |
| 525 } |
| 526 |
| 527 /** |
| 528 * @param {!Element} element |
| 529 * @param {!Event} event |
| 530 * @return {!Element|!AnchorBox|undefined} |
| 531 */ |
| 532 _getPopoverAnchor(element, event) { |
| 533 var anchor = element.enclosingNodeOrSelfWithClass('webkit-html-resource-link
'); |
| 534 if (!anchor || !anchor.href) |
| 535 return; |
| 536 |
| 537 return anchor; |
| 538 } |
| 539 |
| 540 /** |
| 541 * @param {!WebInspector.DOMNode} node |
| 542 * @param {function()} callback |
| 543 */ |
| 544 _loadDimensionsForNode(node, callback) { |
| 545 if (!node.nodeName() || node.nodeName().toLowerCase() !== 'img') { |
| 546 callback(); |
| 547 return; |
| 548 } |
| 549 |
| 550 node.resolveToObject('', resolvedNode); |
| 551 |
| 552 function resolvedNode(object) { |
| 553 if (!object) { |
| 554 callback(); |
| 555 return; |
| 556 } |
| 557 |
| 558 object.callFunctionJSON(features, undefined, callback); |
| 559 object.release(); |
| 560 |
| 561 /** |
| 562 * @return {!{offsetWidth: number, offsetHeight: number, naturalWidth: num
ber, naturalHeight: number, currentSrc: (string|undefined)}} |
| 563 * @suppressReceiverCheck |
| 564 * @this {!Element} |
| 565 */ |
| 566 function features() { |
| 567 return { |
| 568 offsetWidth: this.offsetWidth, |
| 569 offsetHeight: this.offsetHeight, |
| 570 naturalWidth: this.naturalWidth, |
| 571 naturalHeight: this.naturalHeight, |
| 572 currentSrc: this.currentSrc |
| 573 }; |
| 574 } |
| 575 } |
| 576 } |
| 577 |
| 578 /** |
| 579 * @param {!Element} anchor |
| 580 * @param {!WebInspector.Popover} popover |
| 581 */ |
| 582 _showPopover(anchor, popover) { |
| 583 var listItem = anchor.enclosingNodeOrSelfWithNodeName('li'); |
| 584 var node = /** @type {!WebInspector.ElementsTreeElement} */ (listItem.treeEl
ement).node(); |
| 585 this._loadDimensionsForNode( |
| 586 node, WebInspector.DOMPresentationUtils.buildImagePreviewContents.bind( |
| 587 WebInspector.DOMPresentationUtils, node.target(), anchor.href,
true, showPopover)); |
| 588 |
| 589 /** |
| 590 * @param {!Element=} contents |
| 591 */ |
| 592 function showPopover(contents) { |
| 593 if (!contents) |
| 594 return; |
| 595 popover.setCanShrink(false); |
| 596 popover.showForAnchor(contents, anchor); |
| 597 } |
| 598 } |
| 599 |
| 600 _onmousedown(event) { |
| 601 var element = this._treeElementFromEvent(event); |
| 602 |
| 603 if (!element || element.isEventWithinDisclosureTriangle(event)) |
| 604 return; |
| 605 |
| 606 element.select(); |
| 607 } |
| 608 |
| 609 /** |
| 610 * @param {?TreeElement} treeElement |
| 611 */ |
| 612 setHoverEffect(treeElement) { |
| 613 if (this._previousHoveredElement === treeElement) |
| 614 return; |
| 615 |
| 616 if (this._previousHoveredElement) { |
| 617 this._previousHoveredElement.hovered = false; |
| 618 delete this._previousHoveredElement; |
| 619 } |
| 620 |
| 621 if (treeElement) { |
| 622 treeElement.hovered = true; |
| 623 this._previousHoveredElement = treeElement; |
| 624 } |
| 625 } |
| 626 |
| 627 _onmousemove(event) { |
| 628 var element = this._treeElementFromEvent(event); |
| 629 if (element && this._previousHoveredElement === element) |
| 630 return; |
| 631 |
| 632 this.setHoverEffect(element); |
| 633 |
| 634 if (element instanceof WebInspector.ElementsTreeElement) { |
| 635 this._domModel.highlightDOMNodeWithConfig( |
| 636 element.node().id, {mode: 'all', showInfo: !WebInspector.KeyboardShort
cut.eventHasCtrlOrMeta(event)}); |
| 637 return; |
| 638 } |
| 639 |
| 640 if (element instanceof WebInspector.ElementsTreeOutline.ShortcutTreeElement) |
| 641 this._domModel.highlightDOMNodeWithConfig( |
| 642 undefined, {mode: 'all', showInfo: !WebInspector.KeyboardShortcut.even
tHasCtrlOrMeta(event)}, |
| 643 element.backendNodeId()); |
| 644 } |
| 645 |
| 646 _onmouseleave(event) { |
| 647 this.setHoverEffect(null); |
| 648 WebInspector.DOMModel.hideDOMNodeHighlight(); |
| 649 } |
| 650 |
| 651 _ondragstart(event) { |
| 652 if (!event.target.isComponentSelectionCollapsed()) |
| 653 return false; |
| 654 if (event.target.nodeName === 'A') |
| 655 return false; |
| 656 |
| 657 var treeElement = this._treeElementFromEvent(event); |
| 658 if (!this._isValidDragSourceOrTarget(treeElement)) |
| 659 return false; |
| 660 |
| 661 if (treeElement.node().nodeName() === 'BODY' || treeElement.node().nodeName(
) === 'HEAD') |
| 662 return false; |
| 663 |
| 664 event.dataTransfer.setData('text/plain', treeElement.listItemElement.textCon
tent.replace(/\u200b/g, '')); |
| 665 event.dataTransfer.effectAllowed = 'copyMove'; |
| 666 this._treeElementBeingDragged = treeElement; |
| 667 |
| 668 WebInspector.DOMModel.hideDOMNodeHighlight(); |
| 669 |
| 670 return true; |
| 671 } |
| 672 |
| 673 _ondragover(event) { |
| 674 if (!this._treeElementBeingDragged) |
| 675 return false; |
| 676 |
| 677 var treeElement = this._treeElementFromEvent(event); |
| 678 if (!this._isValidDragSourceOrTarget(treeElement)) |
| 679 return false; |
| 680 |
| 681 var node = treeElement.node(); |
| 682 while (node) { |
| 683 if (node === this._treeElementBeingDragged._node) |
| 684 return false; |
| 685 node = node.parentNode; |
| 686 } |
| 687 |
| 688 treeElement.listItemElement.classList.add('elements-drag-over'); |
| 689 this._dragOverTreeElement = treeElement; |
| 690 event.preventDefault(); |
| 691 event.dataTransfer.dropEffect = 'move'; |
| 692 return false; |
| 693 } |
| 694 |
| 695 _ondragleave(event) { |
| 696 this._clearDragOverTreeElementMarker(); |
| 697 event.preventDefault(); |
| 698 return false; |
| 699 } |
| 700 |
| 701 /** |
| 702 * @param {?TreeElement} treeElement |
| 703 * @return {boolean} |
| 704 */ |
| 705 _isValidDragSourceOrTarget(treeElement) { |
| 706 if (!treeElement) |
| 707 return false; |
| 708 |
| 709 if (!(treeElement instanceof WebInspector.ElementsTreeElement)) |
| 710 return false; |
| 711 var elementsTreeElement = /** @type {!WebInspector.ElementsTreeElement} */ (
treeElement); |
| 712 |
| 713 var node = elementsTreeElement.node(); |
| 714 if (!node.parentNode || node.parentNode.nodeType() !== Node.ELEMENT_NODE) |
| 715 return false; |
| 716 |
| 717 return true; |
| 718 } |
| 719 |
| 720 _ondrop(event) { |
| 721 event.preventDefault(); |
| 722 var treeElement = this._treeElementFromEvent(event); |
| 723 if (treeElement) |
| 724 this._doMove(treeElement); |
| 725 } |
| 726 |
| 727 /** |
| 728 * @param {!TreeElement} treeElement |
| 729 */ |
| 730 _doMove(treeElement) { |
| 731 if (!this._treeElementBeingDragged) |
| 732 return; |
| 733 |
| 734 var parentNode; |
| 735 var anchorNode; |
| 736 |
| 737 if (treeElement.isClosingTag()) { |
| 738 // Drop onto closing tag -> insert as last child. |
| 739 parentNode = treeElement.node(); |
| 740 } else { |
| 741 var dragTargetNode = treeElement.node(); |
| 742 parentNode = dragTargetNode.parentNode; |
| 743 anchorNode = dragTargetNode; |
| 744 } |
| 745 |
| 746 var wasExpanded = this._treeElementBeingDragged.expanded; |
| 747 this._treeElementBeingDragged._node.moveTo( |
| 748 parentNode, anchorNode, this.selectNodeAfterEdit.bind(this, wasExpanded)
); |
| 749 |
| 750 delete this._treeElementBeingDragged; |
| 751 } |
| 752 |
| 753 _ondragend(event) { |
| 754 event.preventDefault(); |
| 755 this._clearDragOverTreeElementMarker(); |
| 756 delete this._treeElementBeingDragged; |
| 757 } |
| 758 |
| 759 _clearDragOverTreeElementMarker() { |
| 760 if (this._dragOverTreeElement) { |
| 761 this._dragOverTreeElement.listItemElement.classList.remove('elements-drag-
over'); |
| 762 delete this._dragOverTreeElement; |
| 763 } |
| 764 } |
| 765 |
| 766 _contextMenuEventFired(event) { |
| 767 var treeElement = this._treeElementFromEvent(event); |
| 768 if (treeElement instanceof WebInspector.ElementsTreeElement) |
| 769 this.showContextMenu(treeElement, event); |
| 770 } |
| 771 |
| 772 /** |
| 773 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 774 * @param {!Event} event |
| 775 */ |
| 776 showContextMenu(treeElement, event) { |
| 777 if (WebInspector.isEditing()) |
| 778 return; |
| 779 |
| 780 var contextMenu = new WebInspector.ContextMenu(event); |
| 781 var isPseudoElement = !!treeElement.node().pseudoType(); |
| 782 var isTag = treeElement.node().nodeType() === Node.ELEMENT_NODE && !isPseudo
Element; |
| 783 var textNode = event.target.enclosingNodeOrSelfWithClass('webkit-html-text-n
ode'); |
| 784 if (textNode && textNode.classList.contains('bogus')) |
| 785 textNode = null; |
| 786 var commentNode = event.target.enclosingNodeOrSelfWithClass('webkit-html-com
ment'); |
| 787 contextMenu.appendApplicableItems(event.target); |
| 788 if (textNode) { |
| 789 contextMenu.appendSeparator(); |
| 790 treeElement.populateTextContextMenu(contextMenu, textNode); |
| 791 } else if (isTag) { |
| 792 contextMenu.appendSeparator(); |
| 793 treeElement.populateTagContextMenu(contextMenu, event); |
| 794 } else if (commentNode) { |
| 795 contextMenu.appendSeparator(); |
| 796 treeElement.populateNodeContextMenu(contextMenu); |
| 797 } else if (isPseudoElement) { |
| 798 treeElement.populateScrollIntoView(contextMenu); |
| 799 } |
| 800 |
| 801 contextMenu.appendApplicableItems(treeElement.node()); |
| 802 contextMenu.show(); |
| 803 } |
| 804 |
| 805 runPendingUpdates() { |
| 806 this._updateModifiedNodes(); |
| 807 } |
| 808 |
| 809 handleShortcut(event) { |
| 810 var node = this.selectedDOMNode(); |
| 811 if (!node) |
| 812 return; |
| 813 var treeElement = node[this._treeElementSymbol]; |
| 814 if (!treeElement) |
| 815 return; |
| 816 |
| 817 if (WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) && node.parentNo
de) { |
| 818 if (event.key === 'ArrowUp' && node.previousSibling) { |
| 819 node.moveTo(node.parentNode, node.previousSibling, this.selectNodeAfterE
dit.bind(this, treeElement.expanded)); |
| 820 event.handled = true; |
| 821 return; |
| 822 } |
| 823 if (event.key === 'ArrowDown' && node.nextSibling) { |
| 824 node.moveTo( |
| 825 node.parentNode, node.nextSibling.nextSibling, this.selectNodeAfterE
dit.bind(this, treeElement.expanded)); |
| 826 event.handled = true; |
| 827 return; |
| 828 } |
| 829 } |
| 830 } |
| 831 |
| 832 /** |
| 833 * @param {!WebInspector.DOMNode} node |
| 834 * @param {boolean=} startEditing |
| 835 * @param {function()=} callback |
| 836 */ |
| 837 toggleEditAsHTML(node, startEditing, callback) { |
| 838 var treeElement = node[this._treeElementSymbol]; |
| 839 if (!treeElement || !treeElement.hasEditableNode()) |
| 840 return; |
| 841 |
| 842 if (node.pseudoType()) |
| 843 return; |
| 844 |
| 845 var parentNode = node.parentNode; |
| 846 var index = node.index; |
| 847 var wasExpanded = treeElement.expanded; |
| 848 |
| 849 treeElement.toggleEditAsHTML(editingFinished.bind(this), startEditing); |
| 850 |
| 851 /** |
| 852 * @this {WebInspector.ElementsTreeOutline} |
| 853 * @param {boolean} success |
| 854 */ |
| 855 function editingFinished(success) { |
| 856 if (callback) |
| 857 callback(); |
| 858 if (!success) |
| 859 return; |
| 860 |
| 861 // Select it and expand if necessary. We force tree update so that it proc
esses dom events and is up to date. |
| 862 this.runPendingUpdates(); |
| 863 |
| 864 var newNode = parentNode ? parentNode.children()[index] || parentNode : nu
ll; |
| 865 if (!newNode) |
| 866 return; |
| 867 |
| 868 this.selectDOMNode(newNode, true); |
| 869 |
| 870 if (wasExpanded) { |
| 871 var newTreeItem = this.findTreeElement(newNode); |
| 872 if (newTreeItem) |
| 873 newTreeItem.expand(); |
| 874 } |
| 875 } |
| 876 } |
| 877 |
| 878 /** |
| 879 * @param {boolean} wasExpanded |
| 880 * @param {?Protocol.Error} error |
| 881 * @param {!DOMAgent.NodeId=} nodeId |
| 882 * @return {?WebInspector.ElementsTreeElement} nodeId |
| 883 */ |
| 884 selectNodeAfterEdit(wasExpanded, error, nodeId) { |
| 885 if (error) |
| 886 return null; |
| 887 |
| 888 // Select it and expand if necessary. We force tree update so that it proces
ses dom events and is up to date. |
| 889 this.runPendingUpdates(); |
| 890 |
| 891 var newNode = nodeId ? this._domModel.nodeForId(nodeId) : null; |
| 892 if (!newNode) |
| 893 return null; |
| 894 |
| 895 this.selectDOMNode(newNode, true); |
| 896 |
| 897 var newTreeItem = this.findTreeElement(newNode); |
| 898 if (wasExpanded) { |
| 899 if (newTreeItem) |
| 900 newTreeItem.expand(); |
| 901 } |
| 902 return newTreeItem; |
| 903 } |
| 904 |
| 905 /** |
| 906 * Runs a script on the node's remote object that toggles a class name on |
| 907 * the node and injects a stylesheet into the head of the node's document |
| 908 * containing a rule to set "visibility: hidden" on the class and all it's |
| 909 * ancestors. |
| 910 * |
| 911 * @param {!WebInspector.DOMNode} node |
| 912 * @param {function(?WebInspector.RemoteObject, boolean=)=} userCallback |
| 913 */ |
| 914 toggleHideElement(node, userCallback) { |
| 915 var pseudoType = node.pseudoType(); |
| 916 var effectiveNode = pseudoType ? node.parentNode : node; |
| 917 if (!effectiveNode) |
| 918 return; |
| 919 |
| 920 var hidden = node.marker('hidden-marker'); |
| 921 |
| 922 function resolvedNode(object) { |
| 923 if (!object) |
| 924 return; |
| 925 |
| 926 /** |
| 927 * @param {?string} pseudoType |
| 928 * @param {boolean} hidden |
| 929 * @suppressGlobalPropertiesCheck |
| 930 * @suppressReceiverCheck |
| 931 * @this {!Element} |
| 932 */ |
| 933 function toggleClassAndInjectStyleRule(pseudoType, hidden) { |
| 934 const classNamePrefix = '__web-inspector-hide'; |
| 935 const classNameSuffix = '-shortcut__'; |
| 936 const styleTagId = '__web-inspector-hide-shortcut-style__'; |
| 937 var selectors = []; |
| 938 selectors.push('.__web-inspector-hide-shortcut__'); |
| 939 selectors.push('.__web-inspector-hide-shortcut__ *'); |
| 940 selectors.push('.__web-inspector-hidebefore-shortcut__::before'); |
| 941 selectors.push('.__web-inspector-hideafter-shortcut__::after'); |
| 942 var selector = selectors.join(', '); |
| 943 var ruleBody = ' visibility: hidden !important;'; |
| 944 var rule = '\n' + selector + '\n{\n' + ruleBody + '\n}\n'; |
| 945 var className = classNamePrefix + (pseudoType || '') + classNameSuffix; |
| 946 this.classList.toggle(className, hidden); |
| 947 |
| 948 var localRoot = this; |
| 949 while (localRoot.parentNode) |
| 950 localRoot = localRoot.parentNode; |
| 951 if (localRoot.nodeType === Node.DOCUMENT_NODE) |
| 952 localRoot = document.head; |
| 953 |
| 954 var style = localRoot.querySelector('style#' + styleTagId); |
| 955 if (style) |
| 956 return; |
| 957 |
| 958 style = document.createElement('style'); |
| 959 style.id = styleTagId; |
| 960 style.type = 'text/css'; |
| 961 style.textContent = rule; |
| 962 |
| 963 localRoot.appendChild(style); |
| 964 } |
| 965 |
| 966 object.callFunction(toggleClassAndInjectStyleRule, [{value: pseudoType}, {
value: !hidden}], userCallback); |
| 967 object.release(); |
| 968 node.setMarker('hidden-marker', hidden ? null : true); |
| 969 } |
| 970 |
| 971 effectiveNode.resolveToObject('', resolvedNode); |
| 972 } |
| 973 |
| 974 /** |
| 975 * @param {!WebInspector.DOMNode} node |
| 976 * @return {boolean} |
| 977 */ |
| 978 isToggledToHidden(node) { |
| 979 return !!node.marker('hidden-marker'); |
| 980 } |
| 981 |
| 982 _reset() { |
| 983 this.rootDOMNode = null; |
| 984 this.selectDOMNode(null, false); |
| 985 this._popoverHelper.hidePopover(); |
| 986 delete this._clipboardNodeData; |
| 987 WebInspector.DOMModel.hideDOMNodeHighlight(); |
| 988 this._updateRecords.clear(); |
| 989 } |
| 990 |
| 991 wireToDOMModel() { |
| 992 this._domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbol] = this; |
| 993 this._domModel.addEventListener(WebInspector.DOMModel.Events.NodeInserted, t
his._nodeInserted, this); |
| 994 this._domModel.addEventListener(WebInspector.DOMModel.Events.NodeRemoved, th
is._nodeRemoved, this); |
| 995 this._domModel.addEventListener(WebInspector.DOMModel.Events.AttrModified, t
his._attributeModified, this); |
| 996 this._domModel.addEventListener(WebInspector.DOMModel.Events.AttrRemoved, th
is._attributeRemoved, this); |
| 997 this._domModel.addEventListener( |
| 998 WebInspector.DOMModel.Events.CharacterDataModified, this._characterDataM
odified, this); |
| 999 this._domModel.addEventListener(WebInspector.DOMModel.Events.DocumentUpdated
, this._documentUpdated, this); |
| 1000 this._domModel.addEventListener( |
| 1001 WebInspector.DOMModel.Events.ChildNodeCountUpdated, this._childNodeCount
Updated, this); |
| 1002 this._domModel.addEventListener( |
| 1003 WebInspector.DOMModel.Events.DistributedNodesChanged, this._distributedN
odesChanged, this); |
| 1004 } |
| 1005 |
| 1006 unwireFromDOMModel() { |
| 1007 this._domModel.removeEventListener(WebInspector.DOMModel.Events.NodeInserted
, this._nodeInserted, this); |
| 1008 this._domModel.removeEventListener(WebInspector.DOMModel.Events.NodeRemoved,
this._nodeRemoved, this); |
| 1009 this._domModel.removeEventListener(WebInspector.DOMModel.Events.AttrModified
, this._attributeModified, this); |
| 1010 this._domModel.removeEventListener(WebInspector.DOMModel.Events.AttrRemoved,
this._attributeRemoved, this); |
| 1011 this._domModel.removeEventListener( |
| 1012 WebInspector.DOMModel.Events.CharacterDataModified, this._characterDataM
odified, this); |
| 1013 this._domModel.removeEventListener(WebInspector.DOMModel.Events.DocumentUpda
ted, this._documentUpdated, this); |
| 1014 this._domModel.removeEventListener( |
| 1015 WebInspector.DOMModel.Events.ChildNodeCountUpdated, this._childNodeCount
Updated, this); |
| 1016 this._domModel.removeEventListener( |
| 1017 WebInspector.DOMModel.Events.DistributedNodesChanged, this._distributedN
odesChanged, this); |
| 1018 delete this._domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbol]; |
| 1019 } |
| 1020 |
| 1021 /** |
| 1022 * @param {!WebInspector.DOMNode} node |
| 1023 * @return {!WebInspector.ElementsTreeOutline.UpdateRecord} |
| 1024 */ |
| 1025 _addUpdateRecord(node) { |
| 1026 var record = this._updateRecords.get(node); |
| 1027 if (!record) { |
| 1028 record = new WebInspector.ElementsTreeOutline.UpdateRecord(); |
| 1029 this._updateRecords.set(node, record); |
| 1030 } |
| 1031 return record; |
| 1032 } |
| 1033 |
| 1034 /** |
| 1035 * @param {!WebInspector.DOMNode} node |
| 1036 * @return {?WebInspector.ElementsTreeOutline.UpdateRecord} |
| 1037 */ |
| 1038 _updateRecordForHighlight(node) { |
| 1039 if (!this._visible) |
| 1040 return null; |
| 1041 return this._updateRecords.get(node) || null; |
| 1042 } |
| 1043 |
| 1044 /** |
| 1045 * @param {!WebInspector.Event} event |
| 1046 */ |
| 1047 _documentUpdated(event) { |
| 1048 var inspectedRootDocument = event.data; |
| 1049 |
| 1050 this._reset(); |
| 1051 |
| 1052 if (!inspectedRootDocument) |
| 1053 return; |
| 1054 |
| 1055 this.rootDOMNode = inspectedRootDocument; |
| 1056 } |
| 1057 |
| 1058 /** |
| 1059 * @param {!WebInspector.Event} event |
| 1060 */ |
| 1061 _attributeModified(event) { |
| 1062 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); |
| 1063 this._addUpdateRecord(node).attributeModified(event.data.name); |
| 1064 this._updateModifiedNodesSoon(); |
| 1065 } |
| 1066 |
| 1067 /** |
| 1068 * @param {!WebInspector.Event} event |
| 1069 */ |
| 1070 _attributeRemoved(event) { |
| 1071 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); |
| 1072 this._addUpdateRecord(node).attributeRemoved(event.data.name); |
| 1073 this._updateModifiedNodesSoon(); |
| 1074 } |
| 1075 |
| 1076 /** |
| 1077 * @param {!WebInspector.Event} event |
| 1078 */ |
| 1079 _characterDataModified(event) { |
| 1080 var node = /** @type {!WebInspector.DOMNode} */ (event.data); |
| 1081 this._addUpdateRecord(node).charDataModified(); |
| 1082 // Text could be large and force us to render itself as the child in the tre
e outline. |
| 1083 if (node.parentNode && node.parentNode.firstChild === node.parentNode.lastCh
ild) |
| 1084 this._addUpdateRecord(node.parentNode).childrenModified(); |
| 1085 this._updateModifiedNodesSoon(); |
| 1086 } |
| 1087 |
| 1088 /** |
| 1089 * @param {!WebInspector.Event} event |
| 1090 */ |
| 1091 _nodeInserted(event) { |
| 1092 var node = /** @type {!WebInspector.DOMNode} */ (event.data); |
| 1093 this._addUpdateRecord(/** @type {!WebInspector.DOMNode} */ (node.parentNode)
).nodeInserted(node); |
| 1094 this._updateModifiedNodesSoon(); |
| 1095 } |
| 1096 |
| 1097 /** |
| 1098 * @param {!WebInspector.Event} event |
| 1099 */ |
| 1100 _nodeRemoved(event) { |
| 1101 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); |
| 1102 var parentNode = /** @type {!WebInspector.DOMNode} */ (event.data.parent); |
| 1103 this.resetClipboardIfNeeded(node); |
| 1104 this._addUpdateRecord(parentNode).nodeRemoved(node); |
| 1105 this._updateModifiedNodesSoon(); |
| 1106 } |
| 1107 |
| 1108 /** |
| 1109 * @param {!WebInspector.Event} event |
| 1110 */ |
| 1111 _childNodeCountUpdated(event) { |
| 1112 var node = /** @type {!WebInspector.DOMNode} */ (event.data); |
| 1113 this._addUpdateRecord(node).childrenModified(); |
| 1114 this._updateModifiedNodesSoon(); |
| 1115 } |
| 1116 |
| 1117 /** |
| 1118 * @param {!WebInspector.Event} event |
| 1119 */ |
| 1120 _distributedNodesChanged(event) { |
| 1121 var node = /** @type {!WebInspector.DOMNode} */ (event.data); |
| 1122 this._addUpdateRecord(node).childrenModified(); |
| 1123 this._updateModifiedNodesSoon(); |
| 1124 } |
| 1125 |
| 1126 _updateModifiedNodesSoon() { |
| 1127 if (!this._updateRecords.size) |
| 1128 return; |
| 1129 if (this._updateModifiedNodesTimeout) |
| 1130 return; |
| 1131 this._updateModifiedNodesTimeout = setTimeout(this._updateModifiedNodes.bind
(this), 50); |
| 1132 } |
| 1133 |
| 1134 _updateModifiedNodes() { |
| 1135 if (this._updateModifiedNodesTimeout) { |
| 1136 clearTimeout(this._updateModifiedNodesTimeout); |
| 1137 delete this._updateModifiedNodesTimeout; |
| 1138 } |
| 1139 |
| 1140 var updatedNodes = this._updateRecords.keysArray(); |
| 1141 var hidePanelWhileUpdating = updatedNodes.length > 10; |
| 1142 if (hidePanelWhileUpdating) { |
| 1143 var treeOutlineContainerElement = this.element.parentNode; |
| 1144 var originalScrollTop = treeOutlineContainerElement ? treeOutlineContainer
Element.scrollTop : 0; |
| 1145 this._element.classList.add('hidden'); |
| 1146 } |
| 1147 |
| 1148 if (this._rootDOMNode && this._updateRecords.get(this._rootDOMNode) && |
| 1149 this._updateRecords.get(this._rootDOMNode).hasChangedChildren()) { |
| 1150 // Document's children have changed, perform total update. |
| 1151 this.update(); |
| 1152 } else { |
| 1153 for (var node of this._updateRecords.keys()) { |
| 1154 if (this._updateRecords.get(node).hasChangedChildren()) |
| 1155 this._updateModifiedParentNode(node); |
| 1156 else |
| 1157 this._updateModifiedNode(node); |
| 1158 } |
| 1159 } |
| 1160 |
| 1161 if (hidePanelWhileUpdating) { |
| 1162 this._element.classList.remove('hidden'); |
| 1163 if (originalScrollTop) |
| 1164 treeOutlineContainerElement.scrollTop = originalScrollTop; |
| 1165 } |
| 1166 |
| 1167 this._updateRecords.clear(); |
| 1168 this._fireElementsTreeUpdated(updatedNodes); |
| 1169 } |
| 1170 |
| 1171 _updateModifiedNode(node) { |
| 1172 var treeElement = this.findTreeElement(node); |
| 1173 if (treeElement) |
| 1174 treeElement.updateTitle(this._updateRecordForHighlight(node)); |
| 1175 } |
| 1176 |
| 1177 _updateModifiedParentNode(node) { |
| 1178 var parentTreeElement = this.findTreeElement(node); |
| 1179 if (parentTreeElement) { |
| 1180 parentTreeElement.setExpandable(this._hasVisibleChildren(node)); |
| 1181 parentTreeElement.updateTitle(this._updateRecordForHighlight(node)); |
| 1182 if (parentTreeElement.populated) |
| 1183 this._updateChildren(parentTreeElement); |
| 1184 } |
| 1185 } |
| 1186 |
| 1187 /** |
| 1188 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1189 */ |
| 1190 populateTreeElement(treeElement) { |
| 1191 if (treeElement.childCount() || !treeElement.isExpandable()) |
| 1192 return; |
| 1193 |
| 1194 this._updateModifiedParentNode(treeElement.node()); |
| 1195 } |
| 1196 |
| 1197 /** |
| 1198 * @param {!WebInspector.DOMNode} node |
| 1199 * @param {boolean=} closingTag |
| 1200 * @return {!WebInspector.ElementsTreeElement} |
| 1201 */ |
| 1202 _createElementTreeElement(node, closingTag) { |
| 1203 var treeElement = new WebInspector.ElementsTreeElement(node, closingTag); |
| 1204 treeElement.setExpandable(!closingTag && this._hasVisibleChildren(node)); |
| 1205 if (node.nodeType() === Node.ELEMENT_NODE && node.parentNode && node.parentN
ode.nodeType() === Node.DOCUMENT_NODE && |
| 1206 !node.parentNode.parentNode) |
| 1207 treeElement.setCollapsible(false); |
| 1208 treeElement.selectable = this._selectEnabled; |
| 1209 return treeElement; |
| 1210 } |
| 1211 |
| 1212 /** |
| 1213 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1214 * @param {!WebInspector.DOMNode} child |
| 1215 * @return {?WebInspector.ElementsTreeElement} |
| 1216 */ |
| 1217 _showChild(treeElement, child) { |
| 1218 if (treeElement.isClosingTag()) |
| 1219 return null; |
| 1220 |
| 1221 var index = this._visibleChildren(treeElement.node()).indexOf(child); |
| 1222 if (index === -1) |
| 1223 return null; |
| 1224 |
| 1225 if (index >= treeElement.expandedChildrenLimit()) |
| 1226 this.setExpandedChildrenLimit(treeElement, index + 1); |
| 1227 return /** @type {!WebInspector.ElementsTreeElement} */ (treeElement.childAt
(index)); |
| 1228 } |
| 1229 |
| 1230 /** |
| 1231 * @param {!WebInspector.DOMNode} node |
| 1232 * @return {!Array.<!WebInspector.DOMNode>} visibleChildren |
| 1233 */ |
| 1234 _visibleChildren(node) { |
| 1235 var visibleChildren = WebInspector.ElementsTreeElement.visibleShadowRoots(no
de); |
| 1236 |
| 1237 var importedDocument = node.importedDocument(); |
| 1238 if (importedDocument) |
| 1239 visibleChildren.push(importedDocument); |
| 1240 |
| 1241 var templateContent = node.templateContent(); |
| 1242 if (templateContent) |
| 1243 visibleChildren.push(templateContent); |
| 1244 |
| 1245 var beforePseudoElement = node.beforePseudoElement(); |
| 1246 if (beforePseudoElement) |
| 1247 visibleChildren.push(beforePseudoElement); |
| 1248 |
| 1249 if (node.childNodeCount()) { |
| 1250 var children = node.children(); |
| 1251 if (!this._showHTMLCommentsSetting.get()) |
| 1252 children = children.filter(n => n.nodeType() !== Node.COMMENT_NODE); |
| 1253 visibleChildren = visibleChildren.concat(children); |
| 1254 } |
| 1255 |
| 1256 var afterPseudoElement = node.afterPseudoElement(); |
| 1257 if (afterPseudoElement) |
| 1258 visibleChildren.push(afterPseudoElement); |
| 1259 |
| 1260 return visibleChildren; |
| 1261 } |
| 1262 |
| 1263 /** |
| 1264 * @param {!WebInspector.DOMNode} node |
| 1265 * @return {boolean} |
| 1266 */ |
| 1267 _hasVisibleChildren(node) { |
| 1268 if (node.importedDocument()) |
| 1269 return true; |
| 1270 if (node.templateContent()) |
| 1271 return true; |
| 1272 if (WebInspector.ElementsTreeElement.visibleShadowRoots(node).length) |
| 1273 return true; |
| 1274 if (node.hasPseudoElements()) |
| 1275 return true; |
| 1276 if (node.isInsertionPoint()) |
| 1277 return true; |
| 1278 return !!node.childNodeCount() && !WebInspector.ElementsTreeElement.canShowI
nlineText(node); |
| 1279 } |
| 1280 |
| 1281 /** |
| 1282 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1283 */ |
| 1284 _createExpandAllButtonTreeElement(treeElement) { |
| 1285 var button = createTextButton('', handleLoadAllChildren.bind(this)); |
| 1286 button.value = ''; |
| 1287 var expandAllButtonElement = new TreeElement(button); |
| 1288 expandAllButtonElement.selectable = false; |
| 1289 expandAllButtonElement.expandAllButton = true; |
| 1290 expandAllButtonElement.button = button; |
| 1291 return expandAllButtonElement; |
| 1292 |
| 1293 /** |
| 1294 * @this {WebInspector.ElementsTreeOutline} |
| 1295 * @param {!Event} event |
| 1296 */ |
| 1297 function handleLoadAllChildren(event) { |
| 1298 var visibleChildCount = this._visibleChildren(treeElement.node()).length; |
| 1299 this.setExpandedChildrenLimit( |
| 1300 treeElement, Math.max( |
| 1301 visibleChildCount, treeElement.expandedChildrenLimit(
) + |
| 1302 WebInspector.ElementsTreeElement.InitialChildrenL
imit)); |
| 1303 event.consume(); |
| 1304 } |
| 1305 } |
| 1306 |
| 1307 /** |
| 1308 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1309 * @param {number} expandedChildrenLimit |
| 1310 */ |
| 1311 setExpandedChildrenLimit(treeElement, expandedChildrenLimit) { |
| 1312 if (treeElement.expandedChildrenLimit() === expandedChildrenLimit) |
| 1313 return; |
| 1314 |
| 1315 treeElement.setExpandedChildrenLimit(expandedChildrenLimit); |
| 1316 if (treeElement.treeOutline && !this._treeElementsBeingUpdated.has(treeEleme
nt)) |
| 1317 this._updateModifiedParentNode(treeElement.node()); |
| 1318 } |
| 1319 |
| 1320 /** |
| 1321 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1322 */ |
| 1323 _updateChildren(treeElement) { |
| 1324 if (!treeElement.isExpandable()) { |
| 1325 var selectedTreeElement = treeElement.treeOutline.selectedTreeElement; |
| 1326 if (selectedTreeElement && selectedTreeElement.hasAncestor(treeElement)) |
| 1327 treeElement.select(true); |
| 1328 treeElement.removeChildren(); |
| 1329 return; |
| 1330 } |
| 1331 |
| 1332 console.assert(!treeElement.isClosingTag()); |
| 1333 |
| 1334 treeElement.node().getChildNodes(childNodesLoaded.bind(this)); |
| 1335 |
| 1336 /** |
| 1337 * @param {?Array.<!WebInspector.DOMNode>} children |
| 1338 * @this {WebInspector.ElementsTreeOutline} |
| 1339 */ |
| 1340 function childNodesLoaded(children) { |
| 1341 // FIXME: sort this out, it should not happen. |
| 1342 if (!children) |
| 1343 return; |
| 1344 this._innerUpdateChildren(treeElement); |
| 1345 } |
| 1346 } |
| 1347 |
| 1348 /** |
| 1349 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1350 * @param {!WebInspector.DOMNode} child |
| 1351 * @param {number} index |
| 1352 * @param {boolean=} closingTag |
| 1353 * @return {!WebInspector.ElementsTreeElement} |
| 1354 */ |
| 1355 insertChildElement(treeElement, child, index, closingTag) { |
| 1356 var newElement = this._createElementTreeElement(child, closingTag); |
| 1357 treeElement.insertChild(newElement, index); |
| 1358 return newElement; |
| 1359 } |
| 1360 |
| 1361 /** |
| 1362 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1363 * @param {!WebInspector.ElementsTreeElement} child |
| 1364 * @param {number} targetIndex |
| 1365 */ |
| 1366 _moveChild(treeElement, child, targetIndex) { |
| 1367 if (treeElement.indexOfChild(child) === targetIndex) |
| 1368 return; |
| 1369 var wasSelected = child.selected; |
| 1370 if (child.parent) |
| 1371 child.parent.removeChild(child); |
| 1372 treeElement.insertChild(child, targetIndex); |
| 1373 if (wasSelected) |
| 1374 child.select(); |
| 1375 } |
| 1376 |
| 1377 /** |
| 1378 * @param {!WebInspector.ElementsTreeElement} treeElement |
| 1379 */ |
| 1380 _innerUpdateChildren(treeElement) { |
| 1381 if (this._treeElementsBeingUpdated.has(treeElement)) |
| 1382 return; |
| 1383 |
| 1384 this._treeElementsBeingUpdated.add(treeElement); |
| 1385 |
| 1386 var node = treeElement.node(); |
| 1387 var visibleChildren = this._visibleChildren(node); |
| 1388 var visibleChildrenSet = new Set(visibleChildren); |
| 1389 |
| 1390 // Remove any tree elements that no longer have this node as their parent an
d save |
| 1391 // all existing elements that could be reused. This also removes closing tag
element. |
| 1392 var existingTreeElements = new Map(); |
| 1393 for (var i = treeElement.childCount() - 1; i >= 0; --i) { |
| 1394 var existingTreeElement = treeElement.childAt(i); |
| 1395 if (!(existingTreeElement instanceof WebInspector.ElementsTreeElement)) { |
| 1396 // Remove expand all button and shadow host toolbar. |
| 1397 treeElement.removeChildAtIndex(i); |
| 1398 continue; |
| 1399 } |
| 1400 var elementsTreeElement = /** @type {!WebInspector.ElementsTreeElement} */
(existingTreeElement); |
| 1401 var existingNode = elementsTreeElement.node(); |
| 1402 |
| 1403 if (visibleChildrenSet.has(existingNode)) { |
| 1404 existingTreeElements.set(existingNode, existingTreeElement); |
| 1405 continue; |
| 1406 } |
| 1407 |
| 1408 treeElement.removeChildAtIndex(i); |
| 1409 } |
| 1410 |
| 1411 for (var i = 0; i < visibleChildren.length && i < treeElement.expandedChildr
enLimit(); ++i) { |
| 1412 var child = visibleChildren[i]; |
| 1413 var existingTreeElement = existingTreeElements.get(child) || this.findTree
Element(child); |
| 1414 if (existingTreeElement && existingTreeElement !== treeElement) { |
| 1415 // If an existing element was found, just move it. |
| 1416 this._moveChild(treeElement, existingTreeElement, i); |
| 1417 } else { |
| 1418 // No existing element found, insert a new element. |
| 1419 var newElement = this.insertChildElement(treeElement, child, i); |
| 1420 if (this._updateRecordForHighlight(node) && treeElement.expanded) |
| 1421 WebInspector.ElementsTreeElement.animateOnDOMUpdate(newElement); |
| 1422 // If a node was inserted in the middle of existing list dynamically we
might need to increase the limit. |
| 1423 if (treeElement.childCount() > treeElement.expandedChildrenLimit()) |
| 1424 this.setExpandedChildrenLimit(treeElement, treeElement.expandedChildre
nLimit() + 1); |
| 1425 } |
| 1426 } |
| 1427 |
| 1428 // Update expand all button. |
| 1429 var expandedChildCount = treeElement.childCount(); |
| 1430 if (visibleChildren.length > expandedChildCount) { |
| 1431 var targetButtonIndex = expandedChildCount; |
| 1432 if (!treeElement.expandAllButtonElement) |
| 1433 treeElement.expandAllButtonElement = this._createExpandAllButtonTreeElem
ent(treeElement); |
| 1434 treeElement.insertChild(treeElement.expandAllButtonElement, targetButtonIn
dex); |
| 1435 treeElement.expandAllButtonElement.button.textContent = |
| 1436 WebInspector.UIString('Show All Nodes (%d More)', visibleChildren.leng
th - expandedChildCount); |
| 1437 } else if (treeElement.expandAllButtonElement) { |
| 1438 delete treeElement.expandAllButtonElement; |
| 1439 } |
| 1440 |
| 1441 // Insert shortcuts to distrubuted children. |
| 1442 if (node.isInsertionPoint()) { |
| 1443 for (var distributedNode of node.distributedNodes()) |
| 1444 treeElement.appendChild(new WebInspector.ElementsTreeOutline.ShortcutTre
eElement(distributedNode)); |
| 1445 } |
| 1446 |
| 1447 // Insert close tag. |
| 1448 if (node.nodeType() === Node.ELEMENT_NODE && treeElement.isExpandable()) |
| 1449 this.insertChildElement(treeElement, node, treeElement.childCount(), true)
; |
| 1450 |
| 1451 this._treeElementsBeingUpdated.delete(treeElement); |
| 1452 } |
| 1453 |
| 1454 /** |
| 1455 * @param {!WebInspector.Event} event |
| 1456 */ |
| 1457 _markersChanged(event) { |
| 1458 var node = /** @type {!WebInspector.DOMNode} */ (event.data); |
| 1459 var treeElement = node[this._treeElementSymbol]; |
| 1460 if (treeElement) |
| 1461 treeElement.updateDecorations(); |
| 1462 } |
| 88 }; | 1463 }; |
| 89 | 1464 |
| 90 WebInspector.ElementsTreeOutline._treeOutlineSymbol = Symbol("treeOutline"); | 1465 WebInspector.ElementsTreeOutline._treeOutlineSymbol = Symbol('treeOutline'); |
| 91 | 1466 |
| 92 /** | |
| 93 * @param {!WebInspector.DOMModel} domModel | |
| 94 * @return {?WebInspector.ElementsTreeOutline} | |
| 95 */ | |
| 96 WebInspector.ElementsTreeOutline.forDOMModel = function(domModel) | |
| 97 { | |
| 98 return domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbol] || null
; | |
| 99 }; | |
| 100 | 1467 |
| 101 /** @typedef {{node: !WebInspector.DOMNode, isCut: boolean}} */ | 1468 /** @typedef {{node: !WebInspector.DOMNode, isCut: boolean}} */ |
| 102 WebInspector.ElementsTreeOutline.ClipboardData; | 1469 WebInspector.ElementsTreeOutline.ClipboardData; |
| 103 | 1470 |
| 104 /** @enum {symbol} */ | 1471 /** @enum {symbol} */ |
| 105 WebInspector.ElementsTreeOutline.Events = { | 1472 WebInspector.ElementsTreeOutline.Events = { |
| 106 SelectedNodeChanged: Symbol("SelectedNodeChanged"), | 1473 SelectedNodeChanged: Symbol('SelectedNodeChanged'), |
| 107 ElementsTreeUpdated: Symbol("ElementsTreeUpdated") | 1474 ElementsTreeUpdated: Symbol('ElementsTreeUpdated') |
| 108 }; | 1475 }; |
| 109 | 1476 |
| 110 /** | 1477 /** |
| 111 * @const | 1478 * @const |
| 112 * @type {!Object.<string, string>} | 1479 * @type {!Object.<string, string>} |
| 113 */ | 1480 */ |
| 114 WebInspector.ElementsTreeOutline.MappedCharToEntity = { | 1481 WebInspector.ElementsTreeOutline.MappedCharToEntity = { |
| 115 "\u00a0": "nbsp", | 1482 '\u00a0': 'nbsp', |
| 116 "\u0093": "#147", // <control> | 1483 '\u0093': '#147', // <control> |
| 117 "\u00ad": "shy", | 1484 '\u00ad': 'shy', |
| 118 "\u2002": "ensp", | 1485 '\u2002': 'ensp', |
| 119 "\u2003": "emsp", | 1486 '\u2003': 'emsp', |
| 120 "\u2009": "thinsp", | 1487 '\u2009': 'thinsp', |
| 121 "\u200a": "#8202", // Hairspace | 1488 '\u200a': '#8202', // Hairspace |
| 122 "\u200b": "#8203", // ZWSP | 1489 '\u200b': '#8203', // ZWSP |
| 123 "\u200c": "zwnj", | 1490 '\u200c': 'zwnj', |
| 124 "\u200d": "zwj", | 1491 '\u200d': 'zwj', |
| 125 "\u200e": "lrm", | 1492 '\u200e': 'lrm', |
| 126 "\u200f": "rlm", | 1493 '\u200f': 'rlm', |
| 127 "\u202a": "#8234", // LRE | 1494 '\u202a': '#8234', // LRE |
| 128 "\u202b": "#8235", // RLE | 1495 '\u202b': '#8235', // RLE |
| 129 "\u202c": "#8236", // PDF | 1496 '\u202c': '#8236', // PDF |
| 130 "\u202d": "#8237", // LRO | 1497 '\u202d': '#8237', // LRO |
| 131 "\u202e": "#8238", // RLO | 1498 '\u202e': '#8238', // RLO |
| 132 "\ufeff": "#65279" // BOM | 1499 '\ufeff': '#65279' // BOM |
| 133 }; | 1500 }; |
| 134 | 1501 |
| 135 WebInspector.ElementsTreeOutline.prototype = { | 1502 /** |
| 136 _onShowHTMLCommentsChange: function() | 1503 * @unrestricted |
| 137 { | 1504 */ |
| 138 var selectedNode = this.selectedDOMNode(); | 1505 WebInspector.ElementsTreeOutline.UpdateRecord = class { |
| 139 if (selectedNode && selectedNode.nodeType() === Node.COMMENT_NODE && !th
is._showHTMLCommentsSetting.get()) | 1506 /** |
| 140 this.selectDOMNode(selectedNode.parentNode); | 1507 * @param {string} attrName |
| 141 this.update(); | 1508 */ |
| 142 }, | 1509 attributeModified(attrName) { |
| 1510 if (this._removedAttributes && this._removedAttributes.has(attrName)) |
| 1511 this._removedAttributes.delete(attrName); |
| 1512 if (!this._modifiedAttributes) |
| 1513 this._modifiedAttributes = /** @type {!Set.<string>} */ (new Set()); |
| 1514 this._modifiedAttributes.add(attrName); |
| 1515 } |
| 1516 |
| 1517 /** |
| 1518 * @param {string} attrName |
| 1519 */ |
| 1520 attributeRemoved(attrName) { |
| 1521 if (this._modifiedAttributes && this._modifiedAttributes.has(attrName)) |
| 1522 this._modifiedAttributes.delete(attrName); |
| 1523 if (!this._removedAttributes) |
| 1524 this._removedAttributes = /** @type {!Set.<string>} */ (new Set()); |
| 1525 this._removedAttributes.add(attrName); |
| 1526 } |
| 1527 |
| 1528 /** |
| 1529 * @param {!WebInspector.DOMNode} node |
| 1530 */ |
| 1531 nodeInserted(node) { |
| 1532 this._hasChangedChildren = true; |
| 1533 } |
| 1534 |
| 1535 nodeRemoved(node) { |
| 1536 this._hasChangedChildren = true; |
| 1537 this._hasRemovedChildren = true; |
| 1538 } |
| 1539 |
| 1540 charDataModified() { |
| 1541 this._charDataModified = true; |
| 1542 } |
| 1543 |
| 1544 childrenModified() { |
| 1545 this._hasChangedChildren = true; |
| 1546 } |
| 1547 |
| 1548 /** |
| 1549 * @param {string} attributeName |
| 1550 * @return {boolean} |
| 1551 */ |
| 1552 isAttributeModified(attributeName) { |
| 1553 return this._modifiedAttributes && this._modifiedAttributes.has(attributeNam
e); |
| 1554 } |
| 1555 |
| 1556 /** |
| 1557 * @return {boolean} |
| 1558 */ |
| 1559 hasRemovedAttributes() { |
| 1560 return !!this._removedAttributes && !!this._removedAttributes.size; |
| 1561 } |
| 1562 |
| 1563 /** |
| 1564 * @return {boolean} |
| 1565 */ |
| 1566 isCharDataModified() { |
| 1567 return !!this._charDataModified; |
| 1568 } |
| 1569 |
| 1570 /** |
| 1571 * @return {boolean} |
| 1572 */ |
| 1573 hasChangedChildren() { |
| 1574 return !!this._hasChangedChildren; |
| 1575 } |
| 1576 |
| 1577 /** |
| 1578 * @return {boolean} |
| 1579 */ |
| 1580 hasRemovedChildren() { |
| 1581 return !!this._hasRemovedChildren; |
| 1582 } |
| 1583 }; |
| 1584 |
| 1585 /** |
| 1586 * @implements {WebInspector.Renderer} |
| 1587 * @unrestricted |
| 1588 */ |
| 1589 WebInspector.ElementsTreeOutline.Renderer = class { |
| 1590 /** |
| 1591 * @override |
| 1592 * @param {!Object} object |
| 1593 * @return {!Promise.<!Element>} |
| 1594 */ |
| 1595 render(object) { |
| 1596 return new Promise(renderPromise); |
| 143 | 1597 |
| 144 /** | 1598 /** |
| 145 * @return {symbol} | 1599 * @param {function(!Element)} resolve |
| 1600 * @param {function(!Error)} reject |
| 146 */ | 1601 */ |
| 147 treeElementSymbol: function() | 1602 function renderPromise(resolve, reject) { |
| 148 { | 1603 if (object instanceof WebInspector.DOMNode) { |
| 149 return this._treeElementSymbol; | 1604 onNodeResolved(/** @type {!WebInspector.DOMNode} */ (object)); |
| 150 }, | 1605 } else if (object instanceof WebInspector.DeferredDOMNode) { |
| 151 | 1606 (/** @type {!WebInspector.DeferredDOMNode} */ (object)).resolve(onNodeRe
solved); |
| 152 focus: function() | 1607 } else if (object instanceof WebInspector.RemoteObject) { |
| 153 { | 1608 var domModel = WebInspector.DOMModel.fromTarget((/** @type {!WebInspecto
r.RemoteObject} */ (object)).target()); |
| 154 this._element.focus(); | 1609 if (domModel) |
| 155 }, | 1610 domModel.pushObjectAsNodeToFrontend(object, onNodeResolved); |
| 156 | 1611 else |
| 157 /** | 1612 reject(new Error('No dom model for given JS object target found.')); |
| 158 * @param {boolean} wrap | 1613 } else { |
| 159 */ | 1614 reject(new Error('Can\'t reveal not a node.')); |
| 160 setWordWrap: function(wrap) | 1615 } |
| 161 { | 1616 |
| 162 this._element.classList.toggle("elements-tree-nowrap", !wrap); | 1617 /** |
| 163 }, | 1618 * @param {?WebInspector.DOMNode} node |
| 164 | 1619 */ |
| 165 /** | 1620 function onNodeResolved(node) { |
| 166 * @return {!WebInspector.DOMModel} | 1621 if (!node) { |
| 167 */ | 1622 reject(new Error('Could not resolve node.')); |
| 168 domModel: function() | 1623 return; |
| 169 { | |
| 170 return this._domModel; | |
| 171 }, | |
| 172 | |
| 173 /** | |
| 174 * @param {?WebInspector.InplaceEditor.Controller} multilineEditing | |
| 175 */ | |
| 176 setMultilineEditing: function(multilineEditing) | |
| 177 { | |
| 178 this._multilineEditing = multilineEditing; | |
| 179 }, | |
| 180 | |
| 181 /** | |
| 182 * @return {number} | |
| 183 */ | |
| 184 visibleWidth: function() | |
| 185 { | |
| 186 return this._visibleWidth; | |
| 187 }, | |
| 188 | |
| 189 /** | |
| 190 * @param {number} width | |
| 191 */ | |
| 192 setVisibleWidth: function(width) | |
| 193 { | |
| 194 this._visibleWidth = width; | |
| 195 if (this._multilineEditing) | |
| 196 this._multilineEditing.setWidth(this._visibleWidth); | |
| 197 }, | |
| 198 | |
| 199 /** | |
| 200 * @param {?WebInspector.ElementsTreeOutline.ClipboardData} data | |
| 201 */ | |
| 202 _setClipboardData: function(data) | |
| 203 { | |
| 204 if (this._clipboardNodeData) { | |
| 205 var treeElement = this.findTreeElement(this._clipboardNodeData.node)
; | |
| 206 if (treeElement) | |
| 207 treeElement.setInClipboard(false); | |
| 208 delete this._clipboardNodeData; | |
| 209 } | 1624 } |
| 210 | 1625 var treeOutline = new WebInspector.ElementsTreeOutline(node.domModel(),
false, false); |
| 211 if (data) { | 1626 treeOutline.rootDOMNode = node; |
| 212 var treeElement = this.findTreeElement(data.node); | 1627 if (!treeOutline.firstChild().isExpandable()) |
| 213 if (treeElement) | 1628 treeOutline._element.classList.add('single-node'); |
| 214 treeElement.setInClipboard(true); | 1629 treeOutline.setVisible(true); |
| 215 this._clipboardNodeData = data; | 1630 treeOutline.element.treeElementForTest = treeOutline.firstChild(); |
| 216 } | 1631 resolve(treeOutline.element); |
| 217 }, | 1632 } |
| 218 | 1633 } |
| 219 /** | 1634 } |
| 220 * @param {!WebInspector.DOMNode} removedNode | 1635 }; |
| 221 */ | 1636 |
| 222 resetClipboardIfNeeded: function(removedNode) | 1637 /** |
| 223 { | 1638 * @unrestricted |
| 224 if (this._clipboardNodeData && this._clipboardNodeData.node === removedN
ode) | 1639 */ |
| 225 this._setClipboardData(null); | 1640 WebInspector.ElementsTreeOutline.ShortcutTreeElement = class extends TreeElement
{ |
| 226 }, | 1641 /** |
| 227 | 1642 * @param {!WebInspector.DOMNodeShortcut} nodeShortcut |
| 228 /** | 1643 */ |
| 229 * @param {!Event} event | 1644 constructor(nodeShortcut) { |
| 230 */ | 1645 super(''); |
| 231 _onBeforeCopy: function(event) | 1646 this.listItemElement.createChild('div', 'selection fill'); |
| 232 { | 1647 var title = this.listItemElement.createChild('span', 'elements-tree-shortcut
-title'); |
| 233 event.handled = true; | 1648 var text = nodeShortcut.nodeName.toLowerCase(); |
| 234 }, | 1649 if (nodeShortcut.nodeType === Node.ELEMENT_NODE) |
| 235 | 1650 text = '<' + text + '>'; |
| 236 /** | 1651 title.textContent = '\u21AA ' + text; |
| 237 * @param {boolean} isCut | 1652 |
| 238 * @param {!Event} event | 1653 var link = WebInspector.DOMPresentationUtils.linkifyDeferredNodeReference(no
deShortcut.deferredNode); |
| 239 */ | 1654 this.listItemElement.createTextChild(' '); |
| 240 _onCopyOrCut: function(isCut, event) | 1655 link.classList.add('elements-tree-shortcut-link'); |
| 241 { | 1656 link.textContent = WebInspector.UIString('reveal'); |
| 242 this._setClipboardData(null); | 1657 this.listItemElement.appendChild(link); |
| 243 var originalEvent = event["original"]; | 1658 this._nodeShortcut = nodeShortcut; |
| 244 | 1659 } |
| 245 // Don't prevent the normal copy if the user has a selection. | 1660 |
| 246 if (!originalEvent.target.isComponentSelectionCollapsed()) | 1661 /** |
| 247 return; | 1662 * @return {boolean} |
| 248 | 1663 */ |
| 249 // Do not interfere with text editing. | 1664 get hovered() { |
| 250 if (WebInspector.isEditing()) | 1665 return this._hovered; |
| 251 return; | 1666 } |
| 252 | 1667 |
| 253 var targetNode = this.selectedDOMNode(); | 1668 /** |
| 254 if (!targetNode) | 1669 * @param {boolean} x |
| 255 return; | 1670 */ |
| 256 | 1671 set hovered(x) { |
| 257 originalEvent.clipboardData.clearData(); | 1672 if (this._hovered === x) |
| 258 event.handled = true; | 1673 return; |
| 259 | 1674 this._hovered = x; |
| 260 this.performCopyOrCut(isCut, targetNode); | 1675 this.listItemElement.classList.toggle('hovered', x); |
| 261 }, | 1676 } |
| 262 | 1677 |
| 263 /** | 1678 /** |
| 264 * @param {boolean} isCut | 1679 * @return {number} |
| 265 * @param {?WebInspector.DOMNode} node | 1680 */ |
| 266 */ | 1681 backendNodeId() { |
| 267 performCopyOrCut: function(isCut, node) | 1682 return this._nodeShortcut.deferredNode.backendNodeId(); |
| 268 { | 1683 } |
| 269 if (isCut && (node.isShadowRoot() || node.ancestorUserAgentShadowRoot())
) | 1684 |
| 270 return; | 1685 /** |
| 271 | 1686 * @override |
| 272 node.copyNode(); | 1687 * @param {boolean=} selectedByUser |
| 273 this._setClipboardData({ node: node, isCut: isCut }); | 1688 * @return {boolean} |
| 274 }, | 1689 */ |
| 275 | 1690 onselect(selectedByUser) { |
| 276 /** | 1691 if (!selectedByUser) |
| 277 * @param {!WebInspector.DOMNode} targetNode | 1692 return true; |
| 278 * @return {boolean} | 1693 this._nodeShortcut.deferredNode.highlight(); |
| 279 */ | 1694 this._nodeShortcut.deferredNode.resolve(resolved.bind(this)); |
| 280 canPaste: function(targetNode) | |
| 281 { | |
| 282 if (targetNode.isShadowRoot() || targetNode.ancestorUserAgentShadowRoot(
)) | |
| 283 return false; | |
| 284 | |
| 285 if (!this._clipboardNodeData) | |
| 286 return false; | |
| 287 | |
| 288 var node = this._clipboardNodeData.node; | |
| 289 if (this._clipboardNodeData.isCut && (node === targetNode || node.isAnce
stor(targetNode))) | |
| 290 return false; | |
| 291 | |
| 292 if (targetNode.target() !== node.target()) | |
| 293 return false; | |
| 294 return true; | |
| 295 }, | |
| 296 | |
| 297 /** | |
| 298 * @param {!WebInspector.DOMNode} targetNode | |
| 299 */ | |
| 300 pasteNode: function(targetNode) | |
| 301 { | |
| 302 if (this.canPaste(targetNode)) | |
| 303 this._performPaste(targetNode); | |
| 304 }, | |
| 305 | |
| 306 /** | |
| 307 * @param {!Event} event | |
| 308 */ | |
| 309 _onPaste: function(event) | |
| 310 { | |
| 311 // Do not interfere with text editing. | |
| 312 if (WebInspector.isEditing()) | |
| 313 return; | |
| 314 | |
| 315 var targetNode = this.selectedDOMNode(); | |
| 316 if (!targetNode || !this.canPaste(targetNode)) | |
| 317 return; | |
| 318 | |
| 319 event.handled = true; | |
| 320 this._performPaste(targetNode); | |
| 321 }, | |
| 322 | |
| 323 /** | |
| 324 * @param {!WebInspector.DOMNode} targetNode | |
| 325 */ | |
| 326 _performPaste: function(targetNode) | |
| 327 { | |
| 328 if (this._clipboardNodeData.isCut) { | |
| 329 this._clipboardNodeData.node.moveTo(targetNode, null, expandCallback
.bind(this)); | |
| 330 this._setClipboardData(null); | |
| 331 } else { | |
| 332 this._clipboardNodeData.node.copyTo(targetNode, null, expandCallback
.bind(this)); | |
| 333 } | |
| 334 | |
| 335 /** | |
| 336 * @param {?Protocol.Error} error | |
| 337 * @param {!DOMAgent.NodeId} nodeId | |
| 338 * @this {WebInspector.ElementsTreeOutline} | |
| 339 */ | |
| 340 function expandCallback(error, nodeId) | |
| 341 { | |
| 342 if (error) | |
| 343 return; | |
| 344 var pastedNode = this._domModel.nodeForId(nodeId); | |
| 345 if (!pastedNode) | |
| 346 return; | |
| 347 this.selectDOMNode(pastedNode); | |
| 348 } | |
| 349 }, | |
| 350 | |
| 351 /** | |
| 352 * @param {boolean} visible | |
| 353 */ | |
| 354 setVisible: function(visible) | |
| 355 { | |
| 356 this._visible = visible; | |
| 357 if (!this._visible) { | |
| 358 this._popoverHelper.hidePopover(); | |
| 359 if (this._multilineEditing) | |
| 360 this._multilineEditing.cancel(); | |
| 361 return; | |
| 362 } | |
| 363 | |
| 364 this.runPendingUpdates(); | |
| 365 if (this._selectedDOMNode) | |
| 366 this._revealAndSelectNode(this._selectedDOMNode, false); | |
| 367 }, | |
| 368 | |
| 369 get rootDOMNode() | |
| 370 { | |
| 371 return this._rootDOMNode; | |
| 372 }, | |
| 373 | |
| 374 set rootDOMNode(x) | |
| 375 { | |
| 376 if (this._rootDOMNode === x) | |
| 377 return; | |
| 378 | |
| 379 this._rootDOMNode = x; | |
| 380 | |
| 381 this._isXMLMimeType = x && x.isXMLNode(); | |
| 382 | |
| 383 this.update(); | |
| 384 }, | |
| 385 | |
| 386 get isXMLMimeType() | |
| 387 { | |
| 388 return this._isXMLMimeType; | |
| 389 }, | |
| 390 | |
| 391 /** | |
| 392 * @return {?WebInspector.DOMNode} | |
| 393 */ | |
| 394 selectedDOMNode: function() | |
| 395 { | |
| 396 return this._selectedDOMNode; | |
| 397 }, | |
| 398 | |
| 399 /** | 1695 /** |
| 400 * @param {?WebInspector.DOMNode} node | 1696 * @param {?WebInspector.DOMNode} node |
| 401 * @param {boolean=} focus | 1697 * @this {WebInspector.ElementsTreeOutline.ShortcutTreeElement} |
| 402 */ | 1698 */ |
| 403 selectDOMNode: function(node, focus) | 1699 function resolved(node) { |
| 404 { | 1700 if (node) { |
| 405 if (this._selectedDOMNode === node) { | 1701 this.treeOutline._selectedDOMNode = node; |
| 406 this._revealAndSelectNode(node, !focus); | 1702 this.treeOutline._selectedNodeChanged(); |
| 407 return; | 1703 } |
| 408 } | 1704 } |
| 409 | 1705 return true; |
| 410 this._selectedDOMNode = node; | 1706 } |
| 411 this._revealAndSelectNode(node, !focus); | |
| 412 | |
| 413 // The _revealAndSelectNode() method might find a different element if t
here is inlined text, | |
| 414 // and the select() call would change the selectedDOMNode and reenter th
is setter. So to | |
| 415 // avoid calling _selectedNodeChanged() twice, first check if _selectedD
OMNode is the same | |
| 416 // node as the one passed in. | |
| 417 if (this._selectedDOMNode === node) | |
| 418 this._selectedNodeChanged(!!focus); | |
| 419 }, | |
| 420 | |
| 421 /** | |
| 422 * @return {boolean} | |
| 423 */ | |
| 424 editing: function() | |
| 425 { | |
| 426 var node = this.selectedDOMNode(); | |
| 427 if (!node) | |
| 428 return false; | |
| 429 var treeElement = this.findTreeElement(node); | |
| 430 if (!treeElement) | |
| 431 return false; | |
| 432 return treeElement.isEditing() || false; | |
| 433 }, | |
| 434 | |
| 435 update: function() | |
| 436 { | |
| 437 var selectedNode = this.selectedDOMNode(); | |
| 438 this.removeChildren(); | |
| 439 if (!this.rootDOMNode) | |
| 440 return; | |
| 441 | |
| 442 if (this._includeRootDOMNode) { | |
| 443 var treeElement = this._createElementTreeElement(this.rootDOMNode); | |
| 444 this.appendChild(treeElement); | |
| 445 } else { | |
| 446 // FIXME: this could use findTreeElement to reuse a tree element if
it already exists | |
| 447 var children = this._visibleChildren(this.rootDOMNode); | |
| 448 for (var child of children) { | |
| 449 var treeElement = this._createElementTreeElement(child); | |
| 450 this.appendChild(treeElement); | |
| 451 } | |
| 452 } | |
| 453 | |
| 454 if (selectedNode) | |
| 455 this._revealAndSelectNode(selectedNode, true); | |
| 456 }, | |
| 457 | |
| 458 /** | |
| 459 * @param {boolean} focus | |
| 460 */ | |
| 461 _selectedNodeChanged: function(focus) | |
| 462 { | |
| 463 this.dispatchEventToListeners(WebInspector.ElementsTreeOutline.Events.Se
lectedNodeChanged, {node: this._selectedDOMNode, focus: focus}); | |
| 464 }, | |
| 465 | |
| 466 /** | |
| 467 * @param {!Array.<!WebInspector.DOMNode>} nodes | |
| 468 */ | |
| 469 _fireElementsTreeUpdated: function(nodes) | |
| 470 { | |
| 471 this.dispatchEventToListeners(WebInspector.ElementsTreeOutline.Events.El
ementsTreeUpdated, nodes); | |
| 472 }, | |
| 473 | |
| 474 /** | |
| 475 * @param {!WebInspector.DOMNode} node | |
| 476 * @return {?WebInspector.ElementsTreeElement} | |
| 477 */ | |
| 478 findTreeElement: function(node) | |
| 479 { | |
| 480 var treeElement = this._lookUpTreeElement(node); | |
| 481 if (!treeElement && node.nodeType() === Node.TEXT_NODE) { | |
| 482 // The text node might have been inlined if it was short, so try to
find the parent element. | |
| 483 treeElement = this._lookUpTreeElement(node.parentNode); | |
| 484 } | |
| 485 | |
| 486 return /** @type {?WebInspector.ElementsTreeElement} */ (treeElement); | |
| 487 }, | |
| 488 | |
| 489 /** | |
| 490 * @param {?WebInspector.DOMNode} node | |
| 491 * @return {?TreeElement} | |
| 492 */ | |
| 493 _lookUpTreeElement: function(node) | |
| 494 { | |
| 495 if (!node) | |
| 496 return null; | |
| 497 | |
| 498 var cachedElement = node[this._treeElementSymbol]; | |
| 499 if (cachedElement) | |
| 500 return cachedElement; | |
| 501 | |
| 502 // Walk up the parent pointers from the desired node | |
| 503 var ancestors = []; | |
| 504 for (var currentNode = node.parentNode; currentNode; currentNode = curre
ntNode.parentNode) { | |
| 505 ancestors.push(currentNode); | |
| 506 if (currentNode[this._treeElementSymbol]) // stop climbing as soon
as we hit | |
| 507 break; | |
| 508 } | |
| 509 | |
| 510 if (!currentNode) | |
| 511 return null; | |
| 512 | |
| 513 // Walk down to populate each ancestor's children, to fill in the tree a
nd the cache. | |
| 514 for (var i = ancestors.length - 1; i >= 0; --i) { | |
| 515 var treeElement = ancestors[i][this._treeElementSymbol]; | |
| 516 if (treeElement) | |
| 517 treeElement.onpopulate(); // fill the cache with the children o
f treeElement | |
| 518 } | |
| 519 | |
| 520 return node[this._treeElementSymbol]; | |
| 521 }, | |
| 522 | |
| 523 /** | |
| 524 * @param {!WebInspector.DOMNode} node | |
| 525 * @return {?WebInspector.ElementsTreeElement} | |
| 526 */ | |
| 527 createTreeElementFor: function(node) | |
| 528 { | |
| 529 var treeElement = this.findTreeElement(node); | |
| 530 if (treeElement) | |
| 531 return treeElement; | |
| 532 if (!node.parentNode) | |
| 533 return null; | |
| 534 | |
| 535 treeElement = this.createTreeElementFor(node.parentNode); | |
| 536 return treeElement ? this._showChild(treeElement, node) : null; | |
| 537 }, | |
| 538 | |
| 539 set suppressRevealAndSelect(x) | |
| 540 { | |
| 541 if (this._suppressRevealAndSelect === x) | |
| 542 return; | |
| 543 this._suppressRevealAndSelect = x; | |
| 544 }, | |
| 545 | |
| 546 /** | |
| 547 * @param {?WebInspector.DOMNode} node | |
| 548 * @param {boolean} omitFocus | |
| 549 */ | |
| 550 _revealAndSelectNode: function(node, omitFocus) | |
| 551 { | |
| 552 if (this._suppressRevealAndSelect) | |
| 553 return; | |
| 554 | |
| 555 if (!this._includeRootDOMNode && node === this.rootDOMNode && this.rootD
OMNode) | |
| 556 node = this.rootDOMNode.firstChild; | |
| 557 if (!node) | |
| 558 return; | |
| 559 var treeElement = this.createTreeElementFor(node); | |
| 560 if (!treeElement) | |
| 561 return; | |
| 562 | |
| 563 treeElement.revealAndSelect(omitFocus); | |
| 564 }, | |
| 565 | |
| 566 /** | |
| 567 * @return {?TreeElement} | |
| 568 */ | |
| 569 _treeElementFromEvent: function(event) | |
| 570 { | |
| 571 var scrollContainer = this.element.parentElement; | |
| 572 | |
| 573 // We choose this X coordinate based on the knowledge that our list | |
| 574 // items extend at least to the right edge of the outer <ol> container. | |
| 575 // In the no-word-wrap mode the outer <ol> may be wider than the tree co
ntainer | |
| 576 // (and partially hidden), in which case we are left to use only its rig
ht boundary. | |
| 577 var x = scrollContainer.totalOffsetLeft() + scrollContainer.offsetWidth
- 36; | |
| 578 | |
| 579 var y = event.pageY; | |
| 580 | |
| 581 // Our list items have 1-pixel cracks between them vertically. We avoid | |
| 582 // the cracks by checking slightly above and slightly below the mouse | |
| 583 // and seeing if we hit the same element each time. | |
| 584 var elementUnderMouse = this.treeElementFromPoint(x, y); | |
| 585 var elementAboveMouse = this.treeElementFromPoint(x, y - 2); | |
| 586 var element; | |
| 587 if (elementUnderMouse === elementAboveMouse) | |
| 588 element = elementUnderMouse; | |
| 589 else | |
| 590 element = this.treeElementFromPoint(x, y + 2); | |
| 591 | |
| 592 return element; | |
| 593 }, | |
| 594 | |
| 595 /** | |
| 596 * @param {!Element} element | |
| 597 * @param {!Event} event | |
| 598 * @return {!Element|!AnchorBox|undefined} | |
| 599 */ | |
| 600 _getPopoverAnchor: function(element, event) | |
| 601 { | |
| 602 var anchor = element.enclosingNodeOrSelfWithClass("webkit-html-resource-
link"); | |
| 603 if (!anchor || !anchor.href) | |
| 604 return; | |
| 605 | |
| 606 return anchor; | |
| 607 }, | |
| 608 | |
| 609 /** | |
| 610 * @param {!WebInspector.DOMNode} node | |
| 611 * @param {function()} callback | |
| 612 */ | |
| 613 _loadDimensionsForNode: function(node, callback) | |
| 614 { | |
| 615 if (!node.nodeName() || node.nodeName().toLowerCase() !== "img") { | |
| 616 callback(); | |
| 617 return; | |
| 618 } | |
| 619 | |
| 620 node.resolveToObject("", resolvedNode); | |
| 621 | |
| 622 function resolvedNode(object) | |
| 623 { | |
| 624 if (!object) { | |
| 625 callback(); | |
| 626 return; | |
| 627 } | |
| 628 | |
| 629 object.callFunctionJSON(features, undefined, callback); | |
| 630 object.release(); | |
| 631 | |
| 632 /** | |
| 633 * @return {!{offsetWidth: number, offsetHeight: number, naturalWidt
h: number, naturalHeight: number, currentSrc: (string|undefined)}} | |
| 634 * @suppressReceiverCheck | |
| 635 * @this {!Element} | |
| 636 */ | |
| 637 function features() | |
| 638 { | |
| 639 return { offsetWidth: this.offsetWidth, offsetHeight: this.offse
tHeight, naturalWidth: this.naturalWidth, naturalHeight: this.naturalHeight, cur
rentSrc: this.currentSrc }; | |
| 640 } | |
| 641 } | |
| 642 }, | |
| 643 | |
| 644 /** | |
| 645 * @param {!Element} anchor | |
| 646 * @param {!WebInspector.Popover} popover | |
| 647 */ | |
| 648 _showPopover: function(anchor, popover) | |
| 649 { | |
| 650 var listItem = anchor.enclosingNodeOrSelfWithNodeName("li"); | |
| 651 var node = /** @type {!WebInspector.ElementsTreeElement} */ (listItem.tr
eeElement).node(); | |
| 652 this._loadDimensionsForNode(node, WebInspector.DOMPresentationUtils.buil
dImagePreviewContents.bind(WebInspector.DOMPresentationUtils, node.target(), anc
hor.href, true, showPopover)); | |
| 653 | |
| 654 /** | |
| 655 * @param {!Element=} contents | |
| 656 */ | |
| 657 function showPopover(contents) | |
| 658 { | |
| 659 if (!contents) | |
| 660 return; | |
| 661 popover.setCanShrink(false); | |
| 662 popover.showForAnchor(contents, anchor); | |
| 663 } | |
| 664 }, | |
| 665 | |
| 666 _onmousedown: function(event) | |
| 667 { | |
| 668 var element = this._treeElementFromEvent(event); | |
| 669 | |
| 670 if (!element || element.isEventWithinDisclosureTriangle(event)) | |
| 671 return; | |
| 672 | |
| 673 element.select(); | |
| 674 }, | |
| 675 | |
| 676 /** | |
| 677 * @param {?TreeElement} treeElement | |
| 678 */ | |
| 679 setHoverEffect: function(treeElement) | |
| 680 { | |
| 681 if (this._previousHoveredElement === treeElement) | |
| 682 return; | |
| 683 | |
| 684 if (this._previousHoveredElement) { | |
| 685 this._previousHoveredElement.hovered = false; | |
| 686 delete this._previousHoveredElement; | |
| 687 } | |
| 688 | |
| 689 if (treeElement) { | |
| 690 treeElement.hovered = true; | |
| 691 this._previousHoveredElement = treeElement; | |
| 692 } | |
| 693 }, | |
| 694 | |
| 695 _onmousemove: function(event) | |
| 696 { | |
| 697 var element = this._treeElementFromEvent(event); | |
| 698 if (element && this._previousHoveredElement === element) | |
| 699 return; | |
| 700 | |
| 701 this.setHoverEffect(element); | |
| 702 | |
| 703 if (element instanceof WebInspector.ElementsTreeElement) { | |
| 704 this._domModel.highlightDOMNodeWithConfig(element.node().id, { mode:
"all", showInfo: !WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) }); | |
| 705 return; | |
| 706 } | |
| 707 | |
| 708 if (element instanceof WebInspector.ElementsTreeOutline.ShortcutTreeElem
ent) | |
| 709 this._domModel.highlightDOMNodeWithConfig(undefined, { mode: "all",
showInfo: !WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) }, element.ba
ckendNodeId()); | |
| 710 }, | |
| 711 | |
| 712 _onmouseleave: function(event) | |
| 713 { | |
| 714 this.setHoverEffect(null); | |
| 715 WebInspector.DOMModel.hideDOMNodeHighlight(); | |
| 716 }, | |
| 717 | |
| 718 _ondragstart: function(event) | |
| 719 { | |
| 720 if (!event.target.isComponentSelectionCollapsed()) | |
| 721 return false; | |
| 722 if (event.target.nodeName === "A") | |
| 723 return false; | |
| 724 | |
| 725 var treeElement = this._treeElementFromEvent(event); | |
| 726 if (!this._isValidDragSourceOrTarget(treeElement)) | |
| 727 return false; | |
| 728 | |
| 729 if (treeElement.node().nodeName() === "BODY" || treeElement.node().nodeN
ame() === "HEAD") | |
| 730 return false; | |
| 731 | |
| 732 event.dataTransfer.setData("text/plain", treeElement.listItemElement.tex
tContent.replace(/\u200b/g, "")); | |
| 733 event.dataTransfer.effectAllowed = "copyMove"; | |
| 734 this._treeElementBeingDragged = treeElement; | |
| 735 | |
| 736 WebInspector.DOMModel.hideDOMNodeHighlight(); | |
| 737 | |
| 738 return true; | |
| 739 }, | |
| 740 | |
| 741 _ondragover: function(event) | |
| 742 { | |
| 743 if (!this._treeElementBeingDragged) | |
| 744 return false; | |
| 745 | |
| 746 var treeElement = this._treeElementFromEvent(event); | |
| 747 if (!this._isValidDragSourceOrTarget(treeElement)) | |
| 748 return false; | |
| 749 | |
| 750 var node = treeElement.node(); | |
| 751 while (node) { | |
| 752 if (node === this._treeElementBeingDragged._node) | |
| 753 return false; | |
| 754 node = node.parentNode; | |
| 755 } | |
| 756 | |
| 757 treeElement.listItemElement.classList.add("elements-drag-over"); | |
| 758 this._dragOverTreeElement = treeElement; | |
| 759 event.preventDefault(); | |
| 760 event.dataTransfer.dropEffect = "move"; | |
| 761 return false; | |
| 762 }, | |
| 763 | |
| 764 _ondragleave: function(event) | |
| 765 { | |
| 766 this._clearDragOverTreeElementMarker(); | |
| 767 event.preventDefault(); | |
| 768 return false; | |
| 769 }, | |
| 770 | |
| 771 /** | |
| 772 * @param {?TreeElement} treeElement | |
| 773 * @return {boolean} | |
| 774 */ | |
| 775 _isValidDragSourceOrTarget: function(treeElement) | |
| 776 { | |
| 777 if (!treeElement) | |
| 778 return false; | |
| 779 | |
| 780 if (!(treeElement instanceof WebInspector.ElementsTreeElement)) | |
| 781 return false; | |
| 782 var elementsTreeElement = /** @type {!WebInspector.ElementsTreeElement}
*/ (treeElement); | |
| 783 | |
| 784 var node = elementsTreeElement.node(); | |
| 785 if (!node.parentNode || node.parentNode.nodeType() !== Node.ELEMENT_NODE
) | |
| 786 return false; | |
| 787 | |
| 788 return true; | |
| 789 }, | |
| 790 | |
| 791 _ondrop: function(event) | |
| 792 { | |
| 793 event.preventDefault(); | |
| 794 var treeElement = this._treeElementFromEvent(event); | |
| 795 if (treeElement) | |
| 796 this._doMove(treeElement); | |
| 797 }, | |
| 798 | |
| 799 /** | |
| 800 * @param {!TreeElement} treeElement | |
| 801 */ | |
| 802 _doMove: function(treeElement) | |
| 803 { | |
| 804 if (!this._treeElementBeingDragged) | |
| 805 return; | |
| 806 | |
| 807 var parentNode; | |
| 808 var anchorNode; | |
| 809 | |
| 810 if (treeElement.isClosingTag()) { | |
| 811 // Drop onto closing tag -> insert as last child. | |
| 812 parentNode = treeElement.node(); | |
| 813 } else { | |
| 814 var dragTargetNode = treeElement.node(); | |
| 815 parentNode = dragTargetNode.parentNode; | |
| 816 anchorNode = dragTargetNode; | |
| 817 } | |
| 818 | |
| 819 var wasExpanded = this._treeElementBeingDragged.expanded; | |
| 820 this._treeElementBeingDragged._node.moveTo(parentNode, anchorNode, this.
selectNodeAfterEdit.bind(this, wasExpanded)); | |
| 821 | |
| 822 delete this._treeElementBeingDragged; | |
| 823 }, | |
| 824 | |
| 825 _ondragend: function(event) | |
| 826 { | |
| 827 event.preventDefault(); | |
| 828 this._clearDragOverTreeElementMarker(); | |
| 829 delete this._treeElementBeingDragged; | |
| 830 }, | |
| 831 | |
| 832 _clearDragOverTreeElementMarker: function() | |
| 833 { | |
| 834 if (this._dragOverTreeElement) { | |
| 835 this._dragOverTreeElement.listItemElement.classList.remove("elements
-drag-over"); | |
| 836 delete this._dragOverTreeElement; | |
| 837 } | |
| 838 }, | |
| 839 | |
| 840 _contextMenuEventFired: function(event) | |
| 841 { | |
| 842 var treeElement = this._treeElementFromEvent(event); | |
| 843 if (treeElement instanceof WebInspector.ElementsTreeElement) | |
| 844 this.showContextMenu(treeElement, event); | |
| 845 }, | |
| 846 | |
| 847 /** | |
| 848 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 849 * @param {!Event} event | |
| 850 */ | |
| 851 showContextMenu: function(treeElement, event) | |
| 852 { | |
| 853 if (WebInspector.isEditing()) | |
| 854 return; | |
| 855 | |
| 856 var contextMenu = new WebInspector.ContextMenu(event); | |
| 857 var isPseudoElement = !!treeElement.node().pseudoType(); | |
| 858 var isTag = treeElement.node().nodeType() === Node.ELEMENT_NODE && !isPs
eudoElement; | |
| 859 var textNode = event.target.enclosingNodeOrSelfWithClass("webkit-html-te
xt-node"); | |
| 860 if (textNode && textNode.classList.contains("bogus")) | |
| 861 textNode = null; | |
| 862 var commentNode = event.target.enclosingNodeOrSelfWithClass("webkit-html
-comment"); | |
| 863 contextMenu.appendApplicableItems(event.target); | |
| 864 if (textNode) { | |
| 865 contextMenu.appendSeparator(); | |
| 866 treeElement.populateTextContextMenu(contextMenu, textNode); | |
| 867 } else if (isTag) { | |
| 868 contextMenu.appendSeparator(); | |
| 869 treeElement.populateTagContextMenu(contextMenu, event); | |
| 870 } else if (commentNode) { | |
| 871 contextMenu.appendSeparator(); | |
| 872 treeElement.populateNodeContextMenu(contextMenu); | |
| 873 } else if (isPseudoElement) { | |
| 874 treeElement.populateScrollIntoView(contextMenu); | |
| 875 } | |
| 876 | |
| 877 contextMenu.appendApplicableItems(treeElement.node()); | |
| 878 contextMenu.show(); | |
| 879 }, | |
| 880 | |
| 881 runPendingUpdates: function() | |
| 882 { | |
| 883 this._updateModifiedNodes(); | |
| 884 }, | |
| 885 | |
| 886 handleShortcut: function(event) | |
| 887 { | |
| 888 var node = this.selectedDOMNode(); | |
| 889 if (!node) | |
| 890 return; | |
| 891 var treeElement = node[this._treeElementSymbol]; | |
| 892 if (!treeElement) | |
| 893 return; | |
| 894 | |
| 895 if (WebInspector.KeyboardShortcut.eventHasCtrlOrMeta(event) && node.pare
ntNode) { | |
| 896 if (event.key === "ArrowUp" && node.previousSibling) { | |
| 897 node.moveTo(node.parentNode, node.previousSibling, this.selectNo
deAfterEdit.bind(this, treeElement.expanded)); | |
| 898 event.handled = true; | |
| 899 return; | |
| 900 } | |
| 901 if (event.key === "ArrowDown" && node.nextSibling) { | |
| 902 node.moveTo(node.parentNode, node.nextSibling.nextSibling, this.
selectNodeAfterEdit.bind(this, treeElement.expanded)); | |
| 903 event.handled = true; | |
| 904 return; | |
| 905 } | |
| 906 } | |
| 907 }, | |
| 908 | |
| 909 /** | |
| 910 * @param {!WebInspector.DOMNode} node | |
| 911 * @param {boolean=} startEditing | |
| 912 * @param {function()=} callback | |
| 913 */ | |
| 914 toggleEditAsHTML: function(node, startEditing, callback) | |
| 915 { | |
| 916 var treeElement = node[this._treeElementSymbol]; | |
| 917 if (!treeElement || !treeElement.hasEditableNode()) | |
| 918 return; | |
| 919 | |
| 920 if (node.pseudoType()) | |
| 921 return; | |
| 922 | |
| 923 var parentNode = node.parentNode; | |
| 924 var index = node.index; | |
| 925 var wasExpanded = treeElement.expanded; | |
| 926 | |
| 927 treeElement.toggleEditAsHTML(editingFinished.bind(this), startEditing); | |
| 928 | |
| 929 /** | |
| 930 * @this {WebInspector.ElementsTreeOutline} | |
| 931 * @param {boolean} success | |
| 932 */ | |
| 933 function editingFinished(success) | |
| 934 { | |
| 935 if (callback) | |
| 936 callback(); | |
| 937 if (!success) | |
| 938 return; | |
| 939 | |
| 940 // Select it and expand if necessary. We force tree update so that i
t processes dom events and is up to date. | |
| 941 this.runPendingUpdates(); | |
| 942 | |
| 943 var newNode = parentNode ? parentNode.children()[index] || parentNod
e : null; | |
| 944 if (!newNode) | |
| 945 return; | |
| 946 | |
| 947 this.selectDOMNode(newNode, true); | |
| 948 | |
| 949 if (wasExpanded) { | |
| 950 var newTreeItem = this.findTreeElement(newNode); | |
| 951 if (newTreeItem) | |
| 952 newTreeItem.expand(); | |
| 953 } | |
| 954 } | |
| 955 }, | |
| 956 | |
| 957 /** | |
| 958 * @param {boolean} wasExpanded | |
| 959 * @param {?Protocol.Error} error | |
| 960 * @param {!DOMAgent.NodeId=} nodeId | |
| 961 * @return {?WebInspector.ElementsTreeElement} nodeId | |
| 962 */ | |
| 963 selectNodeAfterEdit: function(wasExpanded, error, nodeId) | |
| 964 { | |
| 965 if (error) | |
| 966 return null; | |
| 967 | |
| 968 // Select it and expand if necessary. We force tree update so that it pr
ocesses dom events and is up to date. | |
| 969 this.runPendingUpdates(); | |
| 970 | |
| 971 var newNode = nodeId ? this._domModel.nodeForId(nodeId) : null; | |
| 972 if (!newNode) | |
| 973 return null; | |
| 974 | |
| 975 this.selectDOMNode(newNode, true); | |
| 976 | |
| 977 var newTreeItem = this.findTreeElement(newNode); | |
| 978 if (wasExpanded) { | |
| 979 if (newTreeItem) | |
| 980 newTreeItem.expand(); | |
| 981 } | |
| 982 return newTreeItem; | |
| 983 }, | |
| 984 | |
| 985 /** | |
| 986 * Runs a script on the node's remote object that toggles a class name on | |
| 987 * the node and injects a stylesheet into the head of the node's document | |
| 988 * containing a rule to set "visibility: hidden" on the class and all it's | |
| 989 * ancestors. | |
| 990 * | |
| 991 * @param {!WebInspector.DOMNode} node | |
| 992 * @param {function(?WebInspector.RemoteObject, boolean=)=} userCallback | |
| 993 */ | |
| 994 toggleHideElement: function(node, userCallback) | |
| 995 { | |
| 996 var pseudoType = node.pseudoType(); | |
| 997 var effectiveNode = pseudoType ? node.parentNode : node; | |
| 998 if (!effectiveNode) | |
| 999 return; | |
| 1000 | |
| 1001 var hidden = node.marker("hidden-marker"); | |
| 1002 | |
| 1003 function resolvedNode(object) | |
| 1004 { | |
| 1005 if (!object) | |
| 1006 return; | |
| 1007 | |
| 1008 /** | |
| 1009 * @param {?string} pseudoType | |
| 1010 * @param {boolean} hidden | |
| 1011 * @suppressGlobalPropertiesCheck | |
| 1012 * @suppressReceiverCheck | |
| 1013 * @this {!Element} | |
| 1014 */ | |
| 1015 function toggleClassAndInjectStyleRule(pseudoType, hidden) | |
| 1016 { | |
| 1017 const classNamePrefix = "__web-inspector-hide"; | |
| 1018 const classNameSuffix = "-shortcut__"; | |
| 1019 const styleTagId = "__web-inspector-hide-shortcut-style__"; | |
| 1020 var selectors = []; | |
| 1021 selectors.push(".__web-inspector-hide-shortcut__"); | |
| 1022 selectors.push(".__web-inspector-hide-shortcut__ *"); | |
| 1023 selectors.push(".__web-inspector-hidebefore-shortcut__::before")
; | |
| 1024 selectors.push(".__web-inspector-hideafter-shortcut__::after"); | |
| 1025 var selector = selectors.join(", "); | |
| 1026 var ruleBody = " visibility: hidden !important;"; | |
| 1027 var rule = "\n" + selector + "\n{\n" + ruleBody + "\n}\n"; | |
| 1028 var className = classNamePrefix + (pseudoType || "") + className
Suffix; | |
| 1029 this.classList.toggle(className, hidden); | |
| 1030 | |
| 1031 var localRoot = this; | |
| 1032 while (localRoot.parentNode) | |
| 1033 localRoot = localRoot.parentNode; | |
| 1034 if (localRoot.nodeType === Node.DOCUMENT_NODE) | |
| 1035 localRoot = document.head; | |
| 1036 | |
| 1037 var style = localRoot.querySelector("style#" + styleTagId); | |
| 1038 if (style) | |
| 1039 return; | |
| 1040 | |
| 1041 style = document.createElement("style"); | |
| 1042 style.id = styleTagId; | |
| 1043 style.type = "text/css"; | |
| 1044 style.textContent = rule; | |
| 1045 | |
| 1046 localRoot.appendChild(style); | |
| 1047 } | |
| 1048 | |
| 1049 object.callFunction(toggleClassAndInjectStyleRule, [{ value: pseudoT
ype }, { value: !hidden}], userCallback); | |
| 1050 object.release(); | |
| 1051 node.setMarker("hidden-marker", hidden ? null : true); | |
| 1052 } | |
| 1053 | |
| 1054 effectiveNode.resolveToObject("", resolvedNode); | |
| 1055 }, | |
| 1056 | |
| 1057 /** | |
| 1058 * @param {!WebInspector.DOMNode} node | |
| 1059 * @return {boolean} | |
| 1060 */ | |
| 1061 isToggledToHidden: function(node) | |
| 1062 { | |
| 1063 return !!node.marker("hidden-marker"); | |
| 1064 }, | |
| 1065 | |
| 1066 _reset: function() | |
| 1067 { | |
| 1068 this.rootDOMNode = null; | |
| 1069 this.selectDOMNode(null, false); | |
| 1070 this._popoverHelper.hidePopover(); | |
| 1071 delete this._clipboardNodeData; | |
| 1072 WebInspector.DOMModel.hideDOMNodeHighlight(); | |
| 1073 this._updateRecords.clear(); | |
| 1074 }, | |
| 1075 | |
| 1076 wireToDOMModel: function() | |
| 1077 { | |
| 1078 this._domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbol] = th
is; | |
| 1079 this._domModel.addEventListener(WebInspector.DOMModel.Events.NodeInserte
d, this._nodeInserted, this); | |
| 1080 this._domModel.addEventListener(WebInspector.DOMModel.Events.NodeRemoved
, this._nodeRemoved, this); | |
| 1081 this._domModel.addEventListener(WebInspector.DOMModel.Events.AttrModifie
d, this._attributeModified, this); | |
| 1082 this._domModel.addEventListener(WebInspector.DOMModel.Events.AttrRemoved
, this._attributeRemoved, this); | |
| 1083 this._domModel.addEventListener(WebInspector.DOMModel.Events.CharacterDa
taModified, this._characterDataModified, this); | |
| 1084 this._domModel.addEventListener(WebInspector.DOMModel.Events.DocumentUpd
ated, this._documentUpdated, this); | |
| 1085 this._domModel.addEventListener(WebInspector.DOMModel.Events.ChildNodeCo
untUpdated, this._childNodeCountUpdated, this); | |
| 1086 this._domModel.addEventListener(WebInspector.DOMModel.Events.Distributed
NodesChanged, this._distributedNodesChanged, this); | |
| 1087 }, | |
| 1088 | |
| 1089 unwireFromDOMModel: function() | |
| 1090 { | |
| 1091 this._domModel.removeEventListener(WebInspector.DOMModel.Events.NodeInse
rted, this._nodeInserted, this); | |
| 1092 this._domModel.removeEventListener(WebInspector.DOMModel.Events.NodeRemo
ved, this._nodeRemoved, this); | |
| 1093 this._domModel.removeEventListener(WebInspector.DOMModel.Events.AttrModi
fied, this._attributeModified, this); | |
| 1094 this._domModel.removeEventListener(WebInspector.DOMModel.Events.AttrRemo
ved, this._attributeRemoved, this); | |
| 1095 this._domModel.removeEventListener(WebInspector.DOMModel.Events.Characte
rDataModified, this._characterDataModified, this); | |
| 1096 this._domModel.removeEventListener(WebInspector.DOMModel.Events.Document
Updated, this._documentUpdated, this); | |
| 1097 this._domModel.removeEventListener(WebInspector.DOMModel.Events.ChildNod
eCountUpdated, this._childNodeCountUpdated, this); | |
| 1098 this._domModel.removeEventListener(WebInspector.DOMModel.Events.Distribu
tedNodesChanged, this._distributedNodesChanged, this); | |
| 1099 delete this._domModel[WebInspector.ElementsTreeOutline._treeOutlineSymbo
l]; | |
| 1100 }, | |
| 1101 | |
| 1102 /** | |
| 1103 * @param {!WebInspector.DOMNode} node | |
| 1104 * @return {!WebInspector.ElementsTreeOutline.UpdateRecord} | |
| 1105 */ | |
| 1106 _addUpdateRecord: function(node) | |
| 1107 { | |
| 1108 var record = this._updateRecords.get(node); | |
| 1109 if (!record) { | |
| 1110 record = new WebInspector.ElementsTreeOutline.UpdateRecord(); | |
| 1111 this._updateRecords.set(node, record); | |
| 1112 } | |
| 1113 return record; | |
| 1114 }, | |
| 1115 | |
| 1116 /** | |
| 1117 * @param {!WebInspector.DOMNode} node | |
| 1118 * @return {?WebInspector.ElementsTreeOutline.UpdateRecord} | |
| 1119 */ | |
| 1120 _updateRecordForHighlight: function(node) | |
| 1121 { | |
| 1122 if (!this._visible) | |
| 1123 return null; | |
| 1124 return this._updateRecords.get(node) || null; | |
| 1125 }, | |
| 1126 | |
| 1127 /** | |
| 1128 * @param {!WebInspector.Event} event | |
| 1129 */ | |
| 1130 _documentUpdated: function(event) | |
| 1131 { | |
| 1132 var inspectedRootDocument = event.data; | |
| 1133 | |
| 1134 this._reset(); | |
| 1135 | |
| 1136 if (!inspectedRootDocument) | |
| 1137 return; | |
| 1138 | |
| 1139 this.rootDOMNode = inspectedRootDocument; | |
| 1140 }, | |
| 1141 | |
| 1142 /** | |
| 1143 * @param {!WebInspector.Event} event | |
| 1144 */ | |
| 1145 _attributeModified: function(event) | |
| 1146 { | |
| 1147 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); | |
| 1148 this._addUpdateRecord(node).attributeModified(event.data.name); | |
| 1149 this._updateModifiedNodesSoon(); | |
| 1150 }, | |
| 1151 | |
| 1152 /** | |
| 1153 * @param {!WebInspector.Event} event | |
| 1154 */ | |
| 1155 _attributeRemoved: function(event) | |
| 1156 { | |
| 1157 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); | |
| 1158 this._addUpdateRecord(node).attributeRemoved(event.data.name); | |
| 1159 this._updateModifiedNodesSoon(); | |
| 1160 }, | |
| 1161 | |
| 1162 /** | |
| 1163 * @param {!WebInspector.Event} event | |
| 1164 */ | |
| 1165 _characterDataModified: function(event) | |
| 1166 { | |
| 1167 var node = /** @type {!WebInspector.DOMNode} */ (event.data); | |
| 1168 this._addUpdateRecord(node).charDataModified(); | |
| 1169 // Text could be large and force us to render itself as the child in the
tree outline. | |
| 1170 if (node.parentNode && node.parentNode.firstChild === node.parentNode.la
stChild) | |
| 1171 this._addUpdateRecord(node.parentNode).childrenModified(); | |
| 1172 this._updateModifiedNodesSoon(); | |
| 1173 }, | |
| 1174 | |
| 1175 /** | |
| 1176 * @param {!WebInspector.Event} event | |
| 1177 */ | |
| 1178 _nodeInserted: function(event) | |
| 1179 { | |
| 1180 var node = /** @type {!WebInspector.DOMNode} */ (event.data); | |
| 1181 this._addUpdateRecord(/** @type {!WebInspector.DOMNode} */ (node.parentN
ode)).nodeInserted(node); | |
| 1182 this._updateModifiedNodesSoon(); | |
| 1183 }, | |
| 1184 | |
| 1185 /** | |
| 1186 * @param {!WebInspector.Event} event | |
| 1187 */ | |
| 1188 _nodeRemoved: function(event) | |
| 1189 { | |
| 1190 var node = /** @type {!WebInspector.DOMNode} */ (event.data.node); | |
| 1191 var parentNode = /** @type {!WebInspector.DOMNode} */ (event.data.parent
); | |
| 1192 this.resetClipboardIfNeeded(node); | |
| 1193 this._addUpdateRecord(parentNode).nodeRemoved(node); | |
| 1194 this._updateModifiedNodesSoon(); | |
| 1195 }, | |
| 1196 | |
| 1197 /** | |
| 1198 * @param {!WebInspector.Event} event | |
| 1199 */ | |
| 1200 _childNodeCountUpdated: function(event) | |
| 1201 { | |
| 1202 var node = /** @type {!WebInspector.DOMNode} */ (event.data); | |
| 1203 this._addUpdateRecord(node).childrenModified(); | |
| 1204 this._updateModifiedNodesSoon(); | |
| 1205 }, | |
| 1206 | |
| 1207 /** | |
| 1208 * @param {!WebInspector.Event} event | |
| 1209 */ | |
| 1210 _distributedNodesChanged: function(event) | |
| 1211 { | |
| 1212 var node = /** @type {!WebInspector.DOMNode} */ (event.data); | |
| 1213 this._addUpdateRecord(node).childrenModified(); | |
| 1214 this._updateModifiedNodesSoon(); | |
| 1215 }, | |
| 1216 | |
| 1217 _updateModifiedNodesSoon: function() | |
| 1218 { | |
| 1219 if (!this._updateRecords.size) | |
| 1220 return; | |
| 1221 if (this._updateModifiedNodesTimeout) | |
| 1222 return; | |
| 1223 this._updateModifiedNodesTimeout = setTimeout(this._updateModifiedNodes.
bind(this), 50); | |
| 1224 }, | |
| 1225 | |
| 1226 _updateModifiedNodes: function() | |
| 1227 { | |
| 1228 if (this._updateModifiedNodesTimeout) { | |
| 1229 clearTimeout(this._updateModifiedNodesTimeout); | |
| 1230 delete this._updateModifiedNodesTimeout; | |
| 1231 } | |
| 1232 | |
| 1233 var updatedNodes = this._updateRecords.keysArray(); | |
| 1234 var hidePanelWhileUpdating = updatedNodes.length > 10; | |
| 1235 if (hidePanelWhileUpdating) { | |
| 1236 var treeOutlineContainerElement = this.element.parentNode; | |
| 1237 var originalScrollTop = treeOutlineContainerElement ? treeOutlineCon
tainerElement.scrollTop : 0; | |
| 1238 this._element.classList.add("hidden"); | |
| 1239 } | |
| 1240 | |
| 1241 if (this._rootDOMNode && this._updateRecords.get(this._rootDOMNode) && t
his._updateRecords.get(this._rootDOMNode).hasChangedChildren()) { | |
| 1242 // Document's children have changed, perform total update. | |
| 1243 this.update(); | |
| 1244 } else { | |
| 1245 for (var node of this._updateRecords.keys()) { | |
| 1246 if (this._updateRecords.get(node).hasChangedChildren()) | |
| 1247 this._updateModifiedParentNode(node); | |
| 1248 else | |
| 1249 this._updateModifiedNode(node); | |
| 1250 } | |
| 1251 } | |
| 1252 | |
| 1253 if (hidePanelWhileUpdating) { | |
| 1254 this._element.classList.remove("hidden"); | |
| 1255 if (originalScrollTop) | |
| 1256 treeOutlineContainerElement.scrollTop = originalScrollTop; | |
| 1257 } | |
| 1258 | |
| 1259 this._updateRecords.clear(); | |
| 1260 this._fireElementsTreeUpdated(updatedNodes); | |
| 1261 }, | |
| 1262 | |
| 1263 _updateModifiedNode: function(node) | |
| 1264 { | |
| 1265 var treeElement = this.findTreeElement(node); | |
| 1266 if (treeElement) | |
| 1267 treeElement.updateTitle(this._updateRecordForHighlight(node)); | |
| 1268 }, | |
| 1269 | |
| 1270 _updateModifiedParentNode: function(node) | |
| 1271 { | |
| 1272 var parentTreeElement = this.findTreeElement(node); | |
| 1273 if (parentTreeElement) { | |
| 1274 parentTreeElement.setExpandable(this._hasVisibleChildren(node)); | |
| 1275 parentTreeElement.updateTitle(this._updateRecordForHighlight(node)); | |
| 1276 if (parentTreeElement.populated) | |
| 1277 this._updateChildren(parentTreeElement); | |
| 1278 } | |
| 1279 }, | |
| 1280 | |
| 1281 /** | |
| 1282 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1283 */ | |
| 1284 populateTreeElement: function(treeElement) | |
| 1285 { | |
| 1286 if (treeElement.childCount() || !treeElement.isExpandable()) | |
| 1287 return; | |
| 1288 | |
| 1289 this._updateModifiedParentNode(treeElement.node()); | |
| 1290 }, | |
| 1291 | |
| 1292 /** | |
| 1293 * @param {!WebInspector.DOMNode} node | |
| 1294 * @param {boolean=} closingTag | |
| 1295 * @return {!WebInspector.ElementsTreeElement} | |
| 1296 */ | |
| 1297 _createElementTreeElement: function(node, closingTag) | |
| 1298 { | |
| 1299 var treeElement = new WebInspector.ElementsTreeElement(node, closingTag)
; | |
| 1300 treeElement.setExpandable(!closingTag && this._hasVisibleChildren(node))
; | |
| 1301 if (node.nodeType() === Node.ELEMENT_NODE && node.parentNode && node.par
entNode.nodeType() === Node.DOCUMENT_NODE && !node.parentNode.parentNode) | |
| 1302 treeElement.setCollapsible(false); | |
| 1303 treeElement.selectable = this._selectEnabled; | |
| 1304 return treeElement; | |
| 1305 }, | |
| 1306 | |
| 1307 /** | |
| 1308 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1309 * @param {!WebInspector.DOMNode} child | |
| 1310 * @return {?WebInspector.ElementsTreeElement} | |
| 1311 */ | |
| 1312 _showChild: function(treeElement, child) | |
| 1313 { | |
| 1314 if (treeElement.isClosingTag()) | |
| 1315 return null; | |
| 1316 | |
| 1317 var index = this._visibleChildren(treeElement.node()).indexOf(child); | |
| 1318 if (index === -1) | |
| 1319 return null; | |
| 1320 | |
| 1321 if (index >= treeElement.expandedChildrenLimit()) | |
| 1322 this.setExpandedChildrenLimit(treeElement, index + 1); | |
| 1323 return /** @type {!WebInspector.ElementsTreeElement} */ (treeElement.chi
ldAt(index)); | |
| 1324 }, | |
| 1325 | |
| 1326 /** | |
| 1327 * @param {!WebInspector.DOMNode} node | |
| 1328 * @return {!Array.<!WebInspector.DOMNode>} visibleChildren | |
| 1329 */ | |
| 1330 _visibleChildren: function(node) | |
| 1331 { | |
| 1332 var visibleChildren = WebInspector.ElementsTreeElement.visibleShadowRoot
s(node); | |
| 1333 | |
| 1334 var importedDocument = node.importedDocument(); | |
| 1335 if (importedDocument) | |
| 1336 visibleChildren.push(importedDocument); | |
| 1337 | |
| 1338 var templateContent = node.templateContent(); | |
| 1339 if (templateContent) | |
| 1340 visibleChildren.push(templateContent); | |
| 1341 | |
| 1342 var beforePseudoElement = node.beforePseudoElement(); | |
| 1343 if (beforePseudoElement) | |
| 1344 visibleChildren.push(beforePseudoElement); | |
| 1345 | |
| 1346 if (node.childNodeCount()) { | |
| 1347 var children = node.children(); | |
| 1348 if (!this._showHTMLCommentsSetting.get()) | |
| 1349 children = children.filter(n => n.nodeType() !== Node.COMMENT_NO
DE); | |
| 1350 visibleChildren = visibleChildren.concat(children); | |
| 1351 } | |
| 1352 | |
| 1353 var afterPseudoElement = node.afterPseudoElement(); | |
| 1354 if (afterPseudoElement) | |
| 1355 visibleChildren.push(afterPseudoElement); | |
| 1356 | |
| 1357 return visibleChildren; | |
| 1358 }, | |
| 1359 | |
| 1360 /** | |
| 1361 * @param {!WebInspector.DOMNode} node | |
| 1362 * @return {boolean} | |
| 1363 */ | |
| 1364 _hasVisibleChildren: function(node) | |
| 1365 { | |
| 1366 if (node.importedDocument()) | |
| 1367 return true; | |
| 1368 if (node.templateContent()) | |
| 1369 return true; | |
| 1370 if (WebInspector.ElementsTreeElement.visibleShadowRoots(node).length) | |
| 1371 return true; | |
| 1372 if (node.hasPseudoElements()) | |
| 1373 return true; | |
| 1374 if (node.isInsertionPoint()) | |
| 1375 return true; | |
| 1376 return !!node.childNodeCount() && !WebInspector.ElementsTreeElement.canS
howInlineText(node); | |
| 1377 }, | |
| 1378 | |
| 1379 /** | |
| 1380 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1381 */ | |
| 1382 _createExpandAllButtonTreeElement: function(treeElement) | |
| 1383 { | |
| 1384 var button = createTextButton("", handleLoadAllChildren.bind(this)); | |
| 1385 button.value = ""; | |
| 1386 var expandAllButtonElement = new TreeElement(button); | |
| 1387 expandAllButtonElement.selectable = false; | |
| 1388 expandAllButtonElement.expandAllButton = true; | |
| 1389 expandAllButtonElement.button = button; | |
| 1390 return expandAllButtonElement; | |
| 1391 | |
| 1392 /** | |
| 1393 * @this {WebInspector.ElementsTreeOutline} | |
| 1394 * @param {!Event} event | |
| 1395 */ | |
| 1396 function handleLoadAllChildren(event) | |
| 1397 { | |
| 1398 var visibleChildCount = this._visibleChildren(treeElement.node()).le
ngth; | |
| 1399 this.setExpandedChildrenLimit(treeElement, Math.max(visibleChildCoun
t, treeElement.expandedChildrenLimit() + WebInspector.ElementsTreeElement.Initia
lChildrenLimit)); | |
| 1400 event.consume(); | |
| 1401 } | |
| 1402 }, | |
| 1403 | |
| 1404 /** | |
| 1405 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1406 * @param {number} expandedChildrenLimit | |
| 1407 */ | |
| 1408 setExpandedChildrenLimit: function(treeElement, expandedChildrenLimit) | |
| 1409 { | |
| 1410 if (treeElement.expandedChildrenLimit() === expandedChildrenLimit) | |
| 1411 return; | |
| 1412 | |
| 1413 treeElement.setExpandedChildrenLimit(expandedChildrenLimit); | |
| 1414 if (treeElement.treeOutline && !this._treeElementsBeingUpdated.has(treeE
lement)) | |
| 1415 this._updateModifiedParentNode(treeElement.node()); | |
| 1416 }, | |
| 1417 | |
| 1418 /** | |
| 1419 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1420 */ | |
| 1421 _updateChildren: function(treeElement) | |
| 1422 { | |
| 1423 if (!treeElement.isExpandable()) { | |
| 1424 var selectedTreeElement = treeElement.treeOutline.selectedTreeElemen
t; | |
| 1425 if (selectedTreeElement && selectedTreeElement.hasAncestor(treeEleme
nt)) | |
| 1426 treeElement.select(true); | |
| 1427 treeElement.removeChildren(); | |
| 1428 return; | |
| 1429 } | |
| 1430 | |
| 1431 console.assert(!treeElement.isClosingTag()); | |
| 1432 | |
| 1433 treeElement.node().getChildNodes(childNodesLoaded.bind(this)); | |
| 1434 | |
| 1435 /** | |
| 1436 * @param {?Array.<!WebInspector.DOMNode>} children | |
| 1437 * @this {WebInspector.ElementsTreeOutline} | |
| 1438 */ | |
| 1439 function childNodesLoaded(children) | |
| 1440 { | |
| 1441 // FIXME: sort this out, it should not happen. | |
| 1442 if (!children) | |
| 1443 return; | |
| 1444 this._innerUpdateChildren(treeElement); | |
| 1445 } | |
| 1446 }, | |
| 1447 | |
| 1448 /** | |
| 1449 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1450 * @param {!WebInspector.DOMNode} child | |
| 1451 * @param {number} index | |
| 1452 * @param {boolean=} closingTag | |
| 1453 * @return {!WebInspector.ElementsTreeElement} | |
| 1454 */ | |
| 1455 insertChildElement: function(treeElement, child, index, closingTag) | |
| 1456 { | |
| 1457 var newElement = this._createElementTreeElement(child, closingTag); | |
| 1458 treeElement.insertChild(newElement, index); | |
| 1459 return newElement; | |
| 1460 }, | |
| 1461 | |
| 1462 /** | |
| 1463 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1464 * @param {!WebInspector.ElementsTreeElement} child | |
| 1465 * @param {number} targetIndex | |
| 1466 */ | |
| 1467 _moveChild: function(treeElement, child, targetIndex) | |
| 1468 { | |
| 1469 if (treeElement.indexOfChild(child) === targetIndex) | |
| 1470 return; | |
| 1471 var wasSelected = child.selected; | |
| 1472 if (child.parent) | |
| 1473 child.parent.removeChild(child); | |
| 1474 treeElement.insertChild(child, targetIndex); | |
| 1475 if (wasSelected) | |
| 1476 child.select(); | |
| 1477 }, | |
| 1478 | |
| 1479 /** | |
| 1480 * @param {!WebInspector.ElementsTreeElement} treeElement | |
| 1481 */ | |
| 1482 _innerUpdateChildren: function(treeElement) | |
| 1483 { | |
| 1484 if (this._treeElementsBeingUpdated.has(treeElement)) | |
| 1485 return; | |
| 1486 | |
| 1487 this._treeElementsBeingUpdated.add(treeElement); | |
| 1488 | |
| 1489 var node = treeElement.node(); | |
| 1490 var visibleChildren = this._visibleChildren(node); | |
| 1491 var visibleChildrenSet = new Set(visibleChildren); | |
| 1492 | |
| 1493 // Remove any tree elements that no longer have this node as their paren
t and save | |
| 1494 // all existing elements that could be reused. This also removes closing
tag element. | |
| 1495 var existingTreeElements = new Map(); | |
| 1496 for (var i = treeElement.childCount() - 1; i >= 0; --i) { | |
| 1497 var existingTreeElement = treeElement.childAt(i); | |
| 1498 if (!(existingTreeElement instanceof WebInspector.ElementsTreeElemen
t)) { | |
| 1499 // Remove expand all button and shadow host toolbar. | |
| 1500 treeElement.removeChildAtIndex(i); | |
| 1501 continue; | |
| 1502 } | |
| 1503 var elementsTreeElement = /** @type {!WebInspector.ElementsTreeEleme
nt} */ (existingTreeElement); | |
| 1504 var existingNode = elementsTreeElement.node(); | |
| 1505 | |
| 1506 if (visibleChildrenSet.has(existingNode)) { | |
| 1507 existingTreeElements.set(existingNode, existingTreeElement); | |
| 1508 continue; | |
| 1509 } | |
| 1510 | |
| 1511 treeElement.removeChildAtIndex(i); | |
| 1512 } | |
| 1513 | |
| 1514 for (var i = 0; i < visibleChildren.length && i < treeElement.expandedCh
ildrenLimit(); ++i) { | |
| 1515 var child = visibleChildren[i]; | |
| 1516 var existingTreeElement = existingTreeElements.get(child) || this.fi
ndTreeElement(child); | |
| 1517 if (existingTreeElement && existingTreeElement !== treeElement) { | |
| 1518 // If an existing element was found, just move it. | |
| 1519 this._moveChild(treeElement, existingTreeElement, i); | |
| 1520 } else { | |
| 1521 // No existing element found, insert a new element. | |
| 1522 var newElement = this.insertChildElement(treeElement, child, i); | |
| 1523 if (this._updateRecordForHighlight(node) && treeElement.expanded
) | |
| 1524 WebInspector.ElementsTreeElement.animateOnDOMUpdate(newEleme
nt); | |
| 1525 // If a node was inserted in the middle of existing list dynamic
ally we might need to increase the limit. | |
| 1526 if (treeElement.childCount() > treeElement.expandedChildrenLimit
()) | |
| 1527 this.setExpandedChildrenLimit(treeElement, treeElement.expan
dedChildrenLimit() + 1); | |
| 1528 } | |
| 1529 } | |
| 1530 | |
| 1531 // Update expand all button. | |
| 1532 var expandedChildCount = treeElement.childCount(); | |
| 1533 if (visibleChildren.length > expandedChildCount) { | |
| 1534 var targetButtonIndex = expandedChildCount; | |
| 1535 if (!treeElement.expandAllButtonElement) | |
| 1536 treeElement.expandAllButtonElement = this._createExpandAllButton
TreeElement(treeElement); | |
| 1537 treeElement.insertChild(treeElement.expandAllButtonElement, targetBu
ttonIndex); | |
| 1538 treeElement.expandAllButtonElement.button.textContent = WebInspector
.UIString("Show All Nodes (%d More)", visibleChildren.length - expandedChildCoun
t); | |
| 1539 } else if (treeElement.expandAllButtonElement) { | |
| 1540 delete treeElement.expandAllButtonElement; | |
| 1541 } | |
| 1542 | |
| 1543 // Insert shortcuts to distrubuted children. | |
| 1544 if (node.isInsertionPoint()) { | |
| 1545 for (var distributedNode of node.distributedNodes()) | |
| 1546 treeElement.appendChild(new WebInspector.ElementsTreeOutline.Sho
rtcutTreeElement(distributedNode)); | |
| 1547 } | |
| 1548 | |
| 1549 // Insert close tag. | |
| 1550 if (node.nodeType() === Node.ELEMENT_NODE && treeElement.isExpandable()) | |
| 1551 this.insertChildElement(treeElement, node, treeElement.childCount(),
true); | |
| 1552 | |
| 1553 this._treeElementsBeingUpdated.delete(treeElement); | |
| 1554 }, | |
| 1555 | |
| 1556 /** | |
| 1557 * @param {!WebInspector.Event} event | |
| 1558 */ | |
| 1559 _markersChanged: function(event) | |
| 1560 { | |
| 1561 var node = /** @type {!WebInspector.DOMNode} */ (event.data); | |
| 1562 var treeElement = node[this._treeElementSymbol]; | |
| 1563 if (treeElement) | |
| 1564 treeElement.updateDecorations(); | |
| 1565 }, | |
| 1566 | |
| 1567 __proto__: TreeOutline.prototype | |
| 1568 }; | 1707 }; |
| 1569 | |
| 1570 /** | |
| 1571 * @constructor | |
| 1572 */ | |
| 1573 WebInspector.ElementsTreeOutline.UpdateRecord = function() | |
| 1574 { | |
| 1575 }; | |
| 1576 | |
| 1577 WebInspector.ElementsTreeOutline.UpdateRecord.prototype = { | |
| 1578 /** | |
| 1579 * @param {string} attrName | |
| 1580 */ | |
| 1581 attributeModified: function(attrName) | |
| 1582 { | |
| 1583 if (this._removedAttributes && this._removedAttributes.has(attrName)) | |
| 1584 this._removedAttributes.delete(attrName); | |
| 1585 if (!this._modifiedAttributes) | |
| 1586 this._modifiedAttributes = /** @type {!Set.<string>} */ (new Set()); | |
| 1587 this._modifiedAttributes.add(attrName); | |
| 1588 }, | |
| 1589 | |
| 1590 /** | |
| 1591 * @param {string} attrName | |
| 1592 */ | |
| 1593 attributeRemoved: function(attrName) | |
| 1594 { | |
| 1595 if (this._modifiedAttributes && this._modifiedAttributes.has(attrName)) | |
| 1596 this._modifiedAttributes.delete(attrName); | |
| 1597 if (!this._removedAttributes) | |
| 1598 this._removedAttributes = /** @type {!Set.<string>} */ (new Set()); | |
| 1599 this._removedAttributes.add(attrName); | |
| 1600 }, | |
| 1601 | |
| 1602 /** | |
| 1603 * @param {!WebInspector.DOMNode} node | |
| 1604 */ | |
| 1605 nodeInserted: function(node) | |
| 1606 { | |
| 1607 this._hasChangedChildren = true; | |
| 1608 }, | |
| 1609 | |
| 1610 nodeRemoved: function(node) | |
| 1611 { | |
| 1612 this._hasChangedChildren = true; | |
| 1613 this._hasRemovedChildren = true; | |
| 1614 }, | |
| 1615 | |
| 1616 charDataModified: function() | |
| 1617 { | |
| 1618 this._charDataModified = true; | |
| 1619 }, | |
| 1620 | |
| 1621 childrenModified: function() | |
| 1622 { | |
| 1623 this._hasChangedChildren = true; | |
| 1624 }, | |
| 1625 | |
| 1626 /** | |
| 1627 * @param {string} attributeName | |
| 1628 * @return {boolean} | |
| 1629 */ | |
| 1630 isAttributeModified: function(attributeName) | |
| 1631 { | |
| 1632 return this._modifiedAttributes && this._modifiedAttributes.has(attribut
eName); | |
| 1633 }, | |
| 1634 | |
| 1635 /** | |
| 1636 * @return {boolean} | |
| 1637 */ | |
| 1638 hasRemovedAttributes: function() | |
| 1639 { | |
| 1640 return !!this._removedAttributes && !!this._removedAttributes.size; | |
| 1641 }, | |
| 1642 | |
| 1643 /** | |
| 1644 * @return {boolean} | |
| 1645 */ | |
| 1646 isCharDataModified: function() | |
| 1647 { | |
| 1648 return !!this._charDataModified; | |
| 1649 }, | |
| 1650 | |
| 1651 /** | |
| 1652 * @return {boolean} | |
| 1653 */ | |
| 1654 hasChangedChildren: function() | |
| 1655 { | |
| 1656 return !!this._hasChangedChildren; | |
| 1657 }, | |
| 1658 | |
| 1659 /** | |
| 1660 * @return {boolean} | |
| 1661 */ | |
| 1662 hasRemovedChildren: function() | |
| 1663 { | |
| 1664 return !!this._hasRemovedChildren; | |
| 1665 } | |
| 1666 }; | |
| 1667 | |
| 1668 /** | |
| 1669 * @constructor | |
| 1670 * @implements {WebInspector.Renderer} | |
| 1671 */ | |
| 1672 WebInspector.ElementsTreeOutline.Renderer = function() | |
| 1673 { | |
| 1674 }; | |
| 1675 | |
| 1676 WebInspector.ElementsTreeOutline.Renderer.prototype = { | |
| 1677 /** | |
| 1678 * @override | |
| 1679 * @param {!Object} object | |
| 1680 * @return {!Promise.<!Element>} | |
| 1681 */ | |
| 1682 render: function(object) | |
| 1683 { | |
| 1684 return new Promise(renderPromise); | |
| 1685 | |
| 1686 /** | |
| 1687 * @param {function(!Element)} resolve | |
| 1688 * @param {function(!Error)} reject | |
| 1689 */ | |
| 1690 function renderPromise(resolve, reject) | |
| 1691 { | |
| 1692 if (object instanceof WebInspector.DOMNode) { | |
| 1693 onNodeResolved(/** @type {!WebInspector.DOMNode} */ (object)); | |
| 1694 } else if (object instanceof WebInspector.DeferredDOMNode) { | |
| 1695 (/** @type {!WebInspector.DeferredDOMNode} */ (object)).resolve(
onNodeResolved); | |
| 1696 } else if (object instanceof WebInspector.RemoteObject) { | |
| 1697 var domModel = WebInspector.DOMModel.fromTarget((/** @type {!Web
Inspector.RemoteObject} */ (object)).target()); | |
| 1698 if (domModel) | |
| 1699 domModel.pushObjectAsNodeToFrontend(object, onNodeResolved); | |
| 1700 else | |
| 1701 reject(new Error("No dom model for given JS object target fo
und.")); | |
| 1702 } else { | |
| 1703 reject(new Error("Can't reveal not a node.")); | |
| 1704 } | |
| 1705 | |
| 1706 /** | |
| 1707 * @param {?WebInspector.DOMNode} node | |
| 1708 */ | |
| 1709 function onNodeResolved(node) | |
| 1710 { | |
| 1711 if (!node) { | |
| 1712 reject(new Error("Could not resolve node.")); | |
| 1713 return; | |
| 1714 } | |
| 1715 var treeOutline = new WebInspector.ElementsTreeOutline(node.domM
odel(), false, false); | |
| 1716 treeOutline.rootDOMNode = node; | |
| 1717 if (!treeOutline.firstChild().isExpandable()) | |
| 1718 treeOutline._element.classList.add("single-node"); | |
| 1719 treeOutline.setVisible(true); | |
| 1720 treeOutline.element.treeElementForTest = treeOutline.firstChild(
); | |
| 1721 resolve(treeOutline.element); | |
| 1722 } | |
| 1723 } | |
| 1724 } | |
| 1725 }; | |
| 1726 | |
| 1727 /** | |
| 1728 * @constructor | |
| 1729 * @extends {TreeElement} | |
| 1730 * @param {!WebInspector.DOMNodeShortcut} nodeShortcut | |
| 1731 */ | |
| 1732 WebInspector.ElementsTreeOutline.ShortcutTreeElement = function(nodeShortcut) | |
| 1733 { | |
| 1734 TreeElement.call(this, ""); | |
| 1735 this.listItemElement.createChild("div", "selection fill"); | |
| 1736 var title = this.listItemElement.createChild("span", "elements-tree-shortcut
-title"); | |
| 1737 var text = nodeShortcut.nodeName.toLowerCase(); | |
| 1738 if (nodeShortcut.nodeType === Node.ELEMENT_NODE) | |
| 1739 text = "<" + text + ">"; | |
| 1740 title.textContent = "\u21AA " + text; | |
| 1741 | |
| 1742 var link = WebInspector.DOMPresentationUtils.linkifyDeferredNodeReference(no
deShortcut.deferredNode); | |
| 1743 this.listItemElement.createTextChild(" "); | |
| 1744 link.classList.add("elements-tree-shortcut-link"); | |
| 1745 link.textContent = WebInspector.UIString("reveal"); | |
| 1746 this.listItemElement.appendChild(link); | |
| 1747 this._nodeShortcut = nodeShortcut; | |
| 1748 }; | |
| 1749 | |
| 1750 WebInspector.ElementsTreeOutline.ShortcutTreeElement.prototype = { | |
| 1751 /** | |
| 1752 * @return {boolean} | |
| 1753 */ | |
| 1754 get hovered() | |
| 1755 { | |
| 1756 return this._hovered; | |
| 1757 }, | |
| 1758 | |
| 1759 set hovered(x) | |
| 1760 { | |
| 1761 if (this._hovered === x) | |
| 1762 return; | |
| 1763 this._hovered = x; | |
| 1764 this.listItemElement.classList.toggle("hovered", x); | |
| 1765 }, | |
| 1766 | |
| 1767 /** | |
| 1768 * @return {number} | |
| 1769 */ | |
| 1770 backendNodeId: function() | |
| 1771 { | |
| 1772 return this._nodeShortcut.deferredNode.backendNodeId(); | |
| 1773 }, | |
| 1774 | |
| 1775 /** | |
| 1776 * @override | |
| 1777 * @param {boolean=} selectedByUser | |
| 1778 * @return {boolean} | |
| 1779 */ | |
| 1780 onselect: function(selectedByUser) | |
| 1781 { | |
| 1782 if (!selectedByUser) | |
| 1783 return true; | |
| 1784 this._nodeShortcut.deferredNode.highlight(); | |
| 1785 this._nodeShortcut.deferredNode.resolve(resolved.bind(this)); | |
| 1786 /** | |
| 1787 * @param {?WebInspector.DOMNode} node | |
| 1788 * @this {WebInspector.ElementsTreeOutline.ShortcutTreeElement} | |
| 1789 */ | |
| 1790 function resolved(node) | |
| 1791 { | |
| 1792 if (node) { | |
| 1793 this.treeOutline._selectedDOMNode = node; | |
| 1794 this.treeOutline._selectedNodeChanged(); | |
| 1795 } | |
| 1796 } | |
| 1797 return true; | |
| 1798 }, | |
| 1799 | |
| 1800 __proto__: TreeElement.prototype | |
| 1801 }; | |
| OLD | NEW |