blob: bde64b7273ac7555bd7fbbdc56c3cb71cb3b5ca5 [file] [log] [blame]
/*
* Copyright (C) 2014, 2015 Apple 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. 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 INC. 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.LayoutTimelineView = class LayoutTimelineView extends WebInspector.TimelineView
{
constructor(timeline, extraArguments)
{
super(timeline, extraArguments);
console.assert(timeline.type === WebInspector.TimelineRecord.Type.Layout, timeline);
this.navigationSidebarTreeOutline.element.classList.add("layout");
var columns = {eventType: {}, location: {}, width: {}, height: {}, startTime: {}, totalTime: {}};
columns.eventType.title = WebInspector.UIString("Type");
columns.eventType.width = "15%";
var typeToLabelMap = new Map;
for (var key in WebInspector.LayoutTimelineRecord.EventType) {
var value = WebInspector.LayoutTimelineRecord.EventType[key];
typeToLabelMap.set(value, WebInspector.LayoutTimelineRecord.displayNameForEventType(value));
}
columns.eventType.scopeBar = WebInspector.TimelineDataGrid.createColumnScopeBar("layout", typeToLabelMap);
columns.eventType.hidden = true;
this._scopeBar = columns.eventType.scopeBar;
columns.location.title = WebInspector.UIString("Initiator");
columns.location.width = "25%";
columns.width.title = WebInspector.UIString("Width");
columns.width.width = "8%";
columns.height.title = WebInspector.UIString("Height");
columns.height.width = "8%";
columns.startTime.title = WebInspector.UIString("Start Time");
columns.startTime.width = "8%";
columns.startTime.aligned = "right";
columns.totalTime.title = WebInspector.UIString("Duration");
columns.totalTime.width = "8%";
columns.totalTime.aligned = "right";
for (var column in columns)
columns[column].sortable = true;
this._dataGrid = new WebInspector.LayoutTimelineDataGrid(this.navigationSidebarTreeOutline, columns);
this._dataGrid.addEventListener(WebInspector.TimelineDataGrid.Event.FiltersDidChange, this._dataGridFiltersDidChange, this);
this._dataGrid.addEventListener(WebInspector.DataGrid.Event.SelectedNodeChanged, this._dataGridNodeSelected, this);
this._dataGrid.sortColumnIdentifierSetting = new WebInspector.Setting("layout-timeline-view-sort", "startTime");
this._dataGrid.sortOrderSetting = new WebInspector.Setting("layout-timeline-view-sort-order", WebInspector.DataGrid.SortOrder.Ascending);
this._hoveredTreeElement = null;
this._hoveredDataGridNode = null;
this._showingHighlight = false;
this._showingHighlightForRecord = null;
this._dataGrid.element.addEventListener("mouseover", this._mouseOverDataGrid.bind(this));
this._dataGrid.element.addEventListener("mouseleave", this._mouseLeaveDataGrid.bind(this));
this.navigationSidebarTreeOutline.element.addEventListener("mouseover", this._mouseOverTreeOutline.bind(this));
this.navigationSidebarTreeOutline.element.addEventListener("mouseleave", this._mouseLeaveTreeOutline.bind(this));
this.element.classList.add("layout");
this.addSubview(this._dataGrid);
timeline.addEventListener(WebInspector.Timeline.Event.RecordAdded, this._layoutTimelineRecordAdded, this);
this._pendingRecords = [];
}
// Public
get navigationSidebarTreeOutlineLabel()
{
return WebInspector.UIString("Records");
}
shown()
{
super.shown();
this._updateHighlight();
this._dataGrid.shown();
}
hidden()
{
this._hideHighlightIfNeeded();
this._dataGrid.hidden();
super.hidden();
}
closed()
{
console.assert(this.representedObject instanceof WebInspector.Timeline);
this.representedObject.removeEventListener(null, null, this);
this._dataGrid.closed();
}
filterDidChange()
{
super.filterDidChange();
this._updateHighlight();
}
matchTreeElementAgainstCustomFilters(treeElement)
{
return this._dataGrid.treeElementMatchesActiveScopeFilters(treeElement);
}
reset()
{
super.reset();
this._hideHighlightIfNeeded();
this._dataGrid.reset();
this._pendingRecords = [];
}
// Protected
treeElementPathComponentSelected(event)
{
var dataGridNode = this._dataGrid.dataGridNodeForTreeElement(event.data.pathComponent.generalTreeElement);
if (!dataGridNode)
return;
dataGridNode.revealAndSelect();
}
treeElementDeselected(treeElement)
{
super.treeElementDeselected(treeElement);
this._updateHighlight();
}
treeElementSelected(treeElement, selectedByUser)
{
if (this._dataGrid.shouldIgnoreSelectionEvent())
return;
super.treeElementSelected(treeElement, selectedByUser);
this._updateHighlight();
}
layout()
{
this._processPendingRecords();
}
// Private
_processPendingRecords()
{
if (!this._pendingRecords.length)
return;
for (var layoutTimelineRecord of this._pendingRecords) {
var treeElement = new WebInspector.TimelineRecordTreeElement(layoutTimelineRecord, WebInspector.SourceCodeLocation.NameStyle.Short);
var dataGridNode = new WebInspector.LayoutTimelineDataGridNode(layoutTimelineRecord, this.zeroTime);
this._dataGrid.addRowInSortOrder(treeElement, dataGridNode);
var stack = [{children: layoutTimelineRecord.children, parentTreeElement: treeElement, index: 0}];
while (stack.length) {
var entry = stack.lastValue;
if (entry.index >= entry.children.length) {
stack.pop();
continue;
}
var childRecord = entry.children[entry.index];
console.assert(childRecord.type === WebInspector.TimelineRecord.Type.Layout, childRecord);
var childTreeElement = new WebInspector.TimelineRecordTreeElement(childRecord, WebInspector.SourceCodeLocation.NameStyle.Short);
var layoutDataGridNode = new WebInspector.LayoutTimelineDataGridNode(childRecord, this.zeroTime);
console.assert(entry.parentTreeElement, "entry without parent!");
this._dataGrid.addRowInSortOrder(childTreeElement, layoutDataGridNode, entry.parentTreeElement);
if (childTreeElement && childRecord.children.length)
stack.push({children: childRecord.children, parentTreeElement: childTreeElement, index: 0});
++entry.index;
}
}
this._pendingRecords = [];
}
_layoutTimelineRecordAdded(event)
{
var layoutTimelineRecord = event.data.record;
console.assert(layoutTimelineRecord instanceof WebInspector.LayoutTimelineRecord);
// Only add top-level records, to avoid processing child records multiple times.
if (layoutTimelineRecord.parent instanceof WebInspector.LayoutTimelineRecord)
return;
this._pendingRecords.push(layoutTimelineRecord);
this.needsLayout();
}
_dataGridFiltersDidChange(event)
{
this.timelineSidebarPanel.updateFilter();
}
_dataGridNodeSelected(event)
{
this.dispatchEventToListeners(WebInspector.ContentView.Event.SelectionPathComponentsDidChange);
}
_updateHighlight()
{
var record = this._hoveredOrSelectedRecord();
if (!record) {
this._hideHighlightIfNeeded();
return;
}
this._showHighlightForRecord(record);
}
_showHighlightForRecord(record)
{
if (this._showingHighlightForRecord === record)
return;
this._showingHighlightForRecord = record;
const contentColor = {r: 111, g: 168, b: 220, a: 0.66};
const outlineColor = {r: 255, g: 229, b: 153, a: 0.66};
var quad = record.quad;
if (quad) {
DOMAgent.highlightQuad(quad.toProtocol(), contentColor, outlineColor);
this._showingHighlight = true;
return;
}
// This record doesn't have a highlight, so hide any existing highlight.
if (this._showingHighlight) {
this._showingHighlight = false;
DOMAgent.hideHighlight();
}
}
_hideHighlightIfNeeded()
{
this._showingHighlightForRecord = null;
if (this._showingHighlight) {
this._showingHighlight = false;
DOMAgent.hideHighlight();
}
}
_hoveredOrSelectedRecord()
{
if (this._hoveredDataGridNode)
return this._hoveredDataGridNode.record;
if (this._hoveredTreeElement)
return this._hoveredTreeElement.record;
if (this._dataGrid.selectedNode) {
var treeElement = this._dataGrid.treeElementForDataGridNode(this._dataGrid.selectedNode);
if (treeElement.revealed())
return this._dataGrid.selectedNode.record;
}
return null;
}
_mouseOverDataGrid(event)
{
var hoveredDataGridNode = this._dataGrid.dataGridNodeFromNode(event.target);
if (!hoveredDataGridNode)
return;
this._hoveredDataGridNode = hoveredDataGridNode;
this._updateHighlight();
}
_mouseLeaveDataGrid(event)
{
this._hoveredDataGridNode = null;
this._updateHighlight();
}
_mouseOverTreeOutline(event)
{
var hoveredTreeElement = this.navigationSidebarTreeOutline.treeElementFromNode(event.target);
if (!hoveredTreeElement)
return;
this._hoveredTreeElement = hoveredTreeElement;
this._updateHighlight();
}
_mouseLeaveTreeOutline(event)
{
this._hoveredTreeElement = null;
this._updateHighlight();
}
};