run-at.js (2949B) - View raw
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96/******************************************************************************* uBlock Origin - a comprehensive, efficient content blocker Copyright (C) 2019-present Raymond Hill 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 {http://www.gnu.org/licenses/}. Home: https://github.com/gorhill/uBlock */ import { registerScriptlet } from './base.js'; import { safeSelf } from './safe-self.js'; /* eslint no-prototype-builtins: 0 */ /** * @helperScriptlet run-at.fn * * @description * Execute a function at a specific page-load milestone. * * @param fn * The function to call. * * @param when * An identifier which tells when the function should be executed. * See <https://developer.mozilla.org/en-US/docs/Web/API/Document/readyState>. * * @example * `runAt(( ) => { start(); }, 'interactive')` * * */ export function runAt(fn, when) { const intFromReadyState = state => { const targets = { 'loading': 1, 'asap': 1, 'interactive': 2, 'end': 2, '2': 2, 'complete': 3, 'idle': 3, '3': 3, }; const tokens = Array.isArray(state) ? state : [ state ]; for ( const token of tokens ) { const prop = `${token}`; if ( Object.hasOwn(targets, prop) === false ) { continue; } return targets[prop]; } return 0; }; const runAt = intFromReadyState(when); if ( intFromReadyState(document.readyState) >= runAt ) { fn(); return; } const onStateChange = ( ) => { if ( intFromReadyState(document.readyState) < runAt ) { return; } fn(); safe.removeEventListener.apply(document, args); }; const safe = safeSelf(); const args = [ 'readystatechange', onStateChange, { capture: true } ]; safe.addEventListener.apply(document, args); } registerScriptlet(runAt, { name: 'run-at.fn', dependencies: [ safeSelf, ], }); /******************************************************************************/ export function runAtHtmlElementFn(fn) { if ( document.documentElement ) { fn(); return; } const observer = new MutationObserver(( ) => { observer.disconnect(); fn(); }); observer.observe(document, { childList: true }); } registerScriptlet(runAtHtmlElementFn, { name: 'run-at-html-element.fn', });