@mml-io/networked-dom-web-runner 0.1.3 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/index.js +14 -28
- package/build/index.js.map +2 -2
- package/package.json +4 -5
- package/src/IframeObservableDOMFactory.ts +21 -34
- package/src/RunnerIframe.ts +10 -8
- package/src/index.ts +0 -1
- package/src/message-types.ts +0 -30
package/build/index.js
CHANGED
|
@@ -29,7 +29,7 @@ module.exports = __toCommonJS(src_exports);
|
|
|
29
29
|
__reExport(src_exports, require("@mml-io/networked-dom-document"), module.exports);
|
|
30
30
|
|
|
31
31
|
// runner-iframe-js-text-namespace:/Users/marcuslongmuir/workspace/mml/packages/networked-dom-web-runner/networked-dom-web-runner-iframe/build/index.js
|
|
32
|
-
var build_default = 'var __defProp = Object.defineProperty;\nvar __defProps = Object.defineProperties;\nvar __getOwnPropDescs = Object.getOwnPropertyDescriptors;\nvar __getOwnPropSymbols = Object.getOwnPropertySymbols;\nvar __hasOwnProp = Object.prototype.hasOwnProperty;\nvar __propIsEnum = Object.prototype.propertyIsEnumerable;\nvar __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;\nvar __spreadValues = (a, b) => {\n for (var prop in b ||= {})\n if (__hasOwnProp.call(b, prop))\n __defNormalProp(a, prop, b[prop]);\n if (__getOwnPropSymbols)\n for (var prop of __getOwnPropSymbols(b)) {\n if (__propIsEnum.call(b, prop))\n __defNormalProp(a, prop, b[prop]);\n }\n return a;\n};\nvar __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));\n\n// ../../observable-dom/src/utils.ts\nfunction virtualDomElementToStatic(el) {\n return {\n nodeId: el.nodeId,\n tag: el.tag,\n attributes: el.attributes,\n childNodes: el.childNodes.map((child) => virtualDomElementToStatic(child)),\n textContent: el.textContent\n };\n}\n\n// ../../observable-dom/src/ObservableDom.ts\nvar ObservableDom = class {\n constructor(observableDOMParameters, callback, runnerFactory) {\n this.nodeToNodeId = /* @__PURE__ */ new Map();\n this.nodeIdToNode = /* @__PURE__ */ new Map();\n this.realElementToVirtualElement = /* @__PURE__ */ new Map();\n this.ignoreTextNodes = true;\n this.nextNodeId = 1;\n this.htmlPath = observableDOMParameters.htmlPath;\n this.ignoreTextNodes = observableDOMParameters.ignoreTextNodes;\n this.callback = callback;\n this.documentTimeIntervalTimer = setInterval(() => {\n this.callback({\n documentTime: this.getDocumentTime()\n });\n }, observableDOMParameters.pingIntervalMilliseconds || 5e3);\n this.domRunner = runnerFactory(\n observableDOMParameters.htmlPath,\n observableDOMParameters.htmlContents,\n observableDOMParameters.params,\n (domRunnerMessage) => {\n if (domRunnerMessage.loaded) {\n this.createVirtualDomElementWithChildren(\n this.domRunner.getDocument(),\n null\n );\n const snapshot = virtualDomElementToStatic(\n this.getVirtualDomElementForRealElementOrThrow(\n this.domRunner.getDocument()\n )\n );\n this.callback({\n snapshot,\n documentTime: this.getDocumentTime()\n });\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n this.callback({\n logMessage: domRunnerMessage.logMessage,\n documentTime: this.getDocumentTime()\n });\n }\n }\n );\n }\n addIPCWebsocket(webSocket) {\n return this.domRunner.addIPCWebsocket(webSocket);\n }\n addConnectedUserId(connectionId) {\n this.domRunner.getWindow().dispatchEvent(\n new (this.domRunner.getWindow()).CustomEvent("connected", {\n detail: { connectionId }\n })\n );\n }\n removeConnectedUserId(connectionId) {\n this.domRunner.getWindow().dispatchEvent(\n new (this.domRunner.getWindow()).CustomEvent("disconnected", {\n detail: { connectionId }\n })\n );\n }\n processModificationList(mutationList) {\n const documentEl = this.domRunner.getDocument();\n const documentVirtualDomElement = this.realElementToVirtualElement.get(documentEl);\n if (!documentVirtualDomElement) {\n throw new Error(`document not created in processModificationList`);\n }\n if (mutationList.length > 1) {\n console.error(\n "More than one mutation record received. It is possible that intermediate states are incorrect."\n );\n }\n for (const mutation of mutationList) {\n if (this.isIgnoredElement(mutation.target)) {\n continue;\n }\n if (mutation.type === "attributes" && // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n this.isIgnoredAttribute(mutation.target, mutation.attributeName)) {\n continue;\n }\n this.addKnownNodesInMutation(mutation);\n const firstNonIgnoredPreviousSibling = mutation.previousSibling ? this.getFirstNonIgnoredPreviousSibling(mutation.previousSibling) : null;\n const targetElement = this.getVirtualDomElementForRealElementOrThrow(\n mutation.target\n );\n const addedNodes = [];\n for (const node of mutation.addedNodes) {\n if (this.isIgnoredElement(node)) {\n continue;\n }\n const virtualDomElement = this.getVirtualDomElementForRealElementOrThrow(\n node\n );\n addedNodes.push(virtualDomElementToStatic(virtualDomElement));\n }\n const removedNodeIds = [];\n for (const node of mutation.removedNodes) {\n if (this.isIgnoredElement(node)) {\n continue;\n }\n const virtualDomElement = this.getVirtualDomElementForRealElementOrThrow(\n node\n );\n removedNodeIds.push(virtualDomElement.nodeId);\n }\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: firstNonIgnoredPreviousSibling !== null ? this.getVirtualDomElementForRealElementOrThrow(firstNonIgnoredPreviousSibling).nodeId : null,\n attribute: mutation.attributeName ? {\n attributeName: mutation.attributeName,\n value: mutation.target.getAttribute(mutation.attributeName)\n } : null\n };\n this.callback({\n mutation: mutationRecord,\n documentTime: this.getDocumentTime()\n });\n this.removeKnownNodesInMutation(mutation);\n }\n }\n addKnownNodesInMutation(mutation) {\n const targetNode = mutation.target;\n const virtualDomElement = this.realElementToVirtualElement.get(targetNode);\n if (!virtualDomElement) {\n throw new Error(\n "Unknown node in addKnownNodesInMutation:" + targetNode + "," + mutation.type\n );\n }\n if (mutation.type === "childList") {\n let previousSibling = mutation.previousSibling;\n let index = 0;\n while (previousSibling && this.isIgnoredElement(previousSibling)) {\n previousSibling = previousSibling.previousSibling;\n }\n if (previousSibling) {\n const previousSiblingElement = this.realElementToVirtualElement.get(\n previousSibling\n );\n if (!previousSiblingElement) {\n throw new Error("Unknown previous sibling");\n }\n index = virtualDomElement.childNodes.indexOf(previousSiblingElement);\n if (index === -1) {\n throw new Error("Previous sibling is not currently a child of the parent element");\n }\n index += 1;\n }\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n const childVirtualDomElement = this.createVirtualDomElementWithChildren(\n asElementOrText,\n virtualDomElement\n );\n if (childVirtualDomElement) {\n if (virtualDomElement.childNodes.indexOf(childVirtualDomElement) === -1) {\n virtualDomElement.childNodes.splice(index, 0, childVirtualDomElement);\n index++;\n }\n }\n });\n } else if (mutation.type === "attributes") {\n const attributeName = mutation.attributeName;\n if (this.isIgnoredAttribute(targetNode, attributeName)) {\n return;\n }\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete virtualDomElement.attributes[attributeName];\n } else {\n virtualDomElement.attributes[attributeName] = attributeValue;\n }\n } else if (mutation.type === "characterData") {\n virtualDomElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n }\n removeKnownNodesInMutation(mutation) {\n const targetNode = mutation.target;\n const virtualDomElement = this.realElementToVirtualElement.get(targetNode);\n if (!virtualDomElement) {\n throw new Error("Unknown node in mutation list:" + targetNode + ", " + mutation.type);\n }\n if (mutation.type === "childList") {\n for (const node of mutation.removedNodes) {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n continue;\n }\n const childDomElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDomElement) {\n console.warn(this.htmlPath, "Unknown node in removeKnownNodesInMutation");\n continue;\n } else {\n this.removeVirtualDomElement(childDomElement);\n const index = virtualDomElement.childNodes.indexOf(childDomElement);\n virtualDomElement.childNodes.splice(index, 1);\n }\n }\n return;\n }\n }\n removeVirtualDomElement(virtualDomElement) {\n this.nodeIdToNode.delete(virtualDomElement.nodeId);\n this.nodeToNodeId.delete(virtualDomElement);\n this.realElementToVirtualElement.delete(virtualDomElement.realElement);\n for (const child of virtualDomElement.childNodes) {\n this.removeVirtualDomElement(child);\n }\n }\n createVirtualDomElementWithChildren(node, parent) {\n const virtualElement = this.createVirtualDomElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (node.childNodes) {\n for (let i = 0; i < node.childNodes.length; i++) {\n const child = node.childNodes[i];\n const childVirtualElement = this.createVirtualDomElementWithChildren(\n child,\n virtualElement\n );\n if (childVirtualElement) {\n virtualElement.childNodes.push(childVirtualElement);\n }\n }\n }\n return virtualElement;\n }\n createVirtualDomElement(node, parent) {\n if (this.isIgnoredElement(node)) {\n return null;\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n throw new Error("Node already has a virtual element: " + node.nodeName);\n }\n if (!node) {\n throw new Error("Cannot assign node id to null");\n }\n const attributes = {};\n if (node.attributes) {\n const asHTMLElement = node;\n for (const key of asHTMLElement.getAttributeNames()) {\n const value = asHTMLElement.getAttribute(key);\n if (value === null) {\n throw new Error("Null attribute value for key: " + key);\n }\n if (!this.isIgnoredAttribute(node, key)) {\n attributes[key] = value;\n }\n }\n }\n const nodeId = this.nextNodeId++;\n const virtualElement = {\n nodeId,\n tag: node.nodeName,\n attributes,\n childNodes: [],\n realElement: node,\n parent\n };\n if (node instanceof this.domRunner.getWindow().Text && node.textContent) {\n virtualElement.textContent = node.textContent;\n }\n this.nodeToNodeId.set(virtualElement, nodeId);\n this.nodeIdToNode.set(nodeId, virtualElement);\n this.realElementToVirtualElement.set(node, virtualElement);\n return virtualElement;\n }\n getFirstNonIgnoredPreviousSibling(node) {\n let currentNode = node;\n if (!this.isIgnoredElement(currentNode)) {\n return currentNode;\n }\n while (currentNode && currentNode.previousSibling) {\n currentNode = currentNode.previousSibling;\n if (!this.isIgnoredElement(currentNode)) {\n return currentNode;\n }\n }\n return null;\n }\n getVirtualDomElementForRealElementOrThrow(realElement) {\n const virtualElement = this.realElementToVirtualElement.get(realElement);\n if (!virtualElement) {\n throw new Error(`Virtual element not found for real element`);\n }\n return virtualElement;\n }\n isIgnoredElement(node) {\n if (this.ignoreTextNodes && node instanceof this.domRunner.getWindow().Text) {\n return true;\n } else if (node instanceof this.domRunner.getWindow().HTMLScriptElement) {\n return true;\n } else if (node instanceof this.domRunner.getWindow().Comment) {\n return true;\n }\n return false;\n }\n isIgnoredAttribute(node, attributeName) {\n return attributeName.startsWith("on");\n }\n dispatchRemoteEventFromConnectionId(connectionId, remoteEvent) {\n const domNode = this.nodeIdToNode.get(remoteEvent.nodeId);\n if (!domNode) {\n console.error("Unknown node ID in remote event: " + remoteEvent.nodeId);\n return;\n }\n if (domNode instanceof this.domRunner.getWindow().Text) {\n console.warn("Cannot dispatch remote event to text node");\n return;\n }\n this.domRunner.dispatchRemoteEventFromConnectionId(\n connectionId,\n domNode.realElement,\n remoteEvent\n );\n }\n dispose() {\n clearInterval(this.documentTimeIntervalTimer);\n this.domRunner.dispose();\n }\n getDocumentTime() {\n return this.domRunner.getDocumentTime();\n }\n};\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(htmlPath, htmlContents, params, callback);\n};\nvar documentLoadTime = Date.now();\nvar WebBrowserDOMRunner = class {\n constructor(htmlPath, htmlContents, params, callback) {\n this.htmlPath = htmlPath;\n this.callback = callback;\n for (const level of ["error", "warn", "info", "log"]) {\n const defaultFn = window.console[level];\n window.console[level] = (...args2) => {\n callback({\n logMessage: {\n level,\n content: args2\n }\n });\n defaultFn(...args2);\n };\n }\n window.onerror = (message, source, line, column, error) => {\n callback({\n logMessage: {\n level: "system",\n content: [\n {\n message,\n type: error == null ? void 0 : error.name,\n line,\n column\n }\n ]\n }\n });\n return false;\n };\n let didSendLoad = false;\n this.mutationObserver = new window.MutationObserver((mutationList) => {\n if (!document) {\n return;\n }\n if (!didSendLoad) {\n throw new Error("MutationObserver called before load");\n }\n this.callback({\n mutationList\n });\n });\n window.params = params;\n const finishLoad = () => {\n if (didSendLoad) {\n throw new Error("finishLoad called twice");\n }\n didSendLoad = true;\n this.callback({\n loaded: true\n });\n this.mutationObserver.observe(window.document, {\n attributes: true,\n childList: true,\n subtree: true,\n characterData: true\n });\n };\n if (document.body) {\n setTimeout(finishLoad, 0);\n } else {\n window.addEventListener("DOMContentLoaded", finishLoad);\n }\n }\n addIPCWebsocket() {\n throw new Error("Not implemented.");\n }\n dispatchRemoteEventFromConnectionId(connectionId, realElement, remoteEvent) {\n const bubbles = remoteEvent.bubbles || false;\n const remoteEventObject = new CustomEvent(remoteEvent.name, {\n bubbles,\n detail: __spreadProps(__spreadValues({}, remoteEvent.params), { connectionId })\n });\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log("WebBrowserDOMRunner.dispose");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n if (document.timeline && document.timeline.currentTime) {\n return document.timeline.currentTime;\n }\n return Date.now() - documentLoadTime;\n }\n // TODO - resolve types (Window needs to expose classes such as CustomEvent as properties)\n getWindow() {\n return window;\n }\n};\n\n// src/IframeWebRunner.ts\nfunction setupIframeWebRunner(argsString) {\n const observableDOMParams = JSON.parse(atob(argsString));\n const sendMessageToHandler = (message) => {\n window.parent.postMessage(JSON.stringify(message), "*");\n };\n const observableDOM = new ObservableDom(\n __spreadProps(__spreadValues({}, observableDOMParams), {\n htmlContents: ""\n // This must be empty as the contents are assumed to be provided by the srcdoc\n }),\n (observableDomMessage) => {\n sendMessageToHandler({\n type: "dom",\n message: observableDomMessage\n });\n },\n WebBrowserDOMRunnerFactory\n );\n window.addEventListener("message", (e) => {\n const parsed = JSON.parse(e.data);\n if (parsed.type === "dispatchRemoteEventFromConnectionId") {\n observableDOM.dispatchRemoteEventFromConnectionId(parsed.connectionId, parsed.event);\n }\n });\n}\n\n// src/index.ts\nvar args = window.args;\nsetupIframeWebRunner(args);\n//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../observable-dom/src/utils.ts", "../../../observable-dom/src/ObservableDom.ts", "../src/WebBrowserDOMRunner.ts", "../src/IframeWebRunner.ts", "../src/index.ts"],
  "sourcesContent": ["import { StaticVirtualDomElement } from \"@mml-io/observable-dom-common\";\n\nimport { LiveVirtualDomElement } from \"./ObservableDom\";\n\nexport function virtualDomElementToStatic(el: LiveVirtualDomElement): StaticVirtualDomElement {\n  return {\n    nodeId: el.nodeId,\n    tag: el.tag,\n    attributes: el.attributes,\n    childNodes: el.childNodes.map((child) => virtualDomElementToStatic(child)),\n    textContent: el.textContent,\n  };\n}\n", "import {\n  LogMessage,\n  ObservableDomInterface,\n  ObservableDomMessage,\n  ObservableDOMParameters,\n  RemoteEvent,\n  StaticVirtualDomElement,\n  StaticVirtualDomMutationIdsRecord,\n} from \"@mml-io/observable-dom-common\";\n\nimport { virtualDomElementToStatic } from \"./utils\";\n\nexport type DOMRunnerMessage = {\n  loaded?: boolean;\n  mutationList?: Array<MutationRecord>;\n  logMessage?: LogMessage;\n};\n\nexport type DOMRunnerInterface = {\n  getDocument(): Document;\n  getWindow(): Window & {\n    CustomEvent: typeof CustomEvent;\n    Text: typeof Text;\n    HTMLScriptElement: typeof HTMLScriptElement;\n    Comment: typeof Comment;\n  }; // TODO - Define this without using JSDOM types\n  addIPCWebsocket(webSocket: WebSocket): void;\n  dispatchRemoteEventFromConnectionId(\n    connectionId: number,\n    realElement: Element,\n    remoteEvent: RemoteEvent,\n  ): void;\n  dispose(): void;\n  getDocumentTime(): number;\n};\n\nexport type DOMRunnerFactory = (\n  htmlPath: string,\n  htmlContents: string,\n  params: object,\n  callback: (domRunnerMessage: DOMRunnerMessage) => void,\n) => DOMRunnerInterface;\n\nexport type LiveVirtualDomElement = Omit<StaticVirtualDomElement, \"childNodes\"> & {\n  realElement: Element | Text;\n  childNodes: Array<LiveVirtualDomElement>;\n  parent: LiveVirtualDomElement | null;\n};\n\nexport class ObservableDom implements ObservableDomInterface {\n  private nodeToNodeId = new Map<LiveVirtualDomElement, number>();\n  private nodeIdToNode = new Map<number, LiveVirtualDomElement>();\n  private realElementToVirtualElement = new Map<Element | Text, LiveVirtualDomElement>();\n  private ignoreTextNodes = true;\n  private callback: (message: ObservableDomMessage) => void;\n  private nextNodeId = 1;\n  private htmlPath: string;\n  private domRunner: DOMRunnerInterface;\n\n  private documentTimeIntervalTimer: NodeJS.Timer;\n\n  constructor(\n    observableDOMParameters: ObservableDOMParameters,\n    callback: (message: ObservableDomMessage) => void,\n    runnerFactory: DOMRunnerFactory,\n  ) {\n    this.htmlPath = observableDOMParameters.htmlPath;\n    this.ignoreTextNodes = observableDOMParameters.ignoreTextNodes;\n    this.callback = callback;\n\n    this.documentTimeIntervalTimer = setInterval(() => {\n      this.callback({\n        documentTime: this.getDocumentTime(),\n      });\n    }, observableDOMParameters.pingIntervalMilliseconds || 5000);\n\n    this.domRunner = runnerFactory(\n      observableDOMParameters.htmlPath,\n      observableDOMParameters.htmlContents,\n      observableDOMParameters.params,\n      (domRunnerMessage: DOMRunnerMessage) => {\n        if (domRunnerMessage.loaded) {\n          this.createVirtualDomElementWithChildren(\n            this.domRunner.getDocument() as unknown as Element,\n            null,\n          );\n\n          const snapshot = virtualDomElementToStatic(\n            this.getVirtualDomElementForRealElementOrThrow(\n              this.domRunner.getDocument() as unknown as Element,\n            ),\n          );\n\n          this.callback({\n            snapshot,\n            documentTime: this.getDocumentTime(),\n          });\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          this.callback({\n            logMessage: domRunnerMessage.logMessage,\n            documentTime: this.getDocumentTime(),\n          });\n        }\n      },\n    );\n  }\n\n  public addIPCWebsocket(webSocket: WebSocket) {\n    return this.domRunner.addIPCWebsocket(webSocket);\n  }\n\n  public addConnectedUserId(connectionId: number): void {\n    this.domRunner.getWindow().dispatchEvent(\n      new (this.domRunner.getWindow().CustomEvent)(\"connected\", {\n        detail: { connectionId },\n      }),\n    );\n  }\n\n  public removeConnectedUserId(connectionId: number): void {\n    this.domRunner.getWindow().dispatchEvent(\n      new (this.domRunner.getWindow().CustomEvent)(\"disconnected\", {\n        detail: { connectionId },\n      }),\n    );\n  }\n\n  private processModificationList(mutationList: Array<MutationRecord>): void {\n    const documentEl = this.domRunner.getDocument() as unknown as Element;\n    const documentVirtualDomElement = this.realElementToVirtualElement.get(documentEl);\n    if (!documentVirtualDomElement) {\n      throw new Error(`document not created in processModificationList`);\n    }\n\n    if (mutationList.length > 1) {\n      // TODO - walk back through the records to derive the intermediate states (e.g. if an attribute is later added to\n      //  an element created in an earlier record then it should not have that attribute when the element is added.\n      //  This is important as incorrect attribute sets can affect visibility and expected client performance.\n      console.error(\n        \"More than one mutation record received. It is possible that intermediate states are incorrect.\",\n      );\n    }\n\n    for (const mutation of mutationList) {\n      if (this.isIgnoredElement(mutation.target as Element | Text)) {\n        continue;\n      }\n\n      if (\n        mutation.type === \"attributes\" &&\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        this.isIgnoredAttribute(mutation.target as Element | Text, mutation.attributeName!)\n      ) {\n        continue;\n      }\n\n      this.addKnownNodesInMutation(mutation);\n\n      // Convert the \"real\" DOM MutationRecord into a \"virtual\" DOM MutationRecord that references the VirtualDOMElements\n      // This is done so that the same process for handling mutations can be used for both changes to a live DOM and also\n      // to diffs between DOM snapshots when reloading\n      const firstNonIgnoredPreviousSibling = mutation.previousSibling\n        ? this.getFirstNonIgnoredPreviousSibling(mutation.previousSibling as Element | Text)\n        : null;\n      const targetElement = this.getVirtualDomElementForRealElementOrThrow(\n        mutation.target as Element | Text,\n      );\n      const addedNodes: Array<StaticVirtualDomElement> = [];\n      for (const node of mutation.addedNodes) {\n        if (this.isIgnoredElement(node as Element | Text)) {\n          continue;\n        }\n        const virtualDomElement = this.getVirtualDomElementForRealElementOrThrow(\n          node as Element | Text,\n        );\n        addedNodes.push(virtualDomElementToStatic(virtualDomElement));\n      }\n\n      const removedNodeIds: Array<number> = [];\n      for (const node of mutation.removedNodes) {\n        if (this.isIgnoredElement(node as Element | Text)) {\n          continue;\n        }\n        const virtualDomElement = this.getVirtualDomElementForRealElementOrThrow(\n          node as Element | Text,\n        );\n        removedNodeIds.push(virtualDomElement.nodeId);\n      }\n\n      const mutationRecord: StaticVirtualDomMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId:\n          firstNonIgnoredPreviousSibling !== null\n            ? this.getVirtualDomElementForRealElementOrThrow(firstNonIgnoredPreviousSibling).nodeId\n            : null,\n        attribute: mutation.attributeName\n          ? {\n              attributeName: mutation.attributeName,\n              value: (mutation.target as Element).getAttribute(mutation.attributeName),\n            }\n          : null,\n      };\n\n      this.callback({\n        mutation: mutationRecord,\n        documentTime: this.getDocumentTime(),\n      });\n\n      this.removeKnownNodesInMutation(mutation);\n    }\n  }\n\n  private addKnownNodesInMutation(mutation: MutationRecord): void {\n    const targetNode = mutation.target as Element | Text;\n    const virtualDomElement = this.realElementToVirtualElement.get(targetNode);\n    if (!virtualDomElement) {\n      throw new Error(\n        \"Unknown node in addKnownNodesInMutation:\" + targetNode + \",\" + mutation.type,\n      );\n    }\n    if (mutation.type === \"childList\") {\n      let previousSibling = mutation.previousSibling;\n      let index = 0;\n      while (previousSibling && this.isIgnoredElement(previousSibling as Element | Text)) {\n        previousSibling = previousSibling.previousSibling;\n      }\n      if (previousSibling) {\n        const previousSiblingElement = this.realElementToVirtualElement.get(\n          previousSibling as Element | Text,\n        );\n        if (!previousSiblingElement) {\n          throw new Error(\"Unknown previous sibling\");\n        }\n        index = virtualDomElement.childNodes.indexOf(previousSiblingElement);\n        if (index === -1) {\n          throw new Error(\"Previous sibling is not currently a child of the parent element\");\n        }\n        index += 1;\n      }\n      mutation.addedNodes.forEach((node: Node) => {\n        const asElementOrText = node as Element | Text;\n        const childVirtualDomElement = this.createVirtualDomElementWithChildren(\n          asElementOrText,\n          virtualDomElement,\n        );\n        if (childVirtualDomElement) {\n          if (virtualDomElement.childNodes.indexOf(childVirtualDomElement) === -1) {\n            virtualDomElement.childNodes.splice(index, 0, childVirtualDomElement);\n            index++;\n          }\n        }\n      });\n    } else if (mutation.type === \"attributes\") {\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      const attributeName = mutation.attributeName!;\n      if (this.isIgnoredAttribute(targetNode, attributeName)) {\n        return;\n      }\n      const attributeValue = (targetNode as Element).getAttribute(attributeName);\n      if (attributeValue === null) {\n        delete virtualDomElement.attributes[attributeName];\n      } else {\n        virtualDomElement.attributes[attributeName] = attributeValue;\n      }\n    } else if (mutation.type === \"characterData\") {\n      virtualDomElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n    }\n  }\n\n  private removeKnownNodesInMutation(mutation: MutationRecord): void {\n    const targetNode = mutation.target as Element | Text;\n    const virtualDomElement = this.realElementToVirtualElement.get(targetNode);\n    if (!virtualDomElement) {\n      throw new Error(\"Unknown node in mutation list:\" + targetNode + \", \" + mutation.type);\n    }\n    if (mutation.type === \"childList\") {\n      for (const node of mutation.removedNodes) {\n        const asElementOrText = node as Element | Text;\n        if (this.isIgnoredElement(asElementOrText)) {\n          continue;\n        }\n        const childDomElement = this.realElementToVirtualElement.get(asElementOrText);\n        if (!childDomElement) {\n          console.warn(this.htmlPath, \"Unknown node in removeKnownNodesInMutation\");\n          continue;\n        } else {\n          this.removeVirtualDomElement(childDomElement);\n          const index = virtualDomElement.childNodes.indexOf(childDomElement);\n          virtualDomElement.childNodes.splice(index, 1);\n        }\n      }\n      return;\n    }\n  }\n\n  private removeVirtualDomElement(virtualDomElement: LiveVirtualDomElement): void {\n    this.nodeIdToNode.delete(virtualDomElement.nodeId);\n    this.nodeToNodeId.delete(virtualDomElement);\n    this.realElementToVirtualElement.delete(virtualDomElement.realElement);\n    for (const child of virtualDomElement.childNodes) {\n      this.removeVirtualDomElement(child);\n    }\n  }\n\n  private createVirtualDomElementWithChildren(\n    node: Element | Text,\n    parent: LiveVirtualDomElement | null,\n  ): LiveVirtualDomElement | null {\n    const virtualElement = this.createVirtualDomElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if ((node as Element).childNodes) {\n      for (let i = 0; i < (node as Element).childNodes.length; i++) {\n        const child = (node as Element).childNodes[i];\n        const childVirtualElement = this.createVirtualDomElementWithChildren(\n          child as Element | Text,\n          virtualElement,\n        );\n        if (childVirtualElement) {\n          virtualElement.childNodes.push(childVirtualElement);\n        }\n      }\n    }\n\n    return virtualElement;\n  }\n\n  private createVirtualDomElement(\n    node: Element | Text,\n    parent: LiveVirtualDomElement | null,\n  ): LiveVirtualDomElement | null {\n    if (this.isIgnoredElement(node)) {\n      return null;\n    }\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      throw new Error(\"Node already has a virtual element: \" + node.nodeName);\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const attributes: { [key: string]: string } = {};\n    if ((node as any).attributes) {\n      const asHTMLElement = node as HTMLElement;\n      for (const key of asHTMLElement.getAttributeNames()) {\n        const value = asHTMLElement.getAttribute(key);\n        if (value === null) {\n          throw new Error(\"Null attribute value for key: \" + key);\n        }\n        if (!this.isIgnoredAttribute(node, key)) {\n          attributes[key] = value;\n        }\n      }\n    }\n\n    const nodeId = this.nextNodeId++;\n    const virtualElement: LiveVirtualDomElement = {\n      nodeId,\n      tag: node.nodeName,\n      attributes,\n      childNodes: [],\n      realElement: node,\n      parent,\n    };\n    if (node instanceof this.domRunner.getWindow().Text && node.textContent) {\n      virtualElement.textContent = node.textContent;\n    }\n    this.nodeToNodeId.set(virtualElement, nodeId);\n    this.nodeIdToNode.set(nodeId, virtualElement);\n    this.realElementToVirtualElement.set(node, virtualElement);\n    return virtualElement;\n  }\n\n  private getFirstNonIgnoredPreviousSibling(node: Element | Text): Element | Text | null {\n    let currentNode = node;\n    if (!this.isIgnoredElement(currentNode)) {\n      return currentNode;\n    }\n    while (currentNode && currentNode.previousSibling) {\n      currentNode = currentNode.previousSibling as Element | Text;\n      if (!this.isIgnoredElement(currentNode)) {\n        return currentNode;\n      }\n    }\n    return null;\n  }\n\n  private getVirtualDomElementForRealElementOrThrow(\n    realElement: Element | Text,\n  ): LiveVirtualDomElement {\n    const virtualElement = this.realElementToVirtualElement.get(realElement);\n    if (!virtualElement) {\n      throw new Error(`Virtual element not found for real element`);\n    }\n    return virtualElement;\n  }\n\n  private isIgnoredElement(node: Element | Text): boolean {\n    if (this.ignoreTextNodes && node instanceof this.domRunner.getWindow().Text) {\n      return true;\n    } else if (node instanceof this.domRunner.getWindow().HTMLScriptElement) {\n      return true;\n    } else if (node instanceof this.domRunner.getWindow().Comment) {\n      return true;\n    }\n    return false;\n  }\n\n  private isIgnoredAttribute(node: Element | Text, attributeName: string): boolean {\n    return attributeName.startsWith(\"on\");\n  }\n\n  public dispatchRemoteEventFromConnectionId(connectionId: number, remoteEvent: RemoteEvent): void {\n    const domNode = this.nodeIdToNode.get(remoteEvent.nodeId);\n    if (!domNode) {\n      console.error(\"Unknown node ID in remote event: \" + remoteEvent.nodeId);\n      return;\n    }\n\n    if (domNode instanceof this.domRunner.getWindow().Text) {\n      console.warn(\"Cannot dispatch remote event to text node\");\n      return;\n    }\n\n    this.domRunner.dispatchRemoteEventFromConnectionId(\n      connectionId,\n      domNode.realElement as Element,\n      remoteEvent,\n    );\n  }\n\n  public dispose() {\n    clearInterval(this.documentTimeIntervalTimer);\n    this.domRunner.dispose();\n  }\n\n  private getDocumentTime() {\n    return this.domRunner.getDocumentTime();\n  }\n}\n", "import { RemoteEvent } from \"@mml-io/networked-dom-protocol\";\nimport { DOMRunnerFactory, DOMRunnerInterface, DOMRunnerMessage } from \"@mml-io/observable-dom\";\n\nexport const WebBrowserDOMRunnerFactory: DOMRunnerFactory = (\n  htmlPath: string,\n  htmlContents: string,\n  params: object,\n  callback: (mutationList: DOMRunnerMessage) => void,\n): DOMRunnerInterface => {\n  return new WebBrowserDOMRunner(htmlPath, htmlContents, params, callback);\n};\n\nconst documentLoadTime = Date.now();\n\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private htmlPath: string;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(\n    htmlPath: string,\n    htmlContents: string,\n    params: object,\n    callback: (domRunnerMessage: DOMRunnerMessage) => void,\n  ) {\n    this.htmlPath = htmlPath;\n    this.callback = callback;\n\n    // Forward console messages\n    for (const level of [\"error\", \"warn\", \"info\", \"log\"] as const) {\n      const defaultFn = window.console[level];\n\n      window.console[level] = (...args) => {\n        callback({\n          logMessage: {\n            level,\n            content: args,\n          },\n        });\n        defaultFn(...args);\n      };\n    }\n\n    // Forward uncaught errors\n    window.onerror = (message, source, line, column, error) => {\n      callback({\n        logMessage: {\n          level: \"system\",\n          content: [\n            {\n              message,\n              type: error?.name,\n              line,\n              column,\n            },\n          ],\n        },\n      });\n      return false;\n    };\n\n    let didSendLoad = false;\n\n    this.mutationObserver = new window.MutationObserver((mutationList) => {\n      if (!document) {\n        return;\n      }\n      if (!didSendLoad) {\n        throw new Error(\"MutationObserver called before load\");\n      }\n      this.callback({\n        mutationList,\n      });\n    });\n\n    (window as any).params = params;\n\n    const finishLoad = () => {\n      if (didSendLoad) {\n        throw new Error(\"finishLoad called twice\");\n      }\n      didSendLoad = true;\n      this.callback({\n        loaded: true,\n      });\n      this.mutationObserver.observe(window.document, {\n        attributes: true,\n        childList: true,\n        subtree: true,\n        characterData: true,\n      });\n    };\n    if (document.body) {\n      setTimeout(finishLoad, 0);\n    } else {\n      window.addEventListener(\"DOMContentLoaded\", finishLoad);\n    }\n  }\n\n  addIPCWebsocket(): void {\n    throw new Error(\"Not implemented.\");\n  }\n\n  dispatchRemoteEventFromConnectionId(\n    connectionId: number,\n    realElement: Element,\n    remoteEvent: RemoteEvent,\n  ): void {\n    const bubbles = remoteEvent.bubbles || false;\n    const remoteEventObject = new CustomEvent(remoteEvent.name, {\n      bubbles,\n      detail: { ...remoteEvent.params, connectionId },\n    });\n\n    // Dispatch the event via JavaScript.\n    realElement.dispatchEvent(remoteEventObject);\n  }\n\n  dispose(): void {\n    // TODO - handle dispose\n    console.log(\"WebBrowserDOMRunner.dispose\");\n  }\n\n  getDocument(): Document {\n    return document;\n  }\n\n  getDocumentTime(): number {\n    if (document.timeline && document.timeline.currentTime) {\n      return document.timeline.currentTime;\n    }\n    return Date.now() - documentLoadTime;\n  }\n\n  // TODO - resolve types (Window needs to expose classes such as CustomEvent as properties)\n  getWindow(): any {\n    return window;\n  }\n}\n", "import {\n  FromInstanceMessageTypes,\n  ToInstanceMessageTypes,\n} from \"@mml-io/networked-dom-web-runner/src/message-types\";\nimport { ObservableDom } from \"@mml-io/observable-dom/src/ObservableDom\";\nimport { ObservableDomMessage, ObservableDOMParameters } from \"@mml-io/observable-dom-common\";\n\nimport { WebBrowserDOMRunnerFactory } from \"./WebBrowserDOMRunner\";\n\n// This runs in the iframe that will execute the document script to setup the listening for events messages\nexport function setupIframeWebRunner(argsString: string) {\n  const observableDOMParams = JSON.parse(atob(argsString)) as ObservableDOMParameters;\n\n  const sendMessageToHandler = (message: FromInstanceMessageTypes) => {\n    window.parent.postMessage(JSON.stringify(message), \"*\");\n  };\n\n  const observableDOM = new ObservableDom(\n    {\n      ...observableDOMParams,\n      htmlContents: \"\", // This must be empty as the contents are assumed to be provided by the srcdoc\n    },\n    (observableDomMessage: ObservableDomMessage) => {\n      sendMessageToHandler({\n        type: \"dom\",\n        message: observableDomMessage,\n      });\n    },\n    WebBrowserDOMRunnerFactory,\n  );\n\n  window.addEventListener(\"message\", (e) => {\n    const parsed = JSON.parse(e.data) as ToInstanceMessageTypes;\n    if (parsed.type === \"dispatchRemoteEventFromConnectionId\") {\n      observableDOM.dispatchRemoteEventFromConnectionId(parsed.connectionId, parsed.event);\n    }\n  });\n}\n", "import { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["args"]
}
\n';
|
|
32
|
+
var build_default = 'var __create = Object.create;\nvar __defProp = Object.defineProperty;\nvar __defProps = Object.defineProperties;\nvar __getOwnPropDesc = Object.getOwnPropertyDescriptor;\nvar __getOwnPropDescs = Object.getOwnPropertyDescriptors;\nvar __getOwnPropNames = Object.getOwnPropertyNames;\nvar __getOwnPropSymbols = Object.getOwnPropertySymbols;\nvar __getProtoOf = Object.getPrototypeOf;\nvar __hasOwnProp = Object.prototype.hasOwnProperty;\nvar __propIsEnum = Object.prototype.propertyIsEnumerable;\nvar __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;\nvar __spreadValues = (a, b) => {\n for (var prop in b ||= {})\n if (__hasOwnProp.call(b, prop))\n __defNormalProp(a, prop, b[prop]);\n if (__getOwnPropSymbols)\n for (var prop of __getOwnPropSymbols(b)) {\n if (__propIsEnum.call(b, prop))\n __defNormalProp(a, prop, b[prop]);\n }\n return a;\n};\nvar __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));\nvar __commonJS = (cb, mod) => function __require() {\n return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;\n};\nvar __copyProps = (to, from, except, desc) => {\n if (from && typeof from === "object" || typeof from === "function") {\n for (let key of __getOwnPropNames(from))\n if (!__hasOwnProp.call(to, key) && key !== except)\n __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });\n }\n return to;\n};\nvar __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(\n // If the importer is in node compatibility mode or this is not an ESM\n // file that has been converted to a CommonJS file using a Babel-\n // compatible transform (i.e. "__esModule" has not been set), then set\n // "default" to the CommonJS "module.exports" for node compatibility.\n isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,\n mod\n));\n\n// ../../observable-dom-common/build/index.js\nvar require_build = __commonJS({\n "../../observable-dom-common/build/index.js"(exports, module2) {\n var __defProp2 = Object.defineProperty;\n var __getOwnPropDesc2 = Object.getOwnPropertyDescriptor;\n var __getOwnPropNames2 = Object.getOwnPropertyNames;\n var __hasOwnProp2 = Object.prototype.hasOwnProperty;\n var __export = (target, all) => {\n for (var name in all)\n __defProp2(target, name, { get: all[name], enumerable: true });\n };\n var __copyProps2 = (to, from, except, desc) => {\n if (from && typeof from === "object" || typeof from === "function") {\n for (let key of __getOwnPropNames2(from))\n if (!__hasOwnProp2.call(to, key) && key !== except)\n __defProp2(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc2(from, key)) || desc.enumerable });\n }\n return to;\n };\n var __toCommonJS = (mod) => __copyProps2(__defProp2({}, "__esModule", { value: true }), mod);\n var src_exports = {};\n __export(src_exports, {\n ADD_CONNECTED_USER_ID_MESSAGE_TYPE: () => ADD_CONNECTED_USER_ID_MESSAGE_TYPE2,\n DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE: () => DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE2,\n DOM_MESSAGE_TYPE: () => DOM_MESSAGE_TYPE2,\n REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE: () => REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE2,\n applyMessageToObservableDOMInstance: () => applyMessageToObservableDOMInstance,\n observableDOMInterfaceToMessageSender: () => observableDOMInterfaceToMessageSender\n });\n module2.exports = __toCommonJS(src_exports);\n var ADD_CONNECTED_USER_ID_MESSAGE_TYPE2 = "addConnectedUserId";\n var REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE2 = "removeConnectedUserId";\n var DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE2 = "dispatchRemoteEventFromConnectionId";\n var DOM_MESSAGE_TYPE2 = "dom";\n function applyMessageToObservableDOMInstance(message, instance) {\n if (message.type === ADD_CONNECTED_USER_ID_MESSAGE_TYPE2) {\n instance.addConnectedUserId(message.connectionId);\n } else if (message.type === REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE2) {\n instance.removeConnectedUserId(message.connectionId);\n } else if (message.type === DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE2) {\n instance.dispatchRemoteEventFromConnectionId(message.connectionId, message.event);\n } else {\n console.error("Unknown message type", message);\n }\n }\n function observableDOMInterfaceToMessageSender(sender, dispose) {\n const remoteObservableDOM = {\n addConnectedUserId(connectionId) {\n sender({\n type: ADD_CONNECTED_USER_ID_MESSAGE_TYPE2,\n connectionId\n });\n },\n dispatchRemoteEventFromConnectionId(connectionId, remoteEvent) {\n sender({\n type: DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE2,\n connectionId,\n event: remoteEvent\n });\n },\n dispose() {\n dispose();\n },\n removeConnectedUserId(connectionId) {\n sender({\n type: REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE2,\n connectionId\n });\n }\n };\n return remoteObservableDOM;\n }\n }\n});\n\n// ../../observable-dom/src/utils.ts\nfunction virtualDOMElementToStatic(el) {\n return {\n nodeId: el.nodeId,\n tag: el.tag,\n attributes: el.attributes,\n childNodes: el.childNodes.map((child) => virtualDOMElementToStatic(child)),\n textContent: el.textContent\n };\n}\n\n// ../../observable-dom/src/ObservableDOM.ts\nvar ObservableDOM = class {\n constructor(observableDOMParameters, callback, runnerFactory) {\n this.nodeToNodeId = /* @__PURE__ */ new Map();\n this.nodeIdToNode = /* @__PURE__ */ new Map();\n this.realElementToVirtualElement = /* @__PURE__ */ new Map();\n this.ignoreTextNodes = true;\n this.nextNodeId = 1;\n this.htmlPath = observableDOMParameters.htmlPath;\n this.ignoreTextNodes = observableDOMParameters.ignoreTextNodes;\n this.callback = callback;\n this.documentTimeIntervalTimer = setInterval(() => {\n this.callback(\n {\n documentTime: this.getDocumentTime()\n },\n this\n );\n }, observableDOMParameters.pingIntervalMilliseconds || 5e3);\n this.domRunner = runnerFactory(\n observableDOMParameters.htmlPath,\n observableDOMParameters.htmlContents,\n observableDOMParameters.params,\n (domRunnerMessage) => {\n if (domRunnerMessage.loaded) {\n this.createVirtualDOMElementWithChildren(\n this.domRunner.getDocument(),\n null\n );\n const snapshot = virtualDOMElementToStatic(\n this.getVirtualDOMElementForRealElementOrThrow(\n this.domRunner.getDocument()\n )\n );\n this.callback(\n {\n snapshot,\n documentTime: this.getDocumentTime()\n },\n this\n );\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n this.callback(\n {\n logMessage: domRunnerMessage.logMessage,\n documentTime: this.getDocumentTime()\n },\n this\n );\n }\n }\n );\n }\n addConnectedUserId(connectionId) {\n this.domRunner.getWindow().dispatchEvent(\n new (this.domRunner.getWindow()).CustomEvent("connected", {\n detail: { connectionId }\n })\n );\n }\n removeConnectedUserId(connectionId) {\n this.domRunner.getWindow().dispatchEvent(\n new (this.domRunner.getWindow()).CustomEvent("disconnected", {\n detail: { connectionId }\n })\n );\n }\n processModificationList(mutationList) {\n const documentEl = this.domRunner.getDocument();\n const documentVirtualDOMElement = this.realElementToVirtualElement.get(documentEl);\n if (!documentVirtualDOMElement) {\n throw new Error(`document not created in processModificationList`);\n }\n if (mutationList.length > 1) {\n console.error(\n "More than one mutation record received. It is possible that intermediate states are incorrect."\n );\n }\n for (const mutation of mutationList) {\n if (this.isIgnoredElement(mutation.target)) {\n continue;\n }\n if (mutation.type === "attributes" && // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n this.isIgnoredAttribute(mutation.target, mutation.attributeName)) {\n continue;\n }\n this.addKnownNodesInMutation(mutation);\n const firstNonIgnoredPreviousSibling = mutation.previousSibling ? this.getFirstNonIgnoredPreviousSibling(mutation.previousSibling) : null;\n const targetElement = this.getVirtualDOMElementForRealElementOrThrow(\n mutation.target\n );\n const addedNodes = [];\n for (const node of mutation.addedNodes) {\n if (this.isIgnoredElement(node)) {\n continue;\n }\n const virtualDOMElement = this.getVirtualDOMElementForRealElementOrThrow(\n node\n );\n addedNodes.push(virtualDOMElementToStatic(virtualDOMElement));\n }\n const removedNodeIds = [];\n for (const node of mutation.removedNodes) {\n if (this.isIgnoredElement(node)) {\n continue;\n }\n const virtualDOMElement = this.getVirtualDOMElementForRealElementOrThrow(\n node\n );\n removedNodeIds.push(virtualDOMElement.nodeId);\n }\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: firstNonIgnoredPreviousSibling !== null ? this.getVirtualDOMElementForRealElementOrThrow(firstNonIgnoredPreviousSibling).nodeId : null,\n attribute: mutation.attributeName ? {\n attributeName: mutation.attributeName,\n value: mutation.target.getAttribute(mutation.attributeName)\n } : null\n };\n this.callback(\n {\n mutation: mutationRecord,\n documentTime: this.getDocumentTime()\n },\n this\n );\n this.removeKnownNodesInMutation(mutation);\n }\n }\n addKnownNodesInMutation(mutation) {\n const targetNode = mutation.target;\n const virtualDOMElement = this.realElementToVirtualElement.get(targetNode);\n if (!virtualDOMElement) {\n throw new Error(\n "Unknown node in addKnownNodesInMutation:" + targetNode + "," + mutation.type\n );\n }\n if (mutation.type === "childList") {\n let previousSibling = mutation.previousSibling;\n let index = 0;\n while (previousSibling && this.isIgnoredElement(previousSibling)) {\n previousSibling = previousSibling.previousSibling;\n }\n if (previousSibling) {\n const previousSiblingElement = this.realElementToVirtualElement.get(\n previousSibling\n );\n if (!previousSiblingElement) {\n throw new Error("Unknown previous sibling");\n }\n index = virtualDOMElement.childNodes.indexOf(previousSiblingElement);\n if (index === -1) {\n throw new Error("Previous sibling is not currently a child of the parent element");\n }\n index += 1;\n }\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n asElementOrText,\n virtualDOMElement\n );\n if (childVirtualDOMElement) {\n if (virtualDOMElement.childNodes.indexOf(childVirtualDOMElement) === -1) {\n virtualDOMElement.childNodes.splice(index, 0, childVirtualDOMElement);\n index++;\n }\n }\n });\n } else if (mutation.type === "attributes") {\n const attributeName = mutation.attributeName;\n if (this.isIgnoredAttribute(targetNode, attributeName)) {\n return;\n }\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete virtualDOMElement.attributes[attributeName];\n } else {\n virtualDOMElement.attributes[attributeName] = attributeValue;\n }\n } else if (mutation.type === "characterData") {\n virtualDOMElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n }\n removeKnownNodesInMutation(mutation) {\n const targetNode = mutation.target;\n const virtualDOMElement = this.realElementToVirtualElement.get(targetNode);\n if (!virtualDOMElement) {\n throw new Error("Unknown node in mutation list:" + targetNode + ", " + mutation.type);\n }\n if (mutation.type === "childList") {\n for (const node of mutation.removedNodes) {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n continue;\n }\n const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDOMElement) {\n console.warn(this.htmlPath, "Unknown node in removeKnownNodesInMutation");\n continue;\n } else {\n this.removeVirtualDOMElement(childDOMElement);\n const index = virtualDOMElement.childNodes.indexOf(childDOMElement);\n virtualDOMElement.childNodes.splice(index, 1);\n }\n }\n return;\n }\n }\n removeVirtualDOMElement(virtualDOMElement) {\n this.nodeIdToNode.delete(virtualDOMElement.nodeId);\n this.nodeToNodeId.delete(virtualDOMElement);\n this.realElementToVirtualElement.delete(virtualDOMElement.realElement);\n for (const child of virtualDOMElement.childNodes) {\n this.removeVirtualDOMElement(child);\n }\n }\n createVirtualDOMElementWithChildren(node, parent) {\n const virtualElement = this.createVirtualDOMElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (node.childNodes) {\n for (let i = 0; i < node.childNodes.length; i++) {\n const child = node.childNodes[i];\n const childVirtualElement = this.createVirtualDOMElementWithChildren(\n child,\n virtualElement\n );\n if (childVirtualElement) {\n virtualElement.childNodes.push(childVirtualElement);\n }\n }\n }\n return virtualElement;\n }\n createVirtualDOMElement(node, parent) {\n if (this.isIgnoredElement(node)) {\n return null;\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n throw new Error("Node already has a virtual element: " + node.nodeName);\n }\n if (!node) {\n throw new Error("Cannot assign node id to null");\n }\n const attributes = {};\n if (node.attributes) {\n const asHTMLElement = node;\n for (const key of asHTMLElement.getAttributeNames()) {\n const value = asHTMLElement.getAttribute(key);\n if (value === null) {\n throw new Error("Null attribute value for key: " + key);\n }\n if (!this.isIgnoredAttribute(node, key)) {\n attributes[key] = value;\n }\n }\n }\n const nodeId = this.nextNodeId++;\n const virtualElement = {\n nodeId,\n tag: node.nodeName,\n attributes,\n childNodes: [],\n realElement: node,\n parent\n };\n if (node instanceof this.domRunner.getWindow().Text && node.textContent) {\n virtualElement.textContent = node.textContent;\n }\n this.nodeToNodeId.set(virtualElement, nodeId);\n this.nodeIdToNode.set(nodeId, virtualElement);\n this.realElementToVirtualElement.set(node, virtualElement);\n return virtualElement;\n }\n getFirstNonIgnoredPreviousSibling(node) {\n let currentNode = node;\n if (!this.isIgnoredElement(currentNode)) {\n return currentNode;\n }\n while (currentNode && currentNode.previousSibling) {\n currentNode = currentNode.previousSibling;\n if (!this.isIgnoredElement(currentNode)) {\n return currentNode;\n }\n }\n return null;\n }\n getVirtualDOMElementForRealElementOrThrow(realElement) {\n const virtualElement = this.realElementToVirtualElement.get(realElement);\n if (!virtualElement) {\n throw new Error(`Virtual element not found for real element`);\n }\n return virtualElement;\n }\n isIgnoredElement(node) {\n if (this.ignoreTextNodes && node instanceof this.domRunner.getWindow().Text) {\n return true;\n } else if (node instanceof this.domRunner.getWindow().HTMLScriptElement) {\n return true;\n } else if (node instanceof this.domRunner.getWindow().Comment) {\n return true;\n }\n return false;\n }\n isIgnoredAttribute(node, attributeName) {\n return attributeName.startsWith("on");\n }\n dispatchRemoteEventFromConnectionId(connectionId, remoteEvent) {\n const domNode = this.nodeIdToNode.get(remoteEvent.nodeId);\n if (!domNode) {\n console.error("Unknown node ID in remote event: " + remoteEvent.nodeId);\n return;\n }\n if (domNode instanceof this.domRunner.getWindow().Text) {\n console.warn("Cannot dispatch remote event to text node");\n return;\n }\n this.domRunner.dispatchRemoteEventFromConnectionId(\n connectionId,\n domNode.realElement,\n remoteEvent\n );\n }\n dispose() {\n clearInterval(this.documentTimeIntervalTimer);\n this.domRunner.dispose();\n }\n getDocumentTime() {\n return this.domRunner.getDocumentTime();\n }\n};\n\n// src/IframeWebRunner.ts\nvar import_observable_dom_common = __toESM(require_build());\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(htmlPath, htmlContents, params, callback);\n};\nvar documentLoadTime = Date.now();\nvar WebBrowserDOMRunner = class {\n constructor(htmlPath, htmlContents, params, callback) {\n this.htmlPath = htmlPath;\n this.callback = callback;\n for (const level of ["error", "warn", "info", "log"]) {\n const defaultFn = window.console[level];\n window.console[level] = (...args2) => {\n callback({\n logMessage: {\n level,\n content: args2\n }\n });\n defaultFn(...args2);\n };\n }\n window.onerror = (message, source, line, column, error) => {\n callback({\n logMessage: {\n level: "system",\n content: [\n {\n message,\n type: error == null ? void 0 : error.name,\n line,\n column\n }\n ]\n }\n });\n return false;\n };\n let didSendLoad = false;\n this.mutationObserver = new window.MutationObserver((mutationList) => {\n if (!document) {\n return;\n }\n if (!didSendLoad) {\n throw new Error("MutationObserver called before load");\n }\n this.callback({\n mutationList\n });\n });\n window.params = params;\n const finishLoad = () => {\n if (didSendLoad) {\n throw new Error("finishLoad called twice");\n }\n didSendLoad = true;\n this.callback({\n loaded: true\n });\n this.mutationObserver.observe(window.document, {\n attributes: true,\n childList: true,\n subtree: true,\n characterData: true\n });\n };\n if (document.body) {\n setTimeout(finishLoad, 0);\n } else {\n window.addEventListener("DOMContentLoaded", finishLoad);\n }\n }\n dispatchRemoteEventFromConnectionId(connectionId, realElement, remoteEvent) {\n const bubbles = remoteEvent.bubbles || false;\n const remoteEventObject = new CustomEvent(remoteEvent.name, {\n bubbles,\n detail: __spreadProps(__spreadValues({}, remoteEvent.params), { connectionId })\n });\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log("WebBrowserDOMRunner.dispose");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n if (document.timeline && document.timeline.currentTime) {\n return document.timeline.currentTime;\n }\n return Date.now() - documentLoadTime;\n }\n // TODO - resolve types (Window needs to expose classes such as CustomEvent as properties)\n getWindow() {\n return window;\n }\n};\n\n// src/IframeWebRunner.ts\nfunction setupIframeWebRunner(argsString) {\n const observableDOMParams = JSON.parse(atob(argsString));\n const sendMessageToHandler = (message) => {\n window.parent.postMessage(JSON.stringify(message), "*");\n };\n const observableDOM = new ObservableDOM(\n __spreadProps(__spreadValues({}, observableDOMParams), {\n htmlContents: ""\n // This must be empty as the contents are assumed to be provided by the srcdoc\n }),\n (observableDOMMessage) => {\n sendMessageToHandler({\n type: import_observable_dom_common.DOM_MESSAGE_TYPE,\n message: observableDOMMessage\n });\n },\n WebBrowserDOMRunnerFactory\n );\n window.addEventListener("message", (e) => {\n const parsed = JSON.parse(e.data);\n switch (parsed.type) {\n case import_observable_dom_common.DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE:\n observableDOM.dispatchRemoteEventFromConnectionId(parsed.connectionId, parsed.event);\n break;\n case import_observable_dom_common.ADD_CONNECTED_USER_ID_MESSAGE_TYPE:\n observableDOM.addConnectedUserId(parsed.connectionId);\n break;\n case import_observable_dom_common.REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE:\n observableDOM.removeConnectedUserId(parsed.connectionId);\n break;\n default:\n console.error("Unknown message type", parsed);\n }\n });\n}\n\n// src/index.ts\nvar args = window.args;\nsetupIframeWebRunner(args);\n//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../observable-dom-common/src/index.ts", "../../../observable-dom-common/src/messages.ts", "../../../observable-dom/src/utils.ts", "../../../observable-dom/src/ObservableDOM.ts", "../src/IframeWebRunner.ts", "../src/WebBrowserDOMRunner.ts", "../src/index.ts"],
  "sourcesContent": ["export type { RemoteEvent } from \"@mml-io/networked-dom-protocol\";\n\nexport * from \"./ObservableDOMInterface\";\nexport * from \"./messages\";\n", "import { RemoteEvent } from \"@mml-io/networked-dom-protocol\";\n\nimport { ObservableDOMInterface, ObservableDOMMessage } from \"./ObservableDOMInterface\";\n\nexport const ADD_CONNECTED_USER_ID_MESSAGE_TYPE = \"addConnectedUserId\";\nexport const REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE = \"removeConnectedUserId\";\nexport const DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE =\n  \"dispatchRemoteEventFromConnectionId\";\nexport const DOM_MESSAGE_TYPE = \"dom\";\n\nexport type AddConnectedUserIdMessage = {\n  type: typeof ADD_CONNECTED_USER_ID_MESSAGE_TYPE;\n  connectionId: number;\n};\n\nexport type RemoveConnectedUserIdMessage = {\n  type: typeof REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE;\n  connectionId: number;\n};\n\nexport type DispatchRemoteEventFromConnectionIdMessage = {\n  type: typeof DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE;\n  connectionId: number;\n  event: RemoteEvent;\n};\n\nexport type ToObservableDOMInstanceMessage =\n  | AddConnectedUserIdMessage\n  | RemoveConnectedUserIdMessage\n  | DispatchRemoteEventFromConnectionIdMessage;\n\ntype DOMMessage = {\n  type: typeof DOM_MESSAGE_TYPE;\n  message: ObservableDOMMessage;\n};\n\nexport type FromObservableDOMInstanceMessage = DOMMessage;\n\nexport function applyMessageToObservableDOMInstance(\n  message: ToObservableDOMInstanceMessage,\n  instance: ObservableDOMInterface,\n) {\n  if (message.type === ADD_CONNECTED_USER_ID_MESSAGE_TYPE) {\n    instance.addConnectedUserId(message.connectionId);\n  } else if (message.type === REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE) {\n    instance.removeConnectedUserId(message.connectionId);\n  } else if (message.type === DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE) {\n    instance.dispatchRemoteEventFromConnectionId(message.connectionId, message.event);\n  } else {\n    console.error(\"Unknown message type\", message);\n  }\n}\n\nexport function observableDOMInterfaceToMessageSender(\n  sender: (message: ToObservableDOMInstanceMessage) => void,\n  dispose: () => void,\n) {\n  const remoteObservableDOM: ObservableDOMInterface = {\n    addConnectedUserId(connectionId: number): void {\n      sender({\n        type: ADD_CONNECTED_USER_ID_MESSAGE_TYPE,\n        connectionId,\n      });\n    },\n    dispatchRemoteEventFromConnectionId(connectionId: number, remoteEvent: RemoteEvent): void {\n      sender({\n        type: DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE,\n        connectionId,\n        event: remoteEvent,\n      });\n    },\n    dispose(): void {\n      dispose();\n    },\n    removeConnectedUserId(connectionId: number): void {\n      sender({\n        type: REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE,\n        connectionId,\n      });\n    },\n  };\n  return remoteObservableDOM;\n}\n", "import { StaticVirtualDOMElement } from \"@mml-io/observable-dom-common\";\n\nimport { LiveVirtualDOMElement } from \"./ObservableDOM\";\n\nexport function virtualDOMElementToStatic(el: LiveVirtualDOMElement): StaticVirtualDOMElement {\n  return {\n    nodeId: el.nodeId,\n    tag: el.tag,\n    attributes: el.attributes,\n    childNodes: el.childNodes.map((child) => virtualDOMElementToStatic(child)),\n    textContent: el.textContent,\n  };\n}\n", "import {\n  LogMessage,\n  ObservableDOMInterface,\n  ObservableDOMMessage,\n  ObservableDOMParameters,\n  RemoteEvent,\n  StaticVirtualDOMElement,\n  StaticVirtualDOMMutationIdsRecord,\n} from \"@mml-io/observable-dom-common\";\n\nimport { virtualDOMElementToStatic } from \"./utils\";\n\nexport type DOMRunnerMessage = {\n  loaded?: boolean;\n  mutationList?: Array<MutationRecord>;\n  logMessage?: LogMessage;\n};\n\nexport type DOMRunnerInterface = {\n  getDocument(): Document;\n  getWindow(): Window & {\n    CustomEvent: typeof CustomEvent;\n    Text: typeof Text;\n    HTMLScriptElement: typeof HTMLScriptElement;\n    Comment: typeof Comment;\n  }; // TODO - Define this without using JSDOM types\n  dispatchRemoteEventFromConnectionId(\n    connectionId: number,\n    realElement: Element,\n    remoteEvent: RemoteEvent,\n  ): void;\n  dispose(): void;\n  getDocumentTime(): number;\n};\n\nexport type DOMRunnerFactory = (\n  htmlPath: string,\n  htmlContents: string,\n  params: object,\n  callback: (domRunnerMessage: DOMRunnerMessage) => void,\n) => DOMRunnerInterface;\n\nexport type LiveVirtualDOMElement = Omit<StaticVirtualDOMElement, \"childNodes\"> & {\n  realElement: Element | Text;\n  childNodes: Array<LiveVirtualDOMElement>;\n  parent: LiveVirtualDOMElement | null;\n};\n\nexport class ObservableDOM implements ObservableDOMInterface {\n  private nodeToNodeId = new Map<LiveVirtualDOMElement, number>();\n  private nodeIdToNode = new Map<number, LiveVirtualDOMElement>();\n  private realElementToVirtualElement = new Map<Element | Text, LiveVirtualDOMElement>();\n  private ignoreTextNodes = true;\n  private callback: (message: ObservableDOMMessage, observableDOM: ObservableDOMInterface) => void;\n  private nextNodeId = 1;\n  private htmlPath: string;\n  private domRunner: DOMRunnerInterface;\n\n  private documentTimeIntervalTimer: NodeJS.Timer;\n\n  constructor(\n    observableDOMParameters: ObservableDOMParameters,\n    callback: (message: ObservableDOMMessage, observableDOM: ObservableDOMInterface) => void,\n    runnerFactory: DOMRunnerFactory,\n  ) {\n    this.htmlPath = observableDOMParameters.htmlPath;\n    this.ignoreTextNodes = observableDOMParameters.ignoreTextNodes;\n    this.callback = callback;\n\n    this.documentTimeIntervalTimer = setInterval(() => {\n      this.callback(\n        {\n          documentTime: this.getDocumentTime(),\n        },\n        this,\n      );\n    }, observableDOMParameters.pingIntervalMilliseconds || 5000);\n\n    this.domRunner = runnerFactory(\n      observableDOMParameters.htmlPath,\n      observableDOMParameters.htmlContents,\n      observableDOMParameters.params,\n      (domRunnerMessage: DOMRunnerMessage) => {\n        if (domRunnerMessage.loaded) {\n          this.createVirtualDOMElementWithChildren(\n            this.domRunner.getDocument() as unknown as Element,\n            null,\n          );\n\n          const snapshot = virtualDOMElementToStatic(\n            this.getVirtualDOMElementForRealElementOrThrow(\n              this.domRunner.getDocument() as unknown as Element,\n            ),\n          );\n\n          this.callback(\n            {\n              snapshot,\n              documentTime: this.getDocumentTime(),\n            },\n            this,\n          );\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          this.callback(\n            {\n              logMessage: domRunnerMessage.logMessage,\n              documentTime: this.getDocumentTime(),\n            },\n            this,\n          );\n        }\n      },\n    );\n  }\n\n  public addConnectedUserId(connectionId: number): void {\n    this.domRunner.getWindow().dispatchEvent(\n      new (this.domRunner.getWindow().CustomEvent)(\"connected\", {\n        detail: { connectionId },\n      }),\n    );\n  }\n\n  public removeConnectedUserId(connectionId: number): void {\n    this.domRunner.getWindow().dispatchEvent(\n      new (this.domRunner.getWindow().CustomEvent)(\"disconnected\", {\n        detail: { connectionId },\n      }),\n    );\n  }\n\n  private processModificationList(mutationList: Array<MutationRecord>): void {\n    const documentEl = this.domRunner.getDocument() as unknown as Element;\n    const documentVirtualDOMElement = this.realElementToVirtualElement.get(documentEl);\n    if (!documentVirtualDOMElement) {\n      throw new Error(`document not created in processModificationList`);\n    }\n\n    if (mutationList.length > 1) {\n      // TODO - walk back through the records to derive the intermediate states (e.g. if an attribute is later added to\n      //  an element created in an earlier record then it should not have that attribute when the element is added.\n      //  This is important as incorrect attribute sets can affect visibility and expected client performance.\n      console.error(\n        \"More than one mutation record received. It is possible that intermediate states are incorrect.\",\n      );\n    }\n\n    for (const mutation of mutationList) {\n      if (this.isIgnoredElement(mutation.target as Element | Text)) {\n        continue;\n      }\n\n      if (\n        mutation.type === \"attributes\" &&\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        this.isIgnoredAttribute(mutation.target as Element | Text, mutation.attributeName!)\n      ) {\n        continue;\n      }\n\n      this.addKnownNodesInMutation(mutation);\n\n      // Convert the \"real\" DOM MutationRecord into a \"virtual\" DOM MutationRecord that references the VirtualDOMElements\n      // This is done so that the same process for handling mutations can be used for both changes to a live DOM and also\n      // to diffs between DOM snapshots when reloading\n      const firstNonIgnoredPreviousSibling = mutation.previousSibling\n        ? this.getFirstNonIgnoredPreviousSibling(mutation.previousSibling as Element | Text)\n        : null;\n      const targetElement = this.getVirtualDOMElementForRealElementOrThrow(\n        mutation.target as Element | Text,\n      );\n      const addedNodes: Array<StaticVirtualDOMElement> = [];\n      for (const node of mutation.addedNodes) {\n        if (this.isIgnoredElement(node as Element | Text)) {\n          continue;\n        }\n        const virtualDOMElement = this.getVirtualDOMElementForRealElementOrThrow(\n          node as Element | Text,\n        );\n        addedNodes.push(virtualDOMElementToStatic(virtualDOMElement));\n      }\n\n      const removedNodeIds: Array<number> = [];\n      for (const node of mutation.removedNodes) {\n        if (this.isIgnoredElement(node as Element | Text)) {\n          continue;\n        }\n        const virtualDOMElement = this.getVirtualDOMElementForRealElementOrThrow(\n          node as Element | Text,\n        );\n        removedNodeIds.push(virtualDOMElement.nodeId);\n      }\n\n      const mutationRecord: StaticVirtualDOMMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId:\n          firstNonIgnoredPreviousSibling !== null\n            ? this.getVirtualDOMElementForRealElementOrThrow(firstNonIgnoredPreviousSibling).nodeId\n            : null,\n        attribute: mutation.attributeName\n          ? {\n              attributeName: mutation.attributeName,\n              value: (mutation.target as Element).getAttribute(mutation.attributeName),\n            }\n          : null,\n      };\n\n      this.callback(\n        {\n          mutation: mutationRecord,\n          documentTime: this.getDocumentTime(),\n        },\n        this,\n      );\n\n      this.removeKnownNodesInMutation(mutation);\n    }\n  }\n\n  private addKnownNodesInMutation(mutation: MutationRecord): void {\n    const targetNode = mutation.target as Element | Text;\n    const virtualDOMElement = this.realElementToVirtualElement.get(targetNode);\n    if (!virtualDOMElement) {\n      throw new Error(\n        \"Unknown node in addKnownNodesInMutation:\" + targetNode + \",\" + mutation.type,\n      );\n    }\n    if (mutation.type === \"childList\") {\n      let previousSibling = mutation.previousSibling;\n      let index = 0;\n      while (previousSibling && this.isIgnoredElement(previousSibling as Element | Text)) {\n        previousSibling = previousSibling.previousSibling;\n      }\n      if (previousSibling) {\n        const previousSiblingElement = this.realElementToVirtualElement.get(\n          previousSibling as Element | Text,\n        );\n        if (!previousSiblingElement) {\n          throw new Error(\"Unknown previous sibling\");\n        }\n        index = virtualDOMElement.childNodes.indexOf(previousSiblingElement);\n        if (index === -1) {\n          throw new Error(\"Previous sibling is not currently a child of the parent element\");\n        }\n        index += 1;\n      }\n      mutation.addedNodes.forEach((node: Node) => {\n        const asElementOrText = node as Element | Text;\n        const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n          asElementOrText,\n          virtualDOMElement,\n        );\n        if (childVirtualDOMElement) {\n          if (virtualDOMElement.childNodes.indexOf(childVirtualDOMElement) === -1) {\n            virtualDOMElement.childNodes.splice(index, 0, childVirtualDOMElement);\n            index++;\n          }\n        }\n      });\n    } else if (mutation.type === \"attributes\") {\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      const attributeName = mutation.attributeName!;\n      if (this.isIgnoredAttribute(targetNode, attributeName)) {\n        return;\n      }\n      const attributeValue = (targetNode as Element).getAttribute(attributeName);\n      if (attributeValue === null) {\n        delete virtualDOMElement.attributes[attributeName];\n      } else {\n        virtualDOMElement.attributes[attributeName] = attributeValue;\n      }\n    } else if (mutation.type === \"characterData\") {\n      virtualDOMElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n    }\n  }\n\n  private removeKnownNodesInMutation(mutation: MutationRecord): void {\n    const targetNode = mutation.target as Element | Text;\n    const virtualDOMElement = this.realElementToVirtualElement.get(targetNode);\n    if (!virtualDOMElement) {\n      throw new Error(\"Unknown node in mutation list:\" + targetNode + \", \" + mutation.type);\n    }\n    if (mutation.type === \"childList\") {\n      for (const node of mutation.removedNodes) {\n        const asElementOrText = node as Element | Text;\n        if (this.isIgnoredElement(asElementOrText)) {\n          continue;\n        }\n        const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n        if (!childDOMElement) {\n          console.warn(this.htmlPath, \"Unknown node in removeKnownNodesInMutation\");\n          continue;\n        } else {\n          this.removeVirtualDOMElement(childDOMElement);\n          const index = virtualDOMElement.childNodes.indexOf(childDOMElement);\n          virtualDOMElement.childNodes.splice(index, 1);\n        }\n      }\n      return;\n    }\n  }\n\n  private removeVirtualDOMElement(virtualDOMElement: LiveVirtualDOMElement): void {\n    this.nodeIdToNode.delete(virtualDOMElement.nodeId);\n    this.nodeToNodeId.delete(virtualDOMElement);\n    this.realElementToVirtualElement.delete(virtualDOMElement.realElement);\n    for (const child of virtualDOMElement.childNodes) {\n      this.removeVirtualDOMElement(child);\n    }\n  }\n\n  private createVirtualDOMElementWithChildren(\n    node: Element | Text,\n    parent: LiveVirtualDOMElement | null,\n  ): LiveVirtualDOMElement | null {\n    const virtualElement = this.createVirtualDOMElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if ((node as Element).childNodes) {\n      for (let i = 0; i < (node as Element).childNodes.length; i++) {\n        const child = (node as Element).childNodes[i];\n        const childVirtualElement = this.createVirtualDOMElementWithChildren(\n          child as Element | Text,\n          virtualElement,\n        );\n        if (childVirtualElement) {\n          virtualElement.childNodes.push(childVirtualElement);\n        }\n      }\n    }\n\n    return virtualElement;\n  }\n\n  private createVirtualDOMElement(\n    node: Element | Text,\n    parent: LiveVirtualDOMElement | null,\n  ): LiveVirtualDOMElement | null {\n    if (this.isIgnoredElement(node)) {\n      return null;\n    }\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      throw new Error(\"Node already has a virtual element: \" + node.nodeName);\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const attributes: { [key: string]: string } = {};\n    if ((node as any).attributes) {\n      const asHTMLElement = node as HTMLElement;\n      for (const key of asHTMLElement.getAttributeNames()) {\n        const value = asHTMLElement.getAttribute(key);\n        if (value === null) {\n          throw new Error(\"Null attribute value for key: \" + key);\n        }\n        if (!this.isIgnoredAttribute(node, key)) {\n          attributes[key] = value;\n        }\n      }\n    }\n\n    const nodeId = this.nextNodeId++;\n    const virtualElement: LiveVirtualDOMElement = {\n      nodeId,\n      tag: node.nodeName,\n      attributes,\n      childNodes: [],\n      realElement: node,\n      parent,\n    };\n    if (node instanceof this.domRunner.getWindow().Text && node.textContent) {\n      virtualElement.textContent = node.textContent;\n    }\n    this.nodeToNodeId.set(virtualElement, nodeId);\n    this.nodeIdToNode.set(nodeId, virtualElement);\n    this.realElementToVirtualElement.set(node, virtualElement);\n    return virtualElement;\n  }\n\n  private getFirstNonIgnoredPreviousSibling(node: Element | Text): Element | Text | null {\n    let currentNode = node;\n    if (!this.isIgnoredElement(currentNode)) {\n      return currentNode;\n    }\n    while (currentNode && currentNode.previousSibling) {\n      currentNode = currentNode.previousSibling as Element | Text;\n      if (!this.isIgnoredElement(currentNode)) {\n        return currentNode;\n      }\n    }\n    return null;\n  }\n\n  private getVirtualDOMElementForRealElementOrThrow(\n    realElement: Element | Text,\n  ): LiveVirtualDOMElement {\n    const virtualElement = this.realElementToVirtualElement.get(realElement);\n    if (!virtualElement) {\n      throw new Error(`Virtual element not found for real element`);\n    }\n    return virtualElement;\n  }\n\n  private isIgnoredElement(node: Element | Text): boolean {\n    if (this.ignoreTextNodes && node instanceof this.domRunner.getWindow().Text) {\n      return true;\n    } else if (node instanceof this.domRunner.getWindow().HTMLScriptElement) {\n      return true;\n    } else if (node instanceof this.domRunner.getWindow().Comment) {\n      return true;\n    }\n    return false;\n  }\n\n  private isIgnoredAttribute(node: Element | Text, attributeName: string): boolean {\n    return attributeName.startsWith(\"on\");\n  }\n\n  public dispatchRemoteEventFromConnectionId(connectionId: number, remoteEvent: RemoteEvent): void {\n    const domNode = this.nodeIdToNode.get(remoteEvent.nodeId);\n    if (!domNode) {\n      console.error(\"Unknown node ID in remote event: \" + remoteEvent.nodeId);\n      return;\n    }\n\n    if (domNode instanceof this.domRunner.getWindow().Text) {\n      console.warn(\"Cannot dispatch remote event to text node\");\n      return;\n    }\n\n    this.domRunner.dispatchRemoteEventFromConnectionId(\n      connectionId,\n      domNode.realElement as Element,\n      remoteEvent,\n    );\n  }\n\n  public dispose() {\n    clearInterval(this.documentTimeIntervalTimer);\n    this.domRunner.dispose();\n  }\n\n  private getDocumentTime() {\n    return this.domRunner.getDocumentTime();\n  }\n}\n", "import { ObservableDOM } from \"@mml-io/observable-dom/src/ObservableDOM\";\nimport {\n  ADD_CONNECTED_USER_ID_MESSAGE_TYPE,\n  DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE,\n  DOM_MESSAGE_TYPE,\n  FromObservableDOMInstanceMessage,\n  ObservableDOMMessage,\n  ObservableDOMParameters,\n  REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE,\n  ToObservableDOMInstanceMessage,\n} from \"@mml-io/observable-dom-common\";\n\nimport { WebBrowserDOMRunnerFactory } from \"./WebBrowserDOMRunner\";\n\n// This runs in the iframe that will execute the document script to setup the listening for events messages\nexport function setupIframeWebRunner(argsString: string) {\n  const observableDOMParams = JSON.parse(atob(argsString)) as ObservableDOMParameters;\n\n  const sendMessageToHandler = (message: FromObservableDOMInstanceMessage) => {\n    window.parent.postMessage(JSON.stringify(message), \"*\");\n  };\n\n  const observableDOM = new ObservableDOM(\n    {\n      ...observableDOMParams,\n      htmlContents: \"\", // This must be empty as the contents are assumed to be provided by the srcdoc\n    },\n    (observableDOMMessage: ObservableDOMMessage) => {\n      sendMessageToHandler({\n        type: DOM_MESSAGE_TYPE,\n        message: observableDOMMessage,\n      });\n    },\n    WebBrowserDOMRunnerFactory,\n  );\n\n  window.addEventListener(\"message\", (e) => {\n    const parsed = JSON.parse(e.data) as ToObservableDOMInstanceMessage;\n    switch (parsed.type) {\n      case DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE:\n        observableDOM.dispatchRemoteEventFromConnectionId(parsed.connectionId, parsed.event);\n        break;\n      case ADD_CONNECTED_USER_ID_MESSAGE_TYPE:\n        observableDOM.addConnectedUserId(parsed.connectionId);\n        break;\n      case REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE:\n        observableDOM.removeConnectedUserId(parsed.connectionId);\n        break;\n      default:\n        console.error(\"Unknown message type\", parsed);\n    }\n  });\n}\n", "import { RemoteEvent } from \"@mml-io/networked-dom-protocol\";\nimport { DOMRunnerFactory, DOMRunnerInterface, DOMRunnerMessage } from \"@mml-io/observable-dom\";\n\nexport const WebBrowserDOMRunnerFactory: DOMRunnerFactory = (\n  htmlPath: string,\n  htmlContents: string,\n  params: object,\n  callback: (mutationList: DOMRunnerMessage) => void,\n): DOMRunnerInterface => {\n  return new WebBrowserDOMRunner(htmlPath, htmlContents, params, callback);\n};\n\nconst documentLoadTime = Date.now();\n\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private htmlPath: string;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(\n    htmlPath: string,\n    htmlContents: string,\n    params: object,\n    callback: (domRunnerMessage: DOMRunnerMessage) => void,\n  ) {\n    this.htmlPath = htmlPath;\n    this.callback = callback;\n\n    // Forward console messages\n    for (const level of [\"error\", \"warn\", \"info\", \"log\"] as const) {\n      const defaultFn = window.console[level];\n\n      window.console[level] = (...args) => {\n        callback({\n          logMessage: {\n            level,\n            content: args,\n          },\n        });\n        defaultFn(...args);\n      };\n    }\n\n    // Forward uncaught errors\n    window.onerror = (message, source, line, column, error) => {\n      callback({\n        logMessage: {\n          level: \"system\",\n          content: [\n            {\n              message,\n              type: error?.name,\n              line,\n              column,\n            },\n          ],\n        },\n      });\n      return false;\n    };\n\n    let didSendLoad = false;\n\n    this.mutationObserver = new window.MutationObserver((mutationList) => {\n      if (!document) {\n        return;\n      }\n      if (!didSendLoad) {\n        throw new Error(\"MutationObserver called before load\");\n      }\n      this.callback({\n        mutationList,\n      });\n    });\n\n    (window as any).params = params;\n\n    const finishLoad = () => {\n      if (didSendLoad) {\n        throw new Error(\"finishLoad called twice\");\n      }\n      didSendLoad = true;\n      this.callback({\n        loaded: true,\n      });\n      this.mutationObserver.observe(window.document, {\n        attributes: true,\n        childList: true,\n        subtree: true,\n        characterData: true,\n      });\n    };\n    if (document.body) {\n      setTimeout(finishLoad, 0);\n    } else {\n      window.addEventListener(\"DOMContentLoaded\", finishLoad);\n    }\n  }\n\n  dispatchRemoteEventFromConnectionId(\n    connectionId: number,\n    realElement: Element,\n    remoteEvent: RemoteEvent,\n  ): void {\n    const bubbles = remoteEvent.bubbles || false;\n    const remoteEventObject = new CustomEvent(remoteEvent.name, {\n      bubbles,\n      detail: { ...remoteEvent.params, connectionId },\n    });\n\n    // Dispatch the event via JavaScript.\n    realElement.dispatchEvent(remoteEventObject);\n  }\n\n  dispose(): void {\n    // TODO - handle dispose\n    console.log(\"WebBrowserDOMRunner.dispose\");\n  }\n\n  getDocument(): Document {\n    return document;\n  }\n\n  getDocumentTime(): number {\n    if (document.timeline && document.timeline.currentTime) {\n      return document.timeline.currentTime;\n    }\n    return Date.now() - documentLoadTime;\n  }\n\n  // TODO - resolve types (Window needs to expose classes such as CustomEvent as properties)\n  getWindow(): any {\n    return window;\n  }\n}\n", "import { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["ADD_CONNECTED_USER_ID_MESSAGE_TYPE", "DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE", "DOM_MESSAGE_TYPE", "REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE", "module", "args"]
}
\n';
|
|
33
33
|
|
|
34
34
|
// src/RunnerIframe.ts
|
|
35
35
|
var RunnerIframe = class {
|
|
@@ -197,42 +197,28 @@ var NetworkedDOMWebRunnerClient = class {
|
|
|
197
197
|
};
|
|
198
198
|
|
|
199
199
|
// src/IframeObservableDOMFactory.ts
|
|
200
|
+
var import_observable_dom_common = require("@mml-io/observable-dom-common");
|
|
200
201
|
var IframeObservableDOMFactory = (observableDOMParameters, callback) => {
|
|
201
202
|
const runnerIframe = new RunnerIframe(
|
|
202
203
|
observableDOMParameters,
|
|
203
|
-
(
|
|
204
|
-
|
|
205
|
-
|
|
204
|
+
(message) => {
|
|
205
|
+
switch (message.type) {
|
|
206
|
+
case import_observable_dom_common.DOM_MESSAGE_TYPE:
|
|
207
|
+
callback(message.message, remoteObservableDOM);
|
|
208
|
+
break;
|
|
209
|
+
default:
|
|
210
|
+
console.error("Unknown message type", message.type);
|
|
206
211
|
}
|
|
207
212
|
}
|
|
208
213
|
);
|
|
209
|
-
const remoteObservableDOM =
|
|
210
|
-
|
|
211
|
-
runnerIframe.sendMessageToRunner(
|
|
212
|
-
type: "addConnectedUserId",
|
|
213
|
-
connectionId
|
|
214
|
-
});
|
|
215
|
-
},
|
|
216
|
-
addIPCWebsocket() {
|
|
217
|
-
throw new Error("Not implemented");
|
|
218
|
-
},
|
|
219
|
-
dispatchRemoteEventFromConnectionId(connectionId, remoteEvent) {
|
|
220
|
-
runnerIframe.sendMessageToRunner({
|
|
221
|
-
type: "dispatchRemoteEventFromConnectionId",
|
|
222
|
-
connectionId,
|
|
223
|
-
event: remoteEvent
|
|
224
|
-
});
|
|
214
|
+
const remoteObservableDOM = (0, import_observable_dom_common.observableDOMInterfaceToMessageSender)(
|
|
215
|
+
(message) => {
|
|
216
|
+
runnerIframe.sendMessageToRunner(message);
|
|
225
217
|
},
|
|
226
|
-
|
|
218
|
+
() => {
|
|
227
219
|
runnerIframe.dispose();
|
|
228
|
-
},
|
|
229
|
-
removeConnectedUserId(connectionId) {
|
|
230
|
-
runnerIframe.sendMessageToRunner({
|
|
231
|
-
type: "removeConnectedUserId",
|
|
232
|
-
connectionId
|
|
233
|
-
});
|
|
234
220
|
}
|
|
235
|
-
|
|
221
|
+
);
|
|
236
222
|
return remoteObservableDOM;
|
|
237
223
|
};
|
|
238
224
|
// Annotate the CommonJS export names for ESM import in node:
|
package/build/index.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../src/index.ts", "../src/RunnerIframe.ts", "../src/NetworkedDOMWebRunnerClient.ts", "../src/FakeWebsocket.ts", "../src/IframeObservableDOMFactory.ts"],
|
|
4
|
-
"sourcesContent": ["export * from \"@mml-io/networked-dom-document\";\
|
|
5
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,wBAAc,2CAAd;;;;;;
|
|
4
|
+
"sourcesContent": ["export * from \"@mml-io/networked-dom-document\";\nexport * from \"./RunnerIframe\";\nexport * from \"./NetworkedDOMWebRunnerClient\";\nexport * from \"./FakeWebsocket\";\nexport * from \"./IframeObservableDOMFactory\";\n", "import {\n FromObservableDOMInstanceMessage,\n ObservableDOMParameters,\n ToObservableDOMInstanceMessage,\n} from \"@mml-io/observable-dom-common\";\n// eslint-disable-next-line @typescript-eslint/ban-ts-comment\n// @ts-ignore\n// eslint-disable-next-line import/no-unresolved\nimport runnerText from \"runner-iframe-js-text\";\n\nexport class RunnerIframe {\n private iframe: HTMLIFrameElement;\n private onMessageCallback: (message: FromObservableDOMInstanceMessage) => void;\n private postMessageListener: (messageEvent: MessageEvent) => void;\n\n constructor(\n observableDOMParameters: ObservableDOMParameters,\n onMessageCallback: (message: FromObservableDOMInstanceMessage) => void,\n ) {\n this.iframe = document.createElement(\"iframe\");\n this.iframe.setAttribute(\"sandbox\", \"allow-scripts\");\n this.iframe.style.position = \"fixed\";\n this.iframe.style.top = \"0\";\n this.iframe.style.left = \"0\";\n this.iframe.style.width = \"0\";\n this.iframe.style.height = \"0\";\n this.iframe.style.border = \"none\";\n\n const paramsMinusCode: Partial<ObservableDOMParameters> = {\n ...observableDOMParameters,\n };\n delete paramsMinusCode.htmlContents;\n\n const args = btoa(JSON.stringify(paramsMinusCode));\n\n const isJSDOM = navigator.userAgent.includes(\"jsdom\");\n if (isJSDOM) {\n // srcdoc not supported, so we have to append elements to the iframe's document\n document.body.append(this.iframe);\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n const iframeBody = this.iframe.contentWindow!.document.body;\n const argsScriptElement = document.createElement(\"script\");\n argsScriptElement.innerHTML = `window.args=\"${args}\";`;\n iframeBody.append(argsScriptElement);\n const runnerScriptElement = document.createElement(\"script\");\n runnerScriptElement.innerHTML = runnerText;\n iframeBody.append(runnerScriptElement);\n const contentHolder = document.createElement(\"div\");\n iframeBody.append(contentHolder);\n contentHolder.innerHTML = observableDOMParameters.htmlContents;\n } else {\n this.iframe.setAttribute(\n \"srcdoc\",\n `\n <script>window.args=\"${args}\";</script>\n <script>${runnerText}</script>\n ${observableDOMParameters.htmlContents}\n `,\n );\n document.body.append(this.iframe);\n }\n\n this.onMessageCallback = onMessageCallback;\n\n this.postMessageListener = (e: MessageEvent) => {\n if (e.source === this.iframe.contentWindow || (isJSDOM && e.source === null)) {\n const parsed = JSON.parse(e.data) as FromObservableDOMInstanceMessage;\n onMessageCallback(parsed);\n }\n };\n window.addEventListener(\"message\", this.postMessageListener);\n }\n\n sendMessageToRunner(message: ToObservableDOMInstanceMessage) {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n this.iframe.contentWindow!.postMessage(JSON.stringify(message), \"*\");\n }\n\n dispose() {\n window.removeEventListener(\"message\", this.postMessageListener);\n this.iframe.remove();\n }\n}\n", "import { EditableNetworkedDOM, NetworkedDOM } from \"@mml-io/networked-dom-document\";\nimport { NetworkedDOMWebsocket } from \"@mml-io/networked-dom-web\";\n\nimport { FakeWebsocket } from \"./FakeWebsocket\";\n\nexport class NetworkedDOMWebRunnerClient {\n public readonly element: HTMLDivElement;\n protected remoteDocumentHolder: HTMLElement;\n\n protected connectedState: {\n document: NetworkedDOM | EditableNetworkedDOM;\n domWebsocket: NetworkedDOMWebsocket;\n fakeWebsocket: FakeWebsocket;\n } | null = null;\n private enableEventHandling: boolean;\n\n constructor(enableEventHandling = true) {\n this.enableEventHandling = enableEventHandling;\n this.element = document.createElement(\"div\");\n this.element.style.position = \"relative\";\n this.element.style.width = \"100%\";\n this.element.style.height = \"100%\";\n\n this.remoteDocumentHolder = document.createElement(\"div\");\n this.element.append(this.remoteDocumentHolder);\n }\n\n public disconnect() {\n if (!this.connectedState) {\n return;\n }\n this.connectedState.document.removeWebSocket(\n this.connectedState.fakeWebsocket.serverSideWebsocket as unknown as WebSocket,\n );\n this.connectedState = null;\n }\n\n public dispose() {\n this.disconnect();\n this.remoteDocumentHolder.remove();\n this.element.remove();\n }\n\n public connect(\n document: NetworkedDOM | EditableNetworkedDOM,\n timeCallback?: (time: number) => void,\n ) {\n if (this.connectedState) {\n this.disconnect();\n }\n const fakeWebsocket = new FakeWebsocket(\"networked-dom-v0.1\");\n let overriddenHandler: ((element: HTMLElement, event: CustomEvent) => void) | null = null;\n const eventHandler = (element: HTMLElement, event: CustomEvent) => {\n if (!overriddenHandler) {\n throw new Error(\"overriddenHandler not set\");\n }\n overriddenHandler(element, event);\n };\n\n if (this.enableEventHandling) {\n this.remoteDocumentHolder.addEventListener(\"click\", (event: Event) => {\n eventHandler(event.target as HTMLElement, event as CustomEvent);\n event.stopPropagation();\n event.preventDefault();\n return false;\n });\n }\n\n const domWebsocket = new NetworkedDOMWebsocket(\n \"ws://localhost\",\n () => fakeWebsocket.clientSideWebsocket as unknown as WebSocket,\n this.remoteDocumentHolder,\n timeCallback,\n );\n overriddenHandler = (element: HTMLElement, event: CustomEvent) => {\n domWebsocket.handleEvent(element, event);\n };\n document.addWebSocket(fakeWebsocket.serverSideWebsocket as unknown as WebSocket);\n this.connectedState = {\n document,\n fakeWebsocket,\n domWebsocket,\n };\n }\n}\n", "class WebsocketEnd extends EventTarget {\n private readonly sendCallback: (data: string | ArrayBufferLike | Blob | ArrayBufferView) => void;\n public readonly protocol: string;\n\n constructor(\n protocol: string,\n sendCallback: (data: string | ArrayBufferLike | Blob | ArrayBufferView) => void,\n ) {\n super();\n this.protocol = protocol;\n this.sendCallback = sendCallback;\n }\n\n public close() {\n this.dispatchEvent(new CloseEvent(\"close\"));\n }\n\n public addEventListener<K extends keyof WebSocketEventMap>(\n type: K,\n listener: (this: WebSocket, ev: WebSocketEventMap[K]) => any,\n options?: boolean | AddEventListenerOptions,\n ) {\n if (type === \"open\") {\n listener.bind(this)(new Event(\"open\"));\n return;\n }\n super.addEventListener(type, listener, options);\n }\n\n public send(data: string | ArrayBufferLike | Blob | ArrayBufferView) {\n this.sendCallback(data);\n }\n}\n\nexport class FakeWebsocket {\n public clientSideWebsocket: WebsocketEnd;\n public serverSideWebsocket: WebsocketEnd;\n\n constructor(protocol: string) {\n this.clientSideWebsocket = new WebsocketEnd(protocol, (data) => {\n this.serverSideWebsocket.dispatchEvent(\n new MessageEvent(\"message\", {\n data,\n }),\n );\n });\n\n this.serverSideWebsocket = new WebsocketEnd(protocol, (data) => {\n this.clientSideWebsocket.dispatchEvent(\n new MessageEvent(\"message\", {\n data,\n }),\n );\n });\n }\n}\n", "import { ObservableDOMFactory } from \"@mml-io/networked-dom-document\";\nimport {\n DOM_MESSAGE_TYPE,\n FromObservableDOMInstanceMessage,\n ObservableDOMInterface,\n observableDOMInterfaceToMessageSender,\n ObservableDOMMessage,\n ObservableDOMParameters,\n ToObservableDOMInstanceMessage,\n} from \"@mml-io/observable-dom-common\";\n\nimport { RunnerIframe } from \"./RunnerIframe\";\n\nexport const IframeObservableDOMFactory: ObservableDOMFactory = (\n observableDOMParameters: ObservableDOMParameters,\n callback: (message: ObservableDOMMessage, observableDOM: ObservableDOMInterface) => void,\n) => {\n const runnerIframe = new RunnerIframe(\n observableDOMParameters,\n (message: FromObservableDOMInstanceMessage) => {\n switch (message.type) {\n case DOM_MESSAGE_TYPE:\n callback(message.message, remoteObservableDOM);\n break;\n default:\n console.error(\"Unknown message type\", message.type);\n }\n },\n );\n\n const remoteObservableDOM: ObservableDOMInterface = observableDOMInterfaceToMessageSender(\n (message: ToObservableDOMInstanceMessage) => {\n runnerIframe.sendMessageToRunner(message);\n },\n () => {\n runnerIframe.dispose();\n },\n );\n return remoteObservableDOM;\n};\n"],
|
|
5
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,wBAAc,2CAAd;;;;;;ACUO,IAAM,eAAN,MAAmB;AAAA,EAKxB,YACE,yBACA,mBACA;AACA,SAAK,SAAS,SAAS,cAAc,QAAQ;AAC7C,SAAK,OAAO,aAAa,WAAW,eAAe;AACnD,SAAK,OAAO,MAAM,WAAW;AAC7B,SAAK,OAAO,MAAM,MAAM;AACxB,SAAK,OAAO,MAAM,OAAO;AACzB,SAAK,OAAO,MAAM,QAAQ;AAC1B,SAAK,OAAO,MAAM,SAAS;AAC3B,SAAK,OAAO,MAAM,SAAS;AAE3B,UAAM,kBAAoD;AAAA,MACxD,GAAG;AAAA,IACL;AACA,WAAO,gBAAgB;AAEvB,UAAM,OAAO,KAAK,KAAK,UAAU,eAAe,CAAC;AAEjD,UAAM,UAAU,UAAU,UAAU,SAAS,OAAO;AACpD,QAAI,SAAS;AAEX,eAAS,KAAK,OAAO,KAAK,MAAM;AAEhC,YAAM,aAAa,KAAK,OAAO,cAAe,SAAS;AACvD,YAAM,oBAAoB,SAAS,cAAc,QAAQ;AACzD,wBAAkB,YAAY,gBAAgB;AAC9C,iBAAW,OAAO,iBAAiB;AACnC,YAAM,sBAAsB,SAAS,cAAc,QAAQ;AAC3D,0BAAoB,YAAY;AAChC,iBAAW,OAAO,mBAAmB;AACrC,YAAM,gBAAgB,SAAS,cAAc,KAAK;AAClD,iBAAW,OAAO,aAAa;AAC/B,oBAAc,YAAY,wBAAwB;AAAA,IACpD,OAAO;AACL,WAAK,OAAO;AAAA,QACV;AAAA,QACA;AAAA,6BACqB;AAAA,gBACb;AAAA,QACR,wBAAwB;AAAA;AAAA,MAE1B;AACA,eAAS,KAAK,OAAO,KAAK,MAAM;AAAA,IAClC;AAEA,SAAK,oBAAoB;AAEzB,SAAK,sBAAsB,CAAC,MAAoB;AAC9C,UAAI,EAAE,WAAW,KAAK,OAAO,iBAAkB,WAAW,EAAE,WAAW,MAAO;AAC5E,cAAM,SAAS,KAAK,MAAM,EAAE,IAAI;AAChC,0BAAkB,MAAM;AAAA,MAC1B;AAAA,IACF;AACA,WAAO,iBAAiB,WAAW,KAAK,mBAAmB;AAAA,EAC7D;AAAA,EAEA,oBAAoB,SAAyC;AAE3D,SAAK,OAAO,cAAe,YAAY,KAAK,UAAU,OAAO,GAAG,GAAG;AAAA,EACrE;AAAA,EAEA,UAAU;AACR,WAAO,oBAAoB,WAAW,KAAK,mBAAmB;AAC9D,SAAK,OAAO,OAAO;AAAA,EACrB;AACF;;;ACjFA,+BAAsC;;;ACDtC,IAAM,eAAN,cAA2B,YAAY;AAAA,EAIrC,YACE,UACA,cACA;AACA,UAAM;AACN,SAAK,WAAW;AAChB,SAAK,eAAe;AAAA,EACtB;AAAA,EAEO,QAAQ;AACb,SAAK,cAAc,IAAI,WAAW,OAAO,CAAC;AAAA,EAC5C;AAAA,EAEO,iBACL,MACA,UACA,SACA;AACA,QAAI,SAAS,QAAQ;AACnB,eAAS,KAAK,IAAI,EAAE,IAAI,MAAM,MAAM,CAAC;AACrC;AAAA,IACF;AACA,UAAM,iBAAiB,MAAM,UAAU,OAAO;AAAA,EAChD;AAAA,EAEO,KAAK,MAAyD;AACnE,SAAK,aAAa,IAAI;AAAA,EACxB;AACF;AAEO,IAAM,gBAAN,MAAoB;AAAA,EAIzB,YAAY,UAAkB;AAC5B,SAAK,sBAAsB,IAAI,aAAa,UAAU,CAAC,SAAS;AAC9D,WAAK,oBAAoB;AAAA,QACvB,IAAI,aAAa,WAAW;AAAA,UAC1B;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF,CAAC;AAED,SAAK,sBAAsB,IAAI,aAAa,UAAU,CAAC,SAAS;AAC9D,WAAK,oBAAoB;AAAA,QACvB,IAAI,aAAa,WAAW;AAAA,UAC1B;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF,CAAC;AAAA,EACH;AACF;;;ADlDO,IAAM,8BAAN,MAAkC;AAAA,EAWvC,YAAY,sBAAsB,MAAM;AAPxC,SAAU,iBAIC;AAIT,SAAK,sBAAsB;AAC3B,SAAK,UAAU,SAAS,cAAc,KAAK;AAC3C,SAAK,QAAQ,MAAM,WAAW;AAC9B,SAAK,QAAQ,MAAM,QAAQ;AAC3B,SAAK,QAAQ,MAAM,SAAS;AAE5B,SAAK,uBAAuB,SAAS,cAAc,KAAK;AACxD,SAAK,QAAQ,OAAO,KAAK,oBAAoB;AAAA,EAC/C;AAAA,EAEO,aAAa;AAClB,QAAI,CAAC,KAAK,gBAAgB;AACxB;AAAA,IACF;AACA,SAAK,eAAe,SAAS;AAAA,MAC3B,KAAK,eAAe,cAAc;AAAA,IACpC;AACA,SAAK,iBAAiB;AAAA,EACxB;AAAA,EAEO,UAAU;AACf,SAAK,WAAW;AAChB,SAAK,qBAAqB,OAAO;AACjC,SAAK,QAAQ,OAAO;AAAA,EACtB;AAAA,EAEO,QACLA,WACA,cACA;AACA,QAAI,KAAK,gBAAgB;AACvB,WAAK,WAAW;AAAA,IAClB;AACA,UAAM,gBAAgB,IAAI,cAAc,oBAAoB;AAC5D,QAAI,oBAAiF;AACrF,UAAM,eAAe,CAAC,SAAsB,UAAuB;AACjE,UAAI,CAAC,mBAAmB;AACtB,cAAM,IAAI,MAAM,2BAA2B;AAAA,MAC7C;AACA,wBAAkB,SAAS,KAAK;AAAA,IAClC;AAEA,QAAI,KAAK,qBAAqB;AAC5B,WAAK,qBAAqB,iBAAiB,SAAS,CAAC,UAAiB;AACpE,qBAAa,MAAM,QAAuB,KAAoB;AAC9D,cAAM,gBAAgB;AACtB,cAAM,eAAe;AACrB,eAAO;AAAA,MACT,CAAC;AAAA,IACH;AAEA,UAAM,eAAe,IAAI;AAAA,MACvB;AAAA,MACA,MAAM,cAAc;AAAA,MACpB,KAAK;AAAA,MACL;AAAA,IACF;AACA,wBAAoB,CAAC,SAAsB,UAAuB;AAChE,mBAAa,YAAY,SAAS,KAAK;AAAA,IACzC;AACA,IAAAA,UAAS,aAAa,cAAc,mBAA2C;AAC/E,SAAK,iBAAiB;AAAA,MACpB,UAAAA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AACF;;;AEnFA,mCAQO;AAIA,IAAM,6BAAmD,CAC9D,yBACA,aACG;AACH,QAAM,eAAe,IAAI;AAAA,IACvB;AAAA,IACA,CAAC,YAA8C;AAC7C,cAAQ,QAAQ,MAAM;AAAA,QACpB,KAAK;AACH,mBAAS,QAAQ,SAAS,mBAAmB;AAC7C;AAAA,QACF;AACE,kBAAQ,MAAM,wBAAwB,QAAQ,IAAI;AAAA,MACtD;AAAA,IACF;AAAA,EACF;AAEA,QAAM,0BAA8C;AAAA,IAClD,CAAC,YAA4C;AAC3C,mBAAa,oBAAoB,OAAO;AAAA,IAC1C;AAAA,IACA,MAAM;AACJ,mBAAa,QAAQ;AAAA,IACvB;AAAA,EACF;AACA,SAAO;AACT;",
|
|
6
6
|
"names": ["document"]
|
|
7
7
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@mml-io/networked-dom-web-runner",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"main": "./build/index.js",
|
|
5
5
|
"types": "./build/index.d.ts",
|
|
6
6
|
"files": [
|
|
@@ -18,14 +18,13 @@
|
|
|
18
18
|
"test-iterate": "jest --watch"
|
|
19
19
|
},
|
|
20
20
|
"dependencies": {
|
|
21
|
-
"@mml-io/networked-dom-web": "^0.
|
|
21
|
+
"@mml-io/networked-dom-web": "^0.2.0",
|
|
22
|
+
"@mml-io/observable-dom-common": "^0.2.0"
|
|
22
23
|
},
|
|
23
24
|
"devDependencies": {
|
|
24
25
|
"@mml-io/networked-dom-web-runner-iframe": "file:../networked-dom-web-runner-iframe",
|
|
25
|
-
"@types/three": "0.152.1",
|
|
26
26
|
"jest-canvas-mock": "2.5.1",
|
|
27
27
|
"jest-environment-jsdom": "29.5.0",
|
|
28
|
-
"jest-expect-message": "1.1.3"
|
|
29
|
-
"standardized-audio-context-mock": "9.6.24"
|
|
28
|
+
"jest-expect-message": "1.1.3"
|
|
30
29
|
}
|
|
31
30
|
}
|
|
@@ -1,53 +1,40 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { ObservableDOMFactory } from "@mml-io/networked-dom-document";
|
|
2
2
|
import {
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
DOM_MESSAGE_TYPE,
|
|
4
|
+
FromObservableDOMInstanceMessage,
|
|
5
|
+
ObservableDOMInterface,
|
|
6
|
+
observableDOMInterfaceToMessageSender,
|
|
7
|
+
ObservableDOMMessage,
|
|
5
8
|
ObservableDOMParameters,
|
|
6
|
-
|
|
9
|
+
ToObservableDOMInstanceMessage,
|
|
7
10
|
} from "@mml-io/observable-dom-common";
|
|
8
11
|
|
|
9
|
-
import { FromInstanceMessageTypes } from "./message-types";
|
|
10
12
|
import { RunnerIframe } from "./RunnerIframe";
|
|
11
13
|
|
|
12
|
-
export const IframeObservableDOMFactory:
|
|
14
|
+
export const IframeObservableDOMFactory: ObservableDOMFactory = (
|
|
13
15
|
observableDOMParameters: ObservableDOMParameters,
|
|
14
|
-
callback: (message:
|
|
16
|
+
callback: (message: ObservableDOMMessage, observableDOM: ObservableDOMInterface) => void,
|
|
15
17
|
) => {
|
|
16
18
|
const runnerIframe = new RunnerIframe(
|
|
17
19
|
observableDOMParameters,
|
|
18
|
-
(
|
|
19
|
-
|
|
20
|
-
|
|
20
|
+
(message: FromObservableDOMInstanceMessage) => {
|
|
21
|
+
switch (message.type) {
|
|
22
|
+
case DOM_MESSAGE_TYPE:
|
|
23
|
+
callback(message.message, remoteObservableDOM);
|
|
24
|
+
break;
|
|
25
|
+
default:
|
|
26
|
+
console.error("Unknown message type", message.type);
|
|
21
27
|
}
|
|
22
28
|
},
|
|
23
29
|
);
|
|
24
30
|
|
|
25
|
-
const remoteObservableDOM:
|
|
26
|
-
|
|
27
|
-
runnerIframe.sendMessageToRunner(
|
|
28
|
-
type: "addConnectedUserId",
|
|
29
|
-
connectionId,
|
|
30
|
-
});
|
|
31
|
+
const remoteObservableDOM: ObservableDOMInterface = observableDOMInterfaceToMessageSender(
|
|
32
|
+
(message: ToObservableDOMInstanceMessage) => {
|
|
33
|
+
runnerIframe.sendMessageToRunner(message);
|
|
31
34
|
},
|
|
32
|
-
|
|
33
|
-
throw new Error("Not implemented");
|
|
34
|
-
},
|
|
35
|
-
dispatchRemoteEventFromConnectionId(connectionId: number, remoteEvent: RemoteEvent): void {
|
|
36
|
-
runnerIframe.sendMessageToRunner({
|
|
37
|
-
type: "dispatchRemoteEventFromConnectionId",
|
|
38
|
-
connectionId,
|
|
39
|
-
event: remoteEvent,
|
|
40
|
-
});
|
|
41
|
-
},
|
|
42
|
-
dispose(): void {
|
|
35
|
+
() => {
|
|
43
36
|
runnerIframe.dispose();
|
|
44
37
|
},
|
|
45
|
-
|
|
46
|
-
runnerIframe.sendMessageToRunner({
|
|
47
|
-
type: "removeConnectedUserId",
|
|
48
|
-
connectionId,
|
|
49
|
-
});
|
|
50
|
-
},
|
|
51
|
-
};
|
|
38
|
+
);
|
|
52
39
|
return remoteObservableDOM;
|
|
53
40
|
};
|
package/src/RunnerIframe.ts
CHANGED
|
@@ -1,19 +1,21 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import {
|
|
2
|
+
FromObservableDOMInstanceMessage,
|
|
3
|
+
ObservableDOMParameters,
|
|
4
|
+
ToObservableDOMInstanceMessage,
|
|
5
|
+
} from "@mml-io/observable-dom-common";
|
|
2
6
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
3
7
|
// @ts-ignore
|
|
4
8
|
// eslint-disable-next-line import/no-unresolved
|
|
5
9
|
import runnerText from "runner-iframe-js-text";
|
|
6
10
|
|
|
7
|
-
import { FromInstanceMessageTypes, ToInstanceMessageTypes } from "./message-types";
|
|
8
|
-
|
|
9
11
|
export class RunnerIframe {
|
|
10
12
|
private iframe: HTMLIFrameElement;
|
|
11
|
-
private onMessageCallback: (
|
|
12
|
-
private postMessageListener: (
|
|
13
|
+
private onMessageCallback: (message: FromObservableDOMInstanceMessage) => void;
|
|
14
|
+
private postMessageListener: (messageEvent: MessageEvent) => void;
|
|
13
15
|
|
|
14
16
|
constructor(
|
|
15
17
|
observableDOMParameters: ObservableDOMParameters,
|
|
16
|
-
onMessageCallback: (
|
|
18
|
+
onMessageCallback: (message: FromObservableDOMInstanceMessage) => void,
|
|
17
19
|
) {
|
|
18
20
|
this.iframe = document.createElement("iframe");
|
|
19
21
|
this.iframe.setAttribute("sandbox", "allow-scripts");
|
|
@@ -62,14 +64,14 @@ export class RunnerIframe {
|
|
|
62
64
|
|
|
63
65
|
this.postMessageListener = (e: MessageEvent) => {
|
|
64
66
|
if (e.source === this.iframe.contentWindow || (isJSDOM && e.source === null)) {
|
|
65
|
-
const parsed = JSON.parse(e.data) as
|
|
67
|
+
const parsed = JSON.parse(e.data) as FromObservableDOMInstanceMessage;
|
|
66
68
|
onMessageCallback(parsed);
|
|
67
69
|
}
|
|
68
70
|
};
|
|
69
71
|
window.addEventListener("message", this.postMessageListener);
|
|
70
72
|
}
|
|
71
73
|
|
|
72
|
-
sendMessageToRunner(message:
|
|
74
|
+
sendMessageToRunner(message: ToObservableDOMInstanceMessage) {
|
|
73
75
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
74
76
|
this.iframe.contentWindow!.postMessage(JSON.stringify(message), "*");
|
|
75
77
|
}
|
package/src/index.ts
CHANGED
package/src/message-types.ts
DELETED
|
@@ -1,30 +0,0 @@
|
|
|
1
|
-
import { RemoteEvent } from "@mml-io/networked-dom-protocol";
|
|
2
|
-
import { ObservableDomMessage } from "@mml-io/observable-dom-common";
|
|
3
|
-
|
|
4
|
-
export type AddConnectedUserIdMessage = {
|
|
5
|
-
type: "addConnectedUserId";
|
|
6
|
-
connectionId: number;
|
|
7
|
-
};
|
|
8
|
-
|
|
9
|
-
export type RemoveConnectedUserIdMessage = {
|
|
10
|
-
type: "removeConnectedUserId";
|
|
11
|
-
connectionId: number;
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
export type DispatchRemoteEventFromConnectionIdMessage = {
|
|
15
|
-
type: "dispatchRemoteEventFromConnectionId";
|
|
16
|
-
connectionId: number;
|
|
17
|
-
event: RemoteEvent;
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
export type ToInstanceMessageTypes =
|
|
21
|
-
| AddConnectedUserIdMessage
|
|
22
|
-
| RemoveConnectedUserIdMessage
|
|
23
|
-
| DispatchRemoteEventFromConnectionIdMessage;
|
|
24
|
-
|
|
25
|
-
type DOMMessage = {
|
|
26
|
-
type: "dom";
|
|
27
|
-
message: ObservableDomMessage;
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
export type FromInstanceMessageTypes = DOMMessage;
|