| // Copyright 2021 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| import '../../../ui/components/expandable_list/expandable_list.js'; |
| import '../../../ui/components/report_view/report_view.js'; |
| import '../../../ui/legacy/legacy.js'; |
| import '../../../ui/kit/kit.js'; |
| |
| import * as Common from '../../../core/common/common.js'; |
| import * as i18n from '../../../core/i18n/i18n.js'; |
| import type * as Platform from '../../../core/platform/platform.js'; |
| import * as SDK from '../../../core/sdk/sdk.js'; |
| import * as Protocol from '../../../generated/protocol.js'; |
| import * as Buttons from '../../../ui/components/buttons/buttons.js'; |
| import type * as ExpandableList from '../../../ui/components/expandable_list/expandable_list.js'; |
| import type * as ReportView from '../../../ui/components/report_view/report_view.js'; |
| import * as Components from '../../../ui/legacy/components/utils/utils.js'; |
| import * as UI from '../../../ui/legacy/legacy.js'; |
| import {html, type LitTemplate, nothing, render, type TemplateResult} from '../../../ui/lit/lit.js'; |
| import * as VisualLogging from '../../../ui/visual_logging/visual_logging.js'; |
| |
| import {NotRestoredReasonDescription} from './BackForwardCacheStrings.js'; |
| import backForwardCacheViewStyles from './backForwardCacheView.css.js'; |
| |
| const UIStrings = { |
| /** |
| * @description Title text in back/forward cache view of the Application panel |
| */ |
| mainFrame: 'Main Frame', |
| /** |
| * @description Title text in back/forward cache view of the Application panel |
| */ |
| backForwardCacheTitle: 'Back/forward cache', |
| /** |
| * @description Status text for the status of the main frame |
| */ |
| unavailable: 'unavailable', |
| /** |
| * @description Entry name text in the back/forward cache view of the Application panel |
| */ |
| url: 'URL', |
| /** |
| * @description Status text for the status of the back/forward cache status |
| */ |
| unknown: 'Unknown Status', |
| /** |
| * @description Status text for the status of the back/forward cache status indicating that |
| * the back/forward cache was not used and a normal navigation occurred instead. |
| */ |
| normalNavigation: |
| 'Not served from back/forward cache: to trigger back/forward cache, use Chrome\'s back/forward buttons, or use the test button below to automatically navigate away and back.', |
| /** |
| * @description Status text for the status of the back/forward cache status indicating that |
| * the back/forward cache was used to restore the page instead of reloading it. |
| */ |
| restoredFromBFCache: 'Successfully served from back/forward cache.', |
| /** |
| * @description Label for a list of reasons which prevent the page from being eligible for |
| * back/forward cache. These reasons are actionable i.e. they can be cleaned up to make the |
| * page eligible for back/forward cache. |
| */ |
| pageSupportNeeded: 'Actionable', |
| /** |
| * @description Label for the completion of the back/forward cache test |
| */ |
| testCompleted: 'Back/forward cache test completed.', |
| /** |
| * @description Explanation for actionable items which prevent the page from being eligible |
| * for back/forward cache. |
| */ |
| pageSupportNeededExplanation: |
| 'These reasons are actionable i.e. they can be cleaned up to make the page eligible for back/forward cache.', |
| /** |
| * @description Label for a list of reasons which prevent the page from being eligible for |
| * back/forward cache. These reasons are circumstantial / not actionable i.e. they cannot be |
| * cleaned up by developers to make the page eligible for back/forward cache. |
| */ |
| circumstantial: 'Not Actionable', |
| /** |
| * @description Explanation for circumstantial/non-actionable items which prevent the page from being eligible |
| * for back/forward cache. |
| */ |
| circumstantialExplanation: |
| 'These reasons are not actionable i.e. caching was prevented by something outside of the direct control of the page.', |
| /** |
| * @description Label for a list of reasons which prevent the page from being eligible for |
| * back/forward cache. These reasons are pending support by chrome i.e. in a future version |
| * of chrome they will not prevent back/forward cache usage anymore. |
| */ |
| supportPending: 'Pending Support', |
| /** |
| * @description Label for the button to test whether BFCache is available for the page |
| */ |
| runTest: 'Test back/forward cache', |
| /** |
| * @description Label for the disabled button while the test is running |
| */ |
| runningTest: 'Running test', |
| /** |
| * @description Link Text about explanation of back/forward cache |
| */ |
| learnMore: 'Learn more: back/forward cache eligibility', |
| /** |
| * @description Link Text about unload handler |
| */ |
| neverUseUnload: 'Learn more: Never use unload handler', |
| /** |
| * @description Explanation for 'pending support' items which prevent the page from being eligible |
| * for back/forward cache. |
| */ |
| supportPendingExplanation: |
| 'Chrome support for these reasons is pending i.e. they will not prevent the page from being eligible for back/forward cache in a future version of Chrome.', |
| /** |
| * @description Text that precedes displaying a link to the extension which blocked the page from being eligible for back/forward cache. |
| */ |
| blockingExtensionId: 'Extension id: ', |
| /** |
| * @description Label for the 'Frames' section of the back/forward cache view, which shows a frame tree of the |
| * page with reasons why the frames can't be cached. |
| */ |
| framesTitle: 'Frames', |
| /** |
| * @description Top level summary of the total number of issues found in a single frame. |
| */ |
| issuesInSingleFrame: '{n, plural, =1 {# issue found in 1 frame.} other {# issues found in 1 frame.}}', |
| /** |
| * @description Top level summary of the total number of issues found and the number of frames they were found in. |
| * 'm' is never less than 2. |
| * @example {3} m |
| */ |
| issuesInMultipleFrames: '{n, plural, =1 {# issue found in {m} frames.} other {# issues found in {m} frames.}}', |
| /** |
| * @description Shows the number of frames with a particular issue. |
| */ |
| framesPerIssue: '{n, plural, =1 {# frame} other {# frames}}', |
| /** |
| * @description Title for a frame in the frame tree that doesn't have a URL. Placeholder indicates which number frame with a blank URL it is. |
| * @example {3} PH1 |
| */ |
| blankURLTitle: 'Blank URL [{PH1}]', |
| /** |
| * @description Shows the number of files with a particular issue. |
| */ |
| filesPerIssue: '{n, plural, =1 {# file} other {# files}}', |
| } as const; |
| |
| const str_ = i18n.i18n.registerUIStrings('panels/application/components/BackForwardCacheView.ts', UIStrings); |
| const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); |
| const {widgetConfig} = UI.Widget; |
| |
| const enum ScreenStatusType { |
| RUNNING = 'Running', |
| RESULT = 'Result', |
| } |
| |
| function renderMainFrameInformation( |
| frame: SDK.ResourceTreeModel.ResourceTreeFrame|null, |
| frameTreeData: {node: FrameTreeNodeData, frameCount: number, issueCount: number}|undefined, |
| reasonToFramesMap: Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>, screenStatus: ScreenStatusType, |
| navigateAwayAndBack: () => Promise<void>): TemplateResult { |
| if (!frame) { |
| // clang-format of |
| return html` |
| <devtools-report-key> |
| ${i18nString(UIStrings.mainFrame)} |
| </devtools-report-key> |
| <devtools-report-value> |
| ${i18nString(UIStrings.unavailable)} |
| </devtools-report-value>`; |
| // clang-format on |
| } |
| const isTestRunning = (screenStatus === ScreenStatusType.RUNNING); |
| // Prevent running BFCache test on the DevTools window itself via DevTools on DevTools |
| const isTestingForbidden = Common.ParsedURL.schemeIs(frame.url, 'devtools:'); |
| // clang-format off |
| return html` |
| ${renderBackForwardCacheStatus(frame.backForwardCacheDetails.restoredFromCache)} |
| <devtools-report-key>${i18nString(UIStrings.url)}</devtools-report-key> |
| <devtools-report-value>${frame.url}</devtools-report-value> |
| ${maybeRenderFrameTree(frameTreeData)} |
| <devtools-report-section> |
| <devtools-button |
| aria-label=${i18nString(UIStrings.runTest)} |
| .disabled=${isTestRunning || isTestingForbidden} |
| .spinner=${isTestRunning} |
| .variant=${Buttons.Button.Variant.PRIMARY} |
| @click=${navigateAwayAndBack} |
| jslog=${VisualLogging.action('back-forward-cache.run-test').track({click: true})}> |
| ${isTestRunning ? html` |
| ${i18nString(UIStrings.runningTest)}`:` |
| ${i18nString(UIStrings.runTest)} |
| `} |
| </devtools-button> |
| </devtools-report-section> |
| <devtools-report-divider> |
| </devtools-report-divider> |
| ${maybeRenderExplanations(frame.backForwardCacheDetails.explanations, |
| frame.backForwardCacheDetails.explanationsTree, |
| reasonToFramesMap)} |
| <devtools-report-section> |
| <x-link href="https://web.dev/bfcache/" class="link" |
| jslog=${VisualLogging.action('learn-more.eligibility').track({click: true})}> |
| ${i18nString(UIStrings.learnMore)} |
| </x-link> |
| </devtools-report-section>`; |
| // clang-format on |
| } |
| |
| function maybeRenderFrameTree( |
| frameTreeData: {node: FrameTreeNodeData, frameCount: number, issueCount: number}|undefined): LitTemplate { |
| if (!frameTreeData || (frameTreeData.frameCount === 0 && frameTreeData.issueCount === 0)) { |
| return nothing; |
| } |
| |
| function renderFrameTreeNode(node: FrameTreeNodeData): TemplateResult { |
| // clang-format off |
| return html` |
| <li role="treeitem" class="text-ellipsis"> |
| ${node.iconName ? html` |
| <devtools-icon class="inline-icon extra-large" .name=${node.iconName} style="margin-bottom: -3px;"> |
| </devtools-icon> |
| ` : nothing} |
| ${node.text} |
| ${node.children?.length ? html` |
| <ul role="group" hidden> |
| ${node.children.map(child => renderFrameTreeNode(child))} |
| </ul>` : nothing} |
| </li>`; |
| // clang-format on |
| } |
| |
| let title = ''; |
| // The translation pipeline does not support nested plurals. We avoid this |
| // here by pulling out the logic for one of the plurals into code instead. |
| if (frameTreeData.frameCount === 1) { |
| title = i18nString(UIStrings.issuesInSingleFrame, {n: frameTreeData.issueCount}); |
| } else { |
| title = i18nString(UIStrings.issuesInMultipleFrames, {n: frameTreeData.issueCount, m: frameTreeData.frameCount}); |
| } |
| // clang-format off |
| return html` |
| <devtools-report-key jslog=${VisualLogging.section('frames')}>${i18nString(UIStrings.framesTitle)}</devtools-report-key> |
| <devtools-report-value> |
| <devtools-tree .template=${html` |
| <ul role="tree"> |
| <li role="treeitem" class="text-ellipsis"> |
| ${title} |
| <ul role="group"> |
| ${renderFrameTreeNode(frameTreeData.node)} |
| </ul> |
| </li> |
| </ul> |
| `}> |
| </devtools-tree> |
| </devtools-report-value>`; |
| // clang-format on |
| } |
| |
| function renderBackForwardCacheStatus(status: boolean|undefined): TemplateResult { |
| switch (status) { |
| case true: |
| // clang-format off |
| return html` |
| <devtools-report-section autofocus tabindex="-1"> |
| <div class="status extra-large"> |
| <devtools-icon class="inline-icon extra-large" name="check-circle" style="color: var(--icon-checkmark-green);"> |
| </devtools-icon> |
| </div> |
| ${i18nString(UIStrings.restoredFromBFCache)} |
| </devtools-report-section>`; |
| // clang-format on |
| case false: |
| // clang-format off |
| return html` |
| <devtools-report-section autofocus tabindex="-1"> |
| <div class="status"> |
| <devtools-icon class="inline-icon extra-large" name="clear"> |
| </devtools-icon> |
| </div> |
| ${i18nString(UIStrings.normalNavigation)} |
| </devtools-report-section>`; |
| // clang-format on |
| } |
| // clang-format off |
| return html` |
| <devtools-report-section autofocus tabindex="-1"> |
| ${i18nString(UIStrings.unknown)} |
| </devtools-report-section>`; |
| // clang-format on |
| } |
| |
| function maybeRenderExplanations( |
| explanations: Protocol.Page.BackForwardCacheNotRestoredExplanation[], |
| explanationTree: Protocol.Page.BackForwardCacheNotRestoredExplanationTree|undefined, |
| reasonToFramesMap: Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>): LitTemplate { |
| if (explanations.length === 0) { |
| return nothing; |
| } |
| |
| const pageSupportNeeded = explanations.filter( |
| explanation => explanation.type === Protocol.Page.BackForwardCacheNotRestoredReasonType.PageSupportNeeded); |
| const supportPending = explanations.filter( |
| explanation => explanation.type === Protocol.Page.BackForwardCacheNotRestoredReasonType.SupportPending); |
| const circumstantial = explanations.filter( |
| explanation => explanation.type === Protocol.Page.BackForwardCacheNotRestoredReasonType.Circumstantial); |
| |
| // Disabled until https://crbug.com/1079231 is fixed. |
| // clang-format off |
| return html` |
| ${renderExplanations(i18nString(UIStrings.pageSupportNeeded), i18nString(UIStrings.pageSupportNeededExplanation), pageSupportNeeded, reasonToFramesMap)} |
| ${renderExplanations(i18nString(UIStrings.supportPending), i18nString(UIStrings.supportPendingExplanation), supportPending, reasonToFramesMap)} |
| ${renderExplanations(i18nString(UIStrings.circumstantial), i18nString(UIStrings.circumstantialExplanation), circumstantial, reasonToFramesMap)}`; |
| // clang-format on |
| } |
| |
| function renderExplanations( |
| category: Platform.UIString.LocalizedString, explainerText: Platform.UIString.LocalizedString, |
| explanations: Protocol.Page.BackForwardCacheNotRestoredExplanation[], |
| reasonToFramesMap: Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>): TemplateResult { |
| // Disabled until https://crbug.com/1079231 is fixed. |
| // clang-format off |
| return html` |
| ${explanations.length > 0 ? html` |
| <devtools-report-section-header> |
| ${category} |
| <div class="help-outline-icon"> |
| <devtools-icon class="inline-icon medium" name="help" title=${explainerText}> |
| </devtools-icon> |
| </div> |
| </devtools-report-section-header> |
| ${explanations.map(explanation => renderReason(explanation, reasonToFramesMap.get(explanation.reason)))} |
| ` : nothing}`; |
| // clang-format on |
| } |
| |
| function maybeRenderReasonContext(explanation: Protocol.Page.BackForwardCacheNotRestoredExplanation): LitTemplate { |
| if (explanation.reason === |
| Protocol.Page.BackForwardCacheNotRestoredReason.EmbedderExtensionSentMessageToCachedFrame && |
| explanation.context) { |
| const link = 'chrome://extensions/?id=' + explanation.context as Platform.DevToolsPath.UrlString; |
| // clang-format off |
| return html`${i18nString(UIStrings.blockingExtensionId)} |
| <devtools-link .href=${link}>${explanation.context}</devtools-link>`; |
| // clang-format on |
| } |
| return nothing; |
| } |
| |
| function renderFramesPerReason(frames: string[]|undefined): LitTemplate { |
| if (frames === undefined || frames.length === 0) { |
| return nothing; |
| } |
| const rows = [html`<div>${i18nString(UIStrings.framesPerIssue, {n: frames.length})}</div>`]; |
| rows.push(...frames.map(url => html`<div class="text-ellipsis" title=${url} |
| jslog=${VisualLogging.treeItem()}>${url}</div>`)); |
| return html` |
| <div class="details-list" |
| jslog=${VisualLogging.tree('frames-per-issue')}> |
| <devtools-expandable-list .data=${{ |
| rows, |
| title: i18nString(UIStrings.framesPerIssue, {n: frames.length}), |
| } as ExpandableList.ExpandableList.ExpandableListData} |
| jslog=${VisualLogging.treeItem()}></devtools-expandable-list> |
| </div> |
| `; |
| } |
| |
| function maybeRenderDeepLinkToUnload(explanation: Protocol.Page.BackForwardCacheNotRestoredExplanation): LitTemplate { |
| if (explanation.reason === Protocol.Page.BackForwardCacheNotRestoredReason.UnloadHandlerExistsInMainFrame || |
| explanation.reason === Protocol.Page.BackForwardCacheNotRestoredReason.UnloadHandlerExistsInSubFrame) { |
| return html` |
| <x-link href="https://web.dev/bfcache/#never-use-the-unload-event" class="link" |
| jslog=${VisualLogging.action('learn-more.never-use-unload').track({ |
| click: true, |
| })}> |
| ${i18nString(UIStrings.neverUseUnload)} |
| </x-link>`; |
| } |
| return nothing; |
| } |
| |
| function maybeRenderJavaScriptDetails(details: Protocol.Page.BackForwardCacheBlockingDetails[]|undefined): LitTemplate { |
| if (details === undefined || details.length === 0) { |
| return nothing; |
| } |
| const maxLengthForDisplayedURLs = 50; |
| const rows = [html`<div>${i18nString(UIStrings.filesPerIssue, {n: details.length})}</div>`]; |
| rows.push(...details.map(detail => html` |
| <devtools-widget .widgetConfig=${widgetConfig(Components.Linkifier.ScriptLocationLink, { |
| sourceURL: detail.url as Platform.DevToolsPath.UrlString, |
| lineNumber: detail.lineNumber, |
| options: { |
| columnNumber: detail.columnNumber, |
| showColumnNumber: true, |
| inlineFrameIndex: 0, |
| maxLength: maxLengthForDisplayedURLs, |
| } |
| })}></devtools-widget>`)); |
| return html` |
| <div class="details-list"> |
| <devtools-expandable-list .data=${ |
| {rows} as ExpandableList.ExpandableList.ExpandableListData}></devtools-expandable-list> |
| </div> |
| `; |
| } |
| |
| function renderReason( |
| explanation: Protocol.Page.BackForwardCacheNotRestoredExplanation, frames: string[]|undefined): TemplateResult { |
| // clang-format off |
| return html` |
| <devtools-report-section> |
| ${(explanation.reason in NotRestoredReasonDescription) ? |
| html` |
| <div class="circled-exclamation-icon"> |
| <devtools-icon class="inline-icon medium" style="color: var(--icon-warning)" name="warning"> |
| </devtools-icon> |
| </div> |
| <div> |
| ${NotRestoredReasonDescription[explanation.reason].name()} |
| ${maybeRenderDeepLinkToUnload(explanation)} |
| ${maybeRenderReasonContext(explanation)} |
| </div>` : |
| nothing} |
| </devtools-report-section> |
| <div class="gray-text"> |
| ${explanation.reason} |
| </div> |
| ${maybeRenderJavaScriptDetails(explanation.details)} |
| ${renderFramesPerReason(frames)}`; |
| // clang-format on |
| } |
| |
| interface ViewInput { |
| frame: SDK.ResourceTreeModel.ResourceTreeFrame|null; |
| frameTreeData: {node: FrameTreeNodeData, frameCount: number, issueCount: number}|undefined; |
| reasonToFramesMap: Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>; |
| screenStatus: ScreenStatusType; |
| navigateAwayAndBack: () => Promise<void>; |
| } |
| |
| type View = (input: ViewInput, output: undefined, target: HTMLElement) => void; |
| |
| const DEFAULT_VIEW: View = (input, output, target) => { |
| // Disabled until https://crbug.com/1079231 is fixed. |
| // clang-format off |
| render(html` |
| <style>${backForwardCacheViewStyles}</style> |
| <devtools-report .data=${ |
| {reportTitle: i18nString(UIStrings.backForwardCacheTitle)} as ReportView.ReportView.ReportData |
| } jslog=${VisualLogging.pane('back-forward-cache')}> |
| |
| ${renderMainFrameInformation(input.frame, input.frameTreeData, input.reasonToFramesMap, input.screenStatus, input.navigateAwayAndBack)} |
| </devtools-report> |
| `, target); |
| // clang-format on |
| }; |
| |
| export class BackForwardCacheView extends UI.Widget.Widget { |
| #screenStatus = ScreenStatusType.RESULT; |
| #historyIndex = 0; |
| #view: View; |
| |
| constructor(view = DEFAULT_VIEW) { |
| super({useShadowDom: true, delegatesFocus: true}); |
| this.#view = view; |
| this.#getMainResourceTreeModel()?.addEventListener( |
| SDK.ResourceTreeModel.Events.PrimaryPageChanged, this.requestUpdate, this); |
| this.#getMainResourceTreeModel()?.addEventListener( |
| SDK.ResourceTreeModel.Events.BackForwardCacheDetailsUpdated, this.requestUpdate, this); |
| this.requestUpdate(); |
| } |
| |
| #getMainResourceTreeModel(): SDK.ResourceTreeModel.ResourceTreeModel|null { |
| const mainTarget = SDK.TargetManager.TargetManager.instance().primaryPageTarget(); |
| return mainTarget?.model(SDK.ResourceTreeModel.ResourceTreeModel) || null; |
| } |
| |
| #getMainFrame(): SDK.ResourceTreeModel.ResourceTreeFrame|null { |
| return this.#getMainResourceTreeModel()?.mainFrame || null; |
| } |
| |
| override async performUpdate(): Promise<void> { |
| const reasonToFramesMap = new Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>(); |
| const frame = this.#getMainFrame(); |
| const explanationTree = frame?.backForwardCacheDetails?.explanationsTree; |
| if (explanationTree) { |
| this.#buildReasonToFramesMap(explanationTree, {blankCount: 1}, reasonToFramesMap); |
| } |
| const frameTreeData = this.#buildFrameTreeDataRecursive(explanationTree, {blankCount: 1}); |
| // Override the icon for the outermost frame. |
| frameTreeData.node.iconName = 'frame'; |
| const viewInput: ViewInput = { |
| frame, |
| frameTreeData, |
| reasonToFramesMap, |
| screenStatus: this.#screenStatus, |
| navigateAwayAndBack: this.#navigateAwayAndBack.bind(this), |
| }; |
| this.#view(viewInput, undefined, this.contentElement); |
| } |
| |
| #renderBackForwardCacheTestResult(): void { |
| SDK.TargetManager.TargetManager.instance().removeModelListener( |
| SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.FrameNavigated, |
| this.#renderBackForwardCacheTestResult, this); |
| this.#screenStatus = ScreenStatusType.RESULT; |
| this.requestUpdate(); |
| void this.updateComplete.then(() => { |
| UI.ARIAUtils.LiveAnnouncer.alert(i18nString(UIStrings.testCompleted)); |
| this.contentElement.focus(); |
| }); |
| } |
| |
| async #onNavigatedAway(): Promise<void> { |
| SDK.TargetManager.TargetManager.instance().removeModelListener( |
| SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.FrameNavigated, this.#onNavigatedAway, |
| this); |
| await this.#waitAndGoBackInHistory(50); |
| } |
| |
| async #waitAndGoBackInHistory(delay: number): Promise<void> { |
| const mainTarget = SDK.TargetManager.TargetManager.instance().primaryPageTarget(); |
| const resourceTreeModel = mainTarget?.model(SDK.ResourceTreeModel.ResourceTreeModel); |
| const historyResults = await resourceTreeModel?.navigationHistory(); |
| if (!resourceTreeModel || !historyResults) { |
| return; |
| } |
| // The navigation history can be delayed. If this is the case we wait and |
| // check again later. Otherwise it would be possible to press the 'Test |
| // BFCache' button again too soon, leading to the browser stepping back in |
| // history without returning to the correct page. |
| if (historyResults.currentIndex === this.#historyIndex) { |
| window.setTimeout(this.#waitAndGoBackInHistory.bind(this, delay * 2), delay); |
| } else { |
| SDK.TargetManager.TargetManager.instance().addModelListener( |
| SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.FrameNavigated, |
| this.#renderBackForwardCacheTestResult, this); |
| resourceTreeModel.navigateToHistoryEntry(historyResults.entries[historyResults.currentIndex - 1]); |
| } |
| } |
| |
| async #navigateAwayAndBack(): Promise<void> { |
| // Checking BFCache Compatibility |
| |
| const mainTarget = SDK.TargetManager.TargetManager.instance().primaryPageTarget(); |
| const resourceTreeModel = mainTarget?.model(SDK.ResourceTreeModel.ResourceTreeModel); |
| const historyResults = await resourceTreeModel?.navigationHistory(); |
| if (!resourceTreeModel || !historyResults) { |
| return; |
| } |
| this.#historyIndex = historyResults.currentIndex; |
| this.#screenStatus = ScreenStatusType.RUNNING; |
| this.requestUpdate(); |
| |
| // This event listener is removed inside of onNavigatedAway(). |
| SDK.TargetManager.TargetManager.instance().addModelListener( |
| SDK.ResourceTreeModel.ResourceTreeModel, SDK.ResourceTreeModel.Events.FrameNavigated, this.#onNavigatedAway, |
| this); |
| |
| // We can know whether the current page can use BFCache |
| // as the browser navigates to another unrelated page and goes back to the current page. |
| // We chose "chrome://terms" because it must be cross-site. |
| // Ideally, We want to have our own testing page like "chrome: //bfcache-test". |
| void resourceTreeModel.navigate('chrome://terms' as Platform.DevToolsPath.UrlString); |
| } |
| |
| // Builds a subtree of the frame tree, conaining only frames with BFCache issues and their ancestors. |
| // Returns the root node, the number of frames in the subtree, and the number of issues in the subtree. |
| #buildFrameTreeDataRecursive( |
| explanationTree: Protocol.Page.BackForwardCacheNotRestoredExplanationTree|undefined, |
| nextBlankURLCount: {blankCount: number}): {node: FrameTreeNodeData, frameCount: number, issueCount: number} { |
| if (!explanationTree) { |
| return {node: {text: ''}, frameCount: 0, issueCount: 0}; |
| } |
| let frameCount = 1; |
| let issueCount = 0; |
| const children: FrameTreeNodeData[] = []; |
| |
| let nodeUrlText = ''; |
| if (explanationTree.url.length) { |
| nodeUrlText = explanationTree.url; |
| } else { |
| nodeUrlText = i18nString(UIStrings.blankURLTitle, {PH1: nextBlankURLCount.blankCount}); |
| nextBlankURLCount.blankCount += 1; |
| } |
| |
| for (const explanation of explanationTree.explanations) { |
| const child = {text: explanation.reason}; |
| issueCount += 1; |
| children.push(child); |
| } |
| for (const child of explanationTree.children) { |
| const frameTreeData = this.#buildFrameTreeDataRecursive(child, nextBlankURLCount); |
| if (frameTreeData.issueCount > 0) { |
| children.push(frameTreeData.node); |
| issueCount += frameTreeData.issueCount; |
| frameCount += frameTreeData.frameCount; |
| } |
| } |
| |
| let node: FrameTreeNodeData = { |
| text: `(${issueCount}) ${nodeUrlText}`, |
| }; |
| if (children.length) { |
| node = {...node, children}; |
| node.iconName = 'iframe'; |
| } else if (!explanationTree.url.length) { |
| // If the current node increased the blank count, but it has no children and |
| // is therefore not shown, decrement the blank count again. |
| nextBlankURLCount.blankCount -= 1; |
| } |
| return {node, frameCount, issueCount}; |
| } |
| |
| #buildReasonToFramesMap( |
| explanationTree: Protocol.Page.BackForwardCacheNotRestoredExplanationTree, |
| nextBlankURLCount: {blankCount: number}, |
| outputMap: Map<Protocol.Page.BackForwardCacheNotRestoredReason, string[]>): void { |
| let url = explanationTree.url; |
| if (url.length === 0) { |
| url = i18nString(UIStrings.blankURLTitle, {PH1: nextBlankURLCount.blankCount}); |
| nextBlankURLCount.blankCount += 1; |
| } |
| explanationTree.explanations.forEach(explanation => { |
| let frames: string[]|undefined = outputMap.get(explanation.reason); |
| if (frames === undefined) { |
| frames = [url]; |
| outputMap.set(explanation.reason, frames); |
| } else { |
| frames.push(url); |
| } |
| }); |
| explanationTree.children.map(child => { |
| this.#buildReasonToFramesMap(child, nextBlankURLCount, outputMap); |
| }); |
| } |
| } |
| |
| interface FrameTreeNodeData { |
| text: string; |
| iconName?: string; |
| children?: FrameTreeNodeData[]; |
| } |