blob: 9b604c2001dd2c85236661513ea87bb165f4f774 [file] [log] [blame]
/*
* Copyright (C) 2007 Apple Inc. All rights reserved.
* Copyright (C) 2014 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 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.
*/
/**
* @unrestricted
*/
export class PropertiesWidget extends UI.ThrottledWidget {
constructor() {
super(true /* isWebComponent */);
this.registerRequiredCSS('elements/propertiesWidget.css');
SDK.targetManager.addModelListener(SDK.DOMModel, SDK.DOMModel.Events.AttrModified, this._onNodeChange, this);
SDK.targetManager.addModelListener(SDK.DOMModel, SDK.DOMModel.Events.AttrRemoved, this._onNodeChange, this);
SDK.targetManager.addModelListener(
SDK.DOMModel, SDK.DOMModel.Events.CharacterDataModified, this._onNodeChange, this);
SDK.targetManager.addModelListener(
SDK.DOMModel, SDK.DOMModel.Events.ChildNodeCountUpdated, this._onNodeChange, this);
UI.context.addFlavorChangeListener(SDK.DOMNode, this._setNode, this);
this._node = UI.context.flavor(SDK.DOMNode);
this._treeOutline = new ObjectUI.ObjectPropertiesSectionsTreeOutline({readOnly: true});
this._treeOutline.setShowSelectionOnKeyboardFocus(/* show */ true, /* preventTabOrder */ false);
this._expandController = new ObjectUI.ObjectPropertiesSectionsTreeExpandController(this._treeOutline);
this.contentElement.appendChild(this._treeOutline.element);
this._treeOutline.addEventListener(UI.TreeOutline.Events.ElementExpanded, () => {
Host.userMetrics.actionTaken(Host.UserMetrics.Action.DOMPropertiesExpanded);
});
this.update();
}
/**
* @param {!Common.Event} event
*/
_setNode(event) {
this._node = /** @type {?SDK.DOMNode} */ (event.data);
this.update();
}
/**
* @override
* @protected
* @return {!Promise<undefined>}
*/
async doUpdate() {
if (this._lastRequestedNode) {
this._lastRequestedNode.domModel().runtimeModel().releaseObjectGroup(_objectGroupName);
delete this._lastRequestedNode;
}
if (!this._node) {
this.contentElement.removeChildren();
return;
}
this._lastRequestedNode = this._node;
const object = await this._node.resolveToObject(_objectGroupName);
if (!object) {
return;
}
const result = await object.callFunction(protoList);
object.release();
if (!result.object || result.wasThrown) {
return;
}
const propertiesResult = await result.object.getOwnProperties(false /* generatePreview */);
result.object.release();
if (!propertiesResult || !propertiesResult.properties) {
return;
}
const properties = propertiesResult.properties;
this._treeOutline.removeChildren();
let selected = false;
// Get array of property user-friendly names.
for (let i = 0; i < properties.length; ++i) {
if (!parseInt(properties[i].name, 10)) {
continue;
}
const property = properties[i].value;
let title = property.description;
title = title.replace(/Prototype$/, '');
const section = this._createSectionTreeElement(property, title);
this._treeOutline.appendChild(section);
if (!selected) {
section.select(/* omitFocus= */ true, /* selectedByUser= */ false);
selected = true;
}
}
/**
* @suppressReceiverCheck
* @this {*}
*/
function protoList() {
let proto = this;
const result = {__proto__: null};
let counter = 1;
while (proto) {
result[counter++] = proto;
proto = proto.__proto__;
}
return result;
}
}
/**
* @param {!SDK.RemoteObject} property
* @param {string} title
* @returns {!ObjectUI.ObjectPropertiesSection.RootElement}
*/
_createSectionTreeElement(property, title) {
const titleElement = createElementWithClass('span', 'tree-element-title');
titleElement.textContent = title;
const section = new ObjectUI.ObjectPropertiesSection.RootElement(property);
section.title = titleElement;
this._expandController.watchSection(title, section);
return section;
}
/**
* @param {!Common.Event} event
*/
_onNodeChange(event) {
if (!this._node) {
return;
}
const data = event.data;
const node = /** @type {!SDK.DOMNode} */ (data instanceof SDK.DOMNode ? data : data.node);
if (this._node !== node) {
return;
}
this.update();
}
}
export const _objectGroupName = 'properties-sidebar-pane';