content-hooks-web.js 2.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859
  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 history, dispatchEvent, CustomEvent, fetch, addEventListener */
  24. (() => {
  25. const FETCH_REQUEST_EVENT = "single-file-request-fetch";
  26. const FETCH_RESPONSE__EVENT = "single-file-response-fetch";
  27. const pushState = history.pushState;
  28. let warningDisplayed;
  29. history.pushState = function (state, title, url) {
  30. if (!warningDisplayed) {
  31. warningDisplayed = true;
  32. console.warn("SingleFile is hooking the history.pushState API to detect navigation."); // eslint-disable-line no-console
  33. }
  34. try {
  35. dispatchEvent(new CustomEvent("single-file-push-state", { detail: { state, title, url } }));
  36. } catch (error) {
  37. // ignored
  38. }
  39. pushState.call(history, state, title, url);
  40. };
  41. history.pushState.toString = function () { return "function pushState() { [native code] }"; };
  42. addEventListener(FETCH_REQUEST_EVENT, async event => {
  43. const url = event.detail;
  44. let detail;
  45. try {
  46. const response = await fetch(url, { cache: "force-cache" });
  47. detail = { url, response: await response.arrayBuffer(), headers: response.headers, status: response.status };
  48. } catch (error) {
  49. detail = { url, error: error.toString() };
  50. }
  51. dispatchEvent(new CustomEvent(FETCH_RESPONSE__EVENT, { detail }));
  52. });
  53. })();