blob: 444988051292b4716ce856c143f4b800d679dc5b [file] [log] [blame]
/*
* Copyright (C) 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.NetworkGridContentView = class NetworkGridContentView extends WebInspector.ContentView
{
constructor(representedObject, extraArguments)
{
console.assert(extraArguments);
console.assert(extraArguments.networkSidebarPanel instanceof WebInspector.NetworkSidebarPanel);
super(representedObject);
WebInspector.Frame.addEventListener(WebInspector.Frame.Event.MainResourceDidChange, this._mainResourceDidChange, this);
this._networkSidebarPanel = extraArguments.networkSidebarPanel;
this._contentTreeOutline = this._networkSidebarPanel.contentTreeOutline;
this._contentTreeOutline.addEventListener(WebInspector.TreeOutline.Event.SelectionDidChange, this._treeSelectionDidChange, this);
let columns = {domain: {}, type: {}, method: {}, scheme: {}, statusCode: {}, cached: {}, protocol: {}, priority: {}, remoteAddress: {}, connectionIdentifier: {}, size: {}, transferSize: {}, requestSent: {}, latency: {}, duration: {}, graph: {}};
columns.domain.title = columns.domain.tooltip = WebInspector.UIString("Domain");
columns.domain.width = "10%";
columns.type.title = columns.type.tooltip = WebInspector.UIString("Type");
columns.type.width = "6%";
columns.method.title = columns.method.tooltip = WebInspector.UIString("Method");
columns.method.width = "5%";
columns.scheme.title = columns.scheme.tooltip = WebInspector.UIString("Scheme");
columns.scheme.width = "5%";
columns.statusCode.title = columns.statusCode.tooltip = WebInspector.UIString("Status");
columns.statusCode.width = "5%";
columns.cached.title = columns.cached.tooltip = WebInspector.UIString("Cached");
columns.cached.width = "8%";
columns.protocol.title = columns.protocol.tooltip = WebInspector.UIString("Protocol");
columns.protocol.width = "5%";
columns.protocol.hidden = true;
columns.priority.title = columns.priority.tooltip = WebInspector.UIString("Priority");
columns.priority.width = "5%";
columns.priority.hidden = true;
columns.remoteAddress.title = columns.remoteAddress.tooltip = WebInspector.UIString("IP Address");
columns.remoteAddress.width = "8%";
columns.remoteAddress.hidden = true;
columns.connectionIdentifier.title = columns.connectionIdentifier.tooltip = WebInspector.UIString("Connection ID");
columns.connectionIdentifier.width = "5%";
columns.connectionIdentifier.hidden = true;
columns.connectionIdentifier.aligned = "right";
columns.size.title = columns.size.tooltip = WebInspector.UIString("Size");
columns.size.width = "6%";
columns.size.aligned = "right";
columns.transferSize.title = columns.transferSize.tooltip = WebInspector.UIString("Transferred");
columns.transferSize.width = "8%";
columns.transferSize.aligned = "right";
columns.requestSent.title = columns.requestSent.tooltip = WebInspector.UIString("Start Time");
columns.requestSent.width = "9%";
columns.requestSent.aligned = "right";
columns.latency.title = columns.latency.tooltip = WebInspector.UIString("Latency");
columns.latency.width = "9%";
columns.latency.aligned = "right";
columns.duration.title = columns.duration.tooltip = WebInspector.UIString("Duration");
columns.duration.width = "9%";
columns.duration.aligned = "right";
for (let column in columns)
columns[column].sortable = true;
this._timelineRuler = new WebInspector.TimelineRuler;
this._timelineRuler.allowsClippedLabels = true;
columns.graph.title = WebInspector.UIString("Timeline");
columns.graph.width = "20%";
columns.graph.headerView = this._timelineRuler;
columns.graph.sortable = false;
// COMPATIBILITY(iOS 10.3): Network load metrics were not previously available.
if (!NetworkAgent.hasEventParameter("loadingFinished", "metrics")) {
delete columns.protocol;
delete columns.priority;
delete columns.remoteAddress;
delete columns.connectionIdentifier;
}
this._dataGrid = new WebInspector.TimelineDataGrid(columns, this._contentTreeOutline);
this._dataGrid.addEventListener(WebInspector.DataGrid.Event.SelectedNodeChanged, this._dataGridNodeSelected, this);
this._dataGrid.sortDelegate = this;
this._dataGrid.sortColumnIdentifier = "requestSent";
this._dataGrid.sortOrder = WebInspector.DataGrid.SortOrder.Ascending;
this._dataGrid.createSettings("network-grid-content-view");
this.element.classList.add("network-grid");
this.addSubview(this._dataGrid);
let networkTimeline = WebInspector.timelineManager.persistentNetworkTimeline;
networkTimeline.addEventListener(WebInspector.Timeline.Event.RecordAdded, this._networkTimelineRecordAdded, this);
networkTimeline.addEventListener(WebInspector.Timeline.Event.Reset, this._networkTimelineReset, this);
// COMPATIBILITY (iOS 10.3): Network.setDisableResourceCaching did not exist.
if (window.NetworkAgent && NetworkAgent.setResourceCachingDisabled) {
let toolTipForDisableResourceCache = WebInspector.UIString("Ignore the resource cache when loading resources");
let activatedToolTipForDisableResourceCache = WebInspector.UIString("Use the resource cache when loading resources");
this._disableResourceCacheNavigationItem = new WebInspector.ActivateButtonNavigationItem("disable-resource-cache", toolTipForDisableResourceCache, activatedToolTipForDisableResourceCache, "Images/IgnoreCaches.svg", 16, 16);
this._disableResourceCacheNavigationItem.activated = WebInspector.resourceCachingDisabledSetting.value;
this._disableResourceCacheNavigationItem.addEventListener(WebInspector.ButtonNavigationItem.Event.Clicked, this._toggleDisableResourceCache, this);
WebInspector.resourceCachingDisabledSetting.addEventListener(WebInspector.Setting.Event.Changed, this._resourceCachingDisabledSettingChanged, this);
}
this._clearNetworkItemsNavigationItem = new WebInspector.ButtonNavigationItem("clear-network-items", WebInspector.UIString("Clear Network Items (%s)").format(WebInspector.clearKeyboardShortcut.displayName), "Images/NavigationItemClear.svg", 16, 16);
this._clearNetworkItemsNavigationItem.addEventListener(WebInspector.ButtonNavigationItem.Event.Clicked, () => this.reset());
this._pendingRecords = [];
this._loadingResourceCount = 0;
this._lastRecordEndTime = NaN;
this._lastUpdateTimestamp = NaN;
this._startTime = NaN;
this._endTime = NaN;
this._scheduledCurrentTimeUpdateIdentifier = undefined;
}
// Public
get secondsPerPixel() { return this._timelineRuler.secondsPerPixel; }
get startTime() { return this._startTime; }
get currentTime() { return this.endTime || this.startTime; }
get endTime() { return this._endTime; }
get zeroTime() { return this.startTime; }
get selectionPathComponents()
{
if (!this._contentTreeOutline.selectedTreeElement || this._contentTreeOutline.selectedTreeElement.hidden)
return null;
var pathComponent = new WebInspector.GeneralTreeElementPathComponent(this._contentTreeOutline.selectedTreeElement);
pathComponent.addEventListener(WebInspector.HierarchicalPathComponent.Event.SiblingWasSelected, this._treeElementPathComponentSelected, this);
return [pathComponent];
}
get navigationItems()
{
let items = [];
if (this._disableResourceCacheNavigationItem)
items.push(this._disableResourceCacheNavigationItem);
items.push(this._clearNetworkItemsNavigationItem);
return items;
}
shown()
{
super.shown();
this._dataGrid.shown();
this._dataGrid.updateLayout(WebInspector.View.LayoutReason.Resize);
if (this._loadingResourceCount && !this._scheduledCurrentTimeUpdateIdentifier)
this._startUpdatingCurrentTime();
}
hidden()
{
this._dataGrid.hidden();
if (this._scheduledCurrentTimeUpdateIdentifier)
this._stopUpdatingCurrentTime();
super.hidden();
}
closed()
{
super.closed();
this._dataGrid.closed();
}
reset()
{
this._contentTreeOutline.removeChildren();
this._dataGrid.reset();
if (this._scheduledCurrentTimeUpdateIdentifier)
this._stopUpdatingCurrentTime();
this._pendingRecords = [];
this._loadingResourceCount = 0;
this._lastRecordEndTime = NaN;
this._lastUpdateTimestamp = NaN;
this._startTime = NaN;
this._endTime = NaN;
this._timelineRuler.startTime = 0;
this._timelineRuler.endTime = 0;
}
// Protected
layout()
{
if (isNaN(this.startTime) || isNaN(this.endTime))
return;
let oldZeroTime = this._timelineRuler.zeroTime;
let oldStartTime = this._timelineRuler.startTime;
let oldEndTime = this._timelineRuler.endTime;
this._timelineRuler.zeroTime = this.zeroTime;
this._timelineRuler.startTime = this.startTime;
if (this.startTime >= this.endTime)
return;
if (!this._scheduledCurrentTimeUpdateIdentifier) {
this._timelineRuler.endTime = this.endTime;
this._endTime = this._lastRecordEndTime + WebInspector.TimelineRecordBar.MinimumWidthPixels * this.secondsPerPixel;
}
this._timelineRuler.endTime = this.endTime;
// We only need to refresh the graphs when the any of the times change.
if (this.zeroTime !== oldZeroTime || this.startTime !== oldStartTime || this.endTime !== oldEndTime) {
for (let dataGridNode of this._dataGrid.children)
dataGridNode.refreshGraph();
}
this._processPendingRecords();
}
handleClearShortcut(event)
{
this.reset();
}
// TimelineDataGrid sort delegate
dataGridSortComparator(sortColumnIdentifier, sortDirection, node1, node2)
{
if (sortColumnIdentifier === "priority")
return WebInspector.Resource.comparePriority(node1.data.priority, node2.data.priority) * sortDirection;
return null;
}
// Private
_resourceCachingDisabledSettingChanged()
{
this._disableResourceCacheNavigationItem.activated = WebInspector.resourceCachingDisabledSetting.value;
}
_toggleDisableResourceCache()
{
WebInspector.resourceCachingDisabledSetting.value = !WebInspector.resourceCachingDisabledSetting.value;
}
_processPendingRecords()
{
if (!this._pendingRecords.length)
return;
for (var resourceTimelineRecord of this._pendingRecords) {
// Skip the record if it already exists in the tree.
var treeElement = this._contentTreeOutline.findTreeElement(resourceTimelineRecord.resource);
if (treeElement)
continue;
treeElement = new WebInspector.ResourceTreeElement(resourceTimelineRecord.resource);
const includesGraph = false;
const shouldShowPopover = true;
let dataGridNode = new WebInspector.ResourceTimelineDataGridNode(resourceTimelineRecord, includesGraph, this, shouldShowPopover);
this._dataGrid.addRowInSortOrder(treeElement, dataGridNode);
}
this._pendingRecords = [];
}
_mainResourceDidChange(event)
{
let frame = event.target;
if (!frame.isMainFrame() || WebInspector.settings.clearNetworkOnNavigate.value)
return;
for (let dataGridNode of this._dataGrid.children)
dataGridNode.element.classList.add("preserved");
}
_networkTimelineReset(event)
{
this.reset();
}
_networkTimelineRecordAdded(event)
{
let resourceTimelineRecord = event.data.record;
console.assert(resourceTimelineRecord instanceof WebInspector.ResourceTimelineRecord);
let update = (event) => {
if (event.target[WebInspector.NetworkGridContentView.ResourceDidFinishOrFail])
return;
event.target.removeEventListener(null, null, this);
event.target[WebInspector.NetworkGridContentView.ResourceDidFinishOrFail] = true;
this._loadingResourceCount--;
if (this._loadingResourceCount)
return;
this._lastRecordEndTime = resourceTimelineRecord.endTime;
this._endTime = Math.max(this._lastRecordEndTime, this._endTime);
if (this._scheduledCurrentTimeUpdateIdentifier)
this.debounce(150)._stopUpdatingCurrentTime();
};
this._pendingRecords.push(resourceTimelineRecord);
this.needsLayout();
let resource = resourceTimelineRecord.resource;
if (resource.finished || resource.failed || resource.canceled)
return;
resource[WebInspector.NetworkGridContentView.ResourceDidFinishOrFail] = false;
resource.addEventListener(WebInspector.Resource.Event.LoadingDidFinish, update, this);
resource.addEventListener(WebInspector.Resource.Event.LoadingDidFail, update, this);
this._loadingResourceCount++;
if (this._scheduledCurrentTimeUpdateIdentifier)
return;
if (isNaN(this._startTime))
this._startTime = this._endTime = resourceTimelineRecord.startTime;
// FIXME: <https://webkit.org/b/153634> Web Inspector: some background tabs think they are the foreground tab and do unnecessary work
if (!(WebInspector.tabBrowser.selectedTabContentView instanceof WebInspector.NetworkTabContentView))
return;
this._startUpdatingCurrentTime();
}
_treeElementPathComponentSelected(event)
{
var dataGridNode = this._dataGrid.dataGridNodeForTreeElement(event.data.pathComponent.generalTreeElement);
if (!dataGridNode)
return;
dataGridNode.revealAndSelect();
}
_treeSelectionDidChange(event)
{
this.dispatchEventToListeners(WebInspector.ContentView.Event.SelectionPathComponentsDidChange);
if (!this._networkSidebarPanel.canShowDifferentContentView())
return;
let treeElement = event.data.selectedElement;
if (treeElement instanceof WebInspector.ResourceTreeElement) {
WebInspector.showRepresentedObject(treeElement.representedObject);
return;
}
console.error("Unknown tree element", treeElement);
}
_dataGridNodeSelected(event)
{
this.dispatchEventToListeners(WebInspector.ContentView.Event.SelectionPathComponentsDidChange);
}
_update(timestamp)
{
console.assert(this._scheduledCurrentTimeUpdateIdentifier);
if (!isNaN(this._lastUpdateTimestamp)) {
let timespanSinceLastUpdate = (timestamp - this._lastUpdateTimestamp) / 1000 || 0;
this._endTime += timespanSinceLastUpdate;
this.updateLayout();
}
this._lastUpdateTimestamp = timestamp;
this._scheduledCurrentTimeUpdateIdentifier = requestAnimationFrame(this._updateCallback);
}
_startUpdatingCurrentTime()
{
console.assert(!this._scheduledCurrentTimeUpdateIdentifier);
if (this._scheduledCurrentTimeUpdateIdentifier)
return;
// Don't update the current time if the Inspector is not visible, as the requestAnimationFrames won't work.
if (!WebInspector.visible)
return;
if (!this._updateCallback)
this._updateCallback = this._update.bind(this);
this._scheduledCurrentTimeUpdateIdentifier = requestAnimationFrame(this._updateCallback);
}
_stopUpdatingCurrentTime()
{
console.assert(this._scheduledCurrentTimeUpdateIdentifier);
if (!this._scheduledCurrentTimeUpdateIdentifier)
return;
this._stopUpdatingCurrentTime.cancelDebounce();
cancelAnimationFrame(this._scheduledCurrentTimeUpdateIdentifier);
this._scheduledCurrentTimeUpdateIdentifier = undefined;
this.needsLayout();
}
};
WebInspector.NetworkGridContentView.ResourceDidFinishOrFail = Symbol("ResourceDidFinishOrFail");