content-lazy-loader.js 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465
  1. /*
  2. * Copyright 2018 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * SingleFile is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU Lesser General Public License as published by
  9. * the Free Software Foundation, either version 3 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * SingleFile is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public License
  18. * along with SingleFile. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. /* global browser, document, timeout, MutationObserver */
  21. this.lazyLoader = this.lazyLoader || (() => {
  22. const LAZY_LOADING_TIMEOUT = 1000;
  23. const MAX_LAZY_LOADING_TIMEOUT = 30000;
  24. return { process };
  25. async function process() {
  26. const scriptURL = browser.runtime.getURL("lib/lazy/web-lazy-loader-before.js");
  27. const scriptBeforeElement = document.createElement("script");
  28. scriptBeforeElement.src = scriptURL;
  29. document.body.appendChild(scriptBeforeElement);
  30. let timeoutId, maxTimeoutId;
  31. const promise = new Promise(resolve => {
  32. scriptBeforeElement.onload = () => scriptBeforeElement.remove();
  33. const observer = new MutationObserver(() => timeoutId = deferLazyLoadEnd(timeoutId, maxTimeoutId, observer, resolve));
  34. observer.observe(document, { attributeFilter: ["src", "srcset"], subtree: true });
  35. timeoutId = deferLazyLoadEnd(timeoutId, maxTimeoutId, observer, resolve);
  36. maxTimeoutId = timeout.set(() => {
  37. timeout.clear(timeoutId);
  38. lazyLoadEnd(maxTimeoutId, observer, resolve);
  39. }, MAX_LAZY_LOADING_TIMEOUT);
  40. });
  41. return promise;
  42. }
  43. function deferLazyLoadEnd(timeoutId, maxTimeoutId, observer, resolve) {
  44. timeout.clear(timeoutId);
  45. return timeout.set(() => lazyLoadEnd(maxTimeoutId, observer, resolve), LAZY_LOADING_TIMEOUT);
  46. }
  47. function lazyLoadEnd(maxTimeoutId, observer, resolve) {
  48. timeout.clear(maxTimeoutId);
  49. timeout.set(resolve, LAZY_LOADING_TIMEOUT);
  50. const scriptURL = browser.runtime.getURL("lib/lazy/web-lazy-loader-after.js");
  51. const scriptAfterElement = document.createElement("script");
  52. scriptAfterElement.src = scriptURL;
  53. document.body.appendChild(scriptAfterElement);
  54. scriptAfterElement.onload = () => scriptAfterElement.remove();
  55. observer.disconnect();
  56. }
  57. })();