yomichan/ext/bg/js/options.js

344 lines
9.8 KiB
JavaScript
Raw Normal View History

2017-07-16 13:14:28 -07:00
/*
2020-01-01 12:00:00 -05:00
* Copyright (C) 2016-2020 Alex Yatskov <alex@foosoft.net>
2017-07-16 13:14:28 -07:00
* Author: Alex Yatskov <alex@foosoft.net>
*
* 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/>.
2017-07-16 13:14:28 -07:00
*/
2020-03-10 22:30:36 -04:00
/* global
* utilStringHashCode
*/
2017-07-16 13:14:28 -07:00
2019-09-07 19:50:58 -04:00
/*
* Generic options functions
*/
function optionsGenericApplyUpdates(options, updates) {
2019-09-06 18:21:20 -04:00
const targetVersion = updates.length;
const currentVersion = options.version;
if (typeof currentVersion === 'number' && Number.isFinite(currentVersion)) {
for (let i = Math.max(0, Math.floor(currentVersion)); i < targetVersion; ++i) {
const update = updates[i];
if (update !== null) {
update(options);
}
}
}
options.version = targetVersion;
return options;
}
2019-09-07 19:50:58 -04:00
/*
* Per-profile options
*/
const profileOptionsVersionUpdates = [
2019-09-06 18:21:20 -04:00
null,
null,
null,
null,
(options) => {
options.general.audioSource = options.general.audioPlayback ? 'jpod101' : 'disabled';
},
(options) => {
options.general.showGuide = false;
},
(options) => {
options.scanning.modifier = options.scanning.requireShift ? 'shift' : 'none';
},
(options) => {
options.general.resultOutputMode = options.general.groupResults ? 'group' : 'split';
options.anki.fieldTemplates = null;
2019-09-06 18:21:20 -04:00
},
(options) => {
if (utilStringHashCode(options.anki.fieldTemplates) === 1285806040) {
options.anki.fieldTemplates = null;
2019-09-06 18:21:20 -04:00
}
},
(options) => {
if (utilStringHashCode(options.anki.fieldTemplates) === -250091611) {
options.anki.fieldTemplates = null;
2019-09-06 18:21:20 -04:00
}
2019-10-09 22:33:35 -04:00
},
(options) => {
const oldAudioSource = options.general.audioSource;
const disabled = oldAudioSource === 'disabled';
options.audio.enabled = !disabled;
options.audio.volume = options.general.audioVolume;
options.audio.autoPlay = options.general.autoPlayAudio;
options.audio.sources = [disabled ? 'jpod101' : oldAudioSource];
delete options.general.audioSource;
delete options.general.audioVolume;
delete options.general.autoPlayAudio;
2019-12-15 00:07:54 -05:00
},
(options) => {
// Version 12 changes:
// The preferred default value of options.anki.fieldTemplates has been changed to null.
if (utilStringHashCode(options.anki.fieldTemplates) === 1444379824) {
options.anki.fieldTemplates = null;
}
2019-09-06 18:21:20 -04:00
}
];
2019-09-07 19:50:58 -04:00
function profileOptionsCreateDefaults() {
2019-09-05 19:56:29 -04:00
return {
2017-07-16 13:14:28 -07:00
general: {
enable: true,
enableClipboardPopups: false,
resultOutputMode: 'group',
2017-07-16 13:14:28 -07:00
debugInfo: false,
maxResults: 32,
showAdvanced: false,
popupDisplayMode: 'default',
2017-07-16 13:14:28 -07:00
popupWidth: 400,
popupHeight: 250,
2018-10-02 23:27:59 +09:00
popupHorizontalOffset: 0,
popupVerticalOffset: 10,
popupHorizontalOffset2: 10,
popupVerticalOffset2: 0,
popupHorizontalTextPosition: 'below',
popupVerticalTextPosition: 'before',
popupScalingFactor: 1,
popupScaleRelativeToPageZoom: false,
popupScaleRelativeToVisualViewport: true,
2017-10-12 09:59:09 +03:00
showGuide: true,
compactTags: false,
compactGlossaries: false,
2019-07-09 17:52:44 -04:00
mainDictionary: '',
2019-10-12 13:48:23 -04:00
popupTheme: 'default',
popupOuterTheme: 'default',
customPopupCss: '',
customPopupOuterCss: '',
enableWanakana: true,
2020-03-01 14:38:16 -05:00
enableClipboardMonitor: false,
showPitchAccentDownstepNotation: true,
showPitchAccentPositionNotation: true,
showPitchAccentGraph: false
2017-07-16 13:14:28 -07:00
},
2019-10-09 22:33:35 -04:00
audio: {
enabled: true,
sources: ['jpod101'],
2019-10-09 22:33:35 -04:00
volume: 100,
autoPlay: false,
2019-10-12 22:50:22 -04:00
customSourceUrl: '',
textToSpeechVoice: ''
2019-10-09 22:33:35 -04:00
},
2017-07-16 13:14:28 -07:00
scanning: {
middleMouse: true,
2019-02-10 20:44:16 -05:00
touchInputEnabled: true,
2017-07-16 13:14:28 -07:00
selectText: true,
alphanumeric: true,
autoHideResults: false,
delay: 20,
2017-07-16 13:14:28 -07:00
length: 10,
2019-08-31 22:12:21 -04:00
modifier: 'shift',
deepDomScan: false,
popupNestingMaxDepth: 0,
2019-09-26 21:33:33 -07:00
enablePopupSearch: false,
enableOnPopupExpressions: false,
enableOnSearchPage: true,
enableSearchTags: false
2017-07-16 13:14:28 -07:00
},
2019-12-22 13:50:30 -05:00
translation: {
convertHalfWidthCharacters: 'false',
convertNumericCharacters: 'false',
convertAlphabeticCharacters: 'false',
2020-01-23 20:49:41 -05:00
convertHiraganaToKatakana: 'false',
convertKatakanaToHiragana: 'variant'
2019-12-22 13:50:30 -05:00
},
2017-07-16 13:14:28 -07:00
dictionaries: {},
2019-11-05 15:56:45 +02:00
parsing: {
enableScanningParser: true,
2019-11-12 23:57:21 +02:00
enableMecabParser: false,
2019-11-13 20:24:11 +02:00
selectedParser: null,
2020-01-25 18:29:52 +02:00
termSpacing: true,
2019-11-13 20:24:11 +02:00
readingMode: 'hiragana'
2019-11-05 15:56:45 +02:00
},
2017-07-16 13:14:28 -07:00
anki: {
enable: false,
server: 'http://127.0.0.1:8765',
tags: ['yomichan'],
sentenceExt: 200,
2019-08-15 19:56:14 -04:00
screenshot: {format: 'png', quality: 92},
2017-07-16 13:14:28 -07:00
terms: {deck: '', model: '', fields: {}},
2017-09-06 13:18:06 -07:00
kanji: {deck: '', model: '', fields: {}},
fieldTemplates: null
2017-07-16 13:14:28 -07:00
}
};
2019-09-05 19:56:29 -04:00
}
2019-09-07 19:50:58 -04:00
function profileOptionsSetDefaults(options) {
const defaults = profileOptionsCreateDefaults();
2017-07-16 13:14:28 -07:00
const combine = (target, source) => {
for (const key in source) {
if (!hasOwn(target, key)) {
2017-07-16 13:14:28 -07:00
target[key] = source[key];
}
}
};
combine(options, defaults);
combine(options.general, defaults.general);
combine(options.scanning, defaults.scanning);
combine(options.anki, defaults.anki);
combine(options.anki.terms, defaults.anki.terms);
combine(options.anki.kanji, defaults.anki.kanji);
return options;
}
2019-09-07 19:50:58 -04:00
function profileOptionsUpdateVersion(options) {
profileOptionsSetDefaults(options);
return optionsGenericApplyUpdates(options, profileOptionsVersionUpdates);
}
/*
* Global options
*
* Each profile has an array named "conditionGroups", which is an array of condition groups
* which enable the contextual selection of profiles. The structure of the array is as follows:
* [
* {
* conditions: [
* {
* type: "string",
* operator: "string",
* value: "string"
* },
* // ...
* ]
* },
* // ...
* ]
2019-09-07 19:50:58 -04:00
*/
const optionsVersionUpdates = [
(options) => {
options.global = {
database: {
prefixWildcardsSupported: false
}
};
}
];
2019-09-07 19:50:58 -04:00
function optionsUpdateVersion(options, defaultProfileOptions) {
// Ensure profiles is an array
if (!Array.isArray(options.profiles)) {
options.profiles = [];
}
// Remove invalid
const profiles = options.profiles;
for (let i = profiles.length - 1; i >= 0; --i) {
if (!isObject(profiles[i])) {
2019-09-07 19:50:58 -04:00
profiles.splice(i, 1);
}
}
// Require at least one profile
if (profiles.length === 0) {
profiles.push({
name: 'Default',
options: defaultProfileOptions,
conditionGroups: []
2019-09-07 19:50:58 -04:00
});
}
// Ensure profileCurrent is valid
const profileCurrent = options.profileCurrent;
if (!(
typeof profileCurrent === 'number' &&
Number.isFinite(profileCurrent) &&
Math.floor(profileCurrent) === profileCurrent &&
profileCurrent >= 0 &&
profileCurrent < profiles.length
)) {
options.profileCurrent = 0;
}
// Update profile options
for (const profile of profiles) {
if (!Array.isArray(profile.conditionGroups)) {
profile.conditionGroups = [];
}
2019-09-07 19:50:58 -04:00
profile.options = profileOptionsUpdateVersion(profile.options);
}
// Version
if (typeof options.version !== 'number') {
options.version = 0;
}
2019-09-07 19:50:58 -04:00
// Generic updates
return optionsGenericApplyUpdates(options, optionsVersionUpdates);
2017-07-16 13:14:28 -07:00
}
function optionsLoad() {
return new Promise((resolve, reject) => {
2019-11-26 22:01:54 -05:00
chrome.storage.local.get(['options'], (store) => {
2019-09-05 20:35:04 -04:00
const error = chrome.runtime.lastError;
if (error) {
2019-10-07 21:04:58 -04:00
reject(new Error(error));
2019-09-05 20:35:04 -04:00
} else {
resolve(store.options);
}
});
2019-11-26 22:01:54 -05:00
}).then((optionsStr) => {
2019-09-05 20:35:04 -04:00
if (typeof optionsStr === 'string') {
const options = JSON.parse(optionsStr);
if (isObject(options)) {
2019-09-05 20:35:04 -04:00
return options;
}
}
return {};
}).catch(() => {
2017-07-16 13:14:28 -07:00
return {};
2019-11-26 22:01:54 -05:00
}).then((options) => {
2019-09-07 19:50:58 -04:00
return (
Array.isArray(options.profiles) ?
optionsUpdateVersion(options, {}) :
optionsUpdateVersion({}, options)
);
2017-07-16 13:14:28 -07:00
});
}
function optionsSave(options) {
return new Promise((resolve, reject) => {
chrome.storage.local.set({options: JSON.stringify(options)}, () => {
const error = chrome.runtime.lastError;
if (error) {
2019-10-07 21:04:58 -04:00
reject(new Error(error));
} else {
resolve();
}
});
2017-07-16 13:14:28 -07:00
});
}
2019-12-14 23:06:44 -05:00
function optionsGetDefault() {
return optionsUpdateVersion({}, {});
}