2019-10-12 16:59:51 +00:00
|
|
|
/*
|
2020-01-01 17:00:00 +00:00
|
|
|
* Copyright (C) 2019-2020 Alex Yatskov <alex@foosoft.net>
|
2019-10-12 16:59:51 +00: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 17:00:31 +00:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2019-10-12 16:59:51 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
class SettingsPopupPreview {
|
|
|
|
constructor() {
|
|
|
|
this.frontend = null;
|
|
|
|
this.apiOptionsGetOld = apiOptionsGet;
|
2019-10-13 16:10:00 +00:00
|
|
|
this.popupInjectOuterStylesheetOld = Popup.injectOuterStylesheet;
|
2019-10-12 16:59:51 +00:00
|
|
|
this.popupShown = false;
|
2019-10-12 21:59:56 +00:00
|
|
|
this.themeChangeTimeout = null;
|
2019-12-12 02:27:42 +00:00
|
|
|
this.textSource = null;
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static create() {
|
|
|
|
const instance = new SettingsPopupPreview();
|
|
|
|
instance.prepare();
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
async prepare() {
|
|
|
|
// Setup events
|
|
|
|
window.addEventListener('message', (e) => this.onMessage(e), false);
|
|
|
|
|
|
|
|
const themeDarkCheckbox = document.querySelector('#theme-dark-checkbox');
|
|
|
|
if (themeDarkCheckbox !== null) {
|
|
|
|
themeDarkCheckbox.addEventListener('change', () => this.onThemeDarkCheckboxChanged(themeDarkCheckbox), false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overwrite API functions
|
|
|
|
window.apiOptionsGet = (...args) => this.apiOptionsGet(...args);
|
|
|
|
|
|
|
|
// Overwrite frontend
|
2019-12-21 18:27:32 +00:00
|
|
|
const popupHost = new PopupProxyHost();
|
|
|
|
await popupHost.prepare();
|
|
|
|
|
|
|
|
const popup = popupHost.createPopup(null, 0);
|
2019-12-21 18:19:31 +00:00
|
|
|
popup.setChildrenSupported(false);
|
|
|
|
|
|
|
|
this.frontend = new Frontend(popup);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
|
|
|
this.frontend.setEnabled = function () {};
|
|
|
|
this.frontend.searchClear = function () {};
|
|
|
|
|
2019-12-21 18:19:31 +00:00
|
|
|
await this.frontend.prepare();
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2019-10-13 16:10:00 +00:00
|
|
|
// Overwrite popup
|
|
|
|
Popup.injectOuterStylesheet = (...args) => this.popupInjectOuterStylesheet(...args);
|
|
|
|
|
2019-10-12 16:59:51 +00:00
|
|
|
// Update search
|
|
|
|
this.updateSearch();
|
|
|
|
}
|
|
|
|
|
|
|
|
async apiOptionsGet(...args) {
|
|
|
|
const options = await this.apiOptionsGetOld(...args);
|
|
|
|
options.general.enable = true;
|
|
|
|
options.general.debugInfo = false;
|
|
|
|
options.general.popupWidth = 400;
|
|
|
|
options.general.popupHeight = 250;
|
|
|
|
options.general.popupHorizontalOffset = 0;
|
|
|
|
options.general.popupVerticalOffset = 10;
|
|
|
|
options.general.popupHorizontalOffset2 = 10;
|
|
|
|
options.general.popupVerticalOffset2 = 0;
|
|
|
|
options.general.popupHorizontalTextPosition = 'below';
|
|
|
|
options.general.popupVerticalTextPosition = 'before';
|
|
|
|
options.scanning.selectText = false;
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
2019-10-13 16:10:00 +00:00
|
|
|
popupInjectOuterStylesheet(...args) {
|
|
|
|
// This simulates the stylesheet priorities when injecting using the web extension API.
|
|
|
|
const result = this.popupInjectOuterStylesheetOld(...args);
|
|
|
|
|
|
|
|
const outerStylesheet = Popup.outerStylesheet;
|
|
|
|
const node = document.querySelector('#client-css');
|
|
|
|
if (node !== null && outerStylesheet !== null) {
|
|
|
|
node.parentNode.insertBefore(outerStylesheet, node);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-10-12 16:59:51 +00:00
|
|
|
onMessage(e) {
|
|
|
|
const {action, params} = e.data;
|
2019-12-12 02:45:39 +00:00
|
|
|
const handler = SettingsPopupPreview._messageHandlers.get(action);
|
|
|
|
if (typeof handler !== 'function') { return; }
|
|
|
|
|
|
|
|
handler(this, params);
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
onThemeDarkCheckboxChanged(node) {
|
|
|
|
document.documentElement.classList.toggle('dark', node.checked);
|
2019-10-12 21:59:56 +00:00
|
|
|
if (this.themeChangeTimeout !== null) {
|
|
|
|
clearTimeout(this.themeChangeTimeout);
|
|
|
|
}
|
|
|
|
this.themeChangeTimeout = setTimeout(() => {
|
|
|
|
this.themeChangeTimeout = null;
|
|
|
|
this.frontend.popup.updateTheme();
|
|
|
|
}, 300);
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
setText(text) {
|
|
|
|
const exampleText = document.querySelector('#example-text');
|
|
|
|
if (exampleText === null) { return; }
|
|
|
|
|
|
|
|
exampleText.textContent = text;
|
|
|
|
this.updateSearch();
|
|
|
|
}
|
|
|
|
|
2019-10-12 21:21:36 +00:00
|
|
|
setInfoVisible(visible) {
|
|
|
|
const node = document.querySelector('.placeholder-info');
|
|
|
|
if (node === null) { return; }
|
|
|
|
|
|
|
|
node.classList.toggle('placeholder-info-visible', visible);
|
|
|
|
}
|
|
|
|
|
|
|
|
setCustomCss(css) {
|
|
|
|
if (this.frontend === null) { return; }
|
|
|
|
this.frontend.popup.setCustomCss(css);
|
|
|
|
}
|
|
|
|
|
2019-10-13 15:51:59 +00:00
|
|
|
setCustomOuterCss(css) {
|
|
|
|
if (this.frontend === null) { return; }
|
|
|
|
this.frontend.popup.setCustomOuterCss(css, true);
|
|
|
|
}
|
|
|
|
|
2019-10-12 16:59:51 +00:00
|
|
|
async updateSearch() {
|
|
|
|
const exampleText = document.querySelector('#example-text');
|
|
|
|
if (exampleText === null) { return; }
|
|
|
|
|
|
|
|
const textNode = exampleText.firstChild;
|
|
|
|
if (textNode === null) { return; }
|
|
|
|
|
|
|
|
const range = document.createRange();
|
|
|
|
range.selectNode(textNode);
|
2020-01-25 16:18:18 +00:00
|
|
|
const source = new TextSourceRange(range, range.toString(), null, null);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2019-10-25 00:01:04 +00:00
|
|
|
try {
|
2019-12-08 09:10:19 +00:00
|
|
|
await this.frontend.onSearchSource(source, 'script');
|
2019-12-23 19:09:41 +00:00
|
|
|
this.frontend.setCurrentTextSource(source);
|
2019-10-25 00:01:04 +00:00
|
|
|
} finally {
|
|
|
|
source.cleanup();
|
|
|
|
}
|
2019-12-12 02:27:42 +00:00
|
|
|
this.textSource = source;
|
2019-12-12 02:31:21 +00:00
|
|
|
await this.frontend.showContentCompleted();
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2019-12-16 02:48:29 +00:00
|
|
|
if (this.frontend.popup.isVisibleSync()) {
|
2019-10-12 16:59:51 +00:00
|
|
|
this.popupShown = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setInfoVisible(!this.popupShown);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-12 02:45:39 +00:00
|
|
|
SettingsPopupPreview._messageHandlers = new Map([
|
|
|
|
['setText', (self, {text}) => self.setText(text)],
|
|
|
|
['setCustomCss', (self, {css}) => self.setCustomCss(css)],
|
|
|
|
['setCustomOuterCss', (self, {css}) => self.setCustomOuterCss(css)]
|
|
|
|
]);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
|
|
|
SettingsPopupPreview.instance = SettingsPopupPreview.create();
|
|
|
|
|
|
|
|
|
|
|
|
|