content-hooks-web.js 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465
  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. let warningDisplayed;
  34. history.pushState = function (state, title, url) {
  35. if (!warningDisplayed) {
  36. warningDisplayed = true;
  37. console.warn("SingleFile is hooking the history.pushState API to detect navigation."); // eslint-disable-line no-console
  38. }
  39. try {
  40. dispatchEvent.call(window, new CustomEvent("single-file-push-state", { detail: { state, title, url } }));
  41. } catch (error) {
  42. // ignored
  43. }
  44. pushState.call(history, state, title, url);
  45. };
  46. history.pushState.toString = function () { return "function pushState() { [native code] }"; };
  47. addEventListener.call(window, FETCH_REQUEST_EVENT, async event => {
  48. const url = event.detail;
  49. let detail;
  50. try {
  51. const response = await fetch(url, { cache: "force-cache" });
  52. detail = { url, response: await response.arrayBuffer(), headers: Array.from(response.headers), status: response.status };
  53. } catch (error) {
  54. detail = { url, error: error.toString() };
  55. }
  56. dispatchEvent.call(window, new CustomEvent(FETCH_RESPONSE_EVENT, { detail }));
  57. });
  58. })();