frame-tree.js 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  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 browser, window, top, document, HTMLHtmlElement, addEventListener, docHelper, timeout */
  21. this.FrameTree = this.FrameTree || (() => {
  22. const MESSAGE_PREFIX = "__FrameTree__";
  23. const TIMEOUT_INIT_REQUEST_MESSAGE = 500;
  24. const TIMEOUT_DATA_RESPONSE_MESSAGE = 500;
  25. const FrameTree = { getFramesData };
  26. let framesData, dataRequestCallbacks, initResponseSent;
  27. if (window == top) {
  28. browser.runtime.onMessage.addListener(onTopBackgroundMessage);
  29. }
  30. browser.runtime.onMessage.addListener(onBackgroundMessage);
  31. addEventListener("message", onFrameWindowMessage, false);
  32. return FrameTree;
  33. async function getFramesData(options) {
  34. await Promise.all(framesData.map(async frameData => {
  35. return new Promise(resolve => {
  36. dataRequestCallbacks.set(frameData.windowId, resolve);
  37. if (frameData.sameDomain) {
  38. top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "getDataRequest", windowId: frameData.windowId, options: { removeHiddenElements: options.removeHiddenElements, compressHTML: options.compressHTML } }), "*");
  39. } else {
  40. browser.runtime.sendMessage({
  41. method: "FrameTree.getDataRequest",
  42. windowId: frameData.windowId,
  43. options: { removeHiddenElements: options.removeHiddenElements, compressHTML: options.compressHTML }
  44. });
  45. }
  46. frameData.getDataResponseTimeout = timeout.set(() => top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "getDataResponse", windowId: frameData.windowId }), "*"), TIMEOUT_DATA_RESPONSE_MESSAGE);
  47. });
  48. }));
  49. return framesData.sort((frame1, frame2) => frame2.windowId.split(".").length - frame1.windowId.split(".").length);
  50. }
  51. function onTopBackgroundMessage(message) {
  52. if (message.method == "FrameTree.initRequest" && document.documentElement instanceof HTMLHtmlElement) {
  53. dataRequestCallbacks = new Map();
  54. framesData = [];
  55. initResponseSent = false;
  56. initRequest(message);
  57. }
  58. if (message.method == "FrameTree.getDataResponse") {
  59. getDataResponse(message);
  60. }
  61. }
  62. function onBackgroundMessage(message) {
  63. if (message.method == "FrameTree.getDataRequest" && FrameTree.windowId == message.windowId) {
  64. const docData = docHelper.preProcessDoc(document, window, message.options);
  65. browser.runtime.sendMessage({
  66. method: "FrameTree.getDataResponse",
  67. windowId: message.windowId,
  68. tabId: message.tabId,
  69. content: docHelper.serialize(document),
  70. emptyStyleRulesText: docData.emptyStyleRulesText,
  71. canvasData: docData.canvasData,
  72. baseURI: document.baseURI,
  73. title: document.title
  74. });
  75. docHelper.postProcessDoc(document, message.options);
  76. }
  77. }
  78. function onFrameWindowMessage(event) {
  79. if (typeof event.data == "string" && event.data.startsWith(MESSAGE_PREFIX + "::")) {
  80. const message = JSON.parse(event.data.substring(MESSAGE_PREFIX.length + 2));
  81. if (message.method == "initRequest") {
  82. initRequest(message);
  83. } else if (message.method == "initResponse") {
  84. initResponse(message);
  85. } else if (message.method == "getDataResponse") {
  86. getDataResponse(message);
  87. }
  88. }
  89. }
  90. function initRequest(message) {
  91. FrameTree.windowId = message.windowId;
  92. const frameElements = document.querySelectorAll("iframe, frame, object[type=\"text/html\"][data]");
  93. if (frameElements.length) {
  94. setFramesWinId(MESSAGE_PREFIX, frameElements, message.options, FrameTree.windowId, window);
  95. } else {
  96. top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "initResponse", framesData: [], windowId: FrameTree.windowId }), "*");
  97. }
  98. }
  99. function initResponse(message) {
  100. if (window == top) {
  101. if (message.framesData) {
  102. message.framesData = message.framesData instanceof Array ? message.framesData : JSON.parse(message.framesData);
  103. framesData = framesData.concat(message.framesData);
  104. const frameData = framesData.find(frameData => frameData.windowId == message.windowId);
  105. if (message.windowId != "0") {
  106. frameData.processed = true;
  107. }
  108. const pendingCount = framesData.filter(frameData => !frameData.processed).length;
  109. if (!pendingCount && !initResponseSent) {
  110. initResponseSent = true;
  111. browser.runtime.sendMessage({ method: "FrameTree.initResponse" });
  112. }
  113. }
  114. } else {
  115. FrameTree.windowId = message.windowId;
  116. }
  117. }
  118. function setFramesWinId(MESSAGE_PREFIX, frameElements, options, windowId, win) {
  119. const framesData = [];
  120. if (win != top) {
  121. win.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "initResponse", windowId }), "*");
  122. }
  123. frameElements.forEach((frameElement, frameIndex) => {
  124. let src, sameDomain;
  125. try {
  126. sameDomain = Boolean(frameElement.contentDocument && frameElement.contentWindow && top.addEventListener && top);
  127. src = frameElement.src;
  128. } catch (error) {
  129. /* ignored */
  130. }
  131. framesData.push({ sameDomain, src, windowId: windowId + "." + frameIndex });
  132. });
  133. top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "initResponse", framesData, windowId }), "*");
  134. frameElements.forEach((frameElement, frameIndex) => {
  135. const frameWinId = windowId + "." + frameIndex;
  136. frameElement.setAttribute(docHelper.WIN_ID_ATTRIBUTE_NAME, frameWinId);
  137. let frameDoc, frameWindow, topWindow;
  138. let content, emptyStyleRulesText, canvasData;
  139. try {
  140. frameDoc = frameElement.contentDocument;
  141. frameWindow = frameElement.contentWindow;
  142. topWindow = top.addEventListener && top;
  143. } catch (error) {
  144. /* ignored */
  145. }
  146. if (frameWindow && frameDoc && topWindow) {
  147. setFramesWinId(MESSAGE_PREFIX, frameDoc.querySelectorAll("iframe, frame, object[type=\"text/html\"][data]"), options, frameWinId, frameWindow);
  148. topWindow.addEventListener("message", onMessage, false);
  149. const docData = docHelper.preProcessDoc(frameDoc, frameWindow, options);
  150. content = docHelper.serialize(frameDoc);
  151. emptyStyleRulesText = docData.emptyStyleRulesText;
  152. canvasData = docData.canvasData;
  153. docHelper.postProcessDoc(frameDoc, options);
  154. } else if (frameWindow) {
  155. frameWindow.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "initRequest", windowId: frameWinId, frameIndex, options }), "*");
  156. timeout.set(() => top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "initResponse", framesData: [], windowId: frameWinId, frameIndex }), "*"), TIMEOUT_INIT_REQUEST_MESSAGE);
  157. }
  158. function onMessage(event) {
  159. if (typeof event.data == "string" && event.data.startsWith(MESSAGE_PREFIX + "::")) {
  160. const message = JSON.parse(event.data.substring(MESSAGE_PREFIX.length + 2));
  161. if (message.method == "getDataRequest" && message.windowId == frameWinId) {
  162. topWindow.removeEventListener("message", onMessage, false);
  163. top.postMessage(MESSAGE_PREFIX + "::" + JSON.stringify({ method: "getDataResponse", windowId: message.windowId, content, baseURI: frameDoc.baseURI, title: document.title, emptyStyleRulesText, canvasData }), "*");
  164. }
  165. }
  166. }
  167. });
  168. }
  169. function getDataResponse(message) {
  170. delete message.tabId;
  171. delete message.method;
  172. const frameData = framesData.find(frameData => frameData.windowId == message.windowId);
  173. timeout.clear(frameData.getDataResponseTimeout);
  174. frameData.content = message.content;
  175. frameData.baseURI = message.baseURI;
  176. frameData.title = message.title;
  177. frameData.emptyStyleRulesText = message.emptyStyleRulesText;
  178. frameData.canvasData = message.canvasData;
  179. dataRequestCallbacks.get(message.windowId)(message);
  180. }
  181. })();