editor.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. /*
  2. * Copyright 2010-2020 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 browser */
  24. import * as config from "./config.js";
  25. import * as tabs from "./tabs.js";
  26. const MAX_CONTENT_SIZE = 32 * (1024 * 1024);
  27. const EDITOR_PAGE_URL = "/extension/ui/pages/editor.html";
  28. const tabsData = new Map();
  29. const partialContents = new Map();
  30. const EDITOR_URL = browser.runtime.getURL(EDITOR_PAGE_URL);
  31. export {
  32. onMessage,
  33. onTabRemoved,
  34. isEditor,
  35. open,
  36. EDITOR_URL
  37. };
  38. async function open({ tabIndex, content, filename }, options) {
  39. const createTabProperties = { active: true, url: EDITOR_PAGE_URL };
  40. if (tabIndex != null) {
  41. createTabProperties.index = tabIndex;
  42. }
  43. const tab = await tabs.create(createTabProperties);
  44. tabsData.set(tab.id, { content, filename, options });
  45. }
  46. function onTabRemoved(tabId) {
  47. tabsData.delete(tabId);
  48. }
  49. function isEditor(tab) {
  50. return tab.url == EDITOR_URL;
  51. }
  52. async function onMessage(message, sender) {
  53. if (message.method.endsWith(".getTabData")) {
  54. const tab = sender.tab;
  55. const tabData = tabsData.get(tab.id);
  56. if (tabData) {
  57. const content = JSON.stringify(tabData);
  58. for (let blockIndex = 0; blockIndex * MAX_CONTENT_SIZE < content.length; blockIndex++) {
  59. const message = {
  60. method: "editor.setTabData"
  61. };
  62. message.truncated = content.length > MAX_CONTENT_SIZE;
  63. if (message.truncated) {
  64. message.finished = (blockIndex + 1) * MAX_CONTENT_SIZE > content.length;
  65. message.content = content.substring(blockIndex * MAX_CONTENT_SIZE, (blockIndex + 1) * MAX_CONTENT_SIZE);
  66. } else {
  67. message.content = content;
  68. }
  69. await tabs.sendMessage(tab.id, message);
  70. }
  71. }
  72. }
  73. if (message.method.endsWith(".open")) {
  74. let contents;
  75. const tab = sender.tab;
  76. if (message.truncated) {
  77. contents = partialContents.get(tab.id);
  78. if (!contents) {
  79. contents = [];
  80. partialContents.set(tab.id, contents);
  81. }
  82. contents.push(message.content);
  83. if (message.finished) {
  84. partialContents.delete(tab.id);
  85. }
  86. } else if (message.content) {
  87. contents = [message.content];
  88. }
  89. if (!message.truncated || message.finished) {
  90. const options = await config.getOptions(tab && tab.url);
  91. const updateTabProperties = { url: EDITOR_PAGE_URL };
  92. await tabs.update(tab.id, updateTabProperties);
  93. tabsData.set(tab.id, { content: contents.join(""), filename: message.filename, options });
  94. }
  95. }
  96. }