chrome-devtools-frontend
Version:
Chrome DevTools UI
168 lines (145 loc) • 6.47 kB
text/typescript
// Copyright 2017 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 Platform from '../../core/platform/platform.js';
import * as Workspace from '../../models/workspace/workspace.js';
import * as WorkspaceDiff from '../../models/workspace_diff/workspace_diff.js';
import * as IconButton from '../../ui/components/icon_button/icon_button.js';
import * as UI from '../../ui/legacy/legacy.js';
import * as Snippets from '../snippets/snippets.js';
import changesSidebarStyles from './changesSidebar.css.js';
const UIStrings = {
/**
*@description Name of an item from source map
*@example {compile.html} PH1
*/
sFromSourceMap: '{PH1} (from source map)',
};
const str_ = i18n.i18n.registerUIStrings('panels/changes/ChangesSidebar.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
export class ChangesSidebar extends Common.ObjectWrapper.eventMixin<EventTypes, typeof UI.Widget.Widget>(
UI.Widget.Widget) {
private treeoutline: UI.TreeOutline.TreeOutlineInShadow;
private readonly treeElements: Map<Workspace.UISourceCode.UISourceCode, UISourceCodeTreeElement>;
private readonly workspaceDiff: WorkspaceDiff.WorkspaceDiff.WorkspaceDiffImpl;
constructor(workspaceDiff: WorkspaceDiff.WorkspaceDiff.WorkspaceDiffImpl) {
super();
this.treeoutline = new UI.TreeOutline.TreeOutlineInShadow();
this.treeoutline.setFocusable(false);
this.treeoutline.setComparator((a, b) => Platform.StringUtilities.compare(a.titleAsText(), b.titleAsText()));
this.treeoutline.addEventListener(UI.TreeOutline.Events.ElementSelected, this.selectionChanged, this);
UI.ARIAUtils.markAsTablist(this.treeoutline.contentElement);
this.element.appendChild(this.treeoutline.element);
this.treeElements = new Map();
this.workspaceDiff = workspaceDiff;
this.workspaceDiff.modifiedUISourceCodes().forEach(this.addUISourceCode.bind(this));
this.workspaceDiff.addEventListener(
WorkspaceDiff.WorkspaceDiff.Events.ModifiedStatusChanged, this.uiSourceCodeMofiedStatusChanged, this);
}
selectUISourceCode(uiSourceCode: Workspace.UISourceCode.UISourceCode, omitFocus?: boolean|undefined): void {
const treeElement = this.treeElements.get(uiSourceCode);
if (!treeElement) {
return;
}
treeElement.select(omitFocus);
}
selectedUISourceCode(): Workspace.UISourceCode.UISourceCode|null {
// @ts-ignore uiSourceCode seems to be dynamically attached.
return this.treeoutline.selectedTreeElement ? this.treeoutline.selectedTreeElement.uiSourceCode : null;
}
private selectionChanged(): void {
this.dispatchEventToListeners(Events.SelectedUISourceCodeChanged);
}
private uiSourceCodeMofiedStatusChanged(
event: Common.EventTarget.EventTargetEvent<WorkspaceDiff.WorkspaceDiff.ModifiedStatusChangedEvent>): void {
if (event.data.isModified) {
this.addUISourceCode(event.data.uiSourceCode);
} else {
this.removeUISourceCode(event.data.uiSourceCode);
}
}
private removeUISourceCode(uiSourceCode: Workspace.UISourceCode.UISourceCode): void {
const treeElement = this.treeElements.get(uiSourceCode);
this.treeElements.delete(uiSourceCode);
if (this.treeoutline.selectedTreeElement === treeElement) {
const nextElementToSelect = treeElement.previousSibling || treeElement.nextSibling;
if (nextElementToSelect) {
nextElementToSelect.select(true);
} else {
treeElement.deselect();
this.selectionChanged();
}
}
if (treeElement) {
this.treeoutline.removeChild(treeElement);
treeElement.dispose();
}
if (this.treeoutline.rootElement().childCount() === 0) {
this.treeoutline.setFocusable(false);
}
}
private addUISourceCode(uiSourceCode: Workspace.UISourceCode.UISourceCode): void {
const treeElement = new UISourceCodeTreeElement(uiSourceCode);
this.treeElements.set(uiSourceCode, treeElement);
this.treeoutline.setFocusable(true);
this.treeoutline.appendChild(treeElement);
if (!this.treeoutline.selectedTreeElement) {
treeElement.select(true);
}
}
override wasShown(): void {
super.wasShown();
this.treeoutline.registerCSSFiles([changesSidebarStyles]);
}
}
export const enum Events {
SelectedUISourceCodeChanged = 'SelectedUISourceCodeChanged',
}
export type EventTypes = {
[Events.SelectedUISourceCodeChanged]: void,
};
export class UISourceCodeTreeElement extends UI.TreeOutline.TreeElement {
uiSourceCode: Workspace.UISourceCode.UISourceCode;
private readonly eventListeners: Common.EventTarget.EventDescriptor[];
constructor(uiSourceCode: Workspace.UISourceCode.UISourceCode) {
super();
this.uiSourceCode = uiSourceCode;
this.listItemElement.classList.add('navigator-' + uiSourceCode.contentType().name() + '-tree-item');
UI.ARIAUtils.markAsTab(this.listItemElement);
let iconName: 'document'|'snippet' = 'document';
if (Snippets.ScriptSnippetFileSystem.isSnippetsUISourceCode(this.uiSourceCode)) {
iconName = 'snippet';
}
const defaultIcon = new IconButton.Icon.Icon();
defaultIcon.data = {
iconName,
color: 'var(--icon-file-default)',
width: '20px',
height: '20px',
};
this.setLeadingIcons([defaultIcon]);
this.eventListeners = [
uiSourceCode.addEventListener(Workspace.UISourceCode.Events.TitleChanged, this.updateTitle, this),
uiSourceCode.addEventListener(Workspace.UISourceCode.Events.WorkingCopyChanged, this.updateTitle, this),
uiSourceCode.addEventListener(Workspace.UISourceCode.Events.WorkingCopyCommitted, this.updateTitle, this),
];
this.updateTitle();
}
private updateTitle(): void {
let titleText: string = this.uiSourceCode.displayName();
if (this.uiSourceCode.isDirty()) {
titleText = '*' + titleText;
}
this.title = titleText;
let tooltip: Common.UIString.LocalizedString|string = this.uiSourceCode.url();
if (this.uiSourceCode.contentType().isFromSourceMap()) {
tooltip = i18nString(UIStrings.sFromSourceMap, {PH1: this.uiSourceCode.displayName()});
}
this.tooltip = tooltip;
}
dispose(): void {
Common.EventTarget.removeEventListeners(this.eventListeners);
}
}