2019-09-08 13:16:05 -04:00
|
|
|
/*
|
2020-04-10 11:06:55 -07:00
|
|
|
* Copyright (C) 2019-2020 Yomichan Authors
|
2019-09-08 13:16:05 -04:00
|
|
|
*
|
|
|
|
* 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
|
2020-01-01 12:00:31 -05:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2019-09-08 13:16:05 -04:00
|
|
|
*/
|
|
|
|
|
2020-03-10 22:30:36 -04:00
|
|
|
/* global
|
|
|
|
* ConditionsUI
|
|
|
|
* conditionsClearCaches
|
|
|
|
* profileConditionsDescriptor
|
2020-05-09 18:36:00 +03:00
|
|
|
* profileConditionsDescriptorPromise
|
2020-03-10 22:30:36 -04:00
|
|
|
* utilBackgroundIsolate
|
|
|
|
*/
|
2020-02-01 15:00:34 -05:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
class ProfileController {
|
2020-05-30 09:33:13 -04:00
|
|
|
constructor(settingsController) {
|
|
|
|
this._settingsController = settingsController;
|
2020-05-29 19:47:18 -04:00
|
|
|
this._conditionsContainer = null;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async prepare() {
|
|
|
|
$('#profile-target').change(this._onTargetProfileChanged.bind(this));
|
|
|
|
$('#profile-name').change(this._onNameChanged.bind(this));
|
|
|
|
$('#profile-add').click(this._onAdd.bind(this));
|
|
|
|
$('#profile-remove').click(this._onRemove.bind(this));
|
|
|
|
$('#profile-remove-confirm').click(this._onRemoveConfirm.bind(this));
|
|
|
|
$('#profile-copy').click(this._onCopy.bind(this));
|
|
|
|
$('#profile-copy-confirm').click(this._onCopyConfirm.bind(this));
|
|
|
|
$('#profile-move-up').click(() => this._onMove(-1));
|
|
|
|
$('#profile-move-down').click(() => this._onMove(1));
|
|
|
|
$('.profile-form').find('input, select, textarea').not('.profile-form-manual').change(this._onInputChanged.bind(this));
|
2020-05-30 09:33:13 -04:00
|
|
|
|
|
|
|
this._settingsController.on('optionsChanged', this._onOptionsChanged.bind(this));
|
|
|
|
|
|
|
|
this._onOptionsChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Private
|
|
|
|
|
|
|
|
async _onOptionsChanged() {
|
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
|
|
|
await this._formWrite(optionsFull);
|
2020-05-29 19:47:18 -04:00
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
_tryGetIntegerValue(selector, min, max) {
|
|
|
|
const value = parseInt($(selector).val(), 10);
|
|
|
|
return (
|
|
|
|
typeof value === 'number' &&
|
|
|
|
Number.isFinite(value) &&
|
|
|
|
Math.floor(value) === value &&
|
|
|
|
value >= min &&
|
|
|
|
value < max
|
|
|
|
) ? value : null;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _formRead(optionsFull) {
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const profile = optionsFull.profiles[currentProfileIndex];
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
// Current profile
|
|
|
|
const index = this._tryGetIntegerValue('#profile-active', 0, optionsFull.profiles.length);
|
|
|
|
if (index !== null) {
|
|
|
|
optionsFull.profileCurrent = index;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
// Profile name
|
|
|
|
profile.name = $('#profile-name').val();
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _formWrite(optionsFull) {
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const profile = optionsFull.profiles[currentProfileIndex];
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
this._populateSelect($('#profile-active'), optionsFull.profiles, optionsFull.profileCurrent, null);
|
|
|
|
this._populateSelect($('#profile-target'), optionsFull.profiles, currentProfileIndex, null);
|
|
|
|
$('#profile-remove').prop('disabled', optionsFull.profiles.length <= 1);
|
|
|
|
$('#profile-copy').prop('disabled', optionsFull.profiles.length <= 1);
|
|
|
|
$('#profile-move-up').prop('disabled', currentProfileIndex <= 0);
|
|
|
|
$('#profile-move-down').prop('disabled', currentProfileIndex >= optionsFull.profiles.length - 1);
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
$('#profile-name').val(profile.name);
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
if (this._conditionsContainer !== null) {
|
|
|
|
this._conditionsContainer.cleanup();
|
|
|
|
}
|
2019-09-09 20:19:49 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
await profileConditionsDescriptorPromise;
|
|
|
|
this._conditionsContainer = new ConditionsUI.Container(
|
|
|
|
profileConditionsDescriptor,
|
|
|
|
'popupLevel',
|
|
|
|
profile.conditionGroups,
|
|
|
|
$('#profile-condition-groups'),
|
|
|
|
$('#profile-add-condition-group')
|
|
|
|
);
|
|
|
|
this._conditionsContainer.save = () => {
|
2020-05-30 09:33:13 -04:00
|
|
|
this._settingsController.save();
|
2020-05-29 19:47:18 -04:00
|
|
|
conditionsClearCaches(profileConditionsDescriptor);
|
|
|
|
};
|
|
|
|
this._conditionsContainer.isolate = utilBackgroundIsolate;
|
2019-09-09 20:19:49 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
_populateSelect(select, profiles, currentValue, ignoreIndices) {
|
|
|
|
select.empty();
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2019-09-10 21:20:03 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
for (let i = 0; i < profiles.length; ++i) {
|
|
|
|
if (ignoreIndices !== null && ignoreIndices.indexOf(i) >= 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const profile = profiles[i];
|
|
|
|
select.append($(`<option value="${i}">${profile.name}</option>`));
|
2019-09-10 21:20:03 -04:00
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
select.val(`${currentValue}`);
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
_createCopyName(name, profiles, maxUniqueAttempts) {
|
|
|
|
let space, index, prefix, suffix;
|
|
|
|
const match = /^([\w\W]*\(Copy)((\s+)(\d+))?(\)\s*)$/.exec(name);
|
|
|
|
if (match === null) {
|
|
|
|
prefix = `${name} (Copy`;
|
|
|
|
space = '';
|
|
|
|
index = '';
|
|
|
|
suffix = ')';
|
2019-09-08 13:16:05 -04:00
|
|
|
} else {
|
2020-05-29 19:47:18 -04:00
|
|
|
prefix = match[1];
|
|
|
|
suffix = match[5];
|
|
|
|
if (typeof match[2] === 'string') {
|
|
|
|
space = match[3];
|
|
|
|
index = parseInt(match[4], 10) + 1;
|
|
|
|
} else {
|
|
|
|
space = ' ';
|
|
|
|
index = 2;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
let i = 0;
|
|
|
|
while (true) {
|
|
|
|
const newName = `${prefix}${space}${index}${suffix}`;
|
|
|
|
if (i++ >= maxUniqueAttempts || profiles.findIndex((profile) => profile.name === newName) < 0) {
|
|
|
|
return newName;
|
|
|
|
}
|
|
|
|
if (typeof index !== 'number') {
|
|
|
|
index = 2;
|
|
|
|
space = ' ';
|
|
|
|
} else {
|
|
|
|
++index;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onInputChanged(e) {
|
|
|
|
if (!e.originalEvent && !e.isTrigger) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
2020-05-29 19:47:18 -04:00
|
|
|
await this._formRead(optionsFull);
|
2020-05-30 09:33:13 -04:00
|
|
|
await this._settingsController.save();
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onTargetProfileChanged() {
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const index = this._tryGetIntegerValue('#profile-target', 0, optionsFull.profiles.length);
|
|
|
|
if (index === null || currentProfileIndex === index) {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
this._settingsController.profileIndex = index;
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onAdd() {
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const profile = utilBackgroundIsolate(optionsFull.profiles[currentProfileIndex]);
|
|
|
|
profile.name = this._createCopyName(profile.name, optionsFull.profiles, 100);
|
|
|
|
optionsFull.profiles.push(profile);
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
this._settingsController.profileIndex = optionsFull.profiles.length - 1;
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
await this._settingsController.save();
|
2020-05-29 19:47:18 -04:00
|
|
|
}
|
2020-04-26 22:33:50 +03:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onRemove(e) {
|
|
|
|
if (e.shiftKey) {
|
|
|
|
return await this._onRemoveConfirm();
|
|
|
|
}
|
2020-04-26 22:33:50 +03:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFull();
|
2020-05-29 19:47:18 -04:00
|
|
|
if (optionsFull.profiles.length <= 1) {
|
|
|
|
return;
|
|
|
|
}
|
2020-04-26 22:33:50 +03:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const profile = optionsFull.profiles[currentProfileIndex];
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
$('#profile-remove-modal-profile-name').text(profile.name);
|
|
|
|
$('#profile-remove-modal').modal('show');
|
2019-09-10 21:32:07 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onRemoveConfirm() {
|
|
|
|
$('#profile-remove-modal').modal('hide');
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
2020-05-29 19:47:18 -04:00
|
|
|
if (optionsFull.profiles.length <= 1) {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
optionsFull.profiles.splice(currentProfileIndex, 1);
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
if (currentProfileIndex >= optionsFull.profiles.length) {
|
2020-05-30 09:33:13 -04:00
|
|
|
this._settingsController.profileIndex = optionsFull.profiles.length - 1;
|
2020-05-29 19:47:18 -04:00
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
if (optionsFull.profileCurrent >= optionsFull.profiles.length) {
|
|
|
|
optionsFull.profileCurrent = optionsFull.profiles.length - 1;
|
|
|
|
}
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
await this._settingsController.save();
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
_onNameChanged() {
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
$('#profile-active, #profile-target').find(`[value="${currentProfileIndex}"]`).text(this.value);
|
2019-09-08 13:16:05 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onMove(offset) {
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
const index = currentProfileIndex + offset;
|
|
|
|
if (index < 0 || index >= optionsFull.profiles.length) {
|
|
|
|
return;
|
|
|
|
}
|
2020-04-26 22:33:50 +03:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
const profile = optionsFull.profiles[currentProfileIndex];
|
|
|
|
optionsFull.profiles.splice(currentProfileIndex, 1);
|
|
|
|
optionsFull.profiles.splice(index, 0, profile);
|
2019-09-08 13:16:05 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
if (optionsFull.profileCurrent === currentProfileIndex) {
|
|
|
|
optionsFull.profileCurrent = index;
|
|
|
|
}
|
2019-09-10 21:20:03 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
this._settingsController.profileIndex = index;
|
2019-09-10 21:29:01 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
await this._settingsController.save();
|
2019-09-10 21:29:01 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onCopy() {
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
2020-05-29 19:47:18 -04:00
|
|
|
if (optionsFull.profiles.length <= 1) {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-10 21:29:01 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
this._populateSelect($('#profile-copy-source'), optionsFull.profiles, currentProfileIndex === 0 ? 1 : 0, [currentProfileIndex]);
|
|
|
|
$('#profile-copy-modal').modal('show');
|
2019-09-10 21:20:03 -04:00
|
|
|
}
|
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
async _onCopyConfirm() {
|
|
|
|
$('#profile-copy-modal').modal('hide');
|
2019-09-10 21:20:03 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
const optionsFull = await this._settingsController.getOptionsFullMutable();
|
2020-05-29 19:47:18 -04:00
|
|
|
const index = this._tryGetIntegerValue('#profile-copy-source', 0, optionsFull.profiles.length);
|
2020-05-30 09:33:13 -04:00
|
|
|
const currentProfileIndex = this._settingsController.profileIndex;
|
2020-05-29 19:47:18 -04:00
|
|
|
if (index === null || index === currentProfileIndex) {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-10 21:20:03 -04:00
|
|
|
|
2020-05-29 19:47:18 -04:00
|
|
|
const profileOptions = utilBackgroundIsolate(optionsFull.profiles[index].options);
|
|
|
|
optionsFull.profiles[currentProfileIndex].options = profileOptions;
|
2019-09-10 21:20:03 -04:00
|
|
|
|
2020-05-30 09:33:13 -04:00
|
|
|
await this._settingsController.save();
|
2020-05-29 19:47:18 -04:00
|
|
|
}
|
2019-09-10 21:20:03 -04:00
|
|
|
}
|