chrome-devtools-frontend
Version:
Chrome DevTools UI
1,078 lines (954 loc) • 43.3 kB
text/typescript
// Copyright 2021 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/*
* Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
* Copyright (C) 2008, 2009 Anthony Ricaud <rik@webkit.org>
* 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
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
import * as Common from '../../core/common/common.js';
import * as Host from '../../core/host/host.js';
import * as i18n from '../../core/i18n/i18n.js';
import * as Platform from '../../core/platform/platform.js';
import * as SDK from '../../core/sdk/sdk.js';
import * as Bindings from '../../models/bindings/bindings.js';
import * as Logs from '../../models/logs/logs.js';
import * as Workspace from '../../models/workspace/workspace.js';
import * as NetworkForward from '../../panels/network/forward/forward.js';
import * as IconButton from '../../ui/components/icon_button/icon_button.js';
import * as PerfUI from '../../ui/legacy/components/perf_ui/perf_ui.js';
import * as UI from '../../ui/legacy/legacy.js';
import * as MobileThrottling from '../mobile_throttling/mobile_throttling.js';
import * as Search from '../search/search.js';
import {BlockedURLsPane} from './BlockedURLsPane.js';
import {Events, type RequestActivatedEvent} from './NetworkDataGridNode.js';
import {NetworkItemView} from './NetworkItemView.js';
import {NetworkLogView} from './NetworkLogView.js';
import {NetworkOverview} from './NetworkOverview.js';
import networkPanelStyles from './networkPanel.css.js';
import {NetworkSearchScope} from './NetworkSearchScope.js';
import {NetworkTransferTimeCalculator, type NetworkTimeCalculator} from './NetworkTimeCalculator.js';
const UIStrings = {
/**
*@description Text to close something
*/
close: 'Close',
/**
*@description Title of a search bar or tool
*/
search: 'Search',
/**
*@description Tooltip text that appears on the setting to preserve log when hovering over the item
*/
doNotClearLogOnPageReload: 'Do not clear log on page reload / navigation',
/**
*@description Text to preserve the log after refreshing
*/
preserveLog: 'Preserve log',
/**
*@description Text to disable cache while DevTools is open
*/
disableCacheWhileDevtoolsIsOpen: 'Disable cache (while DevTools is open)',
/**
*@description Text in Network Config View of the Network panel
*/
disableCache: 'Disable cache',
/**
*@description Tooltip text that appears when hovering over the largeicon settings gear in show settings pane setting in network panel of the network panel
*/
networkSettings: 'Network settings',
/**
*@description Tooltip for expanding network request row setting
*/
showMoreInformationInRequestRows: 'Show more information in request rows',
/**
*@description Text in Network Panel of the Network panel
*/
useLargeRequestRows: 'Use large request rows',
/**
*@description Tooltip text for network request overview setting
*/
showOverviewOfNetworkRequests: 'Show overview of network requests',
/**
*@description Text in Network Panel of the Network panel
*/
showOverview: 'Show overview',
/**
*@description Tooltip for group by frame network setting
*/
groupRequestsByTopLevelRequest: 'Group requests by top level request frame',
/**
*@description Text in Network Panel of the Network panel
*/
groupByFrame: 'Group by frame',
/**
*@description Tooltip for capture screenshot network setting
*/
captureScreenshotsWhenLoadingA: 'Capture screenshots when loading a page',
/**
*@description Text to take screenshots
*/
captureScreenshots: 'Capture screenshots',
/**
* @description Tooltip text that appears when hovering over the largeicon load button in the
* Network Panel. This action prompts the user to select a HAR file to upload to DevTools.
*/
importHarFile: 'Import `HAR` file...',
/**
* @description Tooltip text that appears when hovering over the largeicon download button in the
* Network Panel. HAR is a file format (HTTP Archive) and should not be translated. This action
* triggers the download of a HAR file.
*/
exportHar: 'Export `HAR`...',
/**
*@description Text for throttling the network
*/
throttling: 'Throttling',
/**
*@description Text in Network Panel of the Network panel
*@example {Ctrl + R} PH1
*/
hitSToReloadAndCaptureFilmstrip: 'Hit {PH1} to reload and capture filmstrip.',
/**
*@description A context menu item in the Network Panel of the Network panel
*/
revealInNetworkPanel: 'Reveal in Network panel',
/**
*@description Text in Network Panel of the Network panel
*/
recordingFrames: 'Recording frames...',
/**
*@description Text in Network Panel of the Network panel
*/
fetchingFrames: 'Fetching frames...',
/**
* @description Text of a button in the Network panel's toolbar that open Network Conditions panel in the drawer.
*/
moreNetworkConditions: 'More network conditions…',
};
const str_ = i18n.i18n.registerUIStrings('panels/network/NetworkPanel.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
let networkPanelInstance: NetworkPanel;
export class NetworkPanel extends UI.Panel.Panel implements UI.ContextMenu.Provider, UI.View.ViewLocationResolver {
private readonly networkLogShowOverviewSetting: Common.Settings.Setting<boolean>;
private readonly networkLogLargeRowsSetting: Common.Settings.Setting<boolean>;
private readonly networkRecordFilmStripSetting: Common.Settings.Setting<boolean>;
private readonly toggleRecordAction: UI.ActionRegistration.Action;
private pendingStopTimer!: number|undefined;
networkItemView: NetworkItemView|null;
private filmStripView: PerfUI.FilmStripView.FilmStripView|null;
private filmStripRecorder: FilmStripRecorder|null;
private currentRequest: SDK.NetworkRequest.NetworkRequest|null;
private readonly panelToolbar: UI.Toolbar.Toolbar;
private readonly rightToolbar: UI.Toolbar.Toolbar;
private readonly filterBar: UI.FilterBar.FilterBar;
private readonly settingsPane: UI.Widget.HBox;
private showSettingsPaneSetting: Common.Settings.Setting<boolean>;
private readonly filmStripPlaceholderElement: HTMLElement;
private readonly overviewPane: PerfUI.TimelineOverviewPane.TimelineOverviewPane;
private readonly networkOverview: NetworkOverview;
private readonly overviewPlaceholderElement: HTMLElement;
private readonly calculator: NetworkTransferTimeCalculator;
private splitWidget: UI.SplitWidget.SplitWidget;
private readonly sidebarLocation: UI.View.TabbedViewLocation;
private readonly progressBarContainer: HTMLDivElement;
networkLogView: NetworkLogView;
private readonly fileSelectorElement: HTMLElement;
private readonly detailsWidget: UI.Widget.VBox;
private readonly closeButtonElement: HTMLDivElement;
private preserveLogSetting: Common.Settings.Setting<boolean>;
recordLogSetting: Common.Settings.Setting<boolean>;
private readonly throttlingSelect: UI.Toolbar.ToolbarComboBox;
private readonly displayScreenshotDelay: number;
constructor(displayScreenshotDelay: number) {
super('network');
this.displayScreenshotDelay = displayScreenshotDelay;
this.networkLogShowOverviewSetting =
Common.Settings.Settings.instance().createSetting('networkLogShowOverview', true);
this.networkLogLargeRowsSetting = Common.Settings.Settings.instance().createSetting('networkLogLargeRows', false);
this.networkRecordFilmStripSetting =
Common.Settings.Settings.instance().createSetting('networkRecordFilmStripSetting', false);
this.toggleRecordAction =
(UI.ActionRegistry.ActionRegistry.instance().action('network.toggle-recording') as
UI.ActionRegistration.Action);
this.networkItemView = null;
this.filmStripView = null;
this.filmStripRecorder = null;
this.currentRequest = null;
const panel = new UI.Widget.VBox();
const networkToolbarContainer = panel.contentElement.createChild('div', 'network-toolbar-container');
this.panelToolbar = new UI.Toolbar.Toolbar('', networkToolbarContainer);
this.panelToolbar.makeWrappable(true);
this.rightToolbar = new UI.Toolbar.Toolbar('', networkToolbarContainer);
this.filterBar = new UI.FilterBar.FilterBar('networkPanel', true);
this.filterBar.show(panel.contentElement);
this.filterBar.addEventListener(UI.FilterBar.FilterBarEvents.Changed, this.handleFilterChanged.bind(this));
this.settingsPane = new UI.Widget.HBox();
this.settingsPane.element.classList.add('network-settings-pane');
this.settingsPane.show(panel.contentElement);
this.showSettingsPaneSetting =
Common.Settings.Settings.instance().createSetting('networkShowSettingsToolbar', false);
this.showSettingsPaneSetting.addChangeListener(this.updateSettingsPaneVisibility.bind(this));
this.updateSettingsPaneVisibility();
this.filmStripPlaceholderElement = panel.contentElement.createChild('div', 'network-film-strip-placeholder');
// Create top overview component.
this.overviewPane = new PerfUI.TimelineOverviewPane.TimelineOverviewPane('network');
this.overviewPane.addEventListener(
PerfUI.TimelineOverviewPane.Events.WindowChanged, this.onWindowChanged.bind(this));
this.overviewPane.element.id = 'network-overview-panel';
this.networkOverview = new NetworkOverview();
this.overviewPane.setOverviewControls([this.networkOverview]);
this.overviewPlaceholderElement = panel.contentElement.createChild('div');
this.calculator = new NetworkTransferTimeCalculator();
this.splitWidget = new UI.SplitWidget.SplitWidget(true, false, 'networkPanelSplitViewState');
this.splitWidget.hideMain();
this.splitWidget.show(panel.contentElement);
panel.setDefaultFocusedChild(this.filterBar);
const initialSidebarWidth = 225;
const splitWidget = new UI.SplitWidget.SplitWidget(true, false, 'networkPanelSidebarState', initialSidebarWidth);
splitWidget.hideSidebar();
splitWidget.enableShowModeSaving();
splitWidget.show(this.element);
this.sidebarLocation = UI.ViewManager.ViewManager.instance().createTabbedLocation(async () => {
void UI.ViewManager.ViewManager.instance().showView('network');
splitWidget.showBoth();
}, 'network-sidebar', true);
const tabbedPane = this.sidebarLocation.tabbedPane();
tabbedPane.setMinimumSize(100, 25);
tabbedPane.element.classList.add('network-tabbed-pane');
tabbedPane.element.addEventListener('keydown', event => {
if (event.key !== Platform.KeyboardUtilities.ESCAPE_KEY) {
return;
}
splitWidget.hideSidebar();
event.consume();
});
const closeSidebar = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.close), 'cross');
closeSidebar.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, () => splitWidget.hideSidebar());
tabbedPane.rightToolbar().appendToolbarItem(closeSidebar);
splitWidget.setSidebarWidget(tabbedPane);
splitWidget.setMainWidget(panel);
splitWidget.setDefaultFocusedChild(panel);
this.setDefaultFocusedChild(splitWidget);
this.progressBarContainer = document.createElement('div');
this.networkLogView =
new NetworkLogView(this.filterBar, this.progressBarContainer, this.networkLogLargeRowsSetting);
this.splitWidget.setSidebarWidget(this.networkLogView);
this.fileSelectorElement =
(UI.UIUtils.createFileSelectorElement(this.networkLogView.onLoadFromFile.bind(this.networkLogView)) as
HTMLElement);
panel.element.appendChild(this.fileSelectorElement);
this.detailsWidget = new UI.Widget.VBox();
this.detailsWidget.element.classList.add('network-details-view');
this.splitWidget.setMainWidget(this.detailsWidget);
this.closeButtonElement = document.createElement('div', {is: 'dt-close-button'});
this.closeButtonElement.addEventListener('click', async () => {
const action = UI.ActionRegistry.ActionRegistry.instance().action('network.hide-request-details');
if (action) {
await action.execute();
}
}, false);
this.closeButtonElement.style.margin = '0 5px';
this.networkLogShowOverviewSetting.addChangeListener(this.toggleShowOverview, this);
this.networkLogLargeRowsSetting.addChangeListener(this.toggleLargerRequests, this);
this.networkRecordFilmStripSetting.addChangeListener(this.toggleRecordFilmStrip, this);
this.preserveLogSetting = Common.Settings.Settings.instance().moduleSetting('network_log.preserve-log');
this.recordLogSetting = Common.Settings.Settings.instance().moduleSetting('network_log.record-log');
this.recordLogSetting.addChangeListener(({data}) => this.toggleRecord(data));
this.throttlingSelect = this.createThrottlingConditionsSelect();
this.setupToolbarButtons(splitWidget);
this.toggleRecord(this.recordLogSetting.get());
this.toggleShowOverview();
this.toggleLargerRequests();
this.toggleRecordFilmStrip();
this.updateUI();
SDK.TargetManager.TargetManager.instance().addModelListener(
SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.WillReloadPage, this.willReloadPage, this,
{scoped: true});
SDK.TargetManager.TargetManager.instance().addModelListener(
SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.Load, this.load, this, {scoped: true});
this.networkLogView.addEventListener(Events.RequestSelected, this.onRequestSelected, this);
this.networkLogView.addEventListener(Events.RequestActivated, this.onRequestActivated, this);
Logs.NetworkLog.NetworkLog.instance().addEventListener(
Logs.NetworkLog.Events.RequestAdded, this.onUpdateRequest, this);
Logs.NetworkLog.NetworkLog.instance().addEventListener(
Logs.NetworkLog.Events.RequestUpdated, this.onUpdateRequest, this);
Logs.NetworkLog.NetworkLog.instance().addEventListener(Logs.NetworkLog.Events.Reset, this.onNetworkLogReset, this);
}
static instance(opts?: {
forceNew: boolean,
displayScreenshotDelay?: number,
}): NetworkPanel {
if (!networkPanelInstance || opts?.forceNew) {
networkPanelInstance = new NetworkPanel(opts?.displayScreenshotDelay ?? 1000);
}
return networkPanelInstance;
}
static revealAndFilter(filters: {
filterType: NetworkForward.UIFilter.FilterType|null,
filterValue: string,
}[]): Promise<void> {
const panel = NetworkPanel.instance();
let filterString = '';
for (const filter of filters) {
if (filter.filterType) {
filterString += `${filter.filterType}:${filter.filterValue} `;
} else {
filterString += `${filter.filterValue} `;
}
}
panel.networkLogView.setTextFilterValue(filterString);
return UI.ViewManager.ViewManager.instance().showView('network');
}
static async selectAndShowRequest(
request: SDK.NetworkRequest.NetworkRequest, tab: NetworkForward.UIRequestLocation.UIRequestTabs,
options?: NetworkForward.UIRequestLocation.FilterOptions): Promise<void> {
const panel = NetworkPanel.instance();
await panel.selectAndActivateRequest(request, tab, options);
}
throttlingSelectForTest(): UI.Toolbar.ToolbarComboBox {
return this.throttlingSelect;
}
private onWindowChanged(event: Common.EventTarget.EventTargetEvent<PerfUI.TimelineOverviewPane.WindowChangedEvent>):
void {
const startTime = Math.max(this.calculator.minimumBoundary(), event.data.startTime / 1000);
const endTime = Math.min(this.calculator.maximumBoundary(), event.data.endTime / 1000);
if (startTime === this.calculator.minimumBoundary() && endTime === this.calculator.maximumBoundary()) {
// Reset the filters for NetworkLogView when the window is reset
// to its boundaries. This clears the filters and allows the users
// to see the incoming requests after they have updated the curtains
// to be in the edges. (ex: by double clicking on the overview grid)
this.networkLogView.setWindow(0, 0);
} else {
this.networkLogView.setWindow(startTime, endTime);
}
}
private async searchToggleClick(): Promise<void> {
const action = UI.ActionRegistry.ActionRegistry.instance().action('network.search');
if (action) {
await action.execute();
}
}
private setupToolbarButtons(splitWidget: UI.SplitWidget.SplitWidget): void {
const searchToggle = new UI.Toolbar.ToolbarToggle(i18nString(UIStrings.search), 'search');
function updateSidebarToggle(): void {
const isSidebarShowing = splitWidget.showMode() !== UI.SplitWidget.ShowMode.OnlyMain;
searchToggle.setToggled(isSidebarShowing);
if (!isSidebarShowing) {
(searchToggle.element as HTMLElement).focus();
}
}
this.panelToolbar.appendToolbarItem(UI.Toolbar.Toolbar.createActionButton(this.toggleRecordAction));
this.panelToolbar.appendToolbarItem(UI.Toolbar.Toolbar.createActionButton(
(UI.ActionRegistry.ActionRegistry.instance().action('network.clear') as UI.ActionRegistration.Action)));
this.panelToolbar.appendSeparator();
this.panelToolbar.appendToolbarItem(this.filterBar.filterButton());
updateSidebarToggle();
splitWidget.addEventListener(UI.SplitWidget.Events.ShowModeChanged, updateSidebarToggle);
searchToggle.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, () => {
void this.searchToggleClick();
});
this.panelToolbar.appendToolbarItem(searchToggle);
this.panelToolbar.appendSeparator();
this.panelToolbar.appendToolbarItem(new UI.Toolbar.ToolbarSettingCheckbox(
this.preserveLogSetting, i18nString(UIStrings.doNotClearLogOnPageReload), i18nString(UIStrings.preserveLog)));
this.panelToolbar.appendSeparator();
const disableCacheCheckbox = new UI.Toolbar.ToolbarSettingCheckbox(
Common.Settings.Settings.instance().moduleSetting('cacheDisabled'),
i18nString(UIStrings.disableCacheWhileDevtoolsIsOpen), i18nString(UIStrings.disableCache));
this.panelToolbar.appendToolbarItem(disableCacheCheckbox);
this.panelToolbar.appendToolbarItem(this.throttlingSelect);
const networkConditionsIcon = new IconButton.Icon.Icon();
networkConditionsIcon.data = {
iconName: 'network-settings',
color: 'var(--icon-default)',
width: '20px',
height: '20px',
};
const networkConditionsButton =
new UI.Toolbar.ToolbarButton(i18nString(UIStrings.moreNetworkConditions), networkConditionsIcon);
networkConditionsButton.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, () => {
void UI.ViewManager.ViewManager.instance().showView('network.config');
}, this);
this.panelToolbar.appendToolbarItem(networkConditionsButton);
this.rightToolbar.appendToolbarItem(new UI.Toolbar.ToolbarItem(this.progressBarContainer));
this.rightToolbar.appendSeparator();
this.rightToolbar.appendToolbarItem(new UI.Toolbar.ToolbarSettingToggle(
this.showSettingsPaneSetting, 'gear', i18nString(UIStrings.networkSettings), 'gear-filled'));
const settingsToolbarLeft = new UI.Toolbar.Toolbar('', this.settingsPane.element);
settingsToolbarLeft.makeVertical();
settingsToolbarLeft.appendToolbarItem(new UI.Toolbar.ToolbarSettingCheckbox(
this.networkLogLargeRowsSetting, i18nString(UIStrings.showMoreInformationInRequestRows),
i18nString(UIStrings.useLargeRequestRows)));
settingsToolbarLeft.appendToolbarItem(new UI.Toolbar.ToolbarSettingCheckbox(
this.networkLogShowOverviewSetting, i18nString(UIStrings.showOverviewOfNetworkRequests),
i18nString(UIStrings.showOverview)));
const settingsToolbarRight = new UI.Toolbar.Toolbar('', this.settingsPane.element);
settingsToolbarRight.makeVertical();
settingsToolbarRight.appendToolbarItem(new UI.Toolbar.ToolbarSettingCheckbox(
Common.Settings.Settings.instance().moduleSetting('network.group-by-frame'),
i18nString(UIStrings.groupRequestsByTopLevelRequest), i18nString(UIStrings.groupByFrame)));
settingsToolbarRight.appendToolbarItem(new UI.Toolbar.ToolbarSettingCheckbox(
this.networkRecordFilmStripSetting, i18nString(UIStrings.captureScreenshotsWhenLoadingA),
i18nString(UIStrings.captureScreenshots)));
this.panelToolbar.appendSeparator();
const importHarButton = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.importHarFile), 'import');
importHarButton.addEventListener(
UI.Toolbar.ToolbarButton.Events.Click, () => this.fileSelectorElement.click(), this);
this.panelToolbar.appendToolbarItem(importHarButton);
const exportHarButton = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.exportHar), 'download');
exportHarButton.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, _event => {
void this.networkLogView.exportAll();
}, this);
this.panelToolbar.appendToolbarItem(exportHarButton);
}
private updateSettingsPaneVisibility(): void {
this.settingsPane.element.classList.toggle('hidden', !this.showSettingsPaneSetting.get());
}
private createThrottlingConditionsSelect(): UI.Toolbar.ToolbarComboBox {
const toolbarItem = new UI.Toolbar.ToolbarComboBox(null, i18nString(UIStrings.throttling));
toolbarItem.setMaxWidth(160);
MobileThrottling.ThrottlingManager.throttlingManager().decorateSelectWithNetworkThrottling(
toolbarItem.selectElement());
return toolbarItem;
}
toggleRecord(toggled: boolean): void {
this.toggleRecordAction.setToggled(toggled);
if (this.recordLogSetting.get() !== toggled) {
this.recordLogSetting.set(toggled);
}
this.networkLogView.setRecording(toggled);
if (!toggled && this.filmStripRecorder) {
this.filmStripRecorder.stopRecording(this.filmStripAvailable.bind(this));
}
}
private filmStripAvailable(filmStripModel: SDK.FilmStripModel.FilmStripModel|null): void {
if (!filmStripModel) {
return;
}
const calculator = this.networkLogView.timeCalculator();
if (this.filmStripView) {
this.filmStripView.setModel(
filmStripModel, calculator.minimumBoundary() * 1000, calculator.boundarySpan() * 1000);
}
this.networkOverview.setFilmStripModel(filmStripModel);
const timestamps = filmStripModel.frames().map(mapTimestamp);
function mapTimestamp(frame: SDK.FilmStripModel.Frame): number {
return frame.timestamp / 1000;
}
this.networkLogView.addFilmStripFrames(timestamps);
}
private onNetworkLogReset(event: Common.EventTarget.EventTargetEvent<Logs.NetworkLog.ResetEvent>): void {
const {clearIfPreserved} = event.data;
BlockedURLsPane.reset();
if (!this.preserveLogSetting.get() || clearIfPreserved) {
this.calculator.reset();
this.overviewPane.reset();
}
if (this.filmStripView) {
this.resetFilmStripView();
}
}
private willReloadPage(): void {
if (this.pendingStopTimer) {
clearTimeout(this.pendingStopTimer);
delete this.pendingStopTimer;
}
if (this.isShowing() && this.filmStripRecorder) {
this.filmStripRecorder.startRecording();
}
}
private load(): void {
if (this.filmStripRecorder && this.filmStripRecorder.isRecording()) {
if (this.pendingStopTimer) {
window.clearTimeout(this.pendingStopTimer);
}
this.pendingStopTimer = window.setTimeout(this.stopFilmStripRecording.bind(this), this.displayScreenshotDelay);
}
}
private stopFilmStripRecording(): void {
if (this.filmStripRecorder) {
this.filmStripRecorder.stopRecording(this.filmStripAvailable.bind(this));
}
delete this.pendingStopTimer;
}
private toggleLargerRequests(): void {
this.updateUI();
}
private toggleShowOverview(): void {
const toggled = this.networkLogShowOverviewSetting.get();
if (toggled) {
this.overviewPane.show(this.overviewPlaceholderElement);
} else {
this.overviewPane.detach();
}
this.doResize();
}
private toggleRecordFilmStrip(): void {
const toggled = this.networkRecordFilmStripSetting.get();
if (toggled && !this.filmStripRecorder) {
this.filmStripView = new PerfUI.FilmStripView.FilmStripView();
this.filmStripView.setMode(PerfUI.FilmStripView.Modes.FrameBased);
this.filmStripView.element.classList.add('network-film-strip');
this.filmStripRecorder = new FilmStripRecorder(this.networkLogView.timeCalculator(), this.filmStripView);
this.filmStripView.show(this.filmStripPlaceholderElement);
this.filmStripView.addEventListener(PerfUI.FilmStripView.Events.FrameSelected, this.onFilmFrameSelected, this);
this.filmStripView.addEventListener(PerfUI.FilmStripView.Events.FrameEnter, this.onFilmFrameEnter, this);
this.filmStripView.addEventListener(PerfUI.FilmStripView.Events.FrameExit, this.onFilmFrameExit, this);
this.resetFilmStripView();
}
if (!toggled && this.filmStripRecorder) {
if (this.filmStripView) {
this.filmStripView.detach();
}
this.filmStripView = null;
this.filmStripRecorder = null;
}
}
private resetFilmStripView(): void {
const reloadShortcut =
UI.ShortcutRegistry.ShortcutRegistry.instance().shortcutsForAction('inspector_main.reload')[0];
if (this.filmStripView) {
this.filmStripView.reset();
if (reloadShortcut) {
this.filmStripView.setStatusText(
i18nString(UIStrings.hitSToReloadAndCaptureFilmstrip, {PH1: reloadShortcut.title()}));
}
}
}
override elementsToRestoreScrollPositionsFor(): Element[] {
return this.networkLogView.elementsToRestoreScrollPositionsFor();
}
override wasShown(): void {
UI.Context.Context.instance().setFlavor(NetworkPanel, this);
this.registerCSSFiles([networkPanelStyles]);
// Record the network tool load time after the panel has loaded.
Host.userMetrics.panelLoaded('network', 'DevTools.Launch.Network');
}
override willHide(): void {
UI.Context.Context.instance().setFlavor(NetworkPanel, null);
}
revealAndHighlightRequest(request: SDK.NetworkRequest.NetworkRequest): void {
this.hideRequestPanel();
if (request) {
this.networkLogView.revealAndHighlightRequest(request);
}
}
revealAndHighlightRequestWithId(request: NetworkForward.NetworkRequestId.NetworkRequestId): void {
this.hideRequestPanel();
if (request) {
this.networkLogView.revealAndHighlightRequestWithId(request);
}
}
async selectAndActivateRequest(
request: SDK.NetworkRequest.NetworkRequest, shownTab?: NetworkForward.UIRequestLocation.UIRequestTabs,
options?: NetworkForward.UIRequestLocation.FilterOptions): Promise<NetworkItemView|null> {
await UI.ViewManager.ViewManager.instance().showView('network');
this.networkLogView.selectRequest(request, options);
this.showRequestPanel(shownTab);
return this.networkItemView;
}
private handleFilterChanged(): void {
this.hideRequestPanel();
}
private onRowSizeChanged(): void {
this.updateUI();
}
private onRequestSelected(event: Common.EventTarget.EventTargetEvent<SDK.NetworkRequest.NetworkRequest|null>): void {
const request = event.data;
this.currentRequest = request;
this.networkOverview.setHighlightedRequest(request);
this.updateNetworkItemView();
}
private onRequestActivated(event: Common.EventTarget.EventTargetEvent<RequestActivatedEvent>): void {
const {showPanel, tab, takeFocus} = event.data;
if (showPanel) {
this.showRequestPanel(tab, takeFocus);
} else {
this.hideRequestPanel();
}
}
private showRequestPanel(shownTab?: NetworkForward.UIRequestLocation.UIRequestTabs, takeFocus?: boolean): void {
if (this.splitWidget.showMode() === UI.SplitWidget.ShowMode.Both && !shownTab && !takeFocus) {
// If panel is already shown, and we are not forcing a specific tab, return.
return;
}
this.clearNetworkItemView();
if (this.currentRequest) {
const networkItemView = this.createNetworkItemView(shownTab);
if (networkItemView && takeFocus) {
networkItemView.focus();
}
}
this.updateUI();
}
hideRequestPanel(): void {
this.clearNetworkItemView();
this.splitWidget.hideMain();
this.updateUI();
}
private updateNetworkItemView(): void {
if (this.splitWidget.showMode() === UI.SplitWidget.ShowMode.Both) {
this.clearNetworkItemView();
this.createNetworkItemView();
this.updateUI();
}
}
private clearNetworkItemView(): void {
if (this.networkItemView) {
this.networkItemView.detach();
this.networkItemView = null;
}
}
private createNetworkItemView(initialTab?: NetworkForward.UIRequestLocation.UIRequestTabs): NetworkItemView
|undefined {
if (!this.currentRequest) {
return;
}
this.networkItemView = new NetworkItemView(this.currentRequest, this.networkLogView.timeCalculator(), initialTab);
this.networkItemView.leftToolbar().appendToolbarItem(new UI.Toolbar.ToolbarItem(this.closeButtonElement));
this.networkItemView.show(this.detailsWidget.element);
this.splitWidget.showBoth();
return this.networkItemView;
}
private updateUI(): void {
if (this.detailsWidget) {
this.detailsWidget.element.classList.toggle(
'network-details-view-tall-header', this.networkLogLargeRowsSetting.get());
}
if (this.networkLogView) {
this.networkLogView.switchViewMode(!this.splitWidget.isResizable());
}
}
appendApplicableItems(this: NetworkPanel, event: Event, contextMenu: UI.ContextMenu.ContextMenu, target: Object):
void {
function reveal(this: NetworkPanel, request: SDK.NetworkRequest.NetworkRequest): void {
void UI.ViewManager.ViewManager.instance()
.showView('network')
.then(this.networkLogView.resetFilter.bind(this.networkLogView))
.then(this.revealAndHighlightRequest.bind(this, request));
}
function appendRevealItem(this: NetworkPanel, request: SDK.NetworkRequest.NetworkRequest): void {
contextMenu.revealSection().appendItem(i18nString(UIStrings.revealInNetworkPanel), reveal.bind(this, request));
}
if ((event.target as Node).isSelfOrDescendant(this.element)) {
return;
}
if (target instanceof SDK.Resource.Resource) {
const resource = (target as SDK.Resource.Resource);
if (resource.request) {
appendRevealItem.call(this, resource.request);
}
return;
}
if (target instanceof Workspace.UISourceCode.UISourceCode) {
const uiSourceCode = (target as Workspace.UISourceCode.UISourceCode);
const resource = Bindings.ResourceUtils.resourceForURL(uiSourceCode.url());
if (resource && resource.request) {
appendRevealItem.call(this, resource.request);
}
return;
}
if (!(target instanceof SDK.NetworkRequest.NetworkRequest)) {
return;
}
const request = (target as SDK.NetworkRequest.NetworkRequest);
if (this.networkItemView && this.networkItemView.isShowing() && this.networkItemView.request() === request) {
return;
}
appendRevealItem.call(this, request);
}
private onFilmFrameSelected(event: Common.EventTarget.EventTargetEvent<number>): void {
const timestamp = event.data;
this.overviewPane.setWindowTimes(0, timestamp);
}
private onFilmFrameEnter(event: Common.EventTarget.EventTargetEvent<number>): void {
const timestamp = event.data;
this.networkOverview.selectFilmStripFrame(timestamp);
this.networkLogView.selectFilmStripFrame(timestamp / 1000);
}
private onFilmFrameExit(): void {
this.networkOverview.clearFilmStripFrame();
this.networkLogView.clearFilmStripFrame();
}
private onUpdateRequest(event: Common.EventTarget.EventTargetEvent<SDK.NetworkRequest.NetworkRequest>): void {
const request = event.data;
this.calculator.updateBoundaries(request);
// FIXME: Unify all time units across the frontend!
this.overviewPane.setBounds(this.calculator.minimumBoundary() * 1000, this.calculator.maximumBoundary() * 1000);
this.networkOverview.updateRequest(request);
this.overviewPane.scheduleUpdate();
}
resolveLocation(locationName: string): UI.View.ViewLocation|null {
if (locationName === 'network-sidebar') {
return this.sidebarLocation;
}
return null;
}
}
let contextMenuProviderInstance: ContextMenuProvider;
export class ContextMenuProvider implements UI.ContextMenu.Provider {
static instance(opts: {
forceNew: boolean|null,
} = {forceNew: null}): ContextMenuProvider {
const {forceNew} = opts;
if (!contextMenuProviderInstance || forceNew) {
contextMenuProviderInstance = new ContextMenuProvider();
}
return contextMenuProviderInstance;
}
appendApplicableItems(event: Event, contextMenu: UI.ContextMenu.ContextMenu, target: Object): void {
NetworkPanel.instance().appendApplicableItems(event, contextMenu, target);
}
}
let requestRevealerInstance: RequestRevealer;
export class RequestRevealer implements Common.Revealer.Revealer {
static instance(opts: {
forceNew: boolean|null,
} = {forceNew: null}): RequestRevealer {
const {forceNew} = opts;
if (!requestRevealerInstance || forceNew) {
requestRevealerInstance = new RequestRevealer();
}
return requestRevealerInstance;
}
reveal(request: Object): Promise<void> {
if (!(request instanceof SDK.NetworkRequest.NetworkRequest)) {
return Promise.reject(new Error('Internal error: not a network request'));
}
const panel = NetworkPanel.instance();
return UI.ViewManager.ViewManager.instance().showView('network').then(
panel.revealAndHighlightRequest.bind(panel, request));
}
}
let requestIdRevealerInstance: RequestIdRevealer;
export class RequestIdRevealer implements Common.Revealer.Revealer {
static instance(opts: {
forceNew: boolean|null,
} = {forceNew: null}): RequestIdRevealer {
const {forceNew} = opts;
if (!requestIdRevealerInstance || forceNew) {
requestIdRevealerInstance = new RequestIdRevealer();
}
return requestIdRevealerInstance;
}
reveal(requestId: Object): Promise<void> {
if (!(requestId instanceof NetworkForward.NetworkRequestId.NetworkRequestId)) {
return Promise.reject(new Error('Internal error: not a network request ID'));
}
const panel = NetworkPanel.instance();
return UI.ViewManager.ViewManager.instance().showView('network').then(
panel.revealAndHighlightRequestWithId.bind(panel, requestId));
}
}
let networkLogWithFilterRevealerInstance: NetworkLogWithFilterRevealer;
export class NetworkLogWithFilterRevealer implements Common.Revealer.Revealer {
static instance(opts: {
forceNew: boolean|null,
} = {forceNew: null}): NetworkLogWithFilterRevealer {
const {forceNew} = opts;
if (!networkLogWithFilterRevealerInstance || forceNew) {
networkLogWithFilterRevealerInstance = new NetworkLogWithFilterRevealer();
}
return networkLogWithFilterRevealerInstance;
}
reveal(request: Object): Promise<void> {
if (!(request instanceof NetworkForward.UIFilter.UIRequestFilter)) {
return Promise.reject(new Error('Internal error: not a UIRequestFilter'));
}
return NetworkPanel.revealAndFilter(request.filters);
}
}
export class FilmStripRecorder implements SDK.TracingManager.TracingManagerClient {
private tracingManager: SDK.TracingManager.TracingManager|null;
private resourceTreeModel: SDK.ResourceTreeModel.ResourceTreeModel|null;
private readonly timeCalculator: NetworkTimeCalculator;
private readonly filmStripView: PerfUI.FilmStripView.FilmStripView;
private tracingModel: SDK.TracingModel.TracingModel|null;
private callback: ((arg0: SDK.FilmStripModel.FilmStripModel|null) => void)|null;
constructor(timeCalculator: NetworkTimeCalculator, filmStripView: PerfUI.FilmStripView.FilmStripView) {
this.tracingManager = null;
this.resourceTreeModel = null;
this.timeCalculator = timeCalculator;
this.filmStripView = filmStripView;
this.tracingModel = null;
this.callback = null;
}
traceEventsCollected(events: SDK.TracingManager.EventPayload[]): void {
if (this.tracingModel) {
this.tracingModel.addEvents(events);
}
}
tracingComplete(): void {
if (!this.tracingModel || !this.tracingManager) {
return;
}
this.tracingModel.tracingComplete();
this.tracingManager = null;
if (this.callback) {
this.callback(
new SDK.FilmStripModel.FilmStripModel(this.tracingModel, this.timeCalculator.minimumBoundary() * 1000));
}
this.callback = null;
if (this.resourceTreeModel) {
this.resourceTreeModel.resumeReload();
}
this.resourceTreeModel = null;
}
tracingBufferUsage(): void {
}
eventsRetrievalProgress(_progress: number): void {
}
startRecording(): void {
this.filmStripView.reset();
this.filmStripView.setStatusText(i18nString(UIStrings.recordingFrames));
const tracingManager =
SDK.TargetManager.TargetManager.instance().scopeTarget()?.model(SDK.TracingManager.TracingManager);
if (this.tracingManager || !tracingManager) {
return;
}
this.tracingManager = tracingManager;
this.resourceTreeModel = this.tracingManager.target().model(SDK.ResourceTreeModel.ResourceTreeModel);
this.tracingModel = new SDK.TracingModel.TracingModel();
void this.tracingManager.start(this, '-*,disabled-by-default-devtools.screenshot', '');
Host.userMetrics.actionTaken(Host.UserMetrics.Action.FilmStripStartedRecording);
}
isRecording(): boolean {
return Boolean(this.tracingManager);
}
stopRecording(callback: (arg0: SDK.FilmStripModel.FilmStripModel|null) => void): void {
if (!this.tracingManager) {
return;
}
this.tracingManager.stop();
if (this.resourceTreeModel) {
this.resourceTreeModel.suspendReload();
}
this.callback = callback;
this.filmStripView.setStatusText(i18nString(UIStrings.fetchingFrames));
}
}
let networkActionDelegateInstance: ActionDelegate;
export class ActionDelegate implements UI.ActionRegistration.ActionDelegate {
static instance(opts: {
forceNew: boolean|null,
}|undefined = {forceNew: null}): ActionDelegate {
const {forceNew} = opts;
if (!networkActionDelegateInstance || forceNew) {
networkActionDelegateInstance = new ActionDelegate();
}
return networkActionDelegateInstance;
}
handleAction(context: UI.Context.Context, actionId: string): boolean {
const panel = UI.Context.Context.instance().flavor(NetworkPanel);
console.assert(Boolean(panel && panel instanceof NetworkPanel));
if (!panel) {
return false;
}
switch (actionId) {
case 'network.toggle-recording': {
panel.toggleRecord(!panel.recordLogSetting.get());
return true;
}
case 'network.hide-request-details': {
if (!panel.networkItemView) {
return false;
}
panel.hideRequestPanel();
panel.networkLogView.resetFocus();
return true;
}
case 'network.search': {
const selection = UI.InspectorView.InspectorView.instance().element.window().getSelection();
if (!selection) {
return false;
}
let queryCandidate = '';
if (selection.rangeCount) {
queryCandidate = selection.toString().replace(/\r?\n.*/, '');
}
void SearchNetworkView.openSearch(queryCandidate);
return true;
}
case 'network.clear': {
Logs.NetworkLog.NetworkLog.instance().reset(true);
return true;
}
}
return false;
}
}
let requestLocationRevealerInstance: RequestLocationRevealer;
export class RequestLocationRevealer implements Common.Revealer.Revealer {
static instance(opts: {
forceNew: boolean|null,
}|undefined = {forceNew: null}): RequestLocationRevealer {
const {forceNew} = opts;
if (!requestLocationRevealerInstance || forceNew) {
requestLocationRevealerInstance = new RequestLocationRevealer();
}
return requestLocationRevealerInstance;
}
async reveal(match: Object): Promise<void> {
const location = match as NetworkForward.UIRequestLocation.UIRequestLocation;
const view =
await NetworkPanel.instance().selectAndActivateRequest(location.request, location.tab, location.filterOptions);
if (!view) {
return;
}
if (location.searchMatch) {
await view.revealResponseBody(location.searchMatch.lineNumber);
}
if (location.header) {
view.revealHeader(location.header.section, location.header.header?.name);
}
}
}
let searchNetworkViewInstance: SearchNetworkView;
export class SearchNetworkView extends Search.SearchView.SearchView {
private constructor() {
super('network');
}
static instance(opts: {
forceNew: boolean|null,
} = {forceNew: null}): SearchNetworkView {
const {forceNew} = opts;
if (!searchNetworkViewInstance || forceNew) {
searchNetworkViewInstance = new SearchNetworkView();
}
return searchNetworkViewInstance;
}
static async openSearch(query: string, searchImmediately?: boolean): Promise<Search.SearchView.SearchView> {
await UI.ViewManager.ViewManager.instance().showView('network.search-network-tab');
const searchView = SearchNetworkView.instance();
void searchView.toggle(query, Boolean(searchImmediately));
return searchView;
}
override createScope(): Search.SearchConfig.SearchScope {
return new NetworkSearchScope();
}
}