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';
|
2020-02-27 20:33:13 -05:00
|
|
|
options.anki.fieldTemplates = null;
|
2019-09-06 18:21:20 -04:00
|
|
|
},
|
|
|
|
(options) => {
|
|
|
|
if (utilStringHashCode(options.anki.fieldTemplates) === 1285806040) {
|
2020-02-27 20:33:13 -05:00
|
|
|
options.anki.fieldTemplates = null;
|
2019-09-06 18:21:20 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
(options) => {
|
|
|
|
if (utilStringHashCode(options.anki.fieldTemplates) === -250091611) {
|
2020-02-27 20:33:13 -05:00
|
|
|
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;
|
|
|
|
}
|
2020-03-15 17:32:31 -04:00
|
|
|
},
|
|
|
|
(options) => {
|
|
|
|
// Version 13 changes:
|
|
|
|
// Default anki field tempaltes updated to include {document-title}.
|
|
|
|
let fieldTemplates = options.anki.fieldTemplates;
|
|
|
|
if (typeof fieldTemplates === 'string') {
|
|
|
|
fieldTemplates += '\n\n{{#*inline "document-title"}}\n {{~context.document.title~}}\n{{/inline}}';
|
|
|
|
options.anki.fieldTemplates = fieldTemplates;
|
|
|
|
}
|
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,
|
2020-01-25 19:00:36 +02:00
|
|
|
enableClipboardPopups: false,
|
2017-09-29 05:41:29 +03:00
|
|
|
resultOutputMode: 'group',
|
2017-07-16 13:14:28 -07:00
|
|
|
debugInfo: false,
|
|
|
|
maxResults: 32,
|
|
|
|
showAdvanced: false,
|
2019-02-14 21:42:59 -05:00
|
|
|
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,
|
2019-08-31 11:51:31 -04:00
|
|
|
popupHorizontalOffset2: 10,
|
|
|
|
popupVerticalOffset2: 0,
|
2019-08-31 11:56:12 -04:00
|
|
|
popupHorizontalTextPosition: 'below',
|
2019-08-31 11:51:31 -04:00
|
|
|
popupVerticalTextPosition: 'before',
|
2019-12-23 17:39:00 -05:00
|
|
|
popupScalingFactor: 1,
|
|
|
|
popupScaleRelativeToPageZoom: false,
|
2019-12-30 12:42:12 -05:00
|
|
|
popupScaleRelativeToVisualViewport: true,
|
2017-10-12 09:59:09 +03:00
|
|
|
showGuide: true,
|
2017-10-15 09:45:00 +03:00
|
|
|
compactTags: false,
|
2017-10-24 16:23:13 +03:00
|
|
|
compactGlossaries: false,
|
2019-07-09 17:52:44 -04:00
|
|
|
mainDictionary: '',
|
2019-10-12 13:48:23 -04:00
|
|
|
popupTheme: 'default',
|
2019-10-12 17:59:56 -04:00
|
|
|
popupOuterTheme: 'default',
|
2019-10-13 11:05:21 -04:00
|
|
|
customPopupCss: '',
|
2019-10-27 20:11:23 +02:00
|
|
|
customPopupOuterCss: '',
|
|
|
|
enableWanakana: true,
|
2020-03-01 14:38:16 -05:00
|
|
|
enableClipboardMonitor: false,
|
|
|
|
showPitchAccentDownstepNotation: true,
|
|
|
|
showPitchAccentPositionNotation: true,
|
2020-04-05 01:43:12 +03:00
|
|
|
showPitchAccentGraph: false,
|
|
|
|
showIframePopupsInRootFrame: false
|
2017-07-16 13:14:28 -07:00
|
|
|
},
|
|
|
|
|
2019-10-09 22:33:35 -04:00
|
|
|
audio: {
|
|
|
|
enabled: true,
|
2019-10-11 23:24:08 -04:00
|
|
|
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,
|
2017-09-16 23:08:43 -07:00
|
|
|
autoHideResults: false,
|
2017-09-17 10:09:48 -07:00
|
|
|
delay: 20,
|
2017-07-16 13:14:28 -07:00
|
|
|
length: 10,
|
2019-08-31 22:12:21 -04:00
|
|
|
modifier: 'shift',
|
2019-08-17 19:32:58 -04:00
|
|
|
deepDomScan: false,
|
2019-08-18 16:48:18 -04:00
|
|
|
popupNestingMaxDepth: 0,
|
2019-09-26 21:33:33 -07:00
|
|
|
enablePopupSearch: false,
|
2019-08-25 21:13:17 -04:00
|
|
|
enableOnPopupExpressions: false,
|
2020-01-26 21:00:19 +02:00
|
|
|
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',
|
2020-01-23 20:28:38 -05:00
|
|
|
convertAlphabeticCharacters: 'false',
|
2020-01-23 20:49:41 -05:00
|
|
|
convertHiraganaToKatakana: 'false',
|
2020-01-23 20:28:38 -05:00
|
|
|
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: {}},
|
2019-12-14 16:59:44 -05:00
|
|
|
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) {
|
2019-11-25 14:19:18 -05:00
|
|
|
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
|
2019-09-09 20:19:49 -04:00
|
|
|
*
|
|
|
|
* 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
|
|
|
*/
|
|
|
|
|
2019-11-23 22:54:06 -05: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) {
|
2019-11-26 17:33:09 -05:00
|
|
|
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',
|
2019-09-09 20:19:49 -04:00
|
|
|
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) {
|
2019-09-09 20:19:49 -04:00
|
|
|
if (!Array.isArray(profile.conditionGroups)) {
|
|
|
|
profile.conditionGroups = [];
|
|
|
|
}
|
2019-09-07 19:50:58 -04:00
|
|
|
profile.options = profileOptionsUpdateVersion(profile.options);
|
|
|
|
}
|
|
|
|
|
2019-11-23 22:54:06 -05:00
|
|
|
// 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);
|
2019-11-26 17:33:09 -05:00
|
|
|
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) {
|
2019-09-07 16:15:18 -04:00
|
|
|
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));
|
2019-09-07 16:15:18 -04:00
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
2017-07-16 13:14:28 -07:00
|
|
|
});
|
|
|
|
}
|
2019-12-14 23:06:44 -05:00
|
|
|
|
|
|
|
function optionsGetDefault() {
|
|
|
|
return optionsUpdateVersion({}, {});
|
|
|
|
}
|