blob: f285fcad32aeda3263a11d1380833795e3ebc190 [file] [log] [blame]
Patrick Hulcea087f622018-05-18 00:37:53 +00001// Copyright 2018 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Tim van der Lippe76961572021-04-06 11:48:07 +01005import * as Common from '../../core/common/common.js';
Tim van der Lippee0247312021-04-01 15:25:30 +01006import * as Host from '../../core/host/host.js';
Tim van der Lippeaa1ed7a2021-03-31 15:38:27 +01007import * as Platform from '../../core/platform/platform.js';
Tim van der Lippee00b92f2021-03-31 17:52:17 +01008import * as SDK from '../../core/sdk/sdk.js';
Tim van der Lippe99aeaf32021-04-09 11:33:34 +01009import * as Workspace from '../../models/workspace/workspace.js';
Connor Clark4a7d8342021-07-19 13:38:37 -070010import * as LighthouseReport from '../../third_party/lighthouse/report/report.js';
Tim van der Lippe339ad262021-04-21 13:23:36 +010011import * as Components from '../../ui/legacy/components/utils/utils.js';
Tim van der Lippeaa61faf2021-04-07 16:32:07 +010012import * as UI from '../../ui/legacy/legacy.js';
Tim van der Lippefddcf402021-04-19 14:00:29 +010013import * as ThemeSupport from '../../ui/legacy/theme_support/theme_support.js';
Tim van der Lippe01e1c462021-04-19 16:04:03 +010014import * as Timeline from '../timeline/timeline.js';
Jack Franklin3a802602022-07-13 08:39:42 +000015import {
16 type RunnerResultArtifacts,
17 type NodeDetailsJSON,
18 type SourceLocationDetailsJSON,
Adam Raine2a9afbc2022-11-30 10:28:58 -080019 type ReportJSON,
Jack Franklin3a802602022-07-13 08:39:42 +000020} from './LighthouseReporterTypes.js';
Tim van der Lippe1e10f852020-10-30 14:35:01 +000021
Paul Lewiscf2ef222019-11-22 14:55:35 +000022const MaxLengthForLinks = 40;
23
Adam Raine2a9afbc2022-11-30 10:28:58 -080024interface RenderReportOpts {
25 beforePrint?: () => void;
26 afterPrint?: () => void;
27}
28
29export class LighthouseReportRenderer {
30 static renderLighthouseReport(lhr: ReportJSON, artifacts?: RunnerResultArtifacts, opts?: RenderReportOpts):
31 HTMLElement {
32 let onViewTrace: (() => Promise<void>)|undefined = undefined;
33 if (artifacts) {
34 onViewTrace = async(): Promise<void> => {
35 const defaultPassTrace = artifacts.traces.defaultPass;
36 Host.userMetrics.actionTaken(Host.UserMetrics.Action.LighthouseViewTrace);
37 await UI.InspectorView.InspectorView.instance().showPanel('timeline');
38 Timeline.TimelinePanel.TimelinePanel.instance().loadFromEvents(defaultPassTrace.traceEvents);
39 };
40 }
41
42 async function onSaveFileOverride(blob: Blob): Promise<void> {
43 const domain = new Common.ParsedURL.ParsedURL(lhr.finalUrl || lhr.finalDisplayedUrl).domain();
44 const sanitizedDomain = domain.replace(/[^a-z0-9.-]+/gi, '_');
45 const timestamp = Platform.DateUtilities.toISO8601Compact(new Date(lhr.fetchTime));
46 const ext = blob.type.match('json') ? '.json' : '.html';
47 const basename = `${sanitizedDomain}-${timestamp}${ext}` as Platform.DevToolsPath.RawPathString;
48 const text = await blob.text();
49 void Workspace.FileManager.FileManager.instance().save(basename, text, true /* forceSaveAs */);
50 }
51
52 async function onPrintOverride(rootEl: HTMLElement): Promise<void> {
53 const clonedReport = rootEl.cloneNode(true);
54 const printWindow = window.open('', '_blank', 'channelmode=1,status=1,resizable=1');
55 if (!printWindow) {
56 return;
57 }
58
59 printWindow.document.body.replaceWith(clonedReport);
60 // Linkified nodes are shadow elements, which aren't exposed via `cloneNode`.
61 await LighthouseReportRenderer.linkifyNodeDetails(clonedReport as HTMLElement);
62
63 opts?.beforePrint?.();
64 printWindow.focus();
65 printWindow.print();
66 printWindow.close();
67 opts?.afterPrint?.();
68 }
69
70 function getStandaloneReportHTML(): string {
71 // @ts-expect-error https://github.com/GoogleChrome/lighthouse/issues/11628
Adam Rainede6c4e52023-02-09 14:10:42 -080072 return Lighthouse.ReportGenerator.ReportGenerator.generateReportHtml(lhr);
Adam Raine2a9afbc2022-11-30 10:28:58 -080073 }
74
75 const reportEl = LighthouseReport.renderReport(lhr, {
76 // Disable dark mode so we can manually adjust it.
77 disableDarkMode: true,
78 onViewTrace,
79 onSaveFileOverride,
80 onPrintOverride,
81 getStandaloneReportHTML,
82 });
83 reportEl.classList.add('lh-devtools');
84
85 const updateDarkModeIfNecessary = (): void => {
86 reportEl.classList.toggle('lh-dark', ThemeSupport.ThemeSupport.instance().themeName() === 'dark');
87 };
88 ThemeSupport.ThemeSupport.instance().addEventListener(
89 ThemeSupport.ThemeChangeEvent.eventName, updateDarkModeIfNecessary);
90 updateDarkModeIfNecessary();
91
92 // @ts-ignore Expose LHR on DOM for e2e tests
93 reportEl._lighthouseResultForTesting = lhr;
94 // @ts-ignore Expose Artifacts on DOM for e2e tests
95 reportEl._lighthouseArtifactsForTesting = artifacts;
96
97 // Linkifying requires the target be loaded. Do not block the report
98 // from rendering, as this is just an embellishment and the main target
99 // could take awhile to load.
100 void LighthouseReportRenderer.waitForMainTargetLoad().then(() => {
101 void LighthouseReportRenderer.linkifyNodeDetails(reportEl);
102 void LighthouseReportRenderer.linkifySourceLocationDetails(reportEl);
103 });
104
105 return reportEl;
Tim van der Lippe4df32c92020-11-06 12:35:05 +0000106 }
Jan Schefflerc5a400f2021-01-22 17:41:47 +0100107
Adam Raine2a9afbc2022-11-30 10:28:58 -0800108 static async waitForMainTargetLoad(): Promise<void> {
109 const mainTarget = SDK.TargetManager.TargetManager.instance().mainFrameTarget();
110 if (!mainTarget) {
Paul Irish8f1e33d2018-05-31 02:29:50 +0000111 return;
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000112 }
Adam Raine2a9afbc2022-11-30 10:28:58 -0800113 const resourceTreeModel = mainTarget.model(SDK.ResourceTreeModel.ResourceTreeModel);
114 if (!resourceTreeModel) {
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000115 return;
116 }
Adam Raine2a9afbc2022-11-30 10:28:58 -0800117 await resourceTreeModel.once(SDK.ResourceTreeModel.Events.Load);
Patrick Hulcea087f622018-05-18 00:37:53 +0000118 }
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000119
Jan Schefflerc5a400f2021-01-22 17:41:47 +0100120 static async linkifyNodeDetails(el: Element): Promise<void> {
Danil Somsikove1ee1b22022-11-07 18:44:57 +0100121 const mainTarget = SDK.TargetManager.TargetManager.instance().mainFrameTarget();
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000122 if (!mainTarget) {
123 return;
124 }
Paul Lewisdaac1062020-03-05 14:37:10 +0000125 const domModel = mainTarget.model(SDK.DOMModel.DOMModel);
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000126 if (!domModel) {
127 return;
128 }
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000129
130 for (const origElement of el.getElementsByClassName('lh-node')) {
Jan Schefflerc5a400f2021-01-22 17:41:47 +0100131 const origHTMLElement = origElement as HTMLElement;
Connor Clark4a7d8342021-07-19 13:38:37 -0700132 const detailsItem = origHTMLElement.dataset as unknown as NodeDetailsJSON;
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000133 if (!detailsItem.path) {
cjamcl@google.com5c46b7e2018-12-04 14:44:47 +0000134 continue;
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000135 }
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000136
137 const nodeId = await domModel.pushNodeByPathToFrontend(detailsItem.path);
138
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000139 if (!nodeId) {
cjamcl@google.com5c46b7e2018-12-04 14:44:47 +0000140 continue;
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000141 }
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000142 const node = domModel.nodeForId(nodeId);
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000143 if (!node) {
cjamcl@google.com5c46b7e2018-12-04 14:44:47 +0000144 continue;
Tim van der Lippe1d6e57a2019-09-30 11:55:34 +0000145 }
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000146
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000147 const element = await Common.Linkifier.Linkifier.linkify(
148 node, {tooltip: detailsItem.snippet, preventKeyboardFocus: undefined});
Tim van der Lippe70842f32020-11-23 16:56:57 +0000149 UI.Tooltip.Tooltip.install(origHTMLElement, '');
Connor Clark49872c02020-12-16 13:39:28 -0600150
151 const screenshotElement = origHTMLElement.querySelector('.lh-element-screenshot');
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000152 origHTMLElement.textContent = '';
Connor Clark49872c02020-12-16 13:39:28 -0600153 if (screenshotElement) {
154 origHTMLElement.append(screenshotElement);
155 }
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000156 origHTMLElement.appendChild(element);
cjamcl@google.comda0e4c62018-11-27 23:52:10 +0000157 }
158 }
cjamcl@google.comf2f8c092019-05-30 22:01:56 +0000159
Jan Schefflerc5a400f2021-01-22 17:41:47 +0100160 static async linkifySourceLocationDetails(el: Element): Promise<void> {
Connor Clark0403a422019-11-18 18:03:18 -0800161 for (const origElement of el.getElementsByClassName('lh-source-location')) {
Jan Schefflerc5a400f2021-01-22 17:41:47 +0100162 const origHTMLElement = origElement as HTMLElement;
Connor Clark4a7d8342021-07-19 13:38:37 -0700163 const detailsItem = origHTMLElement.dataset as SourceLocationDetailsJSON;
Connor Clark0403a422019-11-18 18:03:18 -0800164 if (!detailsItem.sourceUrl || !detailsItem.sourceLine || !detailsItem.sourceColumn) {
165 continue;
166 }
167 const url = detailsItem.sourceUrl;
168 const line = Number(detailsItem.sourceLine);
169 const column = Number(detailsItem.sourceColumn);
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000170 const element = await Components.Linkifier.Linkifier.linkifyURL(url, {
171 lineNumber: line,
172 columnNumber: column,
Brandon Walderman666b40a2021-10-19 13:29:05 -0700173 showColumnNumber: false,
Philip Pfaffe068b01d2021-03-22 10:46:26 +0100174 inlineFrameIndex: 0,
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000175 maxLength: MaxLengthForLinks,
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000176 });
Tim van der Lippe70842f32020-11-23 16:56:57 +0000177 UI.Tooltip.Tooltip.install(origHTMLElement, '');
Tim van der Lippefe3cdc92020-11-06 15:23:13 +0000178 origHTMLElement.textContent = '';
179 origHTMLElement.appendChild(element);
Connor Clark0403a422019-11-18 18:03:18 -0800180 }
181 }
Paul Lewiscf2ef222019-11-22 14:55:35 +0000182}