UNPKG

@quick-game/cli

Version:

Command line interface for rapid qg development

914 lines 41.8 kB
// 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 TraceEngine from '../../models/trace/trace.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 } 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 } 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: 'Big 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: '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: '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; export class NetworkPanel extends UI.Panel.Panel { networkLogShowOverviewSetting; networkLogLargeRowsSetting; networkRecordFilmStripSetting; toggleRecordAction; pendingStopTimer; networkItemView; filmStripView; filmStripRecorder; currentRequest; panelToolbar; rightToolbar; filterBar; settingsPane; showSettingsPaneSetting; filmStripPlaceholderElement; overviewPane; networkOverview; overviewPlaceholderElement; calculator; splitWidget; sidebarLocation; progressBarContainer; networkLogView; fileSelectorElement; detailsWidget; closeButtonElement; preserveLogSetting; recordLogSetting; throttlingSelect; displayScreenshotDelay; constructor(displayScreenshotDelay) { 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'); 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("Changed" /* 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)); 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) { if (!networkPanelInstance || opts?.forceNew) { networkPanelInstance = new NetworkPanel(opts?.displayScreenshotDelay ?? 1000); } return networkPanelInstance; } static revealAndFilter(filters) { 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, tab, options) { const panel = NetworkPanel.instance(); await panel.selectAndActivateRequest(request, tab, options); } throttlingSelectForTest() { return this.throttlingSelect; } onWindowChanged(event) { 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); } } async searchToggleClick() { const action = UI.ActionRegistry.ActionRegistry.instance().action('network.search'); if (action) { await action.execute(); } } setupToolbarButtons(splitWidget) { const searchToggle = new UI.Toolbar.ToolbarToggle(i18nString(UIStrings.search), 'search'); function updateSidebarToggle() { const isSidebarShowing = splitWidget.showMode() !== UI.SplitWidget.ShowMode.OnlyMain; searchToggle.setToggled(isSidebarShowing); if (!isSidebarShowing) { searchToggle.element.focus(); } } this.panelToolbar.appendToolbarItem(UI.Toolbar.Toolbar.createActionButton(this.toggleRecordAction)); this.panelToolbar.appendToolbarItem(UI.Toolbar.Toolbar.createActionButton(UI.ActionRegistry.ActionRegistry.instance().action('network.clear'))); 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); } updateSettingsPaneVisibility() { this.settingsPane.element.classList.toggle('hidden', !this.showSettingsPaneSetting.get()); } createThrottlingConditionsSelect() { const toolbarItem = new UI.Toolbar.ToolbarComboBox(null, i18nString(UIStrings.throttling)); toolbarItem.setMaxWidth(160); MobileThrottling.ThrottlingManager.throttlingManager().decorateSelectWithNetworkThrottling(toolbarItem.selectElement()); return toolbarItem; } toggleRecord(toggled) { 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)); } } filmStripAvailable(filmStrip) { if (this.filmStripView) { this.filmStripView.setModel(filmStrip); } const timestamps = filmStrip.frames.map(frame => { // The network view works in seconds. return TraceEngine.Helpers.Timing.microSecondsToSeconds(frame.screenshotEvent.ts); }); this.networkLogView.addFilmStripFrames(timestamps); } onNetworkLogReset(event) { const { clearIfPreserved } = event.data; BlockedURLsPane.reset(); if (!this.preserveLogSetting.get() || clearIfPreserved) { this.calculator.reset(); this.overviewPane.reset(); } if (this.filmStripView) { this.resetFilmStripView(); } } willReloadPage() { if (this.pendingStopTimer) { clearTimeout(this.pendingStopTimer); delete this.pendingStopTimer; } if (this.isShowing() && this.filmStripRecorder) { this.filmStripRecorder.startRecording(); } } load() { if (this.filmStripRecorder && this.filmStripRecorder.isRecording()) { if (this.pendingStopTimer) { window.clearTimeout(this.pendingStopTimer); } this.pendingStopTimer = window.setTimeout(this.stopFilmStripRecording.bind(this), this.displayScreenshotDelay); } } stopFilmStripRecording() { if (this.filmStripRecorder) { this.filmStripRecorder.stopRecording(this.filmStripAvailable.bind(this)); } delete this.pendingStopTimer; } toggleLargerRequests() { this.updateUI(); } toggleShowOverview() { const toggled = this.networkLogShowOverviewSetting.get(); if (toggled) { this.overviewPane.show(this.overviewPlaceholderElement); } else { this.overviewPane.detach(); } this.doResize(); } toggleRecordFilmStrip() { const toggled = this.networkRecordFilmStripSetting.get(); if (toggled && !this.filmStripRecorder) { this.filmStripView = new PerfUI.FilmStripView.FilmStripView(); 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; } } resetFilmStripView() { 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() })); } } } elementsToRestoreScrollPositionsFor() { return this.networkLogView.elementsToRestoreScrollPositionsFor(); } wasShown() { 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'); } willHide() { UI.Context.Context.instance().setFlavor(NetworkPanel, null); } revealAndHighlightRequest(request) { this.hideRequestPanel(); if (request) { this.networkLogView.revealAndHighlightRequest(request); } } revealAndHighlightRequestWithId(request) { this.hideRequestPanel(); if (request) { this.networkLogView.revealAndHighlightRequestWithId(request); } } async selectAndActivateRequest(request, shownTab, options) { await UI.ViewManager.ViewManager.instance().showView('network'); this.networkLogView.selectRequest(request, options); this.showRequestPanel(shownTab); this.networkLogView.revealAndHighlightRequest(request); return this.networkItemView; } handleFilterChanged() { this.hideRequestPanel(); } onRowSizeChanged() { this.updateUI(); } onRequestSelected(event) { const request = event.data; this.currentRequest = request; this.networkOverview.setHighlightedRequest(request); this.updateNetworkItemView(); } onRequestActivated(event) { const { showPanel, tab, takeFocus } = event.data; if (showPanel) { this.showRequestPanel(tab, takeFocus); } else { this.hideRequestPanel(); } } showRequestPanel(shownTab, takeFocus) { 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() { this.clearNetworkItemView(); this.splitWidget.hideMain(); this.updateUI(); } updateNetworkItemView() { if (this.splitWidget.showMode() === UI.SplitWidget.ShowMode.Both) { this.clearNetworkItemView(); this.createNetworkItemView(); this.updateUI(); } } clearNetworkItemView() { if (this.networkItemView) { this.networkItemView.detach(); this.networkItemView = null; } } createNetworkItemView(initialTab) { 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; } updateUI() { 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(event, contextMenu, target) { function reveal(request) { void UI.ViewManager.ViewManager.instance() .showView('network') .then(this.networkLogView.resetFilter.bind(this.networkLogView)) .then(this.revealAndHighlightRequest.bind(this, request)); } function appendRevealItem(request) { contextMenu.revealSection().appendItem(i18nString(UIStrings.revealInNetworkPanel), reveal.bind(this, request)); } if (event.target.isSelfOrDescendant(this.element)) { return; } if (target instanceof SDK.Resource.Resource) { const resource = target; if (resource.request) { appendRevealItem.call(this, resource.request); } return; } if (target instanceof Workspace.UISourceCode.UISourceCode) { const uiSourceCode = target; 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; if (this.networkItemView && this.networkItemView.isShowing() && this.networkItemView.request() === request) { return; } appendRevealItem.call(this, request); } onFilmFrameSelected(event) { const timestamp = event.data; this.overviewPane.setWindowTimes(0, timestamp); } onFilmFrameEnter(event) { const timestamp = event.data; this.networkOverview.selectFilmStripFrame(timestamp); this.networkLogView.selectFilmStripFrame(timestamp / 1000); } onFilmFrameExit() { this.networkOverview.clearFilmStripFrame(); this.networkLogView.clearFilmStripFrame(); } onUpdateRequest(event) { const request = event.data; this.calculator.updateBoundaries(request); // FIXME: Unify all time units across the frontend! this.overviewPane.setBounds(TraceEngine.Types.Timing.MilliSeconds(this.calculator.minimumBoundary() * 1000), TraceEngine.Types.Timing.MilliSeconds(this.calculator.maximumBoundary() * 1000)); this.networkOverview.updateRequest(request); this.overviewPane.scheduleUpdate(); } resolveLocation(locationName) { if (locationName === 'network-sidebar') { return this.sidebarLocation; } return null; } } let contextMenuProviderInstance; export class ContextMenuProvider { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!contextMenuProviderInstance || forceNew) { contextMenuProviderInstance = new ContextMenuProvider(); } return contextMenuProviderInstance; } appendApplicableItems(event, contextMenu, target) { NetworkPanel.instance().appendApplicableItems(event, contextMenu, target); } } let requestRevealerInstance; export class RequestRevealer { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!requestRevealerInstance || forceNew) { requestRevealerInstance = new RequestRevealer(); } return requestRevealerInstance; } reveal(request) { 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; export class RequestIdRevealer { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!requestIdRevealerInstance || forceNew) { requestIdRevealerInstance = new RequestIdRevealer(); } return requestIdRevealerInstance; } reveal(requestId) { 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; export class NetworkLogWithFilterRevealer { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!networkLogWithFilterRevealerInstance || forceNew) { networkLogWithFilterRevealerInstance = new NetworkLogWithFilterRevealer(); } return networkLogWithFilterRevealerInstance; } reveal(request) { if (!(request instanceof NetworkForward.UIFilter.UIRequestFilter)) { return Promise.reject(new Error('Internal error: not a UIRequestFilter')); } return NetworkPanel.revealAndFilter(request.filters); } } export class FilmStripRecorder { tracingManager; resourceTreeModel; timeCalculator; filmStripView; tracingModel; callback; // Used to fetch screenshots of the page load and show them in the panel. #traceEngine; constructor(timeCalculator, filmStripView) { this.#traceEngine = new TraceEngine.TraceModel.Model({ Screenshots: TraceEngine.Handlers.ModelHandlers.Screenshots, }); this.tracingManager = null; this.resourceTreeModel = null; this.timeCalculator = timeCalculator; this.filmStripView = filmStripView; this.tracingModel = null; this.callback = null; } traceEventsCollected(events) { if (this.tracingModel) { this.tracingModel.addEvents(events); } } async tracingComplete() { if (!this.tracingModel || !this.tracingManager) { return; } this.tracingModel.tracingComplete(); this.tracingManager = null; await this.#traceEngine.parse( // OPP's data layer uses `EventPayload` as the type to represent raw JSON from the trace. // When we pass this into the new data engine, we need to tell TS to use the new TraceEventData type. this.tracingModel.allRawEvents()); const data = this.#traceEngine.traceParsedData(this.#traceEngine.size() - 1); if (!data) { return; } const zeroTimeInSeconds = TraceEngine.Types.Timing.Seconds(this.timeCalculator.minimumBoundary()); const filmStrip = TraceEngine.Extras.FilmStrip.fromTraceData(data, TraceEngine.Helpers.Timing.secondsToMicroseconds(zeroTimeInSeconds)); if (this.callback) { this.callback(filmStrip); } this.callback = null; // Now we have created the film strip and stored the data, we need to reset // the trace processor so that it is ready to record again if the user // refreshes the page. this.#traceEngine.resetProcessor(); if (this.resourceTreeModel) { this.resourceTreeModel.resumeReload(); } this.resourceTreeModel = null; } tracingBufferUsage() { } eventsRetrievalProgress(_progress) { } startRecording() { this.filmStripView.reset(); this.filmStripView.setStatusText(i18nString(UIStrings.recordingFrames)); const tracingManager = SDK.TargetManager.TargetManager.instance().scopeTarget()?.model(TraceEngine.TracingManager.TracingManager); if (this.tracingManager || !tracingManager) { return; } this.tracingManager = tracingManager; this.resourceTreeModel = this.tracingManager.target().model(SDK.ResourceTreeModel.ResourceTreeModel); this.tracingModel = new TraceEngine.Legacy.TracingModel(); void this.tracingManager.start(this, '-*,disabled-by-default-devtools.screenshot', ''); Host.userMetrics.actionTaken(Host.UserMetrics.Action.FilmStripStartedRecording); } isRecording() { return Boolean(this.tracingManager); } stopRecording(callback) { if (!this.tracingManager) { return; } this.tracingManager.stop(); if (this.resourceTreeModel) { this.resourceTreeModel.suspendReload(); } this.callback = callback; this.filmStripView.setStatusText(i18nString(UIStrings.fetchingFrames)); } } let networkActionDelegateInstance; export class ActionDelegate { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!networkActionDelegateInstance || forceNew) { networkActionDelegateInstance = new ActionDelegate(); } return networkActionDelegateInstance; } handleAction(context, actionId) { 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; export class RequestLocationRevealer { static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!requestLocationRevealerInstance || forceNew) { requestLocationRevealerInstance = new RequestLocationRevealer(); } return requestLocationRevealerInstance; } async reveal(match) { const location = match; 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; export class SearchNetworkView extends Search.SearchView.SearchView { constructor() { super('network'); } static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!searchNetworkViewInstance || forceNew) { searchNetworkViewInstance = new SearchNetworkView(); } return searchNetworkViewInstance; } static async openSearch(query, searchImmediately) { await UI.ViewManager.ViewManager.instance().showView('network.search-network-tab'); const searchView = SearchNetworkView.instance(); void searchView.toggle(query, Boolean(searchImmediately)); return searchView; } createScope() { return new NetworkSearchScope(); } } //# sourceMappingURL=NetworkPanel.js.map