| Index: Source/devtools/front_end/network/NetworkPanel.js
|
| diff --git a/Source/devtools/front_end/network/NetworkPanel.js b/Source/devtools/front_end/network/NetworkPanel.js
|
| index 62f46e8b88cab39be978b49e802e05e93368b848..05786004b57492030ee8c06397b246e0c1033334 100644
|
| --- a/Source/devtools/front_end/network/NetworkPanel.js
|
| +++ b/Source/devtools/front_end/network/NetworkPanel.js
|
| @@ -1,7 +1,7 @@
|
| /*
|
| * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
|
| * Copyright (C) 2008, 2009 Anthony Ricaud <rik@webkit.org>
|
| - * Copyright (C) 2010, 2011, 2012, 2013, 2014 Google Inc. All rights reserved.
|
| + * Copyright (C) 2011 Google Inc. All rights reserved.
|
| *
|
| * Redistribution and use in source and binary forms, with or without
|
| * modification, are permitted provided that the following conditions
|
| @@ -30,6 +30,1875 @@
|
|
|
| /**
|
| * @constructor
|
| + * @implements {WebInspector.Searchable}
|
| + * @implements {WebInspector.TargetManager.Observer}
|
| + * @extends {WebInspector.VBox}
|
| + * @param {!WebInspector.FilterBar} filterBar
|
| + * @param {!WebInspector.Setting} coulmnsVisibilitySetting
|
| + */
|
| +WebInspector.NetworkLogView = function(filterBar, coulmnsVisibilitySetting)
|
| +{
|
| + WebInspector.VBox.call(this);
|
| + this.registerRequiredCSS("network/networkLogView.css");
|
| + this.registerRequiredCSS("ui/filter.css");
|
| +
|
| + this._filterBar = filterBar;
|
| + this._coulmnsVisibilitySetting = coulmnsVisibilitySetting;
|
| + this._allowRequestSelection = false;
|
| + /** @type {!Map.<string, !WebInspector.NetworkDataGridNode>} */
|
| + this._nodesByRequestId = new Map();
|
| + /** @type {!Object.<string, boolean>} */
|
| + this._staleRequestIds = {};
|
| + /** @type {number} */
|
| + this._mainRequestLoadTime = -1;
|
| + /** @type {number} */
|
| + this._mainRequestDOMContentLoadedTime = -1;
|
| + this._matchedRequestCount = 0;
|
| + this._highlightedSubstringChanges = [];
|
| +
|
| + /** @type {!Array.<!WebInspector.NetworkLogView.Filter>} */
|
| + this._filters = [];
|
| +
|
| + this._currentMatchedRequestNode = null;
|
| + this._currentMatchedRequestIndex = -1;
|
| +
|
| + this._createStatusbarButtons();
|
| + this._createStatusBarItems();
|
| + this._linkifier = new WebInspector.Linkifier();
|
| +
|
| + this._allowPopover = true;
|
| +
|
| + /** @type {number} */
|
| + this._rowHeight = 0;
|
| +
|
| + this._addFilters();
|
| + this._resetSuggestionBuilder();
|
| + this._initializeView();
|
| + this._toggleRecordButton(true);
|
| +
|
| + WebInspector.targetManager.observeTargets(this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.NetworkManager, WebInspector.NetworkManager.EventTypes.RequestStarted, this._onRequestStarted, this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.NetworkManager, WebInspector.NetworkManager.EventTypes.RequestUpdated, this._onRequestUpdated, this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.NetworkManager, WebInspector.NetworkManager.EventTypes.RequestFinished, this._onRequestUpdated, this);
|
| +
|
| + WebInspector.targetManager.addModelListener(WebInspector.ResourceTreeModel, WebInspector.ResourceTreeModel.EventTypes.WillReloadPage, this._willReloadPage, this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.ResourceTreeModel, WebInspector.ResourceTreeModel.EventTypes.MainFrameNavigated, this._mainFrameNavigated, this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.ResourceTreeModel, WebInspector.ResourceTreeModel.EventTypes.Load, this._loadEventFired, this);
|
| + WebInspector.targetManager.addModelListener(WebInspector.ResourceTreeModel, WebInspector.ResourceTreeModel.EventTypes.DOMContentLoaded, this._domContentLoadedEventFired, this);
|
| +}
|
| +
|
| +WebInspector.NetworkLogView.HTTPSchemas = {"http": true, "https": true, "ws": true, "wss": true};
|
| +WebInspector.NetworkLogView._responseHeaderColumns = ["Cache-Control", "Connection", "Content-Encoding", "Content-Length", "ETag", "Keep-Alive", "Last-Modified", "Server", "Vary"];
|
| +WebInspector.NetworkLogView.defaultColumnsVisibility = {
|
| + method: true, status: true, scheme: false, domain: false, remoteAddress: false, type: true, initiator: true, cookies: false, setCookies: false, size: true, time: true, connectionId: false,
|
| + "Cache-Control": false, "Connection": false, "Content-Encoding": false, "Content-Length": false, "ETag": false, "Keep-Alive": false, "Last-Modified": false, "Server": false, "Vary": false
|
| +};
|
| +WebInspector.NetworkLogView._defaultRefreshDelay = 500;
|
| +
|
| +/** @enum {string} */
|
| +WebInspector.NetworkLogView.FilterType = {
|
| + Domain: "Domain",
|
| + HasResponseHeader: "HasResponseHeader",
|
| + Is: "Is",
|
| + Method: "Method",
|
| + MimeType: "MimeType",
|
| + Scheme: "Scheme",
|
| + SetCookieDomain: "SetCookieDomain",
|
| + SetCookieName: "SetCookieName",
|
| + SetCookieValue: "SetCookieValue",
|
| + StatusCode: "StatusCode"
|
| +};
|
| +
|
| +/** @enum {string} */
|
| +WebInspector.NetworkLogView.IsFilterType = {
|
| + Running: "running"
|
| +};
|
| +
|
| +/** @type {!Array.<string>} */
|
| +WebInspector.NetworkLogView._searchKeys = Object.values(WebInspector.NetworkLogView.FilterType);
|
| +
|
| +/** @type {!Object.<string, string>} */
|
| +WebInspector.NetworkLogView._columnTitles = {
|
| + "name": WebInspector.UIString("Name"),
|
| + "method": WebInspector.UIString("Method"),
|
| + "status": WebInspector.UIString("Status"),
|
| + "scheme": WebInspector.UIString("Scheme"),
|
| + "domain": WebInspector.UIString("Domain"),
|
| + "remoteAddress": WebInspector.UIString("Remote Address"),
|
| + "type": WebInspector.UIString("Type"),
|
| + "initiator": WebInspector.UIString("Initiator"),
|
| + "cookies": WebInspector.UIString("Cookies"),
|
| + "setCookies": WebInspector.UIString("Set-Cookies"),
|
| + "size": WebInspector.UIString("Size"),
|
| + "time": WebInspector.UIString("Time"),
|
| + "connectionId": WebInspector.UIString("Connection Id"),
|
| + "timeline": WebInspector.UIString("Timeline"),
|
| +
|
| + // Response header columns
|
| + "Cache-Control": WebInspector.UIString("Cache-Control"),
|
| + "Connection": WebInspector.UIString("Connection"),
|
| + "Content-Encoding": WebInspector.UIString("Content-Encoding"),
|
| + "Content-Length": WebInspector.UIString("Content-Length"),
|
| + "ETag": WebInspector.UIString("ETag"),
|
| + "Keep-Alive": WebInspector.UIString("Keep-Alive"),
|
| + "Last-Modified": WebInspector.UIString("Last-Modified"),
|
| + "Server": WebInspector.UIString("Server"),
|
| + "Vary": WebInspector.UIString("Vary")
|
| +};
|
| +
|
| +WebInspector.NetworkLogView.prototype = {
|
| + /**
|
| + * @param {!WebInspector.Target} target
|
| + */
|
| + targetAdded: function(target)
|
| + {
|
| + target.networkLog.requests.forEach(this._appendRequest.bind(this));
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Target} target
|
| + */
|
| + targetRemoved: function(target)
|
| + {
|
| + },
|
| +
|
| + /**
|
| + * @return {boolean}
|
| + */
|
| + allowRequestSelection: function()
|
| + {
|
| + return this._allowRequestSelection;
|
| + },
|
| +
|
| + _addFilters: function()
|
| + {
|
| + this._textFilterUI = new WebInspector.TextFilterUI();
|
| + this._textFilterUI.addEventListener(WebInspector.FilterUI.Events.FilterChanged, this._filterChanged, this);
|
| + this._filterBar.addFilter(this._textFilterUI);
|
| +
|
| + var types = [];
|
| + for (var typeId in WebInspector.resourceTypes) {
|
| + var resourceType = WebInspector.resourceTypes[typeId];
|
| + if (resourceType === WebInspector.resourceTypes.TextTrack)
|
| + continue;
|
| + types.push({name: resourceType.name(), label: resourceType.categoryTitle()});
|
| + }
|
| + this._resourceTypeFilterUI = new WebInspector.NamedBitSetFilterUI(types, WebInspector.settings.networkResourceTypeFilters);
|
| + this._resourceTypeFilterUI.addEventListener(WebInspector.FilterUI.Events.FilterChanged, this._filterChanged.bind(this), this);
|
| + this._filterBar.addFilter(this._resourceTypeFilterUI);
|
| +
|
| + var dataURLSetting = WebInspector.settings.networkHideDataURL;
|
| + this._dataURLFilterUI = new WebInspector.CheckboxFilterUI("hide-data-url", WebInspector.UIString("Hide data URLs"), true, dataURLSetting);
|
| + this._dataURLFilterUI.addEventListener(WebInspector.FilterUI.Events.FilterChanged, this._filterChanged.bind(this), this);
|
| + this._filterBar.addFilter(this._dataURLFilterUI);
|
| + },
|
| +
|
| + _resetSuggestionBuilder: function()
|
| + {
|
| + this._suggestionBuilder = new WebInspector.FilterSuggestionBuilder(WebInspector.NetworkLogView._searchKeys);
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.Is, WebInspector.NetworkLogView.IsFilterType.Running);
|
| + this._textFilterUI.setSuggestionBuilder(this._suggestionBuilder);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _filterChanged: function(event)
|
| + {
|
| + this._removeAllNodeHighlights();
|
| + this._parseFilterQuery(this._textFilterUI.value());
|
| + this._filterRequests();
|
| + },
|
| +
|
| + _initializeView: function()
|
| + {
|
| + this.element.id = "network-container";
|
| +
|
| + this._createSortingFunctions();
|
| + this._createCalculators();
|
| + this._createTable();
|
| + this._createTimelineGrid();
|
| + this._summaryBarElement = this.element.createChild("div", "network-summary-bar");
|
| +
|
| + this._updateRowsSize();
|
| +
|
| + this._popoverHelper = new WebInspector.PopoverHelper(this.element, this._getPopoverAnchor.bind(this), this._showPopover.bind(this), this._onHidePopover.bind(this));
|
| + // Enable faster hint.
|
| + this._popoverHelper.setTimeout(250, 250);
|
| +
|
| + this.switchViewMode(true);
|
| + },
|
| +
|
| + /**
|
| + * @return {!Array.<!Element>}
|
| + */
|
| + statusBarItems: function()
|
| + {
|
| + return [
|
| + this._recordButton.element,
|
| + this._clearButton.element,
|
| + this._filterBar.filterButton().element,
|
| + this._largerRequestsButton.element,
|
| + this._preserveLogCheckbox.element,
|
| + this._disableCacheCheckbox.element,
|
| + this._progressBarContainer];
|
| + },
|
| +
|
| + /**
|
| + * @return {boolean}
|
| + */
|
| + usesLargeRows: function()
|
| + {
|
| + return !!WebInspector.settings.resourcesLargeRows.get();
|
| + },
|
| +
|
| + /**
|
| + * @param {boolean} flag
|
| + */
|
| + setAllowPopover: function(flag)
|
| + {
|
| + this._allowPopover = flag;
|
| + },
|
| +
|
| + /**
|
| + * @return {!Array.<!Element>}
|
| + */
|
| + elementsToRestoreScrollPositionsFor: function()
|
| + {
|
| + if (!this._dataGrid) // Not initialized yet.
|
| + return [];
|
| + return [this._dataGrid.scrollContainer];
|
| + },
|
| +
|
| + _createTimelineGrid: function()
|
| + {
|
| + this._timelineGrid = new WebInspector.TimelineGrid();
|
| + this._timelineGrid.element.classList.add("network-timeline-grid");
|
| + this._dataGrid.element.appendChild(this._timelineGrid.element);
|
| + },
|
| +
|
| + _createTable: function()
|
| + {
|
| + var columns = [];
|
| + columns.push({
|
| + id: "name",
|
| + titleDOMFragment: this._makeHeaderFragment(WebInspector.UIString("Name"), WebInspector.UIString("Path")),
|
| + title: WebInspector.NetworkLogView._columnTitles["name"],
|
| + sortable: true,
|
| + weight: 20,
|
| + disclosure: true
|
| + });
|
| +
|
| + columns.push({
|
| + id: "method",
|
| + title: WebInspector.NetworkLogView._columnTitles["method"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + columns.push({
|
| + id: "status",
|
| + titleDOMFragment: this._makeHeaderFragment(WebInspector.UIString("Status"), WebInspector.UIString("Text")),
|
| + title: WebInspector.NetworkLogView._columnTitles["status"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + columns.push({
|
| + id: "scheme",
|
| + title: WebInspector.NetworkLogView._columnTitles["scheme"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + columns.push({
|
| + id: "domain",
|
| + title: WebInspector.NetworkLogView._columnTitles["domain"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + columns.push({
|
| + id: "remoteAddress",
|
| + title: WebInspector.NetworkLogView._columnTitles["remoteAddress"],
|
| + sortable: true,
|
| + weight: 10,
|
| + align: WebInspector.DataGrid.Align.Right
|
| + });
|
| +
|
| + columns.push({
|
| + id: "type",
|
| + title: WebInspector.NetworkLogView._columnTitles["type"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + columns.push({
|
| + id: "initiator",
|
| + title: WebInspector.NetworkLogView._columnTitles["initiator"],
|
| + sortable: true,
|
| + weight: 10
|
| + });
|
| +
|
| + columns.push({
|
| + id: "cookies",
|
| + title: WebInspector.NetworkLogView._columnTitles["cookies"],
|
| + sortable: true,
|
| + weight: 6,
|
| + align: WebInspector.DataGrid.Align.Right
|
| + });
|
| +
|
| + columns.push({
|
| + id: "setCookies",
|
| + title: WebInspector.NetworkLogView._columnTitles["setCookies"],
|
| + sortable: true,
|
| + weight: 6,
|
| + align: WebInspector.DataGrid.Align.Right
|
| + });
|
| +
|
| + columns.push({
|
| + id: "size",
|
| + titleDOMFragment: this._makeHeaderFragment(WebInspector.UIString("Size"), WebInspector.UIString("Content")),
|
| + title: WebInspector.NetworkLogView._columnTitles["size"],
|
| + sortable: true,
|
| + weight: 6,
|
| + align: WebInspector.DataGrid.Align.Right
|
| + });
|
| +
|
| + columns.push({
|
| + id: "time",
|
| + titleDOMFragment: this._makeHeaderFragment(WebInspector.UIString("Time"), WebInspector.UIString("Latency")),
|
| + title: WebInspector.NetworkLogView._columnTitles["time"],
|
| + sortable: true,
|
| + weight: 6,
|
| + align: WebInspector.DataGrid.Align.Right
|
| + });
|
| +
|
| + columns.push({
|
| + id: "connectionId",
|
| + title: WebInspector.NetworkLogView._columnTitles["connectionId"],
|
| + sortable: true,
|
| + weight: 6
|
| + });
|
| +
|
| + var responseHeaderColumns = WebInspector.NetworkLogView._responseHeaderColumns;
|
| + for (var i = 0; i < responseHeaderColumns.length; ++i) {
|
| + var headerName = responseHeaderColumns[i];
|
| + var descriptor = {
|
| + id: headerName,
|
| + title: WebInspector.NetworkLogView._columnTitles[headerName],
|
| + weight: 6
|
| + }
|
| + if (headerName === "Content-Length")
|
| + descriptor.align = WebInspector.DataGrid.Align.Right;
|
| + columns.push(descriptor);
|
| + }
|
| +
|
| + columns.push({
|
| + id: "timeline",
|
| + titleDOMFragment: createDocumentFragment(),
|
| + title: WebInspector.NetworkLogView._columnTitles["timeline"],
|
| + sortable: false,
|
| + weight: 40,
|
| + sort: WebInspector.DataGrid.Order.Ascending
|
| + });
|
| +
|
| + this._dataGrid = new WebInspector.SortableDataGrid(columns);
|
| + this._dataGrid.setStickToBottom(true);
|
| + this._updateColumns();
|
| + this._dataGrid.setName("networkLog");
|
| + this._dataGrid.setResizeMethod(WebInspector.DataGrid.ResizeMethod.Last);
|
| + this._dataGrid.element.classList.add("network-log-grid");
|
| + this._dataGrid.element.addEventListener("contextmenu", this._contextMenu.bind(this), true);
|
| + this._dataGrid.show(this.element);
|
| +
|
| + // Event listeners need to be added _after_ we attach to the document, so that owner document is properly update.
|
| + this._dataGrid.addEventListener(WebInspector.DataGrid.Events.SortingChanged, this._sortItems, this);
|
| + this._dataGrid.addEventListener(WebInspector.DataGrid.Events.ColumnsResized, this._updateDividersIfNeeded, this);
|
| +
|
| + this._patchTimelineHeader();
|
| + this._dataGrid.sortNodes(this._sortingFunctions.startTime, false);
|
| + },
|
| +
|
| + /**
|
| + * @param {string} title
|
| + * @param {string} subtitle
|
| + * @return {!DocumentFragment}
|
| + */
|
| + _makeHeaderFragment: function(title, subtitle)
|
| + {
|
| + var fragment = createDocumentFragment();
|
| + fragment.createTextChild(title);
|
| + var subtitleDiv = fragment.createChild("div", "network-header-subtitle");
|
| + subtitleDiv.createTextChild(subtitle);
|
| + return fragment;
|
| + },
|
| +
|
| + _patchTimelineHeader: function()
|
| + {
|
| + var timelineSorting = createElement("select");
|
| +
|
| + var option = createElement("option");
|
| + option.value = "startTime";
|
| + option.label = WebInspector.UIString("Timeline");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + option = createElement("option");
|
| + option.value = "startTime";
|
| + option.label = WebInspector.UIString("Start Time");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + option = createElement("option");
|
| + option.value = "responseTime";
|
| + option.label = WebInspector.UIString("Response Time");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + option = createElement("option");
|
| + option.value = "endTime";
|
| + option.label = WebInspector.UIString("End Time");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + option = createElement("option");
|
| + option.value = "duration";
|
| + option.label = WebInspector.UIString("Duration");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + option = createElement("option");
|
| + option.value = "latency";
|
| + option.label = WebInspector.UIString("Latency");
|
| + timelineSorting.appendChild(option);
|
| +
|
| + var header = this._dataGrid.headerTableHeader("timeline");
|
| + header.replaceChild(timelineSorting, header.firstChild);
|
| +
|
| + timelineSorting.addEventListener("click", function(event) { event.consume() }, false);
|
| + timelineSorting.addEventListener("change", this._sortByTimeline.bind(this), false);
|
| + this._timelineSortSelector = timelineSorting;
|
| + },
|
| +
|
| + _createSortingFunctions: function()
|
| + {
|
| + this._sortingFunctions = {};
|
| + this._sortingFunctions.name = WebInspector.NetworkDataGridNode.NameComparator;
|
| + this._sortingFunctions.method = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "method", false);
|
| + this._sortingFunctions.status = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "statusCode", false);
|
| + this._sortingFunctions.scheme = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "scheme", false);
|
| + this._sortingFunctions.domain = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "domain", false);
|
| + this._sortingFunctions.remoteAddress = WebInspector.NetworkDataGridNode.RemoteAddressComparator;
|
| + this._sortingFunctions.type = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "mimeType", false);
|
| + this._sortingFunctions.initiator = WebInspector.NetworkDataGridNode.InitiatorComparator;
|
| + this._sortingFunctions.cookies = WebInspector.NetworkDataGridNode.RequestCookiesCountComparator;
|
| + this._sortingFunctions.setCookies = WebInspector.NetworkDataGridNode.ResponseCookiesCountComparator;
|
| + this._sortingFunctions.size = WebInspector.NetworkDataGridNode.SizeComparator;
|
| + this._sortingFunctions.time = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "duration", false);
|
| + this._sortingFunctions.connectionId = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "connectionId", false);
|
| + this._sortingFunctions.timeline = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "startTime", false);
|
| + this._sortingFunctions.startTime = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "startTime", false);
|
| + this._sortingFunctions.endTime = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "endTime", false);
|
| + this._sortingFunctions.responseTime = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "responseReceivedTime", false);
|
| + this._sortingFunctions.duration = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "duration", true);
|
| + this._sortingFunctions.latency = WebInspector.NetworkDataGridNode.RequestPropertyComparator.bind(null, "latency", true);
|
| + },
|
| +
|
| + _createCalculators: function()
|
| + {
|
| + /** @type {!WebInspector.NetworkTransferTimeCalculator} */
|
| + this._timeCalculator = new WebInspector.NetworkTransferTimeCalculator();
|
| + /** @type {!WebInspector.NetworkTransferDurationCalculator} */
|
| + this._durationCalculator = new WebInspector.NetworkTransferDurationCalculator();
|
| +
|
| + /** @type {!Object.<string, !WebInspector.NetworkTimeCalculator>} */
|
| + this._calculators = {};
|
| + this._calculators.timeline = this._timeCalculator;
|
| + this._calculators.startTime = this._timeCalculator;
|
| + this._calculators.endTime = this._timeCalculator;
|
| + this._calculators.responseTime = this._timeCalculator;
|
| + this._calculators.duration = this._durationCalculator;
|
| + this._calculators.latency = this._durationCalculator;
|
| +
|
| + this._calculator = this._timeCalculator;
|
| + },
|
| +
|
| + _sortItems: function()
|
| + {
|
| + this._removeAllNodeHighlights();
|
| + var columnIdentifier = this._dataGrid.sortColumnIdentifier();
|
| + if (columnIdentifier === "timeline") {
|
| + this._sortByTimeline();
|
| + return;
|
| + }
|
| + var sortingFunction = this._sortingFunctions[columnIdentifier];
|
| + if (!sortingFunction)
|
| + return;
|
| +
|
| + this._dataGrid.sortNodes(sortingFunction, !this._dataGrid.isSortOrderAscending());
|
| + this._highlightNthMatchedRequestForSearch(this._updateMatchCountAndFindMatchIndex(this._currentMatchedRequestNode), false);
|
| + this._timelineSortSelector.selectedIndex = 0;
|
| +
|
| + WebInspector.notifications.dispatchEventToListeners(WebInspector.UserMetrics.UserAction, {
|
| + action: WebInspector.UserMetrics.UserActionNames.NetworkSort,
|
| + column: columnIdentifier,
|
| + sortOrder: this._dataGrid.sortOrder()
|
| + });
|
| + },
|
| +
|
| + _sortByTimeline: function()
|
| + {
|
| + this._removeAllNodeHighlights();
|
| + var selectedIndex = this._timelineSortSelector.selectedIndex;
|
| + if (!selectedIndex)
|
| + selectedIndex = 1; // Sort by start time by default.
|
| + var selectedOption = this._timelineSortSelector[selectedIndex];
|
| + var value = selectedOption.value;
|
| +
|
| + this._setCalculator(this._calculators[value]);
|
| + var sortingFunction = this._sortingFunctions[value];
|
| + this._dataGrid.sortNodes(sortingFunction);
|
| + this._highlightNthMatchedRequestForSearch(this._updateMatchCountAndFindMatchIndex(this._currentMatchedRequestNode), false);
|
| + this._dataGrid.markColumnAsSortedBy("timeline", WebInspector.DataGrid.Order.Ascending);
|
| + },
|
| +
|
| + _createStatusBarItems: function()
|
| + {
|
| + this._progressBarContainer = createElement("div");
|
| + this._progressBarContainer.className = "status-bar-item";
|
| + },
|
| +
|
| + _updateSummaryBar: function()
|
| + {
|
| + var requestsNumber = this._nodesByRequestId.size;
|
| +
|
| + if (!requestsNumber) {
|
| + if (this._summaryBarElement._isDisplayingWarning)
|
| + return;
|
| + this._summaryBarElement._isDisplayingWarning = true;
|
| + this._summaryBarElement.removeChildren();
|
| + this._summaryBarElement.createChild("div", "warning-icon-small");
|
| + var text = WebInspector.UIString("No requests captured. Reload the page to see detailed information on the network activity.");
|
| + this._summaryBarElement.createTextChild(text);
|
| + this._summaryBarElement.title = text;
|
| + return;
|
| + }
|
| + delete this._summaryBarElement._isDisplayingWarning;
|
| +
|
| + var transferSize = 0;
|
| + var selectedRequestsNumber = 0;
|
| + var selectedTransferSize = 0;
|
| + var baseTime = -1;
|
| + var maxTime = -1;
|
| + var nodes = this._nodesByRequestId.valuesArray();
|
| + for (var i = 0; i < nodes.length; ++i) {
|
| + var request = nodes[i].request();
|
| + var requestTransferSize = request.transferSize;
|
| + transferSize += requestTransferSize;
|
| + if (!nodes[i]._isFilteredOut) {
|
| + selectedRequestsNumber++;
|
| + selectedTransferSize += requestTransferSize;
|
| + }
|
| + if (request.url === request.target().resourceTreeModel.inspectedPageURL())
|
| + baseTime = request.startTime;
|
| + if (request.endTime > maxTime)
|
| + maxTime = request.endTime;
|
| + }
|
| + var text = "";
|
| + if (selectedRequestsNumber !== requestsNumber) {
|
| + text += String.sprintf(WebInspector.UIString("%d / %d requests"), selectedRequestsNumber, requestsNumber);
|
| + text += " \u2758 " + String.sprintf(WebInspector.UIString("%s / %s transferred"), Number.bytesToString(selectedTransferSize), Number.bytesToString(transferSize));
|
| + } else {
|
| + text += String.sprintf(WebInspector.UIString("%d requests"), requestsNumber);
|
| + text += " \u2758 " + String.sprintf(WebInspector.UIString("%s transferred"), Number.bytesToString(transferSize));
|
| + }
|
| + if (baseTime !== -1 && this._mainRequestLoadTime !== -1 && this._mainRequestDOMContentLoadedTime !== -1 && this._mainRequestDOMContentLoadedTime > baseTime) {
|
| + text += " \u2758 " + String.sprintf(WebInspector.UIString("%s (load: %s, DOMContentLoaded: %s)"),
|
| + Number.secondsToString(maxTime - baseTime),
|
| + Number.secondsToString(this._mainRequestLoadTime - baseTime),
|
| + Number.secondsToString(this._mainRequestDOMContentLoadedTime - baseTime));
|
| + }
|
| + this._summaryBarElement.textContent = text;
|
| + this._summaryBarElement.title = text;
|
| + },
|
| +
|
| + _scheduleRefresh: function()
|
| + {
|
| + if (this._needsRefresh)
|
| + return;
|
| +
|
| + this._needsRefresh = true;
|
| +
|
| + if (this.isShowing() && !this._refreshTimeout)
|
| + this._refreshTimeout = setTimeout(this.refresh.bind(this), WebInspector.NetworkLogView._defaultRefreshDelay);
|
| + },
|
| +
|
| + _updateDividersIfNeeded: function()
|
| + {
|
| + var timelineOffset = this._dataGrid.columnOffset("timeline");
|
| + // Position timline grid location.
|
| + if (timelineOffset)
|
| + this._timelineGrid.element.style.left = timelineOffset + "px";
|
| +
|
| + var calculator = this.calculator();
|
| + var proceed = true;
|
| + if (!this.isShowing()) {
|
| + this._scheduleRefresh();
|
| + proceed = false;
|
| + } else {
|
| + calculator.setDisplayWindow(this._timelineGrid.dividersElement.clientWidth);
|
| + proceed = this._timelineGrid.updateDividers(calculator);
|
| + }
|
| + if (!proceed)
|
| + return;
|
| +
|
| + if (calculator.startAtZero) {
|
| + // If our current sorting method starts at zero, that means it shows all
|
| + // requests starting at the same point, and so onLoad event and DOMContent
|
| + // event lines really wouldn't make much sense here, so don't render them.
|
| + return;
|
| + }
|
| +
|
| + this._timelineGrid.removeEventDividers();
|
| + var loadTimePercent = calculator.computePercentageFromEventTime(this._mainRequestLoadTime);
|
| + if (this._mainRequestLoadTime !== -1 && loadTimePercent >= 0) {
|
| + var loadDivider = createElementWithClass("div", "network-event-divider-padding");
|
| + loadDivider.createChild("div", "network-event-divider network-red-divider");
|
| + loadDivider.title = WebInspector.UIString("Load event");
|
| + loadDivider.style.left = loadTimePercent + "%";
|
| + this._timelineGrid.addEventDivider(loadDivider);
|
| + }
|
| +
|
| + var domLoadTimePrecent = calculator.computePercentageFromEventTime(this._mainRequestDOMContentLoadedTime);
|
| + if (this._mainRequestDOMContentLoadedTime !== -1 && domLoadTimePrecent >= 0) {
|
| + var domContentLoadedDivider = createElementWithClass("div", "network-event-divider-padding");
|
| + domContentLoadedDivider.createChild("div", "network-event-divider network-blue-divider");
|
| + domContentLoadedDivider.title = WebInspector.UIString("DOMContentLoaded event");
|
| + domContentLoadedDivider.style.left = domLoadTimePrecent + "%";
|
| + this._timelineGrid.addEventDivider(domContentLoadedDivider);
|
| + }
|
| + },
|
| +
|
| + _refreshIfNeeded: function()
|
| + {
|
| + if (this._needsRefresh)
|
| + this.refresh();
|
| + },
|
| +
|
| + _invalidateAllItems: function()
|
| + {
|
| + var requestIds = this._nodesByRequestId.keysArray();
|
| + for (var i = 0; i < requestIds.length; ++i)
|
| + this._staleRequestIds[requestIds[i]] = true;
|
| + },
|
| +
|
| + /**
|
| + * @return {!WebInspector.NetworkTimeCalculator}
|
| + */
|
| + calculator: function()
|
| + {
|
| + return this._calculator;
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkTimeCalculator} x
|
| + */
|
| + _setCalculator: function(x)
|
| + {
|
| + if (!x || this._calculator === x)
|
| + return;
|
| +
|
| + this._calculator = x;
|
| + this._calculator.reset();
|
| +
|
| + if (this._calculator.startAtZero)
|
| + this._timelineGrid.hideEventDividers();
|
| + else
|
| + this._timelineGrid.showEventDividers();
|
| +
|
| + this._invalidateAllItems();
|
| + this.refresh();
|
| + },
|
| +
|
| + _createStatusbarButtons: function()
|
| + {
|
| + this._recordButton = new WebInspector.StatusBarButton("", "record-profile-status-bar-item");
|
| + this._recordButton.addEventListener("click", this._onRecordButtonClicked, this);
|
| +
|
| + this._clearButton = new WebInspector.StatusBarButton(WebInspector.UIString("Clear"), "clear-status-bar-item");
|
| + this._clearButton.addEventListener("click", this._reset, this);
|
| +
|
| + this._largerRequestsButton = new WebInspector.StatusBarButton(WebInspector.UIString("Use small resource rows."), "network-larger-resources-status-bar-item");
|
| + this._largerRequestsButton.setToggled(WebInspector.settings.resourcesLargeRows.get());
|
| + this._largerRequestsButton.addEventListener("click", this._toggleLargerRequests, this);
|
| +
|
| + this._preserveLogCheckbox = new WebInspector.StatusBarCheckbox(WebInspector.UIString("Preserve log"));
|
| + this._preserveLogCheckbox.element.title = WebInspector.UIString("Do not clear log on page reload / navigation.");
|
| +
|
| + this._disableCacheCheckbox = new WebInspector.StatusBarCheckbox(WebInspector.UIString("Disable cache"));
|
| + WebInspector.SettingsUI.bindCheckbox(this._disableCacheCheckbox.inputElement, WebInspector.settings.cacheDisabled);
|
| + this._disableCacheCheckbox.element.title = WebInspector.UIString("Disable cache (while DevTools is open).");
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _loadEventFired: function(event)
|
| + {
|
| + if (!this._recordButton.toggled())
|
| + return;
|
| +
|
| + var data = /** @type {number} */ (event.data);
|
| + this._mainRequestLoadTime = data || -1;
|
| + // Schedule refresh to update boundaries and draw the new line.
|
| + this._scheduleRefresh();
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _domContentLoadedEventFired: function(event)
|
| + {
|
| + if (!this._recordButton.toggled())
|
| + return;
|
| + var data = /** @type {number} */ (event.data);
|
| + this._mainRequestDOMContentLoadedTime = data || -1;
|
| + // Schedule refresh to update boundaries and draw the new line.
|
| + this._scheduleRefresh();
|
| + },
|
| +
|
| + wasShown: function()
|
| + {
|
| + this._refreshIfNeeded();
|
| + },
|
| +
|
| + willHide: function()
|
| + {
|
| + this._popoverHelper.hidePopover();
|
| + },
|
| +
|
| + refresh: function()
|
| + {
|
| + this._needsRefresh = false;
|
| + if (this._refreshTimeout) {
|
| + clearTimeout(this._refreshTimeout);
|
| + delete this._refreshTimeout;
|
| + }
|
| +
|
| + this._removeAllNodeHighlights();
|
| + var boundariesChanged = false;
|
| + var calculator = this.calculator();
|
| + if (calculator.updateBoundariesForEventTime) {
|
| + boundariesChanged = calculator.updateBoundariesForEventTime(this._mainRequestLoadTime) || boundariesChanged;
|
| + boundariesChanged = calculator.updateBoundariesForEventTime(this._mainRequestDOMContentLoadedTime) || boundariesChanged;
|
| + }
|
| +
|
| + var dataGrid = this._dataGrid;
|
| + var rootNode = dataGrid.rootNode();
|
| + var nodesToInsert = [];
|
| + for (var requestId in this._staleRequestIds) {
|
| + var node = this._nodesByRequestId.get(requestId);
|
| + if (!node)
|
| + continue;
|
| + if (!node._isFilteredOut)
|
| + rootNode.removeChild(node);
|
| + node._isFilteredOut = !this._applyFilter(node);
|
| + if (!node._isFilteredOut)
|
| + nodesToInsert.push(node);
|
| + }
|
| +
|
| + for (var i = 0; i < nodesToInsert.length; ++i) {
|
| + var node = nodesToInsert[i];
|
| + var request = node.request();
|
| + node.refresh();
|
| + dataGrid.insertChild(node);
|
| + node._isMatchingSearchQuery = this._matchRequest(request);
|
| + if (calculator.updateBoundaries(request))
|
| + boundariesChanged = true;
|
| + }
|
| +
|
| + this._highlightNthMatchedRequestForSearch(this._updateMatchCountAndFindMatchIndex(this._currentMatchedRequestNode), false);
|
| +
|
| + if (boundariesChanged) {
|
| + // The boundaries changed, so all item graphs are stale.
|
| + this._updateDividersIfNeeded();
|
| + var nodes = this._nodesByRequestId.valuesArray();
|
| + for (var i = 0; i < nodes.length; ++i)
|
| + nodes[i].refreshGraph();
|
| + }
|
| +
|
| + this._staleRequestIds = {};
|
| + this._updateSummaryBar();
|
| + },
|
| +
|
| + _onRecordButtonClicked: function()
|
| + {
|
| + if (!this._recordButton.toggled())
|
| + this._reset();
|
| + this._toggleRecordButton(!this._recordButton.toggled());
|
| + },
|
| +
|
| + /**
|
| + * @param {boolean} toggled
|
| + */
|
| + _toggleRecordButton: function(toggled)
|
| + {
|
| + this._recordButton.setToggled(toggled);
|
| + this._recordButton.setTitle(toggled ? WebInspector.UIString("Stop Recording Network Log") : WebInspector.UIString("Record Network Log"));
|
| + },
|
| +
|
| + _reset: function()
|
| + {
|
| + this.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.ViewCleared);
|
| +
|
| + this._clearSearchMatchedList();
|
| + if (this._popoverHelper)
|
| + this._popoverHelper.hidePopover();
|
| +
|
| + if (this._calculator)
|
| + this._calculator.reset();
|
| +
|
| + var nodes = this._nodesByRequestId.valuesArray();
|
| + for (var i = 0; i < nodes.length; ++i)
|
| + nodes[i].dispose();
|
| +
|
| + this._nodesByRequestId.clear();
|
| + this._staleRequestIds = {};
|
| + this._resetSuggestionBuilder();
|
| +
|
| + if (this._dataGrid) {
|
| + this._dataGrid.rootNode().removeChildren();
|
| + this._updateDividersIfNeeded();
|
| + this._updateSummaryBar();
|
| + }
|
| +
|
| + this._mainRequestLoadTime = -1;
|
| + this._mainRequestDOMContentLoadedTime = -1;
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _onRequestStarted: function(event)
|
| + {
|
| + if (this._recordButton.toggled()) {
|
| + var request = /** @type {!WebInspector.NetworkRequest} */ (event.data);
|
| + this._appendRequest(request);
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _appendRequest: function(request)
|
| + {
|
| + var node = new WebInspector.NetworkDataGridNode(this, request);
|
| +
|
| + // In case of redirect request id is reassigned to a redirected
|
| + // request and we need to update _nodesByRequestId and search results.
|
| + var originalRequestNode = this._nodesByRequestId.get(request.requestId);
|
| + if (originalRequestNode)
|
| + this._nodesByRequestId.set(originalRequestNode.request().requestId, originalRequestNode);
|
| + this._nodesByRequestId.set(request.requestId, node);
|
| +
|
| + // Pull all the redirects of the main request upon commit load.
|
| + if (request.redirects) {
|
| + for (var i = 0; i < request.redirects.length; ++i)
|
| + this._refreshRequest(request.redirects[i]);
|
| + }
|
| +
|
| + this._refreshRequest(request);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _onRequestUpdated: function(event)
|
| + {
|
| + var request = /** @type {!WebInspector.NetworkRequest} */ (event.data);
|
| + this._refreshRequest(request);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _refreshRequest: function(request)
|
| + {
|
| + if (!this._nodesByRequestId.get(request.requestId))
|
| + return;
|
| +
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.Domain, request.domain);
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.Method, request.requestMethod);
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.MimeType, request.mimeType);
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.Scheme, "" + request.scheme);
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.StatusCode, "" + request.statusCode);
|
| +
|
| + var responseHeaders = request.responseHeaders;
|
| + for (var i = 0, l = responseHeaders.length; i < l; ++i)
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.HasResponseHeader, responseHeaders[i].name);
|
| + var cookies = request.responseCookies;
|
| + for (var i = 0, l = cookies ? cookies.length : 0; i < l; ++i) {
|
| + var cookie = cookies[i];
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.SetCookieDomain, cookie.domain());
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.SetCookieName, cookie.name());
|
| + this._suggestionBuilder.addItem(WebInspector.NetworkLogView.FilterType.SetCookieValue, cookie.value());
|
| + }
|
| +
|
| + this._staleRequestIds[request.requestId] = true;
|
| + this._scheduleRefresh();
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _willReloadPage: function(event)
|
| + {
|
| + this._recordButton.setToggled(true);
|
| + if (!this._preserveLogCheckbox.checked())
|
| + this._reset();
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.Event} event
|
| + */
|
| + _mainFrameNavigated: function(event)
|
| + {
|
| + if (!this._recordButton.toggled() || this._preserveLogCheckbox.checked())
|
| + return;
|
| +
|
| + var frame = /** @type {!WebInspector.ResourceTreeFrame} */ (event.data);
|
| + var loaderId = frame.loaderId;
|
| +
|
| + // Pick provisional load requests.
|
| + var requestsToPick = [];
|
| + var requests = frame.target().networkLog.requests;
|
| + for (var i = 0; i < requests.length; ++i) {
|
| + var request = requests[i];
|
| + if (request.loaderId === loaderId)
|
| + requestsToPick.push(request);
|
| + }
|
| +
|
| + this._reset();
|
| +
|
| + for (var i = 0; i < requestsToPick.length; ++i)
|
| + this._appendRequest(requestsToPick[i]);
|
| + },
|
| +
|
| + /**
|
| + * @param {boolean} detailed
|
| + */
|
| + switchViewMode: function(detailed)
|
| + {
|
| + if (this._detailedMode === detailed)
|
| + return;
|
| + this._detailedMode = detailed;
|
| +
|
| + if (detailed) {
|
| + if (this._dataGrid.selectedNode)
|
| + this._dataGrid.selectedNode.selected = false;
|
| + } else {
|
| + this._removeAllNodeHighlights();
|
| + this._popoverHelper.hidePopover();
|
| + }
|
| +
|
| + this.element.classList.toggle("brief-mode", !detailed);
|
| + this._updateColumns();
|
| + },
|
| +
|
| + _toggleLargerRequests: function()
|
| + {
|
| + WebInspector.settings.resourcesLargeRows.set(!WebInspector.settings.resourcesLargeRows.get());
|
| + this._updateRowsSize();
|
| + },
|
| +
|
| + /**
|
| + * @return {number}
|
| + */
|
| + rowHeight: function()
|
| + {
|
| + return this._rowHeight;
|
| + },
|
| +
|
| + _updateRowsSize: function()
|
| + {
|
| + var largeRows = this.usesLargeRows();
|
| + this._largerRequestsButton.setToggled(largeRows);
|
| + this._rowHeight = largeRows ? 41 : 21;
|
| + this._largerRequestsButton.setTitle(WebInspector.UIString(largeRows ? "Use small resource rows." : "Use large resource rows."));
|
| + this._dataGrid.element.classList.toggle("small", !largeRows);
|
| + this._timelineGrid.element.classList.toggle("small", !largeRows);
|
| + this.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.RowSizeChanged, { largeRows: largeRows });
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} element
|
| + * @param {!Event} event
|
| + * @return {!Element|!AnchorBox|undefined}
|
| + */
|
| + _getPopoverAnchor: function(element, event)
|
| + {
|
| + if (!this._allowPopover)
|
| + return;
|
| + var anchor = element.enclosingNodeOrSelfWithClass("network-graph-bar") || element.enclosingNodeOrSelfWithClass("network-graph-label");
|
| + if (anchor && anchor.parentElement.request && anchor.parentElement.request.timing)
|
| + return anchor;
|
| + anchor = element.enclosingNodeOrSelfWithClass("network-script-initiated");
|
| + if (anchor && anchor.request) {
|
| + var request = /** @type {!WebInspector.NetworkRequest} */ (anchor.request);
|
| + var initiator = anchor.request.initiator();
|
| + if (initiator && (initiator.stackTrace || initiator.asyncStackTrace))
|
| + return anchor;
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} anchor
|
| + * @param {!WebInspector.Popover} popover
|
| + */
|
| + _showPopover: function(anchor, popover)
|
| + {
|
| + var content;
|
| + if (anchor.classList.contains("network-script-initiated")) {
|
| + content = this._generateScriptInitiatedPopoverContent(anchor.request);
|
| + popover.setCanShrink(true);
|
| + } else {
|
| + content = WebInspector.RequestTimingView.createTimingTable(anchor.parentElement.request);
|
| + popover.setCanShrink(false);
|
| + }
|
| + popover.show(content, anchor);
|
| + },
|
| +
|
| + _onHidePopover: function()
|
| + {
|
| + this._linkifier.reset();
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {!Element}
|
| + */
|
| + _generateScriptInitiatedPopoverContent: function(request)
|
| + {
|
| + var framesTable = createElementWithClass("table", "network-stack-trace");
|
| +
|
| + /**
|
| + * @param {!Array.<!ConsoleAgent.CallFrame>} stackTrace
|
| + * @this {WebInspector.NetworkLogView}
|
| + */
|
| + function appendStackTrace(stackTrace)
|
| + {
|
| + for (var i = 0; i < stackTrace.length; ++i) {
|
| + var stackFrame = stackTrace[i];
|
| + var row = createElement("tr");
|
| + row.createChild("td").textContent = stackFrame.functionName || WebInspector.UIString("(anonymous function)");
|
| + row.createChild("td").textContent = " @ ";
|
| + row.createChild("td").appendChild(this._linkifier.linkifyConsoleCallFrame(request.target(), stackFrame));
|
| + framesTable.appendChild(row);
|
| + }
|
| + }
|
| +
|
| + // Initiator is not null, checked in _getPopoverAnchor.
|
| + var initiator = /** @type {!NetworkAgent.Initiator} */ (request.initiator());
|
| + if (initiator.stackTrace)
|
| + appendStackTrace.call(this, initiator.stackTrace);
|
| +
|
| + var asyncStackTrace = initiator.asyncStackTrace;
|
| + while (asyncStackTrace) {
|
| + var callFrames = asyncStackTrace.callFrames;
|
| + if (!callFrames || !callFrames.length)
|
| + break;
|
| + var row = framesTable.createChild("tr");
|
| + row.createChild("td", "network-async-trace-description").textContent = WebInspector.asyncStackTraceLabel(asyncStackTrace.description);
|
| + row.createChild("td");
|
| + row.createChild("td");
|
| + appendStackTrace.call(this, callFrames);
|
| + asyncStackTrace = asyncStackTrace.asyncStackTrace;
|
| + }
|
| +
|
| + return framesTable;
|
| + },
|
| +
|
| + _updateColumns: function()
|
| + {
|
| + var detailedMode = !!this._detailedMode;
|
| + var visibleColumns = {"name": true};
|
| + if (detailedMode) {
|
| + visibleColumns["timeline"] = true;
|
| + var columnsVisibility = this._coulmnsVisibilitySetting.get();
|
| + for (var columnIdentifier in columnsVisibility)
|
| + visibleColumns[columnIdentifier] = columnsVisibility[columnIdentifier];
|
| + }
|
| +
|
| + this._dataGrid.setColumnsVisiblity(visibleColumns);
|
| + },
|
| +
|
| + /**
|
| + * @param {string} columnIdentifier
|
| + */
|
| + _toggleColumnVisibility: function(columnIdentifier)
|
| + {
|
| + var columnsVisibility = this._coulmnsVisibilitySetting.get();
|
| + columnsVisibility[columnIdentifier] = !columnsVisibility[columnIdentifier];
|
| + this._coulmnsVisibilitySetting.set(columnsVisibility);
|
| +
|
| + this._updateColumns();
|
| + },
|
| +
|
| + /**
|
| + * @return {!Array.<string>}
|
| + */
|
| + _getConfigurableColumnIDs: function()
|
| + {
|
| + if (this._configurableColumnIDs)
|
| + return this._configurableColumnIDs;
|
| +
|
| + var columnTitles = WebInspector.NetworkLogView._columnTitles;
|
| + function compare(id1, id2)
|
| + {
|
| + return columnTitles[id1].compareTo(columnTitles[id2]);
|
| + }
|
| +
|
| + var columnIDs = Object.keys(this._coulmnsVisibilitySetting.get());
|
| + this._configurableColumnIDs = columnIDs.sort(compare);
|
| + return this._configurableColumnIDs;
|
| + },
|
| +
|
| + /**
|
| + * @param {!Event} event
|
| + */
|
| + _contextMenu: function(event)
|
| + {
|
| + var contextMenu = new WebInspector.ContextMenu(event);
|
| +
|
| + if (this._detailedMode && event.target.isSelfOrDescendant(this._dataGrid.headerTableBody)) {
|
| + var columnsVisibility = this._coulmnsVisibilitySetting.get();
|
| + var columnIDs = this._getConfigurableColumnIDs();
|
| + var columnTitles = WebInspector.NetworkLogView._columnTitles;
|
| + for (var i = 0; i < columnIDs.length; ++i) {
|
| + var columnIdentifier = columnIDs[i];
|
| + contextMenu.appendCheckboxItem(columnTitles[columnIdentifier], this._toggleColumnVisibility.bind(this, columnIdentifier), !!columnsVisibility[columnIdentifier]);
|
| + }
|
| + contextMenu.show();
|
| + return;
|
| + }
|
| +
|
| + var gridNode = this._dataGrid.dataGridNodeFromNode(event.target);
|
| + var request = gridNode && gridNode.request();
|
| +
|
| + /**
|
| + * @param {string} url
|
| + */
|
| + function openResourceInNewTab(url)
|
| + {
|
| + InspectorFrontendHost.openInNewTab(url);
|
| + }
|
| +
|
| + if (request) {
|
| + contextMenu.appendItem(WebInspector.openLinkExternallyLabel(), openResourceInNewTab.bind(null, request.url));
|
| + contextMenu.appendSeparator();
|
| + contextMenu.appendItem(WebInspector.copyLinkAddressLabel(), this._copyLocation.bind(this, request));
|
| + if (request.requestHeadersText())
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Copy request headers" : "Copy Request Headers"), this._copyRequestHeaders.bind(this, request));
|
| + if (request.responseHeadersText)
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Copy response headers" : "Copy Response Headers"), this._copyResponseHeaders.bind(this, request));
|
| + if (request.finished)
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Copy response" : "Copy Response"), this._copyResponse.bind(this, request));
|
| + contextMenu.appendItem(WebInspector.UIString("Copy as cURL"), this._copyCurlCommand.bind(this, request));
|
| + }
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Copy all as HAR" : "Copy All as HAR"), this._copyAll.bind(this));
|
| +
|
| + contextMenu.appendSeparator();
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Save as HAR with content" : "Save as HAR with Content"), this._exportAll.bind(this));
|
| +
|
| + contextMenu.appendSeparator();
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Clear browser cache" : "Clear Browser Cache"), this._clearBrowserCache.bind(this));
|
| + contextMenu.appendItem(WebInspector.UIString(WebInspector.useLowerCaseMenuTitles() ? "Clear browser cookies" : "Clear Browser Cookies"), this._clearBrowserCookies.bind(this));
|
| +
|
| + if (request && request.resourceType() === WebInspector.resourceTypes.XHR) {
|
| + contextMenu.appendSeparator();
|
| + contextMenu.appendItem(WebInspector.UIString("Replay XHR"), request.replayXHR.bind(request));
|
| + contextMenu.appendSeparator();
|
| + }
|
| +
|
| + contextMenu.show();
|
| + },
|
| +
|
| + _harRequests: function()
|
| + {
|
| + var requests = this._nodesByRequestId.valuesArray().map(function(node) { return node.request(); });
|
| + var httpRequests = requests.filter(WebInspector.NetworkLogView.HTTPRequestsFilter);
|
| + httpRequests = httpRequests.filter(WebInspector.NetworkLogView.FinishedRequestsFilter);
|
| + return httpRequests.filter(WebInspector.NetworkLogView.NonDevToolsRequestsFilter);
|
| + },
|
| +
|
| + _copyAll: function()
|
| + {
|
| + var harArchive = {
|
| + log: (new WebInspector.HARLog(this._harRequests())).build()
|
| + };
|
| + InspectorFrontendHost.copyText(JSON.stringify(harArchive, null, 2));
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _copyLocation: function(request)
|
| + {
|
| + InspectorFrontendHost.copyText(request.url);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _copyRequestHeaders: function(request)
|
| + {
|
| + InspectorFrontendHost.copyText(request.requestHeadersText());
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _copyResponse: function(request)
|
| + {
|
| + /**
|
| + * @param {?string} content
|
| + */
|
| + function callback(content)
|
| + {
|
| + if (request.contentEncoded)
|
| + content = request.asDataURL();
|
| + InspectorFrontendHost.copyText(content || "");
|
| + }
|
| + request.requestContent(callback);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _copyResponseHeaders: function(request)
|
| + {
|
| + InspectorFrontendHost.copyText(request.responseHeadersText);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + _copyCurlCommand: function(request)
|
| + {
|
| + InspectorFrontendHost.copyText(this._generateCurlCommand(request));
|
| + },
|
| +
|
| + _exportAll: function()
|
| + {
|
| + var filename = WebInspector.targetManager.inspectedPageDomain() + ".har";
|
| + var stream = new WebInspector.FileOutputStream();
|
| + stream.open(filename, openCallback.bind(this));
|
| +
|
| + /**
|
| + * @param {boolean} accepted
|
| + * @this {WebInspector.NetworkLogView}
|
| + */
|
| + function openCallback(accepted)
|
| + {
|
| + if (!accepted)
|
| + return;
|
| + var progressIndicator = new WebInspector.ProgressIndicator();
|
| + this._progressBarContainer.appendChild(progressIndicator.element);
|
| + var harWriter = new WebInspector.HARWriter();
|
| + harWriter.write(stream, this._harRequests(), progressIndicator);
|
| + }
|
| + },
|
| +
|
| + _clearBrowserCache: function()
|
| + {
|
| + if (confirm(WebInspector.UIString("Are you sure you want to clear browser cache?")))
|
| + NetworkAgent.clearBrowserCache();
|
| + },
|
| +
|
| + _clearBrowserCookies: function()
|
| + {
|
| + if (confirm(WebInspector.UIString("Are you sure you want to clear browser cookies?")))
|
| + NetworkAgent.clearBrowserCookies();
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| + _matchRequest: function(request)
|
| + {
|
| + var re = this._searchRegExp;
|
| + if (!re)
|
| + return false;
|
| + return re.test(request.name()) || re.test(request.path());
|
| + },
|
| +
|
| + _clearSearchMatchedList: function()
|
| + {
|
| + this._matchedRequestCount = -1;
|
| + this._currentMatchedRequestNode = null;
|
| + this._removeAllHighlights();
|
| + },
|
| +
|
| + _removeAllHighlights: function()
|
| + {
|
| + this._removeAllNodeHighlights();
|
| + for (var i = 0; i < this._highlightedSubstringChanges.length; ++i)
|
| + WebInspector.revertDomChanges(this._highlightedSubstringChanges[i]);
|
| + this._highlightedSubstringChanges = [];
|
| + },
|
| +
|
| + /**
|
| + * @param {number} n
|
| + * @param {boolean} reveal
|
| + */
|
| + _highlightNthMatchedRequestForSearch: function(n, reveal)
|
| + {
|
| + this._removeAllHighlights();
|
| +
|
| + /** @type {!Array.<!WebInspector.NetworkDataGridNode>} */
|
| + var nodes = this._dataGrid.rootNode().children;
|
| + var matchCount = 0;
|
| + var node = null;
|
| + for (var i = 0; i < nodes.length; ++i) {
|
| + if (nodes[i]._isMatchingSearchQuery) {
|
| + if (matchCount === n) {
|
| + node = nodes[i];
|
| + break;
|
| + }
|
| + matchCount++;
|
| + }
|
| + }
|
| + if (!node) {
|
| + this._currentMatchedRequestNode = null;
|
| + return;
|
| + }
|
| +
|
| + var request = node.request();
|
| + var regExp = this._searchRegExp;
|
| + var nameMatched = request.name().match(regExp);
|
| + var pathMatched = request.path().match(regExp);
|
| + if (!nameMatched && pathMatched && !this._largerRequestsButton.toggled())
|
| + this._toggleLargerRequests();
|
| + if (reveal)
|
| + WebInspector.Revealer.reveal(request);
|
| + var highlightedSubstringChanges = node.highlightMatchedSubstring(regExp);
|
| + this._highlightedSubstringChanges.push(highlightedSubstringChanges);
|
| +
|
| + this._currentMatchedRequestNode = node;
|
| + this._currentMatchedRequestIndex = n;
|
| + this.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.SearchIndexUpdated, n);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.SearchableView.SearchConfig} searchConfig
|
| + * @param {boolean} shouldJump
|
| + * @param {boolean=} jumpBackwards
|
| + */
|
| + performSearch: function(searchConfig, shouldJump, jumpBackwards)
|
| + {
|
| + var query = searchConfig.query;
|
| + var currentMatchedRequestNode = this._currentMatchedRequestNode;
|
| + this._clearSearchMatchedList();
|
| + this._searchRegExp = createPlainTextSearchRegex(query, "i");
|
| +
|
| + /** @type {!Array.<!WebInspector.NetworkDataGridNode>} */
|
| + var nodes = this._dataGrid.rootNode().children;
|
| + for (var i = 0; i < nodes.length; ++i)
|
| + nodes[i]._isMatchingSearchQuery = this._matchRequest(nodes[i].request());
|
| + var newMatchedRequestIndex = this._updateMatchCountAndFindMatchIndex(currentMatchedRequestNode);
|
| + if (!newMatchedRequestIndex && jumpBackwards)
|
| + newMatchedRequestIndex = this._matchedRequestCount - 1;
|
| + this._highlightNthMatchedRequestForSearch(newMatchedRequestIndex, shouldJump);
|
| + },
|
| +
|
| + /**
|
| + * @return {boolean}
|
| + */
|
| + supportsCaseSensitiveSearch: function()
|
| + {
|
| + return false;
|
| + },
|
| +
|
| + /**
|
| + * @return {boolean}
|
| + */
|
| + supportsRegexSearch: function()
|
| + {
|
| + return false;
|
| + },
|
| +
|
| + /**
|
| + * @param {?WebInspector.NetworkDataGridNode} node
|
| + * @return {number}
|
| + */
|
| + _updateMatchCountAndFindMatchIndex: function(node)
|
| + {
|
| + /** @type {!Array.<!WebInspector.NetworkDataGridNode>} */
|
| + var nodes = this._dataGrid.rootNode().children;
|
| + var matchCount = 0;
|
| + var matchIndex = 0;
|
| + for (var i = 0; i < nodes.length; ++i) {
|
| + if (!nodes[i]._isMatchingSearchQuery)
|
| + continue;
|
| + if (node === nodes[i])
|
| + matchIndex = matchCount;
|
| + matchCount++;
|
| + }
|
| + if (this._matchedRequestCount !== matchCount) {
|
| + this._matchedRequestCount = matchCount;
|
| + this.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.SearchCountUpdated, matchCount);
|
| + }
|
| + return matchIndex;
|
| + },
|
| +
|
| + /**
|
| + * @param {number} index
|
| + * @return {number}
|
| + */
|
| + _normalizeSearchResultIndex: function(index)
|
| + {
|
| + return (index + this._matchedRequestCount) % this._matchedRequestCount;
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkDataGridNode} node
|
| + * @return {boolean}
|
| + */
|
| + _applyFilter: function(node)
|
| + {
|
| + var request = node.request();
|
| + var resourceType = request.resourceType();
|
| + if (resourceType === WebInspector.resourceTypes.TextTrack)
|
| + resourceType = WebInspector.resourceTypes.Other;
|
| + if (!this._resourceTypeFilterUI.accept(resourceType.name()))
|
| + return false;
|
| + if (this._dataURLFilterUI.checked() && request.parsedURL.isDataURL())
|
| + return false;
|
| + for (var i = 0; i < this._filters.length; ++i) {
|
| + if (!this._filters[i](request))
|
| + return false;
|
| + }
|
| + return true;
|
| + },
|
| +
|
| + /**
|
| + * @param {string} query
|
| + */
|
| + _parseFilterQuery: function(query)
|
| + {
|
| + var parsedQuery = this._suggestionBuilder.parseQuery(query);
|
| + this._filters = parsedQuery.text.map(this._createTextFilter);
|
| + var filters = parsedQuery.filters;
|
| + var n = parsedQuery.filters.length;
|
| + for (var i = 0; i < n; ++i) {
|
| + var filter = parsedQuery.filters[i];
|
| + var filterType = /** @type {!WebInspector.NetworkLogView.FilterType} */ (filter.type);
|
| + this._filters.push(this._createFilter(filterType, filter.data, filter.negative));
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {string} text
|
| + * @return {!WebInspector.NetworkLogView.Filter}
|
| + */
|
| + _createTextFilter: function(text)
|
| + {
|
| + var regexp = new RegExp(text.escapeForRegExp(), "i");
|
| + return WebInspector.NetworkLogView._requestNameOrPathFilter.bind(null, regexp);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkLogView.FilterType} type
|
| + * @param {string} value
|
| + * @param {boolean} negative
|
| + * @return {!WebInspector.NetworkLogView.Filter}
|
| + */
|
| + _createFilter: function(type, value, negative)
|
| + {
|
| + var filter = this._createSpecialFilter(type, value);
|
| + if (!filter)
|
| + return this._createTextFilter((negative ? "-" : "") + type + ":" + value);
|
| + if (negative)
|
| + return WebInspector.NetworkLogView._negativeFilter.bind(null, filter);
|
| + return filter;
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkLogView.FilterType} type
|
| + * @param {string} value
|
| + * @return {?WebInspector.NetworkLogView.Filter}
|
| + */
|
| + _createSpecialFilter: function(type, value)
|
| + {
|
| + switch (type) {
|
| + case WebInspector.NetworkLogView.FilterType.Domain:
|
| + return WebInspector.NetworkLogView._requestDomainFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.HasResponseHeader:
|
| + return WebInspector.NetworkLogView._requestResponseHeaderFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.Is:
|
| + if (value.toLowerCase() === WebInspector.NetworkLogView.IsFilterType.Running)
|
| + return WebInspector.NetworkLogView._runningRequestFilter;
|
| + break;
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.Method:
|
| + return WebInspector.NetworkLogView._requestMethodFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.MimeType:
|
| + return WebInspector.NetworkLogView._requestMimeTypeFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.Scheme:
|
| + return WebInspector.NetworkLogView._requestSchemeFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.SetCookieDomain:
|
| + return WebInspector.NetworkLogView._requestSetCookieDomainFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.SetCookieName:
|
| + return WebInspector.NetworkLogView._requestSetCookieNameFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.SetCookieValue:
|
| + return WebInspector.NetworkLogView._requestSetCookieValueFilter.bind(null, value);
|
| +
|
| + case WebInspector.NetworkLogView.FilterType.StatusCode:
|
| + return WebInspector.NetworkLogView._statusCodeFilter.bind(null, value);
|
| + }
|
| + return null;
|
| + },
|
| +
|
| + _filterRequests: function()
|
| + {
|
| + this._removeAllHighlights();
|
| + this._invalidateAllItems();
|
| + this.refresh();
|
| + },
|
| +
|
| + jumpToPreviousSearchResult: function()
|
| + {
|
| + if (!this._matchedRequestCount)
|
| + return;
|
| + var index = this._normalizeSearchResultIndex(this._currentMatchedRequestIndex - 1);
|
| + this._highlightNthMatchedRequestForSearch(index, true);
|
| + },
|
| +
|
| + jumpToNextSearchResult: function()
|
| + {
|
| + if (!this._matchedRequestCount)
|
| + return;
|
| + var index = this._normalizeSearchResultIndex(this._currentMatchedRequestIndex + 1);
|
| + this._highlightNthMatchedRequestForSearch(index, true);
|
| + },
|
| +
|
| + searchCanceled: function()
|
| + {
|
| + delete this._searchRegExp;
|
| + this._clearSearchMatchedList();
|
| + this.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.SearchCountUpdated, 0);
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| + revealAndHighlightRequest: function(request)
|
| + {
|
| + this._removeAllNodeHighlights();
|
| +
|
| + var node = this._nodesByRequestId.get(request.requestId);
|
| + if (node) {
|
| + node.reveal();
|
| + this._highlightNode(node);
|
| + }
|
| + },
|
| +
|
| + _removeAllNodeHighlights: function()
|
| + {
|
| + if (this._highlightedNode) {
|
| + this._highlightedNode.element().classList.remove("highlighted-row");
|
| + delete this._highlightedNode;
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkDataGridNode} node
|
| + */
|
| + _highlightNode: function(node)
|
| + {
|
| + WebInspector.runCSSAnimationOnce(node.element(), "highlighted-row");
|
| + this._highlightedNode = node;
|
| + },
|
| +
|
| + /**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {string}
|
| + */
|
| + _generateCurlCommand: function(request)
|
| + {
|
| + var command = ["curl"];
|
| + // These headers are derived from URL (except "version") and would be added by cURL anyway.
|
| + var ignoredHeaders = {"host": 1, "method": 1, "path": 1, "scheme": 1, "version": 1};
|
| +
|
| + function escapeStringWin(str)
|
| + {
|
| + /* Replace quote by double quote (but not by \") because it is
|
| + recognized by both cmd.exe and MS Crt arguments parser.
|
| +
|
| + Replace % by "%" because it could be expanded to an environment
|
| + variable value. So %% becomes "%""%". Even if an env variable ""
|
| + (2 doublequotes) is declared, the cmd.exe will not
|
| + substitute it with its value.
|
| +
|
| + Replace each backslash with double backslash to make sure
|
| + MS Crt arguments parser won't collapse them.
|
| +
|
| + Replace new line outside of quotes since cmd.exe doesn't let
|
| + to do it inside.
|
| + */
|
| + return "\"" + str.replace(/"/g, "\"\"")
|
| + .replace(/%/g, "\"%\"")
|
| + .replace(/\\/g, "\\\\")
|
| + .replace(/[\r\n]+/g, "\"^$&\"") + "\"";
|
| + }
|
| +
|
| + function escapeStringPosix(str)
|
| + {
|
| + function escapeCharacter(x)
|
| + {
|
| + var code = x.charCodeAt(0);
|
| + if (code < 256) {
|
| + // Add leading zero when needed to not care about the next character.
|
| + return code < 16 ? "\\x0" + code.toString(16) : "\\x" + code.toString(16);
|
| + }
|
| + code = code.toString(16);
|
| + return "\\u" + ("0000" + code).substr(code.length, 4);
|
| + }
|
| +
|
| + if (/[^\x20-\x7E]|\'/.test(str)) {
|
| + // Use ANSI-C quoting syntax.
|
| + return "$\'" + str.replace(/\\/g, "\\\\")
|
| + .replace(/\'/g, "\\\'")
|
| + .replace(/\n/g, "\\n")
|
| + .replace(/\r/g, "\\r")
|
| + .replace(/[^\x20-\x7E]/g, escapeCharacter) + "'";
|
| + } else {
|
| + // Use single quote syntax.
|
| + return "'" + str + "'";
|
| + }
|
| + }
|
| +
|
| + // cURL command expected to run on the same platform that DevTools run
|
| + // (it may be different from the inspected page platform).
|
| + var escapeString = WebInspector.isWin() ? escapeStringWin : escapeStringPosix;
|
| +
|
| + command.push(escapeString(request.url).replace(/[[{}\]]/g, "\\$&"));
|
| +
|
| + var inferredMethod = "GET";
|
| + var data = [];
|
| + var requestContentType = request.requestContentType();
|
| + if (requestContentType && requestContentType.startsWith("application/x-www-form-urlencoded") && request.requestFormData) {
|
| + data.push("--data");
|
| + data.push(escapeString(request.requestFormData));
|
| + ignoredHeaders["content-length"] = true;
|
| + inferredMethod = "POST";
|
| + } else if (request.requestFormData) {
|
| + data.push("--data-binary");
|
| + data.push(escapeString(request.requestFormData));
|
| + ignoredHeaders["content-length"] = true;
|
| + inferredMethod = "POST";
|
| + }
|
| +
|
| + if (request.requestMethod !== inferredMethod) {
|
| + command.push("-X");
|
| + command.push(request.requestMethod);
|
| + }
|
| +
|
| + var requestHeaders = request.requestHeaders();
|
| + for (var i = 0; i < requestHeaders.length; i++) {
|
| + var header = requestHeaders[i];
|
| + var name = header.name.replace(/^:/, ""); // Translate SPDY v3 headers to HTTP headers.
|
| + if (name.toLowerCase() in ignoredHeaders)
|
| + continue;
|
| + command.push("-H");
|
| + command.push(escapeString(name + ": " + header.value));
|
| + }
|
| + command = command.concat(data);
|
| + command.push("--compressed");
|
| + return command.join(" ");
|
| + },
|
| +
|
| + __proto__: WebInspector.VBox.prototype
|
| +}
|
| +
|
| +/** @typedef {function(!WebInspector.NetworkRequest): boolean} */
|
| +WebInspector.NetworkLogView.Filter;
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkLogView.Filter} filter
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._negativeFilter = function(filter, request)
|
| +{
|
| + return !filter(request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!RegExp} regex
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestNameOrPathFilter = function(regex, request)
|
| +{
|
| + return regex.test(request.name()) || regex.test(request.path());
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestDomainFilter = function(value, request)
|
| +{
|
| + return request.domain === value;
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._runningRequestFilter = function(request)
|
| +{
|
| + return !request.finished;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestResponseHeaderFilter = function(value, request)
|
| +{
|
| + return request.responseHeaderValue(value) !== undefined;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestMethodFilter = function(value, request)
|
| +{
|
| + return request.requestMethod === value;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestMimeTypeFilter = function(value, request)
|
| +{
|
| + return request.mimeType === value;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestSchemeFilter = function(value, request)
|
| +{
|
| + return request.scheme === value;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestSetCookieDomainFilter = function(value, request)
|
| +{
|
| + var cookies = request.responseCookies;
|
| + for (var i = 0, l = cookies ? cookies.length : 0; i < l; ++i) {
|
| + if (cookies[i].domain() === value)
|
| + return false;
|
| + }
|
| + return false;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestSetCookieNameFilter = function(value, request)
|
| +{
|
| + var cookies = request.responseCookies;
|
| + for (var i = 0, l = cookies ? cookies.length : 0; i < l; ++i) {
|
| + if (cookies[i].name() === value)
|
| + return false;
|
| + }
|
| + return false;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._requestSetCookieValueFilter = function(value, request)
|
| +{
|
| + var cookies = request.responseCookies;
|
| + for (var i = 0, l = cookies ? cookies.length : 0; i < l; ++i) {
|
| + if (cookies[i].value() === value)
|
| + return false;
|
| + }
|
| + return false;
|
| +}
|
| +
|
| +/**
|
| + * @param {string} value
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView._statusCodeFilter = function(value, request)
|
| +{
|
| + return ("" + request.statusCode) === value;
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView.HTTPRequestsFilter = function(request)
|
| +{
|
| + return request.parsedURL.isValid && (request.scheme in WebInspector.NetworkLogView.HTTPSchemas);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView.NonDevToolsRequestsFilter = function(request)
|
| +{
|
| + return !WebInspector.NetworkManager.hasDevToolsRequestHeader(request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + * @return {boolean}
|
| + */
|
| +WebInspector.NetworkLogView.FinishedRequestsFilter = function(request)
|
| +{
|
| + return request.finished;
|
| +}
|
| +
|
| +WebInspector.NetworkLogView.EventTypes = {
|
| + ViewCleared: "ViewCleared",
|
| + RowSizeChanged: "RowSizeChanged",
|
| + RequestSelected: "RequestSelected",
|
| + SearchCountUpdated: "SearchCountUpdated",
|
| + SearchIndexUpdated: "SearchIndexUpdated"
|
| +};
|
| +
|
| +/**
|
| + * @constructor
|
| * @implements {WebInspector.ContextMenu.Provider}
|
| * @implements {WebInspector.Searchable}
|
| * @extends {WebInspector.Panel}
|
| @@ -242,6 +2111,7 @@
|
|
|
| this._networkLogView.switchViewMode(true);
|
| this._networkLogView.setAllowPopover(true);
|
| + this._networkLogView._allowRequestSelection = false;
|
| },
|
|
|
| _toggleViewingRequestMode: function()
|
| @@ -253,6 +2123,7 @@
|
| this.element.classList.add("viewing-resource");
|
| this._splitView.showBoth();
|
| this._networkLogView.setAllowPopover(false);
|
| + this._networkLogView._allowRequestSelection = true;
|
| this._networkLogView.switchViewMode(false);
|
| },
|
|
|
| @@ -727,6 +2598,589 @@
|
| },
|
|
|
| __proto__: WebInspector.NetworkTimeCalculator.prototype
|
| +}
|
| +
|
| +/**
|
| + * @constructor
|
| + * @extends {WebInspector.SortableDataGridNode}
|
| + * @param {!WebInspector.NetworkLogView} parentView
|
| + * @param {!WebInspector.NetworkRequest} request
|
| + */
|
| +WebInspector.NetworkDataGridNode = function(parentView, request)
|
| +{
|
| + WebInspector.SortableDataGridNode.call(this, {});
|
| + this._parentView = parentView;
|
| + this._request = request;
|
| + this._linkifier = new WebInspector.Linkifier();
|
| + this._isFilteredOut = true;
|
| + this._isMatchingSearchQuery = false;
|
| + this._staleGraph = true;
|
| +}
|
| +
|
| +WebInspector.NetworkDataGridNode._hoveredRowSymbol = Symbol("hoveredRow");
|
| +
|
| +WebInspector.NetworkDataGridNode.prototype = {
|
| + /**
|
| + * @return {!WebInspector.NetworkRequest}
|
| + */
|
| + request: function()
|
| + {
|
| + return this._request;
|
| + },
|
| +
|
| + /**
|
| + * @override
|
| + * @return {number}
|
| + */
|
| + nodeSelfHeight: function()
|
| + {
|
| + return this._parentView.rowHeight();
|
| + },
|
| +
|
| + /** override */
|
| + createCells: function()
|
| + {
|
| + this._nameCell = null;
|
| + this._timelineCell = null;
|
| + this._initiatorCell = null;
|
| +
|
| + this._element.classList.toggle("network-error-row", this._isFailed());
|
| + WebInspector.SortableDataGridNode.prototype.createCells.call(this);
|
| +
|
| + this._updateGraph();
|
| + },
|
| +
|
| + /**
|
| + * @override
|
| + * @param {string} columnIdentifier
|
| + * @return {!Element}
|
| + */
|
| + createCell: function(columnIdentifier)
|
| + {
|
| + var cell = this.createTD(columnIdentifier);
|
| + switch (columnIdentifier) {
|
| + case "name": this._renderNameCell(cell); break;
|
| + case "timeline": this._createTimelineBar(cell); break;
|
| + case "method": cell.setTextAndTitle(this._request.requestMethod); break;
|
| + case "status": this._renderStatusCell(cell); break;
|
| + case "scheme": cell.setTextAndTitle(this._request.scheme); break;
|
| + case "domain": cell.setTextAndTitle(this._request.domain); break;
|
| + case "remoteAddress": cell.setTextAndTitle(this._request.remoteAddress()); break;
|
| + case "cookies": cell.setTextAndTitle(this._arrayLength(this._request.requestCookies)); break;
|
| + case "setCookies": cell.setTextAndTitle(this._arrayLength(this._request.responseCookies)); break;
|
| + case "connectionId": cell.setTextAndTitle(this._request.connectionId); break;
|
| + case "type": cell.setTextAndTitle(this._request.mimeType || this._request.requestContentType() || ""); break;
|
| + case "initiator": this._renderInitiatorCell(cell); break;
|
| + case "size": this._renderSizeCell(cell); break;
|
| + case "time": this._renderTimeCell(cell); break;
|
| + default: cell.setTextAndTitle(this._request.responseHeaderValue(columnIdentifier) || ""); break;
|
| + }
|
| +
|
| + return cell;
|
| + },
|
| +
|
| + /**
|
| + * @param {?Array} array
|
| + * @return {string}
|
| + */
|
| + _arrayLength: function(array)
|
| + {
|
| + return array ? "" + array.length : "";
|
| + },
|
| +
|
| + /**
|
| + * @override
|
| + * @protected
|
| + */
|
| + willAttach: function()
|
| + {
|
| + if (this._staleGraph)
|
| + this._updateGraph();
|
| + if (this._initiatorCell && this._request.initiatorInfo().type === WebInspector.NetworkRequest.InitiatorType.Script)
|
| + this._initiatorCell.insertBefore(this._linkifiedInitiatorAnchor, this._initiatorCell.firstChild);
|
| + },
|
| +
|
| + wasDetached: function()
|
| + {
|
| + if (this._linkifiedInitiatorAnchor)
|
| + this._linkifiedInitiatorAnchor.remove();
|
| + },
|
| +
|
| + dispose: function()
|
| + {
|
| + this._linkifier.reset();
|
| + },
|
| +
|
| + _onClick: function()
|
| + {
|
| + if (!this._parentView.allowRequestSelection())
|
| + this.select();
|
| + },
|
| +
|
| + select: function()
|
| + {
|
| + this._parentView.dispatchEventToListeners(WebInspector.NetworkLogView.EventTypes.RequestSelected, this._request);
|
| + WebInspector.SortableDataGridNode.prototype.select.apply(this, arguments);
|
| +
|
| + WebInspector.notifications.dispatchEventToListeners(WebInspector.UserMetrics.UserAction, {
|
| + action: WebInspector.UserMetrics.UserActionNames.NetworkRequestSelected,
|
| + url: this._request.url
|
| + });
|
| + },
|
| +
|
| + /**
|
| + * @param {!RegExp=} regexp
|
| + * @return {!Array.<!Object>}
|
| + */
|
| + highlightMatchedSubstring: function(regexp)
|
| + {
|
| + // Ensure element is created.
|
| + this.element();
|
| + var domChanges = [];
|
| + var matchInfo = this._nameCell.textContent.match(regexp);
|
| + if (matchInfo)
|
| + WebInspector.highlightSearchResult(this._nameCell, matchInfo.index, matchInfo[0].length, domChanges);
|
| + return domChanges;
|
| + },
|
| +
|
| + _openInNewTab: function()
|
| + {
|
| + InspectorFrontendHost.openInNewTab(this._request.url);
|
| + },
|
| +
|
| + get selectable()
|
| + {
|
| + return this._parentView.allowRequestSelection();
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _createTimelineBar: function(cell)
|
| + {
|
| + cell = cell.createChild("div");
|
| + this._timelineCell = cell;
|
| +
|
| + cell.className = "network-graph-side";
|
| +
|
| + this._barAreaElement = cell.createChild("div", "network-graph-bar-area");
|
| + this._barAreaElement.request = this._request;
|
| +
|
| + var type = this._request.resourceType().name();
|
| + var cached = this._request.cached();
|
| +
|
| + this._barLeftElement = this._barAreaElement.createChild("div", "network-graph-bar");
|
| + this._barLeftElement.classList.add(type, "waiting");
|
| + this._barLeftElement.classList.toggle("cached", cached);
|
| +
|
| + this._barRightElement = this._barAreaElement.createChild("div", "network-graph-bar");
|
| + this._barRightElement.classList.add(type);
|
| + this._barRightElement.classList.toggle("cached", cached);
|
| +
|
| + this._labelLeftElement = this._barAreaElement.createChild("div", "network-graph-label");
|
| + this._labelLeftElement.classList.add("waiting");
|
| +
|
| + this._labelRightElement = this._barAreaElement.createChild("div", "network-graph-label");
|
| +
|
| + cell.addEventListener("mouseover", this._onMouseOver.bind(this), false);
|
| + },
|
| +
|
| + /**
|
| + * @param {!Event} event
|
| + */
|
| + _onMouseOver: function(event)
|
| + {
|
| + this._refreshLabelPositions();
|
| + this._parentView[WebInspector.NetworkDataGridNode._hoveredRowSymbol] = this;
|
| + },
|
| +
|
| + /**
|
| + * @return {boolean}
|
| + */
|
| + _isFailed: function()
|
| + {
|
| + return (this._request.failed && !this._request.statusCode) || (this._request.statusCode >= 400);
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _renderNameCell: function(cell)
|
| + {
|
| + this._nameCell = cell;
|
| + cell.addEventListener("click", this._onClick.bind(this), false);
|
| + cell.addEventListener("dblclick", this._openInNewTab.bind(this), false);
|
| + var iconElement;
|
| + if (this._request.resourceType() === WebInspector.resourceTypes.Image) {
|
| + var previewImage = createElementWithClass("img", "image-network-icon-preview");
|
| + this._request.populateImageSource(previewImage);
|
| +
|
| + iconElement = createElementWithClass("div", "icon");
|
| + iconElement.appendChild(previewImage);
|
| + } else {
|
| + iconElement = createElementWithClass("img", "icon");
|
| + }
|
| + iconElement.classList.add(this._request.resourceType().name());
|
| +
|
| + cell.appendChild(iconElement);
|
| + cell.createTextChild(this._request.name());
|
| + this._appendSubtitle(cell, this._request.path());
|
| + cell.title = this._request.url;
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _renderStatusCell: function(cell)
|
| + {
|
| + cell.classList.toggle("network-dim-cell", !this._isFailed() && (this._request.cached() || !this._request.statusCode));
|
| +
|
| + if (this._request.failed && !this._request.canceled) {
|
| + var failText = WebInspector.UIString("(failed)");
|
| + if (this._request.localizedFailDescription) {
|
| + cell.createTextChild(failText);
|
| + this._appendSubtitle(cell, this._request.localizedFailDescription);
|
| + cell.title = failText + " " + this._request.localizedFailDescription;
|
| + } else
|
| + cell.setTextAndTitle(failText);
|
| + } else if (this._request.statusCode) {
|
| + cell.createTextChild("" + this._request.statusCode);
|
| + this._appendSubtitle(cell, this._request.statusText);
|
| + cell.title = this._request.statusCode + " " + this._request.statusText;
|
| + } else if (this._request.parsedURL.isDataURL()) {
|
| + cell.setTextAndTitle(WebInspector.UIString("(data)"));
|
| + } else if (this._request.canceled) {
|
| + cell.setTextAndTitle(WebInspector.UIString("(canceled)"));
|
| + } else if (this._request.finished) {
|
| + cell.setTextAndTitle(WebInspector.UIString("Finished"));
|
| + } else {
|
| + cell.setTextAndTitle(WebInspector.UIString("(pending)"));
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _renderInitiatorCell: function(cell)
|
| + {
|
| + this._initiatorCell = cell;
|
| + var request = this._request;
|
| + var initiator = request.initiatorInfo();
|
| +
|
| + switch (initiator.type) {
|
| + case WebInspector.NetworkRequest.InitiatorType.Parser:
|
| + cell.title = initiator.url + ":" + initiator.lineNumber;
|
| + cell.appendChild(WebInspector.linkifyResourceAsNode(initiator.url, initiator.lineNumber - 1));
|
| + this._appendSubtitle(cell, WebInspector.UIString("Parser"));
|
| + break;
|
| +
|
| + case WebInspector.NetworkRequest.InitiatorType.Redirect:
|
| + cell.title = initiator.url;
|
| + console.assert(request.redirectSource);
|
| + var redirectSource = /** @type {!WebInspector.NetworkRequest} */ (request.redirectSource);
|
| + cell.appendChild(WebInspector.linkifyRequestAsNode(redirectSource));
|
| + this._appendSubtitle(cell, WebInspector.UIString("Redirect"));
|
| + break;
|
| +
|
| + case WebInspector.NetworkRequest.InitiatorType.Script:
|
| + if (!this._linkifiedInitiatorAnchor) {
|
| + this._linkifiedInitiatorAnchor = this._linkifier.linkifyScriptLocation(request.target(), null, initiator.url, initiator.lineNumber - 1, initiator.columnNumber - 1);
|
| + this._linkifiedInitiatorAnchor.title = "";
|
| + }
|
| + cell.appendChild(this._linkifiedInitiatorAnchor);
|
| + this._appendSubtitle(cell, WebInspector.UIString("Script"));
|
| + cell.classList.add("network-script-initiated");
|
| + cell.request = request;
|
| + break;
|
| +
|
| + default:
|
| + cell.title = "";
|
| + cell.classList.add("network-dim-cell");
|
| + cell.setTextAndTitle(WebInspector.UIString("Other"));
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _renderSizeCell: function(cell)
|
| + {
|
| + if (this._request.fetchedViaServiceWorker) {
|
| + cell.setTextAndTitle(WebInspector.UIString("(from ServiceWorker)"));
|
| + cell.classList.add("network-dim-cell");
|
| + } else if (this._request.cached()) {
|
| + cell.setTextAndTitle(WebInspector.UIString("(from cache)"));
|
| + cell.classList.add("network-dim-cell");
|
| + } else {
|
| + var resourceSize = Number.bytesToString(this._request.resourceSize);
|
| + var transferSize = Number.bytesToString(this._request.transferSize);
|
| + cell.setTextAndTitle(transferSize);
|
| + this._appendSubtitle(cell, resourceSize);
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cell
|
| + */
|
| + _renderTimeCell: function(cell)
|
| + {
|
| + if (this._request.duration > 0) {
|
| + cell.setTextAndTitle(Number.secondsToString(this._request.duration));
|
| + this._appendSubtitle(cell, Number.secondsToString(this._request.latency));
|
| + } else {
|
| + cell.classList.add("network-dim-cell");
|
| + cell.setTextAndTitle(WebInspector.UIString("Pending"));
|
| + }
|
| + },
|
| +
|
| + /**
|
| + * @param {!Element} cellElement
|
| + * @param {string} subtitleText
|
| + */
|
| + _appendSubtitle: function(cellElement, subtitleText)
|
| + {
|
| + var subtitleElement = createElement("div");
|
| + subtitleElement.className = "network-cell-subtitle";
|
| + subtitleElement.textContent = subtitleText;
|
| + cellElement.appendChild(subtitleElement);
|
| + },
|
| +
|
| + refreshGraph: function()
|
| + {
|
| + if (!this._timelineCell)
|
| + return;
|
| + this._staleGraph = true;
|
| + if (this.attached())
|
| + this.dataGrid.scheduleUpdate();
|
| + },
|
| +
|
| + _updateGraph: function()
|
| + {
|
| + this._staleGraph = false;
|
| + if (!this._timelineCell)
|
| + return;
|
| +
|
| + var calculator = this._parentView.calculator();
|
| + var percentages = calculator.computeBarGraphPercentages(this._request);
|
| + this._percentages = percentages;
|
| +
|
| + this._barAreaElement.classList.remove("hidden");
|
| +
|
| + this._barLeftElement.style.setProperty("left", percentages.start + "%");
|
| + this._barLeftElement.style.setProperty("right", (100 - percentages.middle) + "%");
|
| +
|
| + this._barRightElement.style.setProperty("left", percentages.middle + "%");
|
| + this._barRightElement.style.setProperty("right", (100 - percentages.end) + "%");
|
| +
|
| + var labels = calculator.computeBarGraphLabels(this._request);
|
| + this._labelLeftElement.textContent = labels.left;
|
| + this._labelRightElement.textContent = labels.right;
|
| +
|
| + var tooltip = (labels.tooltip || "");
|
| + this._barLeftElement.title = tooltip;
|
| + this._labelLeftElement.title = tooltip;
|
| + this._labelRightElement.title = tooltip;
|
| + this._barRightElement.title = tooltip;
|
| +
|
| + if (this._parentView[WebInspector.NetworkDataGridNode._hoveredRowSymbol] === this)
|
| + this._refreshLabelPositions();
|
| + },
|
| +
|
| + _refreshLabelPositions: function()
|
| + {
|
| + if (!this._percentages)
|
| + return;
|
| + this._labelLeftElement.style.removeProperty("left");
|
| + this._labelLeftElement.style.removeProperty("right");
|
| + this._labelLeftElement.classList.remove("before");
|
| + this._labelLeftElement.classList.remove("hidden");
|
| +
|
| + this._labelRightElement.style.removeProperty("left");
|
| + this._labelRightElement.style.removeProperty("right");
|
| + this._labelRightElement.classList.remove("after");
|
| + this._labelRightElement.classList.remove("hidden");
|
| +
|
| + const labelPadding = 10;
|
| + const barRightElementOffsetWidth = this._barRightElement.offsetWidth;
|
| + const barLeftElementOffsetWidth = this._barLeftElement.offsetWidth;
|
| +
|
| + if (this._barLeftElement) {
|
| + var leftBarWidth = barLeftElementOffsetWidth - labelPadding;
|
| + var rightBarWidth = (barRightElementOffsetWidth - barLeftElementOffsetWidth) - labelPadding;
|
| + } else {
|
| + var leftBarWidth = (barLeftElementOffsetWidth - barRightElementOffsetWidth) - labelPadding;
|
| + var rightBarWidth = barRightElementOffsetWidth - labelPadding;
|
| + }
|
| +
|
| + const labelLeftElementOffsetWidth = this._labelLeftElement.offsetWidth;
|
| + const labelRightElementOffsetWidth = this._labelRightElement.offsetWidth;
|
| +
|
| + const labelBefore = (labelLeftElementOffsetWidth > leftBarWidth);
|
| + const labelAfter = (labelRightElementOffsetWidth > rightBarWidth);
|
| + const graphElementOffsetWidth = this._timelineCell.offsetWidth;
|
| +
|
| + if (labelBefore && (graphElementOffsetWidth * (this._percentages.start / 100)) < (labelLeftElementOffsetWidth + 10))
|
| + var leftHidden = true;
|
| +
|
| + if (labelAfter && (graphElementOffsetWidth * ((100 - this._percentages.end) / 100)) < (labelRightElementOffsetWidth + 10))
|
| + var rightHidden = true;
|
| +
|
| + if (barLeftElementOffsetWidth == barRightElementOffsetWidth) {
|
| + // The left/right label data are the same, so a before/after label can be replaced by an on-bar label.
|
| + if (labelBefore && !labelAfter)
|
| + leftHidden = true;
|
| + else if (labelAfter && !labelBefore)
|
| + rightHidden = true;
|
| + }
|
| +
|
| + if (labelBefore) {
|
| + if (leftHidden)
|
| + this._labelLeftElement.classList.add("hidden");
|
| + this._labelLeftElement.style.setProperty("right", (100 - this._percentages.start) + "%");
|
| + this._labelLeftElement.classList.add("before");
|
| + } else {
|
| + this._labelLeftElement.style.setProperty("left", this._percentages.start + "%");
|
| + this._labelLeftElement.style.setProperty("right", (100 - this._percentages.middle) + "%");
|
| + }
|
| +
|
| + if (labelAfter) {
|
| + if (rightHidden)
|
| + this._labelRightElement.classList.add("hidden");
|
| + this._labelRightElement.style.setProperty("left", this._percentages.end + "%");
|
| + this._labelRightElement.classList.add("after");
|
| + } else {
|
| + this._labelRightElement.style.setProperty("left", this._percentages.middle + "%");
|
| + this._labelRightElement.style.setProperty("right", (100 - this._percentages.end) + "%");
|
| + }
|
| + },
|
| +
|
| + __proto__: WebInspector.SortableDataGridNode.prototype
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.NameComparator = function(a, b)
|
| +{
|
| + var aFileName = a._request.name();
|
| + var bFileName = b._request.name();
|
| + if (aFileName > bFileName)
|
| + return 1;
|
| + if (bFileName > aFileName)
|
| + return -1;
|
| + return a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.RemoteAddressComparator = function(a, b)
|
| +{
|
| + var aRemoteAddress = a._request.remoteAddress();
|
| + var bRemoteAddress = b._request.remoteAddress();
|
| + if (aRemoteAddress > bRemoteAddress)
|
| + return 1;
|
| + if (bRemoteAddress > aRemoteAddress)
|
| + return -1;
|
| + return a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.SizeComparator = function(a, b)
|
| +{
|
| + if (b._request.cached() && !a._request.cached())
|
| + return 1;
|
| + if (a._request.cached() && !b._request.cached())
|
| + return -1;
|
| + return (a._request.transferSize - b._request.transferSize) || a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.InitiatorComparator = function(a, b)
|
| +{
|
| + var aInitiator = a._request.initiatorInfo();
|
| + var bInitiator = b._request.initiatorInfo();
|
| +
|
| + if (aInitiator.type < bInitiator.type)
|
| + return -1;
|
| + if (aInitiator.type > bInitiator.type)
|
| + return 1;
|
| +
|
| + if (typeof aInitiator.__source === "undefined")
|
| + aInitiator.__source = WebInspector.displayNameForURL(aInitiator.url);
|
| + if (typeof bInitiator.__source === "undefined")
|
| + bInitiator.__source = WebInspector.displayNameForURL(bInitiator.url);
|
| +
|
| + if (aInitiator.__source < bInitiator.__source)
|
| + return -1;
|
| + if (aInitiator.__source > bInitiator.__source)
|
| + return 1;
|
| +
|
| + if (aInitiator.lineNumber < bInitiator.lineNumber)
|
| + return -1;
|
| + if (aInitiator.lineNumber > bInitiator.lineNumber)
|
| + return 1;
|
| +
|
| + if (aInitiator.columnNumber < bInitiator.columnNumber)
|
| + return -1;
|
| + if (aInitiator.columnNumber > bInitiator.columnNumber)
|
| + return 1;
|
| +
|
| + return a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.RequestCookiesCountComparator = function(a, b)
|
| +{
|
| + var aScore = a._request.requestCookies ? a._request.requestCookies.length : 0;
|
| + var bScore = b._request.requestCookies ? b._request.requestCookies.length : 0;
|
| + return (aScore - bScore) || a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.ResponseCookiesCountComparator = function(a, b)
|
| +{
|
| + var aScore = a._request.responseCookies ? a._request.responseCookies.length : 0;
|
| + var bScore = b._request.responseCookies ? b._request.responseCookies.length : 0;
|
| + return (aScore - bScore) || a._request.indentityCompare(b._request);
|
| +}
|
| +
|
| +/**
|
| + * @param {string} propertyName
|
| + * @param {boolean} revert
|
| + * @param {!WebInspector.NetworkDataGridNode} a
|
| + * @param {!WebInspector.NetworkDataGridNode} b
|
| + * @return {number}
|
| + */
|
| +WebInspector.NetworkDataGridNode.RequestPropertyComparator = function(propertyName, revert, a, b)
|
| +{
|
| + var aValue = a._request[propertyName];
|
| + var bValue = b._request[propertyName];
|
| + if (aValue > bValue)
|
| + return revert ? -1 : 1;
|
| + if (bValue > aValue)
|
| + return revert ? 1 : -1;
|
| + return a._request.indentityCompare(b._request);
|
| }
|
|
|
| WebInspector.NetworkPanel.show = function()
|
|
|