/* * Copyright (C) 2020 Yomichan Authors * * 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 * apiBroadcastTab */ class FrameOffsetForwarder { constructor() { this._started = false; this._cacheMaxSize = 1000; this._frameCache = new Set(); this._unreachableContentWindowCache = new Set(); this._forwardFrameOffset = ( window !== window.parent ? this._forwardFrameOffsetParent.bind(this) : this._forwardFrameOffsetOrigin.bind(this) ); this._windowMessageHandlers = new Map([ ['getFrameOffset', ({offset, uniqueId}, e) => this._onGetFrameOffset(offset, uniqueId, e)] ]); } start() { if (this._started) { return; } window.addEventListener('message', this.onMessage.bind(this), false); this._started = true; } async getOffset() { const uniqueId = yomichan.generateId(16); const frameOffsetPromise = yomichan.getTemporaryListenerResult( chrome.runtime.onMessage, ({action, params}, {resolve}) => { if (action === 'frameOffset' && isObject(params) && params.uniqueId === uniqueId) { resolve(params); } }, 5000 ); window.parent.postMessage({ action: 'getFrameOffset', params: { uniqueId, offset: [0, 0] } }, '*'); 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; if (!this._unreachableContentWindowCache.has(e.source)) { sourceFrame = this._findFrameWithContentWindow(e.source); } if (sourceFrame === null) { // closed shadow root etc. this._addToCache(this._unreachableContentWindowCache, e.source); this._forwardFrameOffsetOrigin(null, uniqueId); return; } const [forwardedX, forwardedY] = offset; const {x, y} = sourceFrame.getBoundingClientRect(); offset = [forwardedX + x, forwardedY + y]; this._forwardFrameOffset(offset, uniqueId); } _findFrameWithContentWindow(contentWindow) { const ELEMENT_NODE = Node.ELEMENT_NODE; for (const elements of this._getFrameElementSources()) { while (elements.length > 0) { const element = elements.shift(); if (element.contentWindow === contentWindow) { this._addToCache(this._frameCache, element); return element; } const shadowRoot = element.shadowRoot; if (shadowRoot) { for (const child of shadowRoot.children) { if (child.nodeType === ELEMENT_NODE) { elements.push(child); } } } for (const child of element.children) { if (child.nodeType === ELEMENT_NODE) { elements.push(child); } } } } return null; } *_getFrameElementSources() { const frameCache = []; for (const frame of this._frameCache) { // removed from DOM if (!frame.isConnected) { this._frameCache.delete(frame); continue; } frameCache.push(frame); } yield frameCache; // will contain duplicates, but frame elements are cheap to handle yield [...document.querySelectorAll('frame, iframe:not(.yomichan-float)')]; yield [document.documentElement]; } _addToCache(cache, value) { let freeSlots = this._cacheMaxSize - cache.size; if (freeSlots <= 0) { for (const cachedValue of cache) { cache.delete(cachedValue); ++freeSlots; if (freeSlots > 0) { break; } } } cache.add(value); } _forwardFrameOffsetParent(offset, uniqueId) { window.parent.postMessage({action: 'getFrameOffset', params: {offset, uniqueId}}, '*'); } _forwardFrameOffsetOrigin(offset, uniqueId) { apiBroadcastTab('frameOffset', {offset, uniqueId}); } }