content-autosave.js 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  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 singlefile, frameTree, browser, window, addEventListener, removeEventListener, document, location, docHelper */
  21. this.singlefile.autosave = this.singlefile.autosave || (async () => {
  22. let listenerAdded, options, autoSaveTimeout;
  23. refresh();
  24. browser.runtime.onMessage.addListener(message => {
  25. if (message.autoSavePage) {
  26. autoSavePage();
  27. }
  28. if (message.autoSaveUnloadEnabled) {
  29. refresh();
  30. }
  31. });
  32. return {};
  33. async function autoSavePage() {
  34. const [autoSaveEnabled, options] = await Promise.all([browser.runtime.sendMessage({ isAutoSaveEnabled: true }), browser.runtime.sendMessage({ getConfig: true })]);
  35. if (autoSaveEnabled) {
  36. if (options.autoSaveDelay && !autoSaveTimeout) {
  37. autoSaveTimeout = setTimeout(() => {
  38. autoSavePage();
  39. }, options.autoSaveDelay * 1000);
  40. } else {
  41. const docData = docHelper.preProcessDoc(document, window, options);
  42. let framesData = [];
  43. if (!options.removeFrames && this.frameTree) {
  44. framesData = await frameTree.getAsync(options);
  45. }
  46. browser.runtime.sendMessage({ processContent: true, content: docHelper.serialize(document, false), canvasData: docData.canvasData, stylesheetContents: docData.stylesheetContents, imageData: docData.imageData, framesData, url: location.href });
  47. docHelper.postProcessDoc(document, window);
  48. singlefile.pageAutoSaved = true;
  49. }
  50. }
  51. }
  52. async function refresh() {
  53. const [autoSaveEnabled, config] = await Promise.all([browser.runtime.sendMessage({ isAutoSaveEnabled: true }), browser.runtime.sendMessage({ getConfig: true })]);
  54. options = config;
  55. enableAutoSaveUnload(autoSaveEnabled && (config.autoSaveUnload || config.autoSaveLoadOrUnload));
  56. }
  57. function enableAutoSaveUnload(enabled) {
  58. if (enabled) {
  59. if (!listenerAdded) {
  60. addEventListener("unload", onUnload);
  61. listenerAdded = true;
  62. }
  63. } else {
  64. removeEventListener("unload", onUnload);
  65. listenerAdded = false;
  66. }
  67. }
  68. function onUnload() {
  69. if (!singlefile.pageAutoSaved) {
  70. const docData = docHelper.preProcessDoc(document, window, options);
  71. browser.runtime.sendMessage({ processContent: true, content: docHelper.serialize(document), canvasData: docData.canvasData, stylesheetContents: docData.stylesheetContents, imageData: docData.imageData, framesData: this.frameTree && !options.removeFrames && frameTree.getSync(options), url: location.href });
  72. }
  73. }
  74. })();