| Index: chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js
|
| ===================================================================
|
| --- chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js (revision 0)
|
| +++ chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js (revision 0)
|
| @@ -0,0 +1,742 @@
|
| +/*
|
| + * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
|
| + * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
|
| + * Copyright (C) 2009 Joseph Pecoraro
|
| + *
|
| + * Redistribution and use in source and binary forms, with or without
|
| + * modification, are permitted provided that the following conditions
|
| + * are met:
|
| + *
|
| + * 1. Redistributions of source code must retain the above copyright
|
| + * notice, this list of conditions and the following disclaimer.
|
| + * 2. Redistributions in binary form must reproduce the above copyright
|
| + * notice, this list of conditions and the following disclaimer in the
|
| + * documentation and/or other materials provided with the distribution.
|
| + * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
|
| + * its contributors may be used to endorse or promote products derived
|
| + * from this software without specific prior written permission.
|
| + *
|
| + * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
|
| + * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
| + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
| + * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
|
| + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
| + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
| + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
| + * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
| + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
| + * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
| + */
|
| +
|
| +WebInspector.ElementsTreeOutline = function() {
|
| + this.element = document.createElement("ol");
|
| + this.element.addEventListener("mousedown", this._onmousedown.bind(this), false);
|
| + this.element.addEventListener("dblclick", this._ondblclick.bind(this), false);
|
| + this.element.addEventListener("mousemove", this._onmousemove.bind(this), false);
|
| + this.element.addEventListener("mouseout", this._onmouseout.bind(this), false);
|
| +
|
| + TreeOutline.call(this, this.element);
|
| +
|
| + this.includeRootDOMNode = true;
|
| + this.selectEnabled = false;
|
| + this.rootDOMNode = null;
|
| + this.focusedDOMNode = null;
|
| +}
|
| +
|
| +WebInspector.ElementsTreeOutline.prototype = {
|
| + get rootDOMNode()
|
| + {
|
| + return this._rootDOMNode;
|
| + },
|
| +
|
| + set rootDOMNode(x)
|
| + {
|
| + if (this._rootDOMNode === x)
|
| + return;
|
| +
|
| + this._rootDOMNode = x;
|
| +
|
| + this.update();
|
| + },
|
| +
|
| + get focusedDOMNode()
|
| + {
|
| + return this._focusedDOMNode;
|
| + },
|
| +
|
| + set focusedDOMNode(x)
|
| + {
|
| + if (this._focusedDOMNode === x) {
|
| + this.revealAndSelectNode(x);
|
| + return;
|
| + }
|
| +
|
| + this._focusedDOMNode = x;
|
| +
|
| + this.revealAndSelectNode(x);
|
| +
|
| + // The revealAndSelectNode() method might find a different element if there is inlined text,
|
| + // and the select() call would change the focusedDOMNode and reenter this setter. So to
|
| + // avoid calling focusedNodeChanged() twice, first check if _focusedDOMNode is the same
|
| + // node as the one passed in.
|
| + if (this._focusedDOMNode === x) {
|
| + this.focusedNodeChanged();
|
| +
|
| + if (x && !this.suppressSelectHighlight) {
|
| + InspectorController.highlightDOMNode(x.id);
|
| +
|
| + if ("_restorePreviousHighlightNodeTimeout" in this)
|
| + clearTimeout(this._restorePreviousHighlightNodeTimeout);
|
| +
|
| + function restoreHighlightToHoveredNode()
|
| + {
|
| + var hoveredNode = WebInspector.hoveredDOMNode;
|
| + if (hoveredNode)
|
| + InspectorController.highlightDOMNode(hoveredNode.id);
|
| + else
|
| + InspectorController.hideDOMNodeHighlight();
|
| + }
|
| +
|
| + this._restorePreviousHighlightNodeTimeout = setTimeout(restoreHighlightToHoveredNode, 2000);
|
| + }
|
| + }
|
| + },
|
| +
|
| + update: function()
|
| + {
|
| + this.removeChildren();
|
| +
|
| + if (!this.rootDOMNode)
|
| + return;
|
| +
|
| + var treeElement;
|
| + if (this.includeRootDOMNode) {
|
| + treeElement = new WebInspector.ElementsTreeElement(this.rootDOMNode);
|
| + treeElement.selectable = this.selectEnabled;
|
| + this.appendChild(treeElement);
|
| + } else {
|
| + // FIXME: this could use findTreeElement to reuse a tree element if it already exists
|
| + var node = (Preferences.ignoreWhitespace ? firstChildSkippingWhitespace.call(this.rootDOMNode) : this.rootDOMNode.firstChild);
|
| + while (node) {
|
| + treeElement = new WebInspector.ElementsTreeElement(node);
|
| + treeElement.selectable = this.selectEnabled;
|
| + this.appendChild(treeElement);
|
| + node = Preferences.ignoreWhitespace ? nextSiblingSkippingWhitespace.call(node) : node.nextSibling;
|
| + }
|
| + }
|
| +
|
| + this.updateSelection();
|
| + },
|
| +
|
| + updateSelection: function()
|
| + {
|
| + if (!this.selectedTreeElement)
|
| + return;
|
| + var element = this.treeOutline.selectedTreeElement;
|
| + element.updateSelection();
|
| + },
|
| +
|
| + focusedNodeChanged: function(forceUpdate) {},
|
| +
|
| + findTreeElement: function(node)
|
| + {
|
| + var treeElement = TreeOutline.prototype.findTreeElement.call(this, node, isAncestorNode, parentNode);
|
| + if (!treeElement && node.nodeType === Node.TEXT_NODE) {
|
| + // The text node might have been inlined if it was short, so try to find the parent element.
|
| + treeElement = TreeOutline.prototype.findTreeElement.call(this, node.parentNode, isAncestorNode, parentNode);
|
| + }
|
| +
|
| + return treeElement;
|
| + },
|
| +
|
| + revealAndSelectNode: function(node)
|
| + {
|
| + if (!node)
|
| + return;
|
| +
|
| + var treeElement = this.findTreeElement(node);
|
| + if (!treeElement)
|
| + return;
|
| +
|
| + treeElement.reveal();
|
| + treeElement.select();
|
| + },
|
| +
|
| + _treeElementFromEvent: function(event)
|
| + {
|
| + var root = this.element;
|
| +
|
| + // We choose this X coordinate based on the knowledge that our list
|
| + // items extend nearly to the right edge of the outer <ol>.
|
| + var x = root.totalOffsetLeft + root.offsetWidth - 20;
|
| +
|
| + var y = event.pageY;
|
| +
|
| + // Our list items have 1-pixel cracks between them vertically. We avoid
|
| + // the cracks by checking slightly above and slightly below the mouse
|
| + // and seeing if we hit the same element each time.
|
| + var elementUnderMouse = this.treeElementFromPoint(x, y);
|
| + var elementAboveMouse = this.treeElementFromPoint(x, y - 2);
|
| + var element;
|
| + if (elementUnderMouse === elementAboveMouse)
|
| + element = elementUnderMouse;
|
| + else
|
| + element = this.treeElementFromPoint(x, y + 2);
|
| +
|
| + return element;
|
| + },
|
| +
|
| + _ondblclick: function(event)
|
| + {
|
| + var element = this._treeElementFromEvent(event);
|
| +
|
| + if (!element || !element.ondblclick)
|
| + return;
|
| +
|
| + element.ondblclick(element, event);
|
| + },
|
| +
|
| + _onmousedown: function(event)
|
| + {
|
| + var element = this._treeElementFromEvent(event);
|
| +
|
| + if (!element || element.isEventWithinDisclosureTriangle(event))
|
| + return;
|
| +
|
| + element.select();
|
| + },
|
| +
|
| + _onmousemove: function(event)
|
| + {
|
| + if (this._previousHoveredElement) {
|
| + this._previousHoveredElement.hovered = false;
|
| + delete this._previousHoveredElement;
|
| + }
|
| +
|
| + var element = this._treeElementFromEvent(event);
|
| + if (element && !element.elementCloseTag) {
|
| + element.hovered = true;
|
| + this._previousHoveredElement = element;
|
| + }
|
| +
|
| + WebInspector.hoveredDOMNode = (element && !element.elementCloseTag ? element.representedObject : null);
|
| + },
|
| +
|
| + _onmouseout: function(event)
|
| + {
|
| + var nodeUnderMouse = document.elementFromPoint(event.pageX, event.pageY);
|
| + if (nodeUnderMouse && nodeUnderMouse.isDescendant(this.element))
|
| + return;
|
| +
|
| + if (this._previousHoveredElement) {
|
| + this._previousHoveredElement.hovered = false;
|
| + delete this._previousHoveredElement;
|
| + }
|
| +
|
| + WebInspector.hoveredDOMNode = null;
|
| + }
|
| +}
|
| +
|
| +WebInspector.ElementsTreeOutline.prototype.__proto__ = TreeOutline.prototype;
|
| +
|
| +WebInspector.ElementsTreeElement = function(node)
|
| +{
|
| + var hasChildren = Preferences.ignoreWhitespace ? (firstChildSkippingWhitespace.call(node) ? true : false) : node.hasChildNodes();
|
| + var titleInfo = nodeTitleInfo.call(node, hasChildren, WebInspector.linkifyURL);
|
| +
|
| + if (titleInfo.hasChildren)
|
| + this.whitespaceIgnored = Preferences.ignoreWhitespace;
|
| +
|
| + // The title will be updated in onattach.
|
| + TreeElement.call(this, "", node, titleInfo.hasChildren);
|
| +
|
| + if (this.representedObject.nodeType == Node.ELEMENT_NODE)
|
| + this._canAddAttributes = true;
|
| +}
|
| +
|
| +WebInspector.ElementsTreeElement.prototype = {
|
| + get highlighted()
|
| + {
|
| + return this._highlighted;
|
| + },
|
| +
|
| + set highlighted(x)
|
| + {
|
| + if (this._highlighted === x)
|
| + return;
|
| +
|
| + this._highlighted = x;
|
| +
|
| + if (this.listItemElement) {
|
| + if (x)
|
| + this.listItemElement.addStyleClass("highlighted");
|
| + else
|
| + this.listItemElement.removeStyleClass("highlighted");
|
| + }
|
| + },
|
| +
|
| + get hovered()
|
| + {
|
| + return this._hovered;
|
| + },
|
| +
|
| + set hovered(x)
|
| + {
|
| + if (this._hovered === x)
|
| + return;
|
| +
|
| + this._hovered = x;
|
| +
|
| + if (this.listItemElement) {
|
| + if (x) {
|
| + this.updateSelection();
|
| + this.listItemElement.addStyleClass("hovered");
|
| + } else
|
| + this.listItemElement.removeStyleClass("hovered");
|
| + if (this._canAddAttributes)
|
| + this.toggleNewAttributeButton();
|
| + }
|
| + },
|
| +
|
| + toggleNewAttributeButton: function()
|
| + {
|
| + function removeWhenEditing(event)
|
| + {
|
| + if (this._addAttributeElement && this._addAttributeElement.parentNode)
|
| + this._addAttributeElement.parentNode.removeChild(this._addAttributeElement);
|
| + delete this._addAttributeElement;
|
| + }
|
| +
|
| + if (!this._addAttributeElement && this._hovered && !this._editing) {
|
| + var span = document.createElement("span");
|
| + span.className = "add-attribute";
|
| + span.textContent = "\u2026";
|
| + span.addEventListener("dblclick", removeWhenEditing.bind(this), false);
|
| + this._addAttributeElement = span;
|
| +
|
| + var tag = this.listItemElement.getElementsByClassName("webkit-html-tag")[0];
|
| + this._insertInLastAttributePosition(tag, span);
|
| + } else if (!this._hovered && this._addAttributeElement) {
|
| + if (this._addAttributeElement.parentNode)
|
| + this._addAttributeElement.parentNode.removeChild(this._addAttributeElement);
|
| + delete this._addAttributeElement;
|
| + }
|
| + },
|
| +
|
| + updateSelection: function()
|
| + {
|
| + var listItemElement = this.listItemElement;
|
| + if (!listItemElement)
|
| + return;
|
| +
|
| + if (document.body.offsetWidth <= 0) {
|
| + // The stylesheet hasn't loaded yet or the window is closed,
|
| + // so we can't calculate what is need. Return early.
|
| + return;
|
| + }
|
| +
|
| + if (!this.selectionElement) {
|
| + this.selectionElement = document.createElement("div");
|
| + this.selectionElement.className = "selection selected";
|
| + listItemElement.insertBefore(this.selectionElement, listItemElement.firstChild);
|
| + }
|
| +
|
| + this.selectionElement.style.height = listItemElement.offsetHeight + "px";
|
| + },
|
| +
|
| + onattach: function()
|
| + {
|
| + this.listItemElement.addEventListener("mousedown", this.onmousedown.bind(this), false);
|
| +
|
| + if (this._highlighted)
|
| + this.listItemElement.addStyleClass("highlighted");
|
| +
|
| + if (this._hovered) {
|
| + this.updateSelection();
|
| + this.listItemElement.addStyleClass("hovered");
|
| + }
|
| +
|
| + this._updateTitle();
|
| +
|
| + this._preventFollowingLinksOnDoubleClick();
|
| + },
|
| +
|
| + _preventFollowingLinksOnDoubleClick: function()
|
| + {
|
| + var links = this.listItemElement.querySelectorAll("li > .webkit-html-tag > .webkit-html-attribute > .webkit-html-external-link, li > .webkit-html-tag > .webkit-html-attribute > .webkit-html-resource-link");
|
| + if (!links)
|
| + return;
|
| +
|
| + for (var i = 0; i < links.length; ++i)
|
| + links[i].preventFollowOnDoubleClick = true;
|
| + },
|
| +
|
| + onpopulate: function()
|
| + {
|
| + if (this.children.length || this.whitespaceIgnored !== Preferences.ignoreWhitespace)
|
| + return;
|
| +
|
| + this.whitespaceIgnored = Preferences.ignoreWhitespace;
|
| +
|
| + this.updateChildren();
|
| + },
|
| +
|
| + updateChildren: function(fullRefresh)
|
| + {
|
| + WebInspector.domAgent.getChildNodesAsync(this.representedObject, this._updateChildren.bind(this, fullRefresh));
|
| + },
|
| +
|
| + _updateChildren: function(fullRefresh)
|
| + {
|
| + if (fullRefresh) {
|
| + var selectedTreeElement = this.treeOutline.selectedTreeElement;
|
| + if (selectedTreeElement && selectedTreeElement.hasAncestor(this))
|
| + this.select();
|
| + this.removeChildren();
|
| + }
|
| +
|
| + var treeElement = this;
|
| + var treeChildIndex = 0;
|
| +
|
| + function updateChildrenOfNode(node)
|
| + {
|
| + var treeOutline = treeElement.treeOutline;
|
| + var child = (Preferences.ignoreWhitespace ? firstChildSkippingWhitespace.call(node) : node.firstChild);
|
| + while (child) {
|
| + var currentTreeElement = treeElement.children[treeChildIndex];
|
| + if (!currentTreeElement || currentTreeElement.representedObject !== child) {
|
| + // Find any existing element that is later in the children list.
|
| + var existingTreeElement = null;
|
| + for (var i = (treeChildIndex + 1); i < treeElement.children.length; ++i) {
|
| + if (treeElement.children[i].representedObject === child) {
|
| + existingTreeElement = treeElement.children[i];
|
| + break;
|
| + }
|
| + }
|
| +
|
| + if (existingTreeElement && existingTreeElement.parent === treeElement) {
|
| + // If an existing element was found and it has the same parent, just move it.
|
| + var wasSelected = existingTreeElement.selected;
|
| + treeElement.removeChild(existingTreeElement);
|
| + treeElement.insertChild(existingTreeElement, treeChildIndex);
|
| + if (wasSelected)
|
| + existingTreeElement.select();
|
| + } else {
|
| + // No existing element found, insert a new element.
|
| + var newElement = new WebInspector.ElementsTreeElement(child);
|
| + newElement.selectable = treeOutline.selectEnabled;
|
| + treeElement.insertChild(newElement, treeChildIndex);
|
| + }
|
| + }
|
| +
|
| + child = Preferences.ignoreWhitespace ? nextSiblingSkippingWhitespace.call(child) : child.nextSibling;
|
| + ++treeChildIndex;
|
| + }
|
| + }
|
| +
|
| + // Remove any tree elements that no longer have this node (or this node's contentDocument) as their parent.
|
| + for (var i = (this.children.length - 1); i >= 0; --i) {
|
| + if ("elementCloseTag" in this.children[i])
|
| + continue;
|
| +
|
| + var currentChild = this.children[i];
|
| + var currentNode = currentChild.representedObject;
|
| + var currentParentNode = currentNode.parentNode;
|
| +
|
| + if (currentParentNode === this.representedObject)
|
| + continue;
|
| +
|
| + var selectedTreeElement = this.treeOutline.selectedTreeElement;
|
| + if (selectedTreeElement && (selectedTreeElement === currentChild || selectedTreeElement.hasAncestor(currentChild)))
|
| + this.select();
|
| +
|
| + this.removeChildAtIndex(i);
|
| + }
|
| +
|
| + updateChildrenOfNode(this.representedObject);
|
| +
|
| + var lastChild = this.children[this.children.length - 1];
|
| + if (this.representedObject.nodeType == Node.ELEMENT_NODE && (!lastChild || !lastChild.elementCloseTag)) {
|
| + var title = "<span class=\"webkit-html-tag close\"></" + this.representedObject.nodeName.toLowerCase().escapeHTML() + "></span>";
|
| + var item = new TreeElement(title, null, false);
|
| + item.selectable = false;
|
| + item.elementCloseTag = true;
|
| + this.appendChild(item);
|
| + }
|
| + },
|
| +
|
| + onexpand: function()
|
| + {
|
| + this.treeOutline.updateSelection();
|
| + },
|
| +
|
| + oncollapse: function()
|
| + {
|
| + this.treeOutline.updateSelection();
|
| + },
|
| +
|
| + onreveal: function()
|
| + {
|
| + if (this.listItemElement)
|
| + this.listItemElement.scrollIntoViewIfNeeded(false);
|
| + },
|
| +
|
| + onselect: function()
|
| + {
|
| + this.treeOutline.focusedDOMNode = this.representedObject;
|
| + this.updateSelection();
|
| + },
|
| +
|
| + onmousedown: function(event)
|
| + {
|
| + if (this._editing)
|
| + return;
|
| +
|
| + // Prevent selecting the nearest word on double click.
|
| + if (event.detail >= 2)
|
| + event.preventDefault();
|
| + },
|
| +
|
| + ondblclick: function(treeElement, event)
|
| + {
|
| + if (this._editing)
|
| + return;
|
| +
|
| + if (this._startEditing(event, treeElement))
|
| + return;
|
| +
|
| + if (this.treeOutline.panel) {
|
| + this.treeOutline.rootDOMNode = this.representedObject.parentNode;
|
| + this.treeOutline.focusedDOMNode = this.representedObject;
|
| + }
|
| +
|
| + if (this.hasChildren && !this.expanded)
|
| + this.expand();
|
| + },
|
| +
|
| + _insertInLastAttributePosition: function(tag, node)
|
| + {
|
| + if (tag.getElementsByClassName("webkit-html-attribute").length > 0)
|
| + tag.insertBefore(node, tag.lastChild);
|
| + else {
|
| + var nodeName = tag.textContent.match(/^<(.*?)>$/)[1];
|
| + tag.textContent = '';
|
| + tag.appendChild(document.createTextNode('<'+nodeName));
|
| + tag.appendChild(node);
|
| + tag.appendChild(document.createTextNode('>'));
|
| + }
|
| + },
|
| +
|
| + _startEditing: function(event, treeElement)
|
| + {
|
| + if (this.treeOutline.focusedDOMNode != this.representedObject)
|
| + return;
|
| +
|
| + if (this.representedObject.nodeType != Node.ELEMENT_NODE && this.representedObject.nodeType != Node.TEXT_NODE)
|
| + return false;
|
| +
|
| + var textNode = event.target.enclosingNodeOrSelfWithClass("webkit-html-text-node");
|
| + if (textNode)
|
| + return this._startEditingTextNode(textNode);
|
| +
|
| + var attribute = event.target.enclosingNodeOrSelfWithClass("webkit-html-attribute");
|
| + if (attribute)
|
| + return this._startEditingAttribute(attribute, event.target);
|
| +
|
| + var newAttribute = event.target.enclosingNodeOrSelfWithClass("add-attribute");
|
| + if (newAttribute)
|
| + return this._addNewAttribute(treeElement.listItemElement);
|
| +
|
| + return false;
|
| + },
|
| +
|
| + _addNewAttribute: function(listItemElement)
|
| + {
|
| + var attr = document.createElement("span");
|
| + attr.className = "webkit-html-attribute";
|
| + attr.style.marginLeft = "2px"; // overrides the .editing margin rule
|
| + attr.style.marginRight = "2px"; // overrides the .editing margin rule
|
| + var name = document.createElement("span");
|
| + name.className = "webkit-html-attribute-name new-attribute";
|
| + name.textContent = " ";
|
| + var value = document.createElement("span");
|
| + value.className = "webkit-html-attribute-value";
|
| + attr.appendChild(name);
|
| + attr.appendChild(value);
|
| +
|
| + var tag = listItemElement.getElementsByClassName("webkit-html-tag")[0];
|
| + this._insertInLastAttributePosition(tag, attr);
|
| + return this._startEditingAttribute(attr, attr);
|
| + },
|
| +
|
| + _triggerEditAttribute: function(attributeName)
|
| + {
|
| + var attributeElements = this.listItemElement.getElementsByClassName("webkit-html-attribute-name");
|
| + for (var i = 0, len = attributeElements.length; i < len; ++i) {
|
| + if (attributeElements[i].textContent === attributeName) {
|
| + for (var elem = attributeElements[i].nextSibling; elem; elem = elem.nextSibling) {
|
| + if (elem.nodeType !== Node.ELEMENT_NODE)
|
| + continue;
|
| +
|
| + if (elem.hasStyleClass("webkit-html-attribute-value"))
|
| + return this._startEditingAttribute(attributeElements[i].parentNode, elem);
|
| + }
|
| + }
|
| + }
|
| + },
|
| +
|
| + _startEditingAttribute: function(attribute, elementForSelection)
|
| + {
|
| + if (WebInspector.isBeingEdited(attribute))
|
| + return true;
|
| +
|
| + var attributeNameElement = attribute.getElementsByClassName("webkit-html-attribute-name")[0];
|
| + if (!attributeNameElement)
|
| + return false;
|
| +
|
| + var attributeName = attributeNameElement.innerText;
|
| +
|
| + function removeZeroWidthSpaceRecursive(node)
|
| + {
|
| + if (node.nodeType === Node.TEXT_NODE) {
|
| + node.nodeValue = node.nodeValue.replace(/\u200B/g, "");
|
| + return;
|
| + }
|
| +
|
| + if (node.nodeType !== Node.ELEMENT_NODE)
|
| + return;
|
| +
|
| + for (var child = node.firstChild; child; child = child.nextSibling)
|
| + removeZeroWidthSpaceRecursive(child);
|
| + }
|
| +
|
| + // Remove zero-width spaces that were added by nodeTitleInfo.
|
| + removeZeroWidthSpaceRecursive(attribute);
|
| +
|
| + this._editing = true;
|
| +
|
| + WebInspector.startEditing(attribute, this._attributeEditingCommitted.bind(this), this._editingCancelled.bind(this), attributeName);
|
| + window.getSelection().setBaseAndExtent(elementForSelection, 0, elementForSelection, 1);
|
| +
|
| + return true;
|
| + },
|
| +
|
| + _startEditingTextNode: function(textNode)
|
| + {
|
| + if (WebInspector.isBeingEdited(textNode))
|
| + return true;
|
| +
|
| + this._editing = true;
|
| +
|
| + WebInspector.startEditing(textNode, this._textNodeEditingCommitted.bind(this), this._editingCancelled.bind(this));
|
| + window.getSelection().setBaseAndExtent(textNode, 0, textNode, 1);
|
| +
|
| + return true;
|
| + },
|
| +
|
| + _attributeEditingCommitted: function(element, newText, oldText, attributeName, moveDirection)
|
| + {
|
| + delete this._editing;
|
| +
|
| + // Before we do anything, determine where we should move
|
| + // next based on the current element's settings
|
| + var moveToAttribute;
|
| + var newAttribute;
|
| + if (moveDirection) {
|
| + var found = false;
|
| + var attributes = this.representedObject.attributes;
|
| + for (var i = 0, len = attributes.length; i < len; ++i) {
|
| + if (attributes[i].name === attributeName) {
|
| + found = true;
|
| + if (moveDirection === "backward" && i > 0)
|
| + moveToAttribute = attributes[i - 1].name;
|
| + else if (moveDirection === "forward" && i < attributes.length - 1)
|
| + moveToAttribute = attributes[i + 1].name;
|
| + else if (moveDirection === "forward" && i === attributes.length - 1)
|
| + newAttribute = true;
|
| + }
|
| + }
|
| +
|
| + if (!found && moveDirection === "backward")
|
| + moveToAttribute = attributes[attributes.length - 1].name;
|
| + else if (!found && moveDirection === "forward" && !/^\s*$/.test(newText))
|
| + newAttribute = true;
|
| + }
|
| +
|
| + function moveToNextAttributeIfNeeded() {
|
| + if (moveToAttribute)
|
| + this._triggerEditAttribute(moveToAttribute);
|
| + else if (newAttribute)
|
| + this._addNewAttribute(this.listItemElement);
|
| + }
|
| +
|
| + var parseContainerElement = document.createElement("span");
|
| + parseContainerElement.innerHTML = "<span " + newText + "></span>";
|
| + var parseElement = parseContainerElement.firstChild;
|
| +
|
| + if (!parseElement) {
|
| + this._editingCancelled(element, attributeName);
|
| + moveToNextAttributeIfNeeded.call(this);
|
| + return;
|
| + }
|
| +
|
| + if (!parseElement.hasAttributes()) {
|
| + this.representedObject.removeAttribute(attributeName);
|
| + this._updateTitle();
|
| + moveToNextAttributeIfNeeded.call(this);
|
| + return;
|
| + }
|
| +
|
| + var foundOriginalAttribute = false;
|
| + for (var i = 0; i < parseElement.attributes.length; ++i) {
|
| + var attr = parseElement.attributes[i];
|
| + foundOriginalAttribute = foundOriginalAttribute || attr.name === attributeName;
|
| + try {
|
| + this.representedObject.setAttribute(attr.name, attr.value);
|
| + } catch(e) {} // ignore invalid attribute (innerHTML doesn't throw errors, but this can)
|
| + }
|
| +
|
| + if (!foundOriginalAttribute)
|
| + this.representedObject.removeAttribute(attributeName);
|
| +
|
| + this._updateTitle();
|
| +
|
| + this.treeOutline.focusedNodeChanged(true);
|
| +
|
| + moveToNextAttributeIfNeeded.call(this);
|
| + },
|
| +
|
| + _textNodeEditingCommitted: function(element, newText)
|
| + {
|
| + delete this._editing;
|
| +
|
| + var textNode;
|
| + if (this.representedObject.nodeType == Node.ELEMENT_NODE) {
|
| + // We only show text nodes inline in elements if the element only
|
| + // has a single child, and that child is a text node.
|
| + textNode = this.representedObject.firstChild;
|
| + } else if (this.representedObject.nodeType == Node.TEXT_NODE)
|
| + textNode = this.representedObject;
|
| +
|
| + textNode.nodeValue = newText;
|
| + this._updateTitle();
|
| + },
|
| +
|
| + _editingCancelled: function(element, context)
|
| + {
|
| + delete this._editing;
|
| +
|
| + this._updateTitle();
|
| + },
|
| +
|
| + _updateTitle: function()
|
| + {
|
| + var title = nodeTitleInfo.call(this.representedObject, this.hasChildren, WebInspector.linkifyURL).title;
|
| + this.title = "<span class=\"highlight\">" + title + "</span>";
|
| + delete this.selectionElement;
|
| + this.updateSelection();
|
| + this._preventFollowingLinksOnDoubleClick();
|
| + },
|
| +}
|
| +
|
| +WebInspector.ElementsTreeElement.prototype.__proto__ = TreeElement.prototype;
|
|
|