UNPKG

chrome-devtools-frontend

Version:
193 lines (170 loc) • 7.4 kB
// Copyright 2021 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. /* * Copyright (C) 2008 Apple Inc. All Rights Reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ import * as Common from '../../core/common/common.js'; import * as i18n from '../../core/i18n/i18n.js'; import * as DataGrid from '../../ui/legacy/components/data_grid/data_grid.js'; import * as UI from '../../ui/legacy/legacy.js'; import {type Database} from './DatabaseModel.js'; const UIStrings = { /** *@description Text in Database Table View of the Application panel */ database: 'Database', /** *@description Text to refresh the page */ refresh: 'Refresh', /** *@description Text in Database Table View of the Application panel */ visibleColumns: 'Visible columns', /** *@description Text in Database Table View of the Application panel *@example {database} PH1 */ theStableIsEmpty: 'The "{PH1}" table is empty.', /** *@description Error msg element text content in Database Table View of the Application panel *@example {database} PH1 */ anErrorOccurredTryingToreadTheS: 'An error occurred trying to read the "{PH1}" table.', }; const str_ = i18n.i18n.registerUIStrings('panels/application/DatabaseTableView.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); export interface VisibleColumnsSetting { [tableName: string]: string; } export class DatabaseTableView extends UI.View.SimpleView { database: Database; tableName: string; private lastVisibleColumns: string; private readonly columnsMap: Map<string, string>; private readonly visibleColumnsSetting: Common.Settings.Setting<VisibleColumnsSetting>; refreshButton: UI.Toolbar.ToolbarButton; private readonly visibleColumnsInput: UI.Toolbar.ToolbarInput; private dataGrid: DataGrid.SortableDataGrid.SortableDataGrid<DataGrid.SortableDataGrid.SortableDataGridNode<unknown>>| null; private emptyWidget?: UI.EmptyWidget.EmptyWidget; constructor(database: Database, tableName: string) { super(i18nString(UIStrings.database)); this.database = database; this.tableName = tableName; this.lastVisibleColumns = ''; this.columnsMap = new Map(); this.element.classList.add('storage-view', 'table'); this.visibleColumnsSetting = Common.Settings.Settings.instance().createSetting('databaseTableViewVisibleColumns', {}); this.refreshButton = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.refresh), 'refresh'); this.refreshButton.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, this.refreshButtonClicked, this); this.visibleColumnsInput = new UI.Toolbar.ToolbarInput(i18nString(UIStrings.visibleColumns), '', 1); this.visibleColumnsInput.addEventListener( UI.Toolbar.ToolbarInput.Event.TextChanged, this.onVisibleColumnsChanged, this); this.dataGrid = null; } override wasShown(): void { this.update(); } override async toolbarItems(): Promise<UI.Toolbar.ToolbarItem[]> { return [this.refreshButton, this.visibleColumnsInput]; } private escapeTableName(tableName: string): string { return tableName.replace(/\"/g, '""'); } update(): void { void this.database.executeSql( 'SELECT rowid, * FROM "' + this.escapeTableName(this.tableName) + '"', this.queryFinished.bind(this), this.queryError.bind(this)); } // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/no-explicit-any private queryFinished(columnNames: string[], values: any[]): void { this.detachChildWidgets(); this.element.removeChildren(); this.dataGrid = DataGrid.SortableDataGrid.SortableDataGrid.create(columnNames, values, i18nString(UIStrings.database)); this.visibleColumnsInput.setVisible(Boolean(this.dataGrid)); if (!this.dataGrid) { this.emptyWidget = new UI.EmptyWidget.EmptyWidget(i18nString(UIStrings.theStableIsEmpty, {PH1: this.tableName})); this.emptyWidget.show(this.element); return; } this.dataGrid.setStriped(true); this.dataGrid.asWidget().show(this.element); this.dataGrid.autoSizeColumns(5); this.columnsMap.clear(); for (let i = 1; i < columnNames.length; ++i) { this.columnsMap.set(columnNames[i], String(i)); } this.lastVisibleColumns = ''; const visibleColumnsText = this.visibleColumnsSetting.get()[this.tableName] || ''; this.visibleColumnsInput.setValue(visibleColumnsText); this.onVisibleColumnsChanged(); } private onVisibleColumnsChanged(): void { if (!this.dataGrid) { return; } const text = this.visibleColumnsInput.value(); const parts = text.split(/[\s,]+/); const matches = new Set<string>(); const columnsVisibility = new Set<string>(); columnsVisibility.add('0'); for (const part of parts) { const mappedColumn = this.columnsMap.get(part); if (mappedColumn !== undefined) { matches.add(part); columnsVisibility.add(mappedColumn); } } const newVisibleColumns = [...matches].sort().join(', '); if (newVisibleColumns.length === 0) { for (const v of this.columnsMap.values()) { columnsVisibility.add(v); } } if (newVisibleColumns === this.lastVisibleColumns) { return; } const visibleColumnsRegistry = this.visibleColumnsSetting.get(); visibleColumnsRegistry[this.tableName] = text; this.visibleColumnsSetting.set(visibleColumnsRegistry); this.dataGrid.setColumnsVisiblity(columnsVisibility); this.lastVisibleColumns = newVisibleColumns; } private queryError(): void { this.detachChildWidgets(); this.element.removeChildren(); const errorMsgElement = document.createElement('div'); errorMsgElement.className = 'storage-table-error'; errorMsgElement.textContent = i18nString(UIStrings.anErrorOccurredTryingToreadTheS, {PH1: this.tableName}); this.element.appendChild(errorMsgElement); } private refreshButtonClicked(): void { this.update(); } }