blob: c6c471074de002127dc1b3dbcf5bd5dd4918ddf1 [file] [log] [blame]
// Copyright 2020 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.
import * as Common from '../../core/common/common.js';
import * as i18n from '../../core/i18n/i18n.js';
import * as SDK from '../../core/sdk/sdk.js';
import * as UI from '../../ui/legacy/legacy.js';
import {ApplicationPanelSidebar} from './ApplicationPanelSidebar.js';
import {ApplicationPanelTreeElement, ExpandableApplicationPanelTreeElement} from './ApplicationPanelTreeElement.js';
import {ResourcesPanel} from './ResourcesPanel.js';
import {ServiceWorkerCacheView} from './ServiceWorkerCacheViews.js';
const UIStrings = {
/**
*@description Text in Application Panel Sidebar of the Application panel
*/
cacheStorage: 'Cache Storage',
/**
*@description A context menu item in the Application Panel Sidebar of the Application panel
*/
refreshCaches: 'Refresh Caches',
/**
*@description Text to delete something
*/
delete: 'Delete',
};
const str_ = i18n.i18n.registerUIStrings('panels/application/ApplicationPanelCacheSection.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
export class ApplicationCacheManifestTreeElement extends ApplicationPanelTreeElement {
private readonly manifestURL: string;
constructor(resourcesPanel: ResourcesPanel, manifestURL: string) {
const title = new Common.ParsedURL.ParsedURL(manifestURL).displayName;
super(resourcesPanel, title, false);
this.tooltip = manifestURL;
this.manifestURL = manifestURL;
}
get itemURL(): string {
return 'appcache://' + this.manifestURL;
}
onselect(selectedByUser: boolean|undefined): boolean {
super.onselect(selectedByUser);
this.resourcesPanel.showCategoryView(this.manifestURL, null);
return false;
}
}
export class ServiceWorkerCacheTreeElement extends ExpandableApplicationPanelTreeElement {
private swCacheModel: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel|null;
private swCacheTreeElements: Set<SWCacheTreeElement>;
constructor(resourcesPanel: ResourcesPanel) {
super(resourcesPanel, i18nString(UIStrings.cacheStorage), 'CacheStorage');
const icon = UI.Icon.Icon.create('mediumicon-database', 'resource-tree-item');
this.setLink('https://developer.chrome.com/docs/devtools/storage/cache/?utm_source=devtools');
this.setLeadingIcons([icon]);
this.swCacheModel = null;
this.swCacheTreeElements = new Set();
}
initialize(model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel|null): void {
this.swCacheTreeElements.clear();
this.swCacheModel = model;
if (model) {
for (const cache of model.caches()) {
this.addCache(model, cache);
}
}
SDK.SDKModel.TargetManager.instance().addModelListener(
SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel, SDK.ServiceWorkerCacheModel.Events.CacheAdded,
this.cacheAdded, this);
SDK.SDKModel.TargetManager.instance().addModelListener(
SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel, SDK.ServiceWorkerCacheModel.Events.CacheRemoved,
this.cacheRemoved, this);
}
onattach(): void {
super.onattach();
this.listItemElement.addEventListener('contextmenu', this.handleContextMenuEvent.bind(this), true);
}
private handleContextMenuEvent(event: MouseEvent): void {
const contextMenu = new UI.ContextMenu.ContextMenu(event);
contextMenu.defaultSection().appendItem(i18nString(UIStrings.refreshCaches), this.refreshCaches.bind(this));
contextMenu.show();
}
private refreshCaches(): void {
if (this.swCacheModel) {
this.swCacheModel.refreshCacheNames();
}
}
private cacheAdded(event: Common.EventTarget.EventTargetEvent): void {
const cache = /** @type {!SDK.ServiceWorkerCacheModel.Cache} */ (event.data.cache);
const model = /** @type {!SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel} */ (event.data.model);
this.addCache(model, cache);
}
private addCache(
model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel, cache: SDK.ServiceWorkerCacheModel.Cache): void {
const swCacheTreeElement = new SWCacheTreeElement(this.resourcesPanel, model, cache);
this.swCacheTreeElements.add(swCacheTreeElement);
this.appendChild(swCacheTreeElement);
}
private cacheRemoved(event: Common.EventTarget.EventTargetEvent): void {
const cache = /** @type {!SDK.ServiceWorkerCacheModel.Cache} */ (event.data.cache);
const model = /** @type {!SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel} */ (event.data.model);
const swCacheTreeElement = this.cacheTreeElement(model, cache);
if (!swCacheTreeElement) {
return;
}
this.removeChild(swCacheTreeElement);
this.swCacheTreeElements.delete(swCacheTreeElement);
this.setExpandable(this.childCount() > 0);
}
private cacheTreeElement(
model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel,
cache: SDK.ServiceWorkerCacheModel.Cache): SWCacheTreeElement|null {
for (const cacheTreeElement of this.swCacheTreeElements) {
if (cacheTreeElement.hasModelAndCache(model, cache)) {
return cacheTreeElement;
}
}
return null;
}
}
export class SWCacheTreeElement extends ApplicationPanelTreeElement {
private readonly model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel;
private cache: SDK.ServiceWorkerCacheModel.Cache;
private view: ServiceWorkerCacheView|null;
constructor(
resourcesPanel: ResourcesPanel, model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel,
cache: SDK.ServiceWorkerCacheModel.Cache) {
super(resourcesPanel, cache.cacheName + ' - ' + cache.securityOrigin, false);
this.model = model;
this.cache = cache;
/** @type {?} */
this.view = null;
const icon = UI.Icon.Icon.create('mediumicon-table', 'resource-tree-item');
this.setLeadingIcons([icon]);
}
get itemURL(): string {
// I don't think this will work at all.
return 'cache://' + this.cache.cacheId;
}
onattach(): void {
super.onattach();
this.listItemElement.addEventListener('contextmenu', this.handleContextMenuEvent.bind(this), true);
}
private handleContextMenuEvent(event: MouseEvent): void {
const contextMenu = new UI.ContextMenu.ContextMenu(event);
contextMenu.defaultSection().appendItem(i18nString(UIStrings.delete), this.clearCache.bind(this));
contextMenu.show();
}
private clearCache(): void {
this.model.deleteCache(this.cache);
}
update(cache: SDK.ServiceWorkerCacheModel.Cache): void {
this.cache = cache;
if (this.view) {
this.view.update(cache);
}
}
onselect(selectedByUser: boolean|undefined): boolean {
super.onselect(selectedByUser);
if (!this.view) {
this.view = new ServiceWorkerCacheView(this.model, this.cache);
}
this.showView(this.view);
return false;
}
hasModelAndCache(
model: SDK.ServiceWorkerCacheModel.ServiceWorkerCacheModel, cache: SDK.ServiceWorkerCacheModel.Cache): boolean {
return this.cache.equals(cache) && this.model === model;
}
}
export class ApplicationCacheFrameTreeElement extends ApplicationPanelTreeElement {
private readonly sidebar: ApplicationPanelSidebar;
readonly frameId: string;
readonly manifestURL: string;
constructor(sidebar: ApplicationPanelSidebar, frame: SDK.ResourceTreeModel.ResourceTreeFrame, manifestURL: string) {
super(sidebar._panel, '', false);
this.sidebar = sidebar;
this.frameId = frame.id;
this.manifestURL = manifestURL;
this.refreshTitles(frame);
const icon = UI.Icon.Icon.create('mediumicon-frame-top', 'navigator-folder-tree-item');
this.setLeadingIcons([icon]);
}
get itemURL(): string {
return 'appcache://' + this.manifestURL + '/' + encodeURI(this.titleAsText());
}
private refreshTitles(frame: SDK.ResourceTreeModel.ResourceTreeFrame): void {
this.title = frame.displayName();
}
frameNavigated(frame: SDK.ResourceTreeModel.ResourceTreeFrame): void {
this.refreshTitles(frame);
}
onselect(selectedByUser: boolean|undefined): boolean {
super.onselect(selectedByUser);
this.sidebar._showApplicationCache(this.frameId);
return false;
}
}