blob: 88fd7f9e4e4841365e6d37e81235594dd2ecd725 [file] [log] [blame]
// 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), 'largeicon-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;
}
wasShown(): void {
this.update();
}
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();
}
}