9fa2ebddc8
* Rename action * Store more state information * Conditional checks for details toggle links * Add support for hiding/showing details via menu
495 lines
19 KiB
JavaScript
495 lines
19 KiB
JavaScript
/*
|
|
* Copyright (C) 2020 Yomichan Authors
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/* global
|
|
* DictionaryDatabase
|
|
* Modal
|
|
* ObjectPropertyAccessor
|
|
* api
|
|
*/
|
|
|
|
class DictionaryEntry {
|
|
constructor(dictionaryController, node, dictionaryInfo) {
|
|
this._dictionaryController = dictionaryController;
|
|
this._node = node;
|
|
this._dictionaryInfo = dictionaryInfo;
|
|
this._eventListeners = new EventListenerCollection();
|
|
this._detailsContainer = null;
|
|
this._hasDetails = false;
|
|
this._hasCounts = false;
|
|
}
|
|
|
|
get node() {
|
|
return this._node;
|
|
}
|
|
|
|
get dictionaryTitle() {
|
|
return this._dictionaryInfo.title;
|
|
}
|
|
|
|
prepare() {
|
|
const node = this._node;
|
|
const {title, revision, prefixWildcardsSupported, version} = this._dictionaryInfo;
|
|
|
|
this._detailsContainer = node.querySelector('.dictionary-details');
|
|
|
|
const enabledCheckbox = node.querySelector('.dictionary-enabled');
|
|
const allowSecondarySearchesCheckbox = node.querySelector('.dictionary-allow-secondary-searches');
|
|
const priorityInput = node.querySelector('.dictionary-priority');
|
|
const deleteButton = node.querySelector('.dictionary-delete-button');
|
|
const menuButton = node.querySelector('.dictionary-menu-button');
|
|
const detailsTable = node.querySelector('.dictionary-details-table');
|
|
const detailsToggleLink = node.querySelector('.dictionary-details-toggle-link');
|
|
const outdatedContainer = node.querySelector('.dictionary-outdated-notification');
|
|
const titleNode = node.querySelector('.dictionary-title');
|
|
const versionNode = node.querySelector('.dictionary-version');
|
|
const wildcardSupportedCheckbox = node.querySelector('.dictionary-prefix-wildcard-searches-supported');
|
|
|
|
const hasDetails = this._setupDetails(detailsTable);
|
|
this._hasDetails = hasDetails;
|
|
|
|
titleNode.textContent = title;
|
|
versionNode.textContent = `rev.${revision}`;
|
|
wildcardSupportedCheckbox.checked = !!prefixWildcardsSupported;
|
|
|
|
outdatedContainer.hidden = (version >= 3);
|
|
if (detailsToggleLink !== null) { detailsToggleLink.hidden = !hasDetails; }
|
|
|
|
enabledCheckbox.dataset.setting = ObjectPropertyAccessor.getPathString(['dictionaries', title, 'enabled']);
|
|
allowSecondarySearchesCheckbox.dataset.setting = ObjectPropertyAccessor.getPathString(['dictionaries', title, 'allowSecondarySearches']);
|
|
priorityInput.dataset.setting = ObjectPropertyAccessor.getPathString(['dictionaries', title, 'priority']);
|
|
|
|
if (deleteButton !== null) {
|
|
this._eventListeners.addEventListener(deleteButton, 'click', this._onDeleteButtonClicked.bind(this), false);
|
|
}
|
|
if (menuButton !== null) {
|
|
this._eventListeners.addEventListener(menuButton, 'menuOpened', this._onMenuOpened.bind(this), false);
|
|
this._eventListeners.addEventListener(menuButton, 'menuClosed', this._onMenuClosed.bind(this), false);
|
|
}
|
|
if (detailsToggleLink !== null && this._detailsContainer !== null) {
|
|
this._eventListeners.addEventListener(detailsToggleLink, 'click', this._onDetailsToggleLinkClicked.bind(this), false);
|
|
}
|
|
this._eventListeners.addEventListener(priorityInput, 'settingChanged', this._onPriorityChanged.bind(this), false);
|
|
}
|
|
|
|
cleanup() {
|
|
this._eventListeners.removeAllEventListeners();
|
|
const node = this._node;
|
|
if (node.parentNode !== null) {
|
|
node.parentNode.removeChild(node);
|
|
}
|
|
}
|
|
|
|
setCounts(counts) {
|
|
const node = this._node.querySelector('.dictionary-counts');
|
|
node.textContent = JSON.stringify({info: this._dictionaryInfo, counts}, null, 4);
|
|
node.hidden = false;
|
|
this._hasCounts = true;
|
|
}
|
|
|
|
// Private
|
|
|
|
_onDeleteButtonClicked(e) {
|
|
e.preventDefault();
|
|
this._delete();
|
|
}
|
|
|
|
_onMenuOpened(e) {
|
|
const {detail: {menu}} = e;
|
|
const showDetails = menu.querySelector('.popup-menu-item[data-menu-action="showDetails"]');
|
|
const hideDetails = menu.querySelector('.popup-menu-item[data-menu-action="hideDetails"]');
|
|
const hasDetails = (this._detailsContainer !== null && (this._hasDetails || this._hasCounts));
|
|
const detailsVisible = (hasDetails && !this._detailsContainer.hidden);
|
|
if (showDetails !== null) {
|
|
showDetails.hidden = detailsVisible;
|
|
showDetails.disabled = !hasDetails;
|
|
}
|
|
if (hideDetails !== null) {
|
|
hideDetails.hidden = !detailsVisible;
|
|
hideDetails.disabled = !hasDetails;
|
|
}
|
|
}
|
|
|
|
_onMenuClosed(e) {
|
|
const {detail: {action}} = e;
|
|
switch (action) {
|
|
case 'delete':
|
|
this._delete();
|
|
break;
|
|
case 'showDetails':
|
|
if (this._detailsContainer !== null) { this._detailsContainer.hidden = false; }
|
|
break;
|
|
case 'hideDetails':
|
|
if (this._detailsContainer !== null) { this._detailsContainer.hidden = true; }
|
|
break;
|
|
}
|
|
}
|
|
|
|
_onDetailsToggleLinkClicked(e) {
|
|
e.preventDefault();
|
|
this._detailsContainer.hidden = !this._detailsContainer.hidden;
|
|
}
|
|
|
|
_onPriorityChanged(e) {
|
|
const {detail: {value}} = e;
|
|
this._node.style.order = `${-value}`;
|
|
}
|
|
|
|
_setupDetails(detailsTable) {
|
|
const targets = [
|
|
['Author', 'author'],
|
|
['URL', 'url'],
|
|
['Description', 'description'],
|
|
['Attribution', 'attribution']
|
|
];
|
|
|
|
const dictionaryInfo = this._dictionaryInfo;
|
|
const fragment = document.createDocumentFragment();
|
|
let any = false;
|
|
for (const [label, key] of targets) {
|
|
const info = dictionaryInfo[key];
|
|
if (typeof info !== 'string') { continue; }
|
|
|
|
const details = this._dictionaryController.instantiateTemplate('dictionary-details-entry');
|
|
details.dataset.type = key;
|
|
details.querySelector('.dictionary-details-entry-label').textContent = `${label}:`;
|
|
details.querySelector('.dictionary-details-entry-info').textContent = info;
|
|
fragment.appendChild(details);
|
|
|
|
any = true;
|
|
}
|
|
|
|
detailsTable.appendChild(fragment);
|
|
return any;
|
|
}
|
|
|
|
_delete() {
|
|
this._dictionaryController.deleteDictionary(this.dictionaryTitle);
|
|
}
|
|
}
|
|
|
|
class DictionaryController {
|
|
constructor(settingsController, modalController, storageController, statusFooter=null) {
|
|
this._settingsController = settingsController;
|
|
this._modalController = modalController;
|
|
this._storageController = storageController;
|
|
this._statusFooter = statusFooter;
|
|
this._dictionaries = null;
|
|
this._dictionaryEntries = [];
|
|
this._databaseStateToken = null;
|
|
this._checkingIntegrity = false;
|
|
this._warningNode = null;
|
|
this._checkIntegrityButton = null;
|
|
this._dictionaryEntryContainer = null;
|
|
this._integrityExtraInfoContainer = null;
|
|
this._deleteDictionaryModal = null;
|
|
this._integrityExtraInfoNode = null;
|
|
this._isDeleting = false;
|
|
}
|
|
|
|
async prepare() {
|
|
this._warningNode = document.querySelector('#dictionary-warning');
|
|
this._checkIntegrityButton = document.querySelector('#dictionary-check-integrity');
|
|
this._dictionaryEntryContainer = document.querySelector('#dictionary-list');
|
|
this._integrityExtraInfoContainer = document.querySelector('#dictionary-list-extra');
|
|
this._deleteDictionaryModal = this._modalController.getModal('dictionary-confirm-delete');
|
|
|
|
yomichan.on('databaseUpdated', this._onDatabaseUpdated.bind(this));
|
|
|
|
document.querySelector('#dictionary-confirm-delete-button').addEventListener('click', this._onDictionaryConfirmDelete.bind(this), false);
|
|
this._checkIntegrityButton.addEventListener('click', this._onCheckIntegrityButtonClick.bind(this), false);
|
|
|
|
await this._onDatabaseUpdated();
|
|
}
|
|
|
|
deleteDictionary(dictionaryTitle) {
|
|
if (this._isDeleting) { return; }
|
|
const modal = this._deleteDictionaryModal;
|
|
modal.node.dataset.dictionaryTitle = dictionaryTitle;
|
|
modal.node.querySelector('#dictionary-confirm-delete-name').textContent = dictionaryTitle;
|
|
modal.setVisible(true);
|
|
}
|
|
|
|
instantiateTemplate(name) {
|
|
return this._settingsController.instantiateTemplate(name);
|
|
}
|
|
|
|
// Private
|
|
|
|
async _onDatabaseUpdated() {
|
|
const token = {};
|
|
this._databaseStateToken = token;
|
|
this._dictionaries = null;
|
|
const dictionaries = await api.getDictionaryInfo();
|
|
if (this._databaseStateToken !== token) { return; }
|
|
this._dictionaries = dictionaries;
|
|
|
|
this._warningNode.hidden = (dictionaries.length > 0);
|
|
this._updateMainDictionarySelectOptions(dictionaries);
|
|
|
|
for (const entry of this._dictionaryEntries) {
|
|
entry.cleanup();
|
|
}
|
|
this._dictionaryEntries = [];
|
|
|
|
await this._ensureDictionarySettings(dictionaries);
|
|
for (const dictionary of dictionaries) {
|
|
this._createDictionaryEntry(dictionary);
|
|
}
|
|
}
|
|
|
|
_onDictionaryConfirmDelete(e) {
|
|
e.preventDefault();
|
|
|
|
const modal = this._deleteDictionaryModal;
|
|
modal.setVisible(false);
|
|
|
|
const title = modal.node.dataset.dictionaryTitle;
|
|
if (typeof title !== 'string') { return; }
|
|
delete modal.node.dataset.dictionaryTitle;
|
|
|
|
this._deleteDictionary(title);
|
|
}
|
|
|
|
_onCheckIntegrityButtonClick(e) {
|
|
e.preventDefault();
|
|
this._checkIntegrity();
|
|
}
|
|
|
|
_updateMainDictionarySelectOptions(dictionaries) {
|
|
for (const select of document.querySelectorAll('[data-setting="general.mainDictionary"]')) {
|
|
const fragment = document.createDocumentFragment();
|
|
|
|
let option = document.createElement('option');
|
|
option.className = 'text-muted';
|
|
option.value = '';
|
|
option.textContent = 'Not selected';
|
|
fragment.appendChild(option);
|
|
|
|
for (const {title, sequenced} of dictionaries) {
|
|
if (!sequenced) { continue; }
|
|
option = document.createElement('option');
|
|
option.value = title;
|
|
option.textContent = title;
|
|
fragment.appendChild(option);
|
|
}
|
|
|
|
select.textContent = ''; // Empty
|
|
select.appendChild(fragment);
|
|
}
|
|
}
|
|
|
|
async _checkIntegrity() {
|
|
if (this._dictionaries === null || this._checkingIntegrity || this._isDeleting) { return; }
|
|
|
|
try {
|
|
this._checkingIntegrity = true;
|
|
this._setButtonsEnabled(false);
|
|
|
|
const token = this._databaseStateToken;
|
|
const dictionaryTitles = this._dictionaries.map(({title}) => title);
|
|
const {counts, total} = await api.getDictionaryCounts(dictionaryTitles, true);
|
|
if (this._databaseStateToken !== token) { return; }
|
|
|
|
for (let i = 0, ii = Math.min(counts.length, this._dictionaryEntries.length); i < ii; ++i) {
|
|
const entry = this._dictionaryEntries[i];
|
|
entry.setCounts(counts[i]);
|
|
}
|
|
|
|
this._setCounts(counts, total);
|
|
} finally {
|
|
this._setButtonsEnabled(true);
|
|
this._checkingIntegrity = false;
|
|
}
|
|
}
|
|
|
|
_setCounts(dictionaryCounts, totalCounts) {
|
|
const remainders = Object.assign({}, totalCounts);
|
|
const keys = Object.keys(remainders);
|
|
|
|
for (const counts of dictionaryCounts) {
|
|
for (const key of keys) {
|
|
remainders[key] -= counts[key];
|
|
}
|
|
}
|
|
|
|
let totalRemainder = 0;
|
|
for (const key of keys) {
|
|
totalRemainder += remainders[key];
|
|
}
|
|
|
|
this._cleanupExtra();
|
|
if (totalRemainder > 0) {
|
|
this.extra = this._createExtra(totalCounts, remainders, totalRemainder);
|
|
}
|
|
}
|
|
|
|
_createExtra(totalCounts, remainders, totalRemainder) {
|
|
const node = this.instantiateTemplate('dictionary-extra');
|
|
this._integrityExtraInfoNode = node;
|
|
|
|
node.querySelector('.dictionary-total-count').textContent = `${totalRemainder} item${totalRemainder !== 1 ? 's' : ''}`;
|
|
|
|
const n = node.querySelector('.dictionary-counts');
|
|
n.textContent = JSON.stringify({counts: totalCounts, remainders}, null, 4);
|
|
n.hidden = false;
|
|
|
|
this._integrityExtraInfoContainer.appendChild(node);
|
|
}
|
|
|
|
_cleanupExtra() {
|
|
const node = this._integrityExtraInfoNode;
|
|
if (node === null) { return; }
|
|
this._integrityExtraInfoNode = null;
|
|
|
|
const parent = node.parentNode;
|
|
if (parent === null) { return; }
|
|
|
|
parent.removeChild(node);
|
|
}
|
|
|
|
_createDictionaryEntry(dictionary) {
|
|
const node = this.instantiateTemplate('dictionary');
|
|
this._dictionaryEntryContainer.appendChild(node);
|
|
|
|
const entry = new DictionaryEntry(this, node, dictionary);
|
|
this._dictionaryEntries.push(entry);
|
|
entry.prepare();
|
|
}
|
|
|
|
async _deleteDictionary(dictionaryTitle) {
|
|
if (this._isDeleting || this._checkingIntegrity) { return; }
|
|
|
|
const index = this._dictionaryEntries.findIndex((entry) => entry.dictionaryTitle === dictionaryTitle);
|
|
if (index < 0) { return; }
|
|
|
|
const storageController = this._storageController;
|
|
const statusFooter = this._statusFooter;
|
|
const {node} = this._dictionaryEntries[index];
|
|
const progressSelector = '.dictionary-delete-progress';
|
|
const progressContainers = [
|
|
...node.querySelectorAll('.progress-container'),
|
|
...document.querySelectorAll(`#dictionaries ${progressSelector}`)
|
|
];
|
|
const progressBars = [
|
|
...node.querySelectorAll('.progress-bar'),
|
|
...document.querySelectorAll(`${progressSelector} .progress-bar`)
|
|
];
|
|
const infoLabels = document.querySelectorAll(`${progressSelector} .progress-info`);
|
|
const statusLabels = document.querySelectorAll(`${progressSelector} .progress-status`);
|
|
const prevention = this._settingsController.preventPageExit();
|
|
try {
|
|
this._isDeleting = true;
|
|
this._setButtonsEnabled(false);
|
|
|
|
const onProgress = ({processed, count, storeCount, storesProcesed}) => {
|
|
const percent = (
|
|
(count > 0 && storesProcesed > 0) ?
|
|
(processed / count) * (storesProcesed / storeCount) * 100.0 :
|
|
0.0
|
|
);
|
|
const cssString = `${percent}%`;
|
|
const statusString = `${percent.toFixed(0)}%`;
|
|
for (const progressBar of progressBars) { progressBar.style.width = cssString; }
|
|
for (const label of statusLabels) { label.textContent = statusString; }
|
|
};
|
|
|
|
onProgress({processed: 0, count: 1, storeCount: 1, storesProcesed: 0});
|
|
|
|
for (const progress of progressContainers) { progress.hidden = false; }
|
|
for (const label of infoLabels) { label.textContent = 'Deleting dictionary...'; }
|
|
if (statusFooter !== null) { statusFooter.setTaskActive(progressSelector, true); }
|
|
|
|
await this._deleteDictionaryInternal(dictionaryTitle, onProgress);
|
|
await this._deleteDictionarySettings(dictionaryTitle);
|
|
} catch (e) {
|
|
yomichan.logError(e);
|
|
} finally {
|
|
prevention.end();
|
|
for (const progress of progressContainers) { progress.hidden = true; }
|
|
if (statusFooter !== null) { statusFooter.setTaskActive(progressSelector, false); }
|
|
this._setButtonsEnabled(true);
|
|
storageController.updateStats();
|
|
this._isDeleting = false;
|
|
}
|
|
}
|
|
|
|
_setButtonsEnabled(value) {
|
|
value = !value;
|
|
for (const node of document.querySelectorAll('.dictionary-database-mutating-input')) {
|
|
node.disabled = value;
|
|
}
|
|
}
|
|
|
|
async _deleteDictionaryInternal(dictionaryTitle, onProgress) {
|
|
const dictionaryDatabase = await this._getPreparedDictionaryDatabase();
|
|
try {
|
|
await dictionaryDatabase.deleteDictionary(dictionaryTitle, {rate: 1000}, onProgress);
|
|
api.triggerDatabaseUpdated('dictionary', 'delete');
|
|
} finally {
|
|
dictionaryDatabase.close();
|
|
}
|
|
}
|
|
|
|
async _getPreparedDictionaryDatabase() {
|
|
const dictionaryDatabase = new DictionaryDatabase();
|
|
await dictionaryDatabase.prepare();
|
|
return dictionaryDatabase;
|
|
}
|
|
|
|
async _deleteDictionarySettings(dictionaryTitle) {
|
|
const optionsFull = await this._settingsController.getOptionsFull();
|
|
const {profiles} = optionsFull;
|
|
const targets = [];
|
|
for (let i = 0, ii = profiles.length; i < ii; ++i) {
|
|
const {options: {dictionaries}} = profiles[i];
|
|
if (Object.prototype.hasOwnProperty.call(dictionaries, dictionaryTitle)) {
|
|
const path = ObjectPropertyAccessor.getPathString(['profiles', i, 'options', 'dictionaries', dictionaryTitle]);
|
|
targets.push({action: 'delete', path});
|
|
}
|
|
}
|
|
await this._settingsController.modifyGlobalSettings(targets);
|
|
}
|
|
|
|
async _ensureDictionarySettings(dictionaries2) {
|
|
const optionsFull = await this._settingsController.getOptionsFull();
|
|
const {profiles} = optionsFull;
|
|
const targets = [];
|
|
for (const {title} of dictionaries2) {
|
|
for (let i = 0, ii = profiles.length; i < ii; ++i) {
|
|
const {options: {dictionaries: dictionaryOptions}} = profiles[i];
|
|
if (Object.prototype.hasOwnProperty.call(dictionaryOptions, title)) { continue; }
|
|
|
|
const path = ObjectPropertyAccessor.getPathString(['profiles', i, 'options', 'dictionaries', title]);
|
|
targets.push({
|
|
action: 'set',
|
|
path,
|
|
value: {
|
|
enabled: false,
|
|
allowSecondarySearches: false,
|
|
priority: 0
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
if (targets.length > 0) {
|
|
await this._settingsController.modifyGlobalSettings(targets);
|
|
}
|
|
}
|
|
}
|