2020-03-19 15:46:05 +00:00
|
|
|
/*
|
2020-04-10 18:06:55 +00:00
|
|
|
* Copyright (C) 2020 Yomichan Authors
|
2020-03-19 15:46:05 +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
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* global
|
|
|
|
* apiForward
|
|
|
|
*/
|
|
|
|
|
|
|
|
class FrameOffsetForwarder {
|
|
|
|
constructor() {
|
2020-03-22 01:29:09 +00:00
|
|
|
this._started = false;
|
|
|
|
|
|
|
|
this._forwardFrameOffset = (
|
|
|
|
window !== window.parent ?
|
2020-03-19 15:46:05 +00:00
|
|
|
this._forwardFrameOffsetParent.bind(this) :
|
2020-03-22 01:29:09 +00:00
|
|
|
this._forwardFrameOffsetOrigin.bind(this)
|
|
|
|
);
|
2020-03-19 15:46:05 +00:00
|
|
|
|
|
|
|
this._windowMessageHandlers = new Map([
|
2020-03-22 01:29:09 +00:00
|
|
|
['getFrameOffset', ({offset, uniqueId}, e) => this._onGetFrameOffset(offset, uniqueId, e)]
|
2020-03-19 15:46:05 +00:00
|
|
|
]);
|
2020-03-22 01:29:09 +00:00
|
|
|
}
|
2020-03-19 15:46:05 +00:00
|
|
|
|
2020-03-22 01:29:09 +00:00
|
|
|
start() {
|
|
|
|
if (this._started) { return; }
|
2020-03-19 15:46:05 +00:00
|
|
|
window.addEventListener('message', this.onMessage.bind(this), false);
|
2020-03-22 01:29:09 +00:00
|
|
|
this._started = true;
|
2020-03-19 15:46:05 +00:00
|
|
|
}
|
|
|
|
|
2020-03-22 12:11:43 +00:00
|
|
|
async getOffset() {
|
2020-03-19 15:46:05 +00:00
|
|
|
const uniqueId = yomichan.generateId(16);
|
|
|
|
|
2020-03-22 02:55:16 +00:00
|
|
|
const frameOffsetPromise = yomichan.getTemporaryListenerResult(
|
|
|
|
chrome.runtime.onMessage,
|
|
|
|
({action, params}, {resolve}) => {
|
|
|
|
if (action === 'frameOffset' && isObject(params) && params.uniqueId === uniqueId) {
|
|
|
|
resolve(params);
|
|
|
|
}
|
2020-03-22 10:45:58 +00:00
|
|
|
},
|
|
|
|
5000
|
2020-03-22 02:55:16 +00:00
|
|
|
);
|
2020-03-19 15:46:05 +00:00
|
|
|
|
|
|
|
window.parent.postMessage({
|
|
|
|
action: 'getFrameOffset',
|
|
|
|
params: {
|
|
|
|
uniqueId,
|
2020-03-22 12:11:43 +00:00
|
|
|
offset: [0, 0]
|
2020-03-19 15:46:05 +00:00
|
|
|
}
|
|
|
|
}, '*');
|
|
|
|
|
|
|
|
const {offset} = await frameOffsetPromise;
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
onMessage(e) {
|
|
|
|
const {action, params} = e.data;
|
|
|
|
const handler = this._windowMessageHandlers.get(action);
|
|
|
|
if (typeof handler !== 'function') { return; }
|
|
|
|
handler(params, e);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onGetFrameOffset(offset, uniqueId, e) {
|
|
|
|
let sourceFrame = null;
|
|
|
|
for (const frame of document.querySelectorAll('frame, iframe:not(.yomichan-float)')) {
|
|
|
|
if (frame.contentWindow !== e.source) { continue; }
|
|
|
|
sourceFrame = frame;
|
|
|
|
break;
|
|
|
|
}
|
2020-04-04 13:48:35 +00:00
|
|
|
if (sourceFrame === null) {
|
|
|
|
this._forwardFrameOffsetOrigin(null, uniqueId);
|
|
|
|
return;
|
|
|
|
}
|
2020-03-19 15:46:05 +00:00
|
|
|
|
|
|
|
const [forwardedX, forwardedY] = offset;
|
|
|
|
const {x, y} = sourceFrame.getBoundingClientRect();
|
|
|
|
offset = [forwardedX + x, forwardedY + y];
|
|
|
|
|
|
|
|
this._forwardFrameOffset(offset, uniqueId);
|
|
|
|
}
|
|
|
|
|
|
|
|
_forwardFrameOffsetParent(offset, uniqueId) {
|
|
|
|
window.parent.postMessage({action: 'getFrameOffset', params: {offset, uniqueId}}, '*');
|
|
|
|
}
|
|
|
|
|
|
|
|
_forwardFrameOffsetOrigin(offset, uniqueId) {
|
|
|
|
apiForward('frameOffset', {offset, uniqueId});
|
|
|
|
}
|
|
|
|
}
|