content-hooks-web.js 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. /*
  2. * Copyright 2010-2019 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * The code in this file is free software: you can redistribute it and/or
  8. * modify it under the terms of the GNU Affero General Public License
  9. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  10. * of the License, or (at your option) any later version.
  11. *
  12. * The code in this file 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 GNU Affero
  15. * General Public License for more details.
  16. *
  17. * As additional permission under GNU AGPL version 3 section 7, you may
  18. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  19. * AGPL normally required by section 4, provided you include this license
  20. * notice and a URL through which recipients can access the Corresponding
  21. * Source.
  22. */
  23. /* global window */
  24. (() => {
  25. const FETCH_REQUEST_EVENT = "single-file-request-fetch";
  26. const FETCH_RESPONSE_EVENT = "single-file-response-fetch";
  27. const history = window.history;
  28. const dispatchEvent = window.dispatchEvent;
  29. const CustomEvent = window.CustomEvent;
  30. const fetch = window.fetch;
  31. const addEventListener = window.addEventListener;
  32. const pushState = history.pushState;
  33. const console = window.console;
  34. let warningDisplayed;
  35. history.pushState = function (state, title, url) {
  36. if (!warningDisplayed) {
  37. warningDisplayed = true;
  38. console.warn("SingleFile is hooking the history.pushState API to detect navigation."); // eslint-disable-line no-console
  39. }
  40. try {
  41. dispatchEvent.call(window, new CustomEvent("single-file-push-state", { detail: { state, title, url } }));
  42. } catch (error) {
  43. // ignored
  44. }
  45. pushState.call(history, state, title, url);
  46. };
  47. history.pushState.toString = function () { return "function pushState() { [native code] }"; };
  48. addEventListener.call(window, FETCH_REQUEST_EVENT, async event => {
  49. const url = event.detail;
  50. let detail;
  51. try {
  52. const response = await fetch(url, { cache: "force-cache" });
  53. detail = { url, response: await response.arrayBuffer(), headers: Array.from(response.headers), status: response.status };
  54. } catch (error) {
  55. detail = { url, error: error.toString() };
  56. }
  57. dispatchEvent.call(window, new CustomEvent(FETCH_RESPONSE_EVENT, { detail }));
  58. });
  59. })();