editor.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  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 }) {
  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 });
  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. const options = await config.getOptions(tabData.url);
  57. if (tabData) {
  58. const content = JSON.stringify(tabData);
  59. for (let blockIndex = 0; blockIndex * MAX_CONTENT_SIZE < content.length; blockIndex++) {
  60. const message = {
  61. method: "editor.setTabData"
  62. };
  63. message.truncated = content.length > MAX_CONTENT_SIZE;
  64. if (message.truncated) {
  65. message.finished = (blockIndex + 1) * MAX_CONTENT_SIZE > content.length;
  66. message.content = content.substring(blockIndex * MAX_CONTENT_SIZE, (blockIndex + 1) * MAX_CONTENT_SIZE);
  67. } else {
  68. message.content = content;
  69. message.options = options;
  70. }
  71. await tabs.sendMessage(tab.id, message);
  72. }
  73. }
  74. }
  75. if (message.method.endsWith(".open")) {
  76. let contents;
  77. const tab = sender.tab;
  78. if (message.truncated) {
  79. contents = partialContents.get(tab.id);
  80. if (!contents) {
  81. contents = [];
  82. partialContents.set(tab.id, contents);
  83. }
  84. contents.push(message.content);
  85. if (message.finished) {
  86. partialContents.delete(tab.id);
  87. }
  88. } else if (message.content) {
  89. contents = [message.content];
  90. }
  91. if (!message.truncated || message.finished) {
  92. const updateTabProperties = { url: EDITOR_PAGE_URL };
  93. await tabs.update(tab.id, updateTabProperties);
  94. tabsData.set(tab.id, { url: tab.url, content: contents.join(""), filename: message.filename });
  95. }
  96. }
  97. }