blob: 58a556ca3e5ca41b2b53f28b355927bf3dcf296f [file] [log] [blame]
Wolfgang Beyer40530b682022-05-17 13:02:011// Copyright 2022 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
5import * as Common from '../../../core/common/common.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:136import * as Host from '../../../core/host/host.js';
Wolfgang Beyer40530b682022-05-17 13:02:017import * as i18n from '../../../core/i18n/i18n.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:138import * as Platform from '../../../core/platform/platform.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:259import * as Root from '../../../core/root/root.js';
Wolfgang Beyer40530b682022-05-17 13:02:0110import * as SDK from '../../../core/sdk/sdk.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:2511import * as Persistence from '../../../models/persistence/persistence.js';
12import * as Workspace from '../../../models/workspace/workspace.js';
Wolfgang Beyer40426d82022-08-03 11:08:0713import * as NetworkForward from '../../../panels/network/forward/forward.js';
Wolfgang Beyerd1dd7962022-05-24 15:48:4814import * as Buttons from '../../../ui/components/buttons/buttons.js';
Wolfgang Beyer40530b682022-05-17 13:02:0115import * as ComponentHelpers from '../../../ui/components/helpers/helpers.js';
Wolfgang Beyer7f495ee2022-07-08 13:45:1316import * as IconButton from '../../../ui/components/icon_button/icon_button.js';
Jack Franklin1f19d522022-07-06 14:00:1817import * as Input from '../../../ui/components/input/input.js';
Wolfgang Beyer40530b682022-05-17 13:02:0118import * as UI from '../../../ui/legacy/legacy.js';
19import * as LitHtml from '../../../ui/lit-html/lit-html.js';
Wolfgang Beyer185fb1f2022-08-01 13:48:2520import * as Sources from '../../sources/sources.js';
Wolfgang Beyerd2286472022-09-02 18:32:3021
Wolfgang Beyer65a82282022-09-05 08:22:3722import {type RequestHeaderSectionData, RequestHeaderSection} from './RequestHeaderSection.js';
Wolfgang Beyerd2286472022-09-02 18:32:3023import {type ResponseHeaderSectionData, ResponseHeaderSection} from './ResponseHeaderSection.js';
Wolfgang Beyer40530b682022-05-17 13:02:0124
25import requestHeadersViewStyles from './RequestHeadersView.css.js';
26
Wolfgang Beyerd1dd7962022-05-24 15:48:4827const RAW_HEADER_CUTOFF = 3000;
Wolfgang Beyer40530b682022-05-17 13:02:0128const {render, html} = LitHtml;
29
30const UIStrings = {
31 /**
32 *@description Text in Request Headers View of the Network panel
33 */
Wolfgang Beyer37fcd282022-06-29 09:34:1734 fromDiskCache: '(from disk cache)',
35 /**
36 *@description Text in Request Headers View of the Network panel
37 */
Wolfgang Beyer40530b682022-05-17 13:02:0138 fromMemoryCache: '(from memory cache)',
39 /**
40 *@description Text in Request Headers View of the Network panel
41 */
Wolfgang Beyer37fcd282022-06-29 09:34:1742 fromPrefetchCache: '(from prefetch cache)',
43 /**
44 *@description Text in Request Headers View of the Network panel
45 */
Wolfgang Beyer40530b682022-05-17 13:02:0146 fromServiceWorker: '(from `service worker`)',
47 /**
48 *@description Text in Request Headers View of the Network panel
49 */
50 fromSignedexchange: '(from signed-exchange)',
51 /**
52 *@description Text in Request Headers View of the Network panel
53 */
Wolfgang Beyer40530b682022-05-17 13:02:0154 fromWebBundle: '(from Web Bundle)',
55 /**
56 *@description Section header for a list of the main aspects of a http request
57 */
58 general: 'General',
59 /**
Wolfgang Beyer185fb1f2022-08-01 13:48:2560 *@description Label for a link from the network panel's headers view to the file in which
61 * header overrides are defined in the sources panel.
62 */
63 headerOverrides: 'Header overrides',
64 /**
Wolfgang Beyer235544c2022-05-24 08:07:4565 *@description Label for a checkbox to switch between raw and parsed headers
66 */
67 raw: 'Raw',
68 /**
69 *@description Text in Request Headers View of the Network panel
70 */
Wolfgang Beyer37fcd282022-06-29 09:34:1771 referrerPolicy: 'Referrer Policy',
Wolfgang Beyer235544c2022-05-24 08:07:4572 /**
Wolfgang Beyer37fcd282022-06-29 09:34:1773 *@description Text in Network Log View Columns of the Network panel
Wolfgang Beyer40530b682022-05-17 13:02:0174 */
Wolfgang Beyer37fcd282022-06-29 09:34:1775 remoteAddress: 'Remote Address',
76 /**
77 *@description Text in Request Headers View of the Network panel
78 */
79 requestHeaders: 'Request Headers',
Wolfgang Beyer40530b682022-05-17 13:02:0180 /**
81 *@description The HTTP method of a request
82 */
83 requestMethod: 'Request Method',
84 /**
Wolfgang Beyer37fcd282022-06-29 09:34:1785 *@description The URL of a request
86 */
87 requestUrl: 'Request URL',
88 /**
Wolfgang Beyer235544c2022-05-24 08:07:4589 *@description A context menu item in the Network Log View Columns of the Network panel
90 */
91 responseHeaders: 'Response Headers',
92 /**
Wolfgang Beyerc8f09372022-09-19 14:27:2793 *@description Title text for a link to the Sources panel to the file containing the header override definitions
94 */
95 revealHeaderOverrides: 'Reveal header override definitions',
96 /**
Wolfgang Beyerd1dd7962022-05-24 15:48:4897 *@description Text to show more content
98 */
99 showMore: 'Show more',
100 /**
Wolfgang Beyer40530b682022-05-17 13:02:01101 *@description HTTP response code
102 */
103 statusCode: 'Status Code',
Wolfgang Beyer40530b682022-05-17 13:02:01104};
105const str_ = i18n.i18n.registerUIStrings('panels/network/components/RequestHeadersView.ts', UIStrings);
106const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
107
108export class RequestHeadersView extends UI.Widget.VBox {
Wolfgang Beyerd2286472022-09-02 18:32:30109 readonly #requestHeadersComponent = new RequestHeadersComponent();
Wolfgang Beyer40530b682022-05-17 13:02:01110 readonly #request: SDK.NetworkRequest.NetworkRequest;
111
112 constructor(request: SDK.NetworkRequest.NetworkRequest) {
113 super();
114 this.#request = request;
Wolfgang Beyerd2286472022-09-02 18:32:30115 this.contentElement.appendChild(this.#requestHeadersComponent);
Wolfgang Beyer40530b682022-05-17 13:02:01116 }
117
118 wasShown(): void {
119 this.#request.addEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
120 this.#request.addEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47121 this.#request.addEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
122 this.#request.addEventListener(SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyer40530b682022-05-17 13:02:01123 this.#refreshHeadersView();
124 }
125
126 willHide(): void {
127 this.#request.removeEventListener(SDK.NetworkRequest.Events.RemoteAddressChanged, this.#refreshHeadersView, this);
128 this.#request.removeEventListener(SDK.NetworkRequest.Events.FinishedLoading, this.#refreshHeadersView, this);
Wolfgang Beyer87150062022-07-29 10:45:47129 this.#request.removeEventListener(SDK.NetworkRequest.Events.RequestHeadersChanged, this.#refreshHeadersView, this);
130 this.#request.removeEventListener(SDK.NetworkRequest.Events.ResponseHeadersChanged, this.#refreshHeadersView, this);
Wolfgang Beyer40530b682022-05-17 13:02:01131 }
132
133 #refreshHeadersView(): void {
Wolfgang Beyerd2286472022-09-02 18:32:30134 this.#requestHeadersComponent.data = {
Wolfgang Beyer40530b682022-05-17 13:02:01135 request: this.#request,
136 };
137 }
Wolfgang Beyer40426d82022-08-03 11:08:07138
139 revealHeader(section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string): void {
Wolfgang Beyerd2286472022-09-02 18:32:30140 this.#requestHeadersComponent.data = {
Wolfgang Beyer40426d82022-08-03 11:08:07141 request: this.#request,
142 toReveal: {section, header: header},
143 };
144 }
Wolfgang Beyer40530b682022-05-17 13:02:01145}
146
147export interface RequestHeadersComponentData {
148 request: SDK.NetworkRequest.NetworkRequest;
Wolfgang Beyer40426d82022-08-03 11:08:07149 toReveal?: {section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string};
Wolfgang Beyer40530b682022-05-17 13:02:01150}
151
152export class RequestHeadersComponent extends HTMLElement {
153 static readonly litTagName = LitHtml.literal`devtools-request-headers`;
154 readonly #shadow = this.attachShadow({mode: 'open'});
155 #request?: Readonly<SDK.NetworkRequest.NetworkRequest>;
Wolfgang Beyer235544c2022-05-24 08:07:45156 #showResponseHeadersText = false;
157 #showRequestHeadersText = false;
Wolfgang Beyerd1dd7962022-05-24 15:48:48158 #showResponseHeadersTextFull = false;
159 #showRequestHeadersTextFull = false;
Wolfgang Beyer40426d82022-08-03 11:08:07160 #toReveal?: {section: NetworkForward.UIRequestLocation.UIHeaderSection, header?: string} = undefined;
Wolfgang Beyer185fb1f2022-08-01 13:48:25161 readonly #workspace = Workspace.Workspace.WorkspaceImpl.instance();
Wolfgang Beyer40530b682022-05-17 13:02:01162
163 set data(data: RequestHeadersComponentData) {
164 this.#request = data.request;
Wolfgang Beyer40426d82022-08-03 11:08:07165 this.#toReveal = data.toReveal;
Wolfgang Beyer40530b682022-05-17 13:02:01166 this.#render();
167 }
168
169 connectedCallback(): void {
170 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles];
Wolfgang Beyer185fb1f2022-08-01 13:48:25171 this.#workspace.addEventListener(
172 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
173 this.#workspace.addEventListener(
174 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
175 }
176
177 disconnectedCallback(): void {
178 this.#workspace.removeEventListener(
179 Workspace.Workspace.Events.UISourceCodeAdded, this.#uiSourceCodeAddedOrRemoved, this);
180 this.#workspace.removeEventListener(
181 Workspace.Workspace.Events.UISourceCodeRemoved, this.#uiSourceCodeAddedOrRemoved, this);
182 }
183
184 #uiSourceCodeAddedOrRemoved(event: Common.EventTarget.EventTargetEvent<Workspace.UISourceCode.UISourceCode>): void {
185 if (this.#getHeaderOverridesFileUrl() === event.data.url()) {
186 this.#render();
187 }
Wolfgang Beyer40530b682022-05-17 13:02:01188 }
189
190 #render(): void {
Wolfgang Beyer3e62bd72022-07-28 14:26:48191 if (!this.#request) {
192 return;
193 }
Wolfgang Beyer40530b682022-05-17 13:02:01194
195 // Disabled until https://crbug.com/1079231 is fixed.
196 // clang-format off
197 render(html`
198 ${this.#renderGeneralSection()}
Wolfgang Beyer235544c2022-05-24 08:07:45199 ${this.#renderResponseHeaders()}
200 ${this.#renderRequestHeaders()}
Wolfgang Beyer40530b682022-05-17 13:02:01201 `, this.#shadow, {host: this});
202 // clang-format on
203 }
204
Wolfgang Beyer3e62bd72022-07-28 14:26:48205 #renderResponseHeaders(): LitHtml.LitTemplate {
206 if (!this.#request) {
207 return LitHtml.nothing;
208 }
Wolfgang Beyer235544c2022-05-24 08:07:45209
210 const toggleShowRaw = (): void => {
211 this.#showResponseHeadersText = !this.#showResponseHeadersText;
212 this.#render();
213 };
214
215 // Disabled until https://crbug.com/1079231 is fixed.
216 // clang-format off
217 return html`
218 <${Category.litTagName}
219 @togglerawevent=${toggleShowRaw}
220 .data=${{
221 name: 'responseHeaders',
222 title: i18nString(UIStrings.responseHeaders),
223 headerCount: this.#request.sortedResponseHeaders.length,
224 checked: this.#request.responseHeadersText ? this.#showResponseHeadersText : undefined,
Wolfgang Beyer185fb1f2022-08-01 13:48:25225 additionalContent: this.#renderHeaderOverridesLink(),
Wolfgang Beyer40426d82022-08-03 11:08:07226 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Response,
Wolfgang Beyer235544c2022-05-24 08:07:45227 } as CategoryData}
228 aria-label=${i18nString(UIStrings.responseHeaders)}
229 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48230 ${this.#showResponseHeadersText ?
231 this.#renderRawHeaders(this.#request.responseHeadersText, true) : html`
Wolfgang Beyerd2286472022-09-02 18:32:30232 <${ResponseHeaderSection.litTagName} .data=${{
233 request: this.#request,
234 toReveal: this.#toReveal,
235 } as ResponseHeaderSectionData}></${ResponseHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45236 `}
237 </${Category.litTagName}>
238 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47239 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45240 }
241
Wolfgang Beyer185fb1f2022-08-01 13:48:25242 #renderHeaderOverridesLink(): LitHtml.LitTemplate {
243 const overrideable = Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.HEADER_OVERRIDES);
244 if (!overrideable || !this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl())) {
245 return LitHtml.nothing;
246 }
247
248 const overridesSetting: Common.Settings.Setting<boolean> =
249 Common.Settings.Settings.instance().moduleSetting('persistenceNetworkOverridesEnabled');
250 // Disabled until https://crbug.com/1079231 is fixed.
251 // clang-format off
252 const fileIcon = overridesSetting.get() ? html`
253 <${IconButton.Icon.Icon.litTagName} class="inline-icon purple-dot" .data=${{
254 iconName: 'file-sync_icon',
255 width: '11px',
256 height: '13px',
257 } as IconButton.Icon.IconData}>
258 </${IconButton.Icon.Icon.litTagName}>` : html`
259 <${IconButton.Icon.Icon.litTagName} class="inline-icon" .data=${{
260 iconName: 'file_icon',
261 color: 'var(--color-text-primary)',
262 width: '12px',
263 height: '12px',
264 } as IconButton.Icon.IconData}>
265 </${IconButton.Icon.Icon.litTagName}>`;
266 // clang-format on
267
268 const revealHeadersFile = (event: Event): void => {
269 event.preventDefault();
270 const uiSourceCode = this.#workspace.uiSourceCodeForURL(this.#getHeaderOverridesFileUrl());
271 if (uiSourceCode) {
272 Sources.SourcesPanel.SourcesPanel.instance().showUISourceCode(uiSourceCode);
273 }
274 };
275
276 return html`
Wolfgang Beyerc8f09372022-09-19 14:27:27277 <x-link @click=${revealHeadersFile} class="link devtools-link" title=${UIStrings.revealHeaderOverrides}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25278 ${fileIcon}${i18nString(UIStrings.headerOverrides)}
279 </x-link>
280 `;
281 }
282
283 #getHeaderOverridesFileUrl(): Platform.DevToolsPath.UrlString {
284 if (!this.#request) {
285 return Platform.DevToolsPath.EmptyUrlString;
286 }
287 const fileUrl = Persistence.NetworkPersistenceManager.NetworkPersistenceManager.instance().fileUrlFromNetworkUrl(
288 this.#request.url(), /* ignoreInactive */ true);
289 return fileUrl.substring(0, fileUrl.lastIndexOf('/')) + '/' +
290 Persistence.NetworkPersistenceManager.HEADERS_FILENAME as Platform.DevToolsPath.UrlString;
291 }
292
Wolfgang Beyer3e62bd72022-07-28 14:26:48293 #renderRequestHeaders(): LitHtml.LitTemplate {
294 if (!this.#request) {
295 return LitHtml.nothing;
296 }
Wolfgang Beyer7f495ee2022-07-08 13:45:13297 const requestHeadersText = this.#request.requestHeadersText();
298
Wolfgang Beyer235544c2022-05-24 08:07:45299 const toggleShowRaw = (): void => {
300 this.#showRequestHeadersText = !this.#showRequestHeadersText;
301 this.#render();
302 };
303
Wolfgang Beyer235544c2022-05-24 08:07:45304 // Disabled until https://crbug.com/1079231 is fixed.
305 // clang-format off
306 return html`
307 <${Category.litTagName}
308 @togglerawevent=${toggleShowRaw}
309 .data=${{
310 name: 'requestHeaders',
311 title: i18nString(UIStrings.requestHeaders),
312 headerCount: this.#request.requestHeaders().length,
313 checked: requestHeadersText? this.#showRequestHeadersText : undefined,
Wolfgang Beyer40426d82022-08-03 11:08:07314 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.Request,
Wolfgang Beyer235544c2022-05-24 08:07:45315 } as CategoryData}
316 aria-label=${i18nString(UIStrings.requestHeaders)}
317 >
Wolfgang Beyerd1dd7962022-05-24 15:48:48318 ${(this.#showRequestHeadersText && requestHeadersText) ?
319 this.#renderRawHeaders(requestHeadersText, false) : html`
Wolfgang Beyer65a82282022-09-05 08:22:37320 <${RequestHeaderSection.litTagName} .data=${{
321 request: this.#request,
322 toReveal: this.#toReveal,
323 } as RequestHeaderSectionData}></${RequestHeaderSection.litTagName}>
Wolfgang Beyer235544c2022-05-24 08:07:45324 `}
325 </${Category.litTagName}>
326 `;
Wolfgang Beyerbcf4a832022-07-25 12:34:47327 // clang-format on
Wolfgang Beyer235544c2022-05-24 08:07:45328 }
329
Wolfgang Beyerd1dd7962022-05-24 15:48:48330 #renderRawHeaders(rawHeadersText: string, forResponseHeaders: boolean): LitHtml.TemplateResult {
331 const trimmed = rawHeadersText.trim();
332 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
333 const isShortened = !showFull && trimmed.length > RAW_HEADER_CUTOFF;
334
335 const showMore = ():void => {
336 if (forResponseHeaders) {
337 this.#showResponseHeadersTextFull = true;
338 } else {
339 this.#showRequestHeadersTextFull = true;
340 }
341 this.#render();
342 };
343
344 const onContextMenuOpen = (event: Event): void => {
345 const showFull = forResponseHeaders ? this.#showResponseHeadersTextFull : this.#showRequestHeadersTextFull;
346 if (!showFull) {
347 const contextMenu = new UI.ContextMenu.ContextMenu(event);
348 const section = contextMenu.newSection();
349 section.appendItem(i18nString(UIStrings.showMore), showMore);
350 void contextMenu.show();
351 }
352 };
353
354 const addContextMenuListener = (el: Element):void => {
355 if (isShortened) {
356 el.addEventListener('contextmenu', onContextMenuOpen);
357 }
358 };
359
360 return html`
361 <div class="row raw-headers-row" on-render=${ComponentHelpers.Directives.nodeRenderedCallback(addContextMenuListener)}>
362 <div class="raw-headers">${isShortened ? trimmed.substring(0, RAW_HEADER_CUTOFF) : trimmed}</div>
363 ${isShortened ? html`
364 <${Buttons.Button.Button.litTagName}
365 .size=${Buttons.Button.Size.SMALL}
366 .variant=${Buttons.Button.Variant.SECONDARY}
367 @click=${showMore}
368 >${i18nString(UIStrings.showMore)}</${Buttons.Button.Button.litTagName}>
369 ` : LitHtml.nothing}
370 </div>
371 `;
372 }
373
Wolfgang Beyer3e62bd72022-07-28 14:26:48374 #renderGeneralSection(): LitHtml.LitTemplate {
375 if (!this.#request) {
376 return LitHtml.nothing;
377 }
Wolfgang Beyer40530b682022-05-17 13:02:01378
Wolfgang Beyer40426d82022-08-03 11:08:07379 const statusClasses = [];
Wolfgang Beyer40530b682022-05-17 13:02:01380 if (this.#request.statusCode < 300 || this.#request.statusCode === 304) {
Wolfgang Beyer40426d82022-08-03 11:08:07381 statusClasses.push('green-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01382 } else if (this.#request.statusCode < 400) {
Wolfgang Beyer40426d82022-08-03 11:08:07383 statusClasses.push('yellow-circle');
384 } else {
385 statusClasses.push('red-circle');
Wolfgang Beyer40530b682022-05-17 13:02:01386 }
387
388 let statusText = this.#request.statusCode + ' ' + this.#request.statusText;
Wolfgang Beyer40530b682022-05-17 13:02:01389 if (this.#request.cachedInMemory()) {
390 statusText += ' ' + i18nString(UIStrings.fromMemoryCache);
Wolfgang Beyer40426d82022-08-03 11:08:07391 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01392 } else if (this.#request.fetchedViaServiceWorker) {
393 statusText += ' ' + i18nString(UIStrings.fromServiceWorker);
Wolfgang Beyer40426d82022-08-03 11:08:07394 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01395 } else if (this.#request.redirectSourceSignedExchangeInfoHasNoErrors()) {
396 statusText += ' ' + i18nString(UIStrings.fromSignedexchange);
Wolfgang Beyer40426d82022-08-03 11:08:07397 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01398 } else if (this.#request.webBundleInnerRequestInfo()) {
399 statusText += ' ' + i18nString(UIStrings.fromWebBundle);
Wolfgang Beyer40426d82022-08-03 11:08:07400 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01401 } else if (this.#request.fromPrefetchCache()) {
402 statusText += ' ' + i18nString(UIStrings.fromPrefetchCache);
Wolfgang Beyer40426d82022-08-03 11:08:07403 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01404 } else if (this.#request.cached()) {
405 statusText += ' ' + i18nString(UIStrings.fromDiskCache);
Wolfgang Beyer40426d82022-08-03 11:08:07406 statusClasses.push('status-with-comment');
Wolfgang Beyer40530b682022-05-17 13:02:01407 }
408
409 // Disabled until https://crbug.com/1079231 is fixed.
410 // clang-format off
411 return html`
Wolfgang Beyer235544c2022-05-24 08:07:45412 <${Category.litTagName}
Wolfgang Beyer40426d82022-08-03 11:08:07413 .data=${{
414 name: 'general',
415 title: i18nString(UIStrings.general),
416 forceOpen: this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General,
417 } as CategoryData}
Wolfgang Beyer235544c2022-05-24 08:07:45418 aria-label=${i18nString(UIStrings.general)}
419 >
Wolfgang Beyer40426d82022-08-03 11:08:07420 ${this.#renderGeneralRow(i18nString(UIStrings.requestUrl), this.#request.url())}
421 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.requestMethod), this.#request.requestMethod) : LitHtml.nothing}
422 ${this.#request.statusCode? this.#renderGeneralRow(i18nString(UIStrings.statusCode), statusText, statusClasses) : LitHtml.nothing}
423 ${this.#request.remoteAddress()? this.#renderGeneralRow(i18nString(UIStrings.remoteAddress), this.#request.remoteAddress()) : LitHtml.nothing}
424 ${this.#request.referrerPolicy()? this.#renderGeneralRow(i18nString(UIStrings.referrerPolicy), String(this.#request.referrerPolicy())) : LitHtml.nothing}
Wolfgang Beyer40530b682022-05-17 13:02:01425 </${Category.litTagName}>
426 `;
427 // clang-format on
428 }
Wolfgang Beyer40426d82022-08-03 11:08:07429
430 #renderGeneralRow(name: Common.UIString.LocalizedString, value: string, classNames?: string[]): LitHtml.LitTemplate {
431 const isHighlighted = this.#toReveal?.section === NetworkForward.UIRequestLocation.UIHeaderSection.General &&
Wolfgang Beyerfd504ec2022-09-06 09:48:11432 name.toLowerCase() === this.#toReveal?.header?.toLowerCase();
Wolfgang Beyer40426d82022-08-03 11:08:07433 return html`
434 <div class="row ${isHighlighted ? 'header-highlight' : ''}">
435 <div class="header-name">${name}:</div>
436 <div
437 class="header-value ${classNames?.join(' ')}"
438 @copy=${(): void => Host.userMetrics.actionTaken(Host.UserMetrics.Action.NetworkPanelCopyValue)}
439 >${value}</div>
440 </div>
441 `;
442 }
Wolfgang Beyer40530b682022-05-17 13:02:01443}
444
Wolfgang Beyer235544c2022-05-24 08:07:45445export class ToggleRawHeadersEvent extends Event {
446 static readonly eventName = 'togglerawevent';
447
448 constructor() {
449 super(ToggleRawHeadersEvent.eventName, {});
450 }
451}
452
Wolfgang Beyer40530b682022-05-17 13:02:01453export interface CategoryData {
454 name: string;
455 title: Common.UIString.LocalizedString;
Wolfgang Beyer235544c2022-05-24 08:07:45456 headerCount?: number;
457 checked?: boolean;
Wolfgang Beyer185fb1f2022-08-01 13:48:25458 additionalContent?: LitHtml.LitTemplate;
Wolfgang Beyer40426d82022-08-03 11:08:07459 forceOpen?: boolean;
Wolfgang Beyer40530b682022-05-17 13:02:01460}
461
462export class Category extends HTMLElement {
463 static readonly litTagName = LitHtml.literal`devtools-request-headers-category`;
464 readonly #shadow = this.attachShadow({mode: 'open'});
465 #expandedSetting?: Common.Settings.Setting<boolean>;
466 #title: Common.UIString.LocalizedString = Common.UIString.LocalizedEmptyString;
Wolfgang Beyer235544c2022-05-24 08:07:45467 #headerCount?: number = undefined;
468 #checked: boolean|undefined = undefined;
Wolfgang Beyer185fb1f2022-08-01 13:48:25469 #additionalContent: LitHtml.LitTemplate|undefined = undefined;
Wolfgang Beyer40426d82022-08-03 11:08:07470 #forceOpen: boolean|undefined = undefined;
Wolfgang Beyer40530b682022-05-17 13:02:01471
472 connectedCallback(): void {
Jack Franklin1f19d522022-07-06 14:00:18473 this.#shadow.adoptedStyleSheets = [requestHeadersViewStyles, Input.checkboxStyles];
Wolfgang Beyer40530b682022-05-17 13:02:01474 }
475
476 set data(data: CategoryData) {
477 this.#title = data.title;
478 this.#expandedSetting =
479 Common.Settings.Settings.instance().createSetting('request-info-' + data.name + '-category-expanded', true);
Wolfgang Beyer235544c2022-05-24 08:07:45480 this.#headerCount = data.headerCount;
481 this.#checked = data.checked;
Wolfgang Beyer185fb1f2022-08-01 13:48:25482 this.#additionalContent = data.additionalContent;
Wolfgang Beyer40426d82022-08-03 11:08:07483 this.#forceOpen = data.forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01484 this.#render();
485 }
486
Wolfgang Beyer235544c2022-05-24 08:07:45487 #onCheckboxToggle(): void {
488 this.dispatchEvent(new ToggleRawHeadersEvent());
489 }
490
Wolfgang Beyer40530b682022-05-17 13:02:01491 #render(): void {
Wolfgang Beyer40426d82022-08-03 11:08:07492 const isOpen = (this.#expandedSetting ? this.#expandedSetting.get() : true) || this.#forceOpen;
Wolfgang Beyer40530b682022-05-17 13:02:01493 // Disabled until https://crbug.com/1079231 is fixed.
494 // clang-format off
495 render(html`
Wolfgang Beyer235544c2022-05-24 08:07:45496 <details ?open=${isOpen} @toggle=${this.#onToggle}>
497 <summary class="header" @keydown=${this.#onSummaryKeyDown}>
Wolfgang Beyer185fb1f2022-08-01 13:48:25498 <div class="header-grid-container">
499 <div>
Wolfgang Beyerd2ea5672022-09-05 07:37:28500 ${this.#title}${this.#headerCount !== undefined ?
Wolfgang Beyer185fb1f2022-08-01 13:48:25501 html`<span class="header-count"> (${this.#headerCount})</span>` :
502 LitHtml.nothing
503 }
504 </div>
505 <div class="hide-when-closed">
506 ${this.#checked !== undefined ? html`
507 <label><input type="checkbox" .checked=${this.#checked} @change=${this.#onCheckboxToggle} />${i18nString(UIStrings.raw)}</label>
508 ` : LitHtml.nothing}
509 </div>
510 <div class="hide-when-closed">${this.#additionalContent}</div>
Wolfgang Beyer235544c2022-05-24 08:07:45511 </summary>
Wolfgang Beyer40530b682022-05-17 13:02:01512 <slot></slot>
513 </details>
514 `, this.#shadow, {host: this});
515 // clang-format on
516 }
517
518 #onSummaryKeyDown(event: KeyboardEvent): void {
519 if (!event.target) {
520 return;
521 }
522 const summaryElement = event.target as HTMLElement;
523 const detailsElement = summaryElement.parentElement as HTMLDetailsElement;
524 if (!detailsElement) {
525 throw new Error('<details> element is not found for a <summary> element');
526 }
527 switch (event.key) {
528 case 'ArrowLeft':
529 detailsElement.open = false;
530 break;
531 case 'ArrowRight':
532 detailsElement.open = true;
533 break;
534 }
535 }
536
537 #onToggle(event: Event): void {
538 this.#expandedSetting?.set((event.target as HTMLDetailsElement).open);
539 }
540}
541
542ComponentHelpers.CustomElements.defineComponent('devtools-request-headers', RequestHeadersComponent);
543ComponentHelpers.CustomElements.defineComponent('devtools-request-headers-category', Category);
544
545declare global {
546 // eslint-disable-next-line @typescript-eslint/no-unused-vars
547 interface HTMLElementTagNameMap {
548 'devtools-request-headers': RequestHeadersComponent;
549 'devtools-request-headers-category': Category;
550 }
551}