2019-10-12 16:59:51 +00:00
|
|
|
/*
|
2021-01-01 19:50:41 +00:00
|
|
|
* Copyright (C) 2019-2021 Yomichan Authors
|
2019-10-12 16:59:51 +00: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 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
|
|
|
*/
|
|
|
|
|
2020-03-11 02:30:36 +00:00
|
|
|
/* global
|
|
|
|
* Frontend
|
|
|
|
* TextSourceRange
|
2020-05-24 17:30:40 +00:00
|
|
|
* api
|
2020-10-20 23:08:17 +00:00
|
|
|
* wanakana
|
2020-03-11 02:30:36 +00:00
|
|
|
*/
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
class PopupPreviewFrame {
|
2021-02-10 03:56:04 +00:00
|
|
|
constructor(tabId, frameId, popupFactory, hotkeyHandler) {
|
|
|
|
this._tabId = tabId;
|
2020-06-21 20:14:05 +00:00
|
|
|
this._frameId = frameId;
|
|
|
|
this._popupFactory = popupFactory;
|
2021-01-18 00:28:42 +00:00
|
|
|
this._hotkeyHandler = hotkeyHandler;
|
2020-05-30 00:29:19 +00:00
|
|
|
this._frontend = null;
|
|
|
|
this._apiOptionsGetOld = null;
|
|
|
|
this._popupShown = false;
|
|
|
|
this._themeChangeTimeout = null;
|
|
|
|
this._textSource = null;
|
|
|
|
this._optionsContext = null;
|
2020-10-20 23:08:17 +00:00
|
|
|
this._exampleText = null;
|
|
|
|
this._exampleTextInput = null;
|
2020-02-17 04:41:17 +00:00
|
|
|
this._targetOrigin = chrome.runtime.getURL('/').replace(/\/$/, '');
|
2020-02-27 01:07:14 +00:00
|
|
|
|
|
|
|
this._windowMessageHandlers = new Map([
|
2020-10-20 23:08:17 +00:00
|
|
|
['setText', this._onSetText.bind(this)],
|
2020-05-30 00:29:19 +00:00
|
|
|
['setCustomCss', this._setCustomCss.bind(this)],
|
|
|
|
['setCustomOuterCss', this._setCustomOuterCss.bind(this)],
|
|
|
|
['updateOptionsContext', this._updateOptionsContext.bind(this)]
|
2020-02-27 01:07:14 +00:00
|
|
|
]);
|
2020-04-26 19:33:50 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
async prepare() {
|
2020-10-20 23:08:17 +00:00
|
|
|
this._exampleText = document.querySelector('#example-text');
|
|
|
|
this._exampleTextInput = document.querySelector('#example-text-input');
|
|
|
|
|
|
|
|
if (this._exampleTextInput !== null && typeof wanakana !== 'undefined') {
|
|
|
|
wanakana.bind(this._exampleTextInput);
|
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
window.addEventListener('message', this._onMessage.bind(this), false);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2020-04-26 19:33:50 +00:00
|
|
|
// Setup events
|
2020-05-30 00:29:19 +00:00
|
|
|
document.querySelector('#theme-dark-checkbox').addEventListener('change', this._onThemeDarkCheckboxChanged.bind(this), false);
|
2020-10-20 23:08:17 +00:00
|
|
|
this._exampleText.addEventListener('click', this._onExampleTextClick.bind(this), false);
|
|
|
|
this._exampleTextInput.addEventListener('blur', this._onExampleTextInputBlur.bind(this), false);
|
|
|
|
this._exampleTextInput.addEventListener('input', this._onExampleTextInputInput.bind(this), false);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
|
|
|
// Overwrite API functions
|
2020-05-30 00:29:19 +00:00
|
|
|
this._apiOptionsGetOld = api.optionsGet.bind(api);
|
|
|
|
api.optionsGet = this._apiOptionsGet.bind(this);
|
2019-10-12 16:59:51 +00:00
|
|
|
|
|
|
|
// Overwrite frontend
|
2020-08-16 20:16:18 +00:00
|
|
|
this._frontend = new Frontend({
|
2021-02-10 03:56:04 +00:00
|
|
|
tabId: this._tabId,
|
2020-08-16 20:16:18 +00:00
|
|
|
frameId: this._frameId,
|
|
|
|
popupFactory: this._popupFactory,
|
|
|
|
depth: 0,
|
|
|
|
parentPopupId: null,
|
|
|
|
parentFrameId: null,
|
|
|
|
useProxyPopup: false,
|
2021-01-31 21:18:17 +00:00
|
|
|
canUseWindowPopup: false,
|
2020-09-26 22:47:58 +00:00
|
|
|
pageType: 'web',
|
2020-11-23 20:23:47 +00:00
|
|
|
allowRootFramePopupProxy: false,
|
2021-01-18 00:28:42 +00:00
|
|
|
childrenSupported: false,
|
|
|
|
hotkeyHandler: this._hotkeyHandler
|
2020-08-16 20:16:18 +00:00
|
|
|
});
|
2020-11-09 03:49:40 +00:00
|
|
|
this._frontend.setOptionsContextOverride(this._optionsContext);
|
2020-05-30 00:29:19 +00:00
|
|
|
await this._frontend.prepare();
|
|
|
|
this._frontend.setDisabledOverride(true);
|
|
|
|
this._frontend.canClearSelection = false;
|
2020-11-23 20:23:47 +00:00
|
|
|
this._frontend.popup.on('customOuterCssChanged', this._onCustomOuterCssChanged.bind(this));
|
2020-06-21 20:14:05 +00:00
|
|
|
|
2019-10-12 16:59:51 +00:00
|
|
|
// Update search
|
2020-05-30 00:29:19 +00:00
|
|
|
this._updateSearch();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Private
|
|
|
|
|
|
|
|
async _apiOptionsGet(...args) {
|
|
|
|
const options = await this._apiOptionsGetOld(...args);
|
2019-10-12 16:59:51 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-11-29 17:00:41 +00:00
|
|
|
_onCustomOuterCssChanged({node, inShadow}) {
|
|
|
|
if (node === null || inShadow) { return; }
|
2019-10-13 16:10:00 +00:00
|
|
|
|
2020-11-26 19:13:53 +00:00
|
|
|
const node2 = document.querySelector('#popup-outer-css');
|
2020-11-23 20:23:47 +00:00
|
|
|
if (node2 === null) { return; }
|
2019-10-13 16:10:00 +00:00
|
|
|
|
2020-11-23 20:23:47 +00:00
|
|
|
// This simulates the stylesheet priorities when injecting using the web extension API.
|
|
|
|
node2.parentNode.insertBefore(node, node2);
|
2019-10-13 16:10:00 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
_onMessage(e) {
|
2020-02-17 04:41:17 +00:00
|
|
|
if (e.origin !== this._targetOrigin) { return; }
|
|
|
|
|
2019-10-12 16:59:51 +00:00
|
|
|
const {action, params} = e.data;
|
2020-02-27 01:07:14 +00:00
|
|
|
const handler = this._windowMessageHandlers.get(action);
|
2019-12-12 02:45:39 +00:00
|
|
|
if (typeof handler !== 'function') { return; }
|
|
|
|
|
2020-02-27 01:07:14 +00:00
|
|
|
handler(params);
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
_onThemeDarkCheckboxChanged(e) {
|
2020-02-27 02:19:22 +00:00
|
|
|
document.documentElement.classList.toggle('dark', e.target.checked);
|
2020-05-30 00:29:19 +00:00
|
|
|
if (this._themeChangeTimeout !== null) {
|
|
|
|
clearTimeout(this._themeChangeTimeout);
|
2019-10-12 21:59:56 +00:00
|
|
|
}
|
2020-05-30 00:29:19 +00:00
|
|
|
this._themeChangeTimeout = setTimeout(() => {
|
|
|
|
this._themeChangeTimeout = null;
|
2020-06-21 20:14:05 +00:00
|
|
|
const popup = this._frontend.popup;
|
|
|
|
if (popup === null) { return; }
|
|
|
|
popup.updateTheme();
|
2019-10-12 21:59:56 +00:00
|
|
|
}, 300);
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
2020-10-20 23:08:17 +00:00
|
|
|
_onExampleTextClick() {
|
|
|
|
if (this._exampleTextInput === null) { return; }
|
|
|
|
const visible = this._exampleTextInput.hidden;
|
|
|
|
this._exampleTextInput.hidden = !visible;
|
|
|
|
if (!visible) { return; }
|
|
|
|
this._exampleTextInput.focus();
|
|
|
|
this._exampleTextInput.select();
|
|
|
|
}
|
|
|
|
|
|
|
|
_onExampleTextInputBlur() {
|
|
|
|
if (this._exampleTextInput === null) { return; }
|
|
|
|
this._exampleTextInput.hidden = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
_onExampleTextInputInput(e) {
|
|
|
|
this._setText(e.currentTarget.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onSetText({text}) {
|
|
|
|
this._setText(text, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
_setText(text, setInput) {
|
|
|
|
if (setInput && this._exampleTextInput !== null) {
|
|
|
|
this._exampleTextInput.value = text;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this._exampleText === null) { return; }
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2020-10-20 23:08:17 +00:00
|
|
|
this._exampleText.textContent = text;
|
2020-05-30 13:31:46 +00:00
|
|
|
if (this._frontend === null) { return; }
|
2020-05-30 00:29:19 +00:00
|
|
|
this._updateSearch();
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
_setInfoVisible(visible) {
|
2019-10-12 21:21:36 +00:00
|
|
|
const node = document.querySelector('.placeholder-info');
|
|
|
|
if (node === null) { return; }
|
|
|
|
|
|
|
|
node.classList.toggle('placeholder-info-visible', visible);
|
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
_setCustomCss({css}) {
|
|
|
|
if (this._frontend === null) { return; }
|
2020-06-21 20:14:05 +00:00
|
|
|
const popup = this._frontend.popup;
|
|
|
|
if (popup === null) { return; }
|
|
|
|
popup.setCustomCss(css);
|
2019-10-12 21:21:36 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
_setCustomOuterCss({css}) {
|
|
|
|
if (this._frontend === null) { return; }
|
2020-06-21 20:14:05 +00:00
|
|
|
const popup = this._frontend.popup;
|
|
|
|
if (popup === null) { return; }
|
|
|
|
popup.setCustomOuterCss(css, false);
|
2019-10-13 15:51:59 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
async _updateOptionsContext({optionsContext}) {
|
|
|
|
this._optionsContext = optionsContext;
|
2020-05-30 13:31:46 +00:00
|
|
|
if (this._frontend === null) { return; }
|
2020-11-09 03:49:40 +00:00
|
|
|
this._frontend.setOptionsContextOverride(optionsContext);
|
2020-05-30 00:29:19 +00:00
|
|
|
await this._frontend.updateOptions();
|
|
|
|
await this._updateSearch();
|
2020-04-26 19:33:50 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
async _updateSearch() {
|
2020-10-20 23:08:17 +00:00
|
|
|
if (this._exampleText === null) { return; }
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2020-10-20 23:08:17 +00:00
|
|
|
const textNode = this._exampleText.firstChild;
|
2019-10-12 16:59:51 +00:00
|
|
|
if (textNode === null) { return; }
|
|
|
|
|
|
|
|
const range = document.createRange();
|
2020-10-21 00:53:18 +00:00
|
|
|
range.selectNodeContents(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 {
|
2020-05-30 00:29:19 +00:00
|
|
|
await this._frontend.setTextSource(source);
|
2019-10-25 00:01:04 +00:00
|
|
|
} finally {
|
|
|
|
source.cleanup();
|
|
|
|
}
|
2020-05-30 00:29:19 +00:00
|
|
|
this._textSource = source;
|
|
|
|
await this._frontend.showContentCompleted();
|
2019-10-12 16:59:51 +00:00
|
|
|
|
2020-06-21 20:14:05 +00:00
|
|
|
const popup = this._frontend.popup;
|
|
|
|
if (popup !== null && popup.isVisibleSync()) {
|
2020-05-30 00:29:19 +00:00
|
|
|
this._popupShown = true;
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:29:19 +00:00
|
|
|
this._setInfoVisible(!this._popupShown);
|
2019-10-12 16:59:51 +00:00
|
|
|
}
|
|
|
|
}
|