@mml-io/networked-dom-web-runner 0.17.1 → 0.18.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 CHANGED
@@ -2,7 +2,7 @@
2
2
  export * from "@mml-io/networked-dom-document";
3
3
 
4
4
  // runner-iframe-js-text-namespace:/home/runner/work/mml/mml/packages/networked-dom-web-runner/networked-dom-web-runner-iframe/build/index.js
5
- var build_default = '// ../../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.loaded = false;\n this.preLoadLogMessages = [];\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.loaded = true;\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 for (const logMessage of this.preLoadLogMessages) {\n this.callback(\n {\n logMessage,\n documentTime: this.getDocumentTime()\n },\n this\n );\n }\n this.preLoadLogMessages = [];\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n if (!this.loaded) {\n this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n return;\n }\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 }\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 const targetNode = mutation.target;\n const targetElement = this.realElementToVirtualElement.get(targetNode);\n if (!targetElement) {\n throw new Error("Unknown node:" + targetNode + "," + mutation.type);\n }\n let firstNonIgnoredPreviousSibling = mutation.previousSibling;\n let insertionIndex = 0;\n while (firstNonIgnoredPreviousSibling && this.isIgnoredElement(firstNonIgnoredPreviousSibling)) {\n firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling;\n }\n let previousSiblingElement = void 0;\n if (firstNonIgnoredPreviousSibling) {\n previousSiblingElement = this.realElementToVirtualElement.get(\n firstNonIgnoredPreviousSibling\n );\n if (!previousSiblingElement) {\n throw new Error("Unknown previous sibling");\n }\n insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n if (insertionIndex === -1) {\n throw new Error("Previous sibling is not currently a child of the parent element");\n }\n insertionIndex += 1;\n }\n const toAdd = [];\n const removedNodeIds = [];\n if (mutation.type === "childList") {\n mutation.removedNodes.forEach((node) => {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n return;\n }\n const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDOMElement) {\n return;\n } else {\n const index = targetElement.childNodes.indexOf(childDOMElement);\n if (index === -1) {\n } else {\n this.removeVirtualDOMElement(childDOMElement);\n removedNodeIds.push(childDOMElement.nodeId);\n const removal = targetElement.childNodes.splice(index, 1);\n if (removal.length !== 1) {\n throw new Error("Removal length not 1");\n } else {\n if (removal[0].nodeId !== childDOMElement.nodeId) {\n throw new Error("Removal node id mismatch");\n }\n }\n }\n }\n });\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n if (asElementOrText.parentNode !== targetNode) {\n } else {\n const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n asElementOrText,\n targetElement\n );\n if (childVirtualDOMElement) {\n toAdd.push(childVirtualDOMElement);\n }\n }\n });\n targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\n } else if (mutation.type === "attributes") {\n const attributeName = mutation.attributeName;\n if (!this.isIgnoredAttribute(targetNode, attributeName)) {\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete targetElement.attributes[attributeName];\n } else {\n targetElement.attributes[attributeName] = attributeValue;\n }\n }\n } else if (mutation.type === "characterData") {\n targetElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n const addedNodes = toAdd.map(virtualDOMElementToStatic);\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: previousSiblingElement ? previousSiblingElement.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 }\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, existing] = this.createVirtualDOMElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (existing) {\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, false];\n }\n if (!node) {\n throw new Error("Cannot assign node id to null");\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n return [existingValue, true];\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, false];\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// ../../observable-dom-common/build/index.js\nvar ADD_CONNECTED_USER_ID_MESSAGE_TYPE = "addConnectedUserId";\nvar REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE = "removeConnectedUserId";\nvar DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE = "dispatchRemoteEventFromConnectionId";\nvar DOM_MESSAGE_TYPE = "dom";\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(params, callback);\n};\nvar documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n documentLoadTime = Date.now() - document.timeline.currentTime;\n}\nvar WebBrowserDOMRunner = class {\n constructor(params, callback) {\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?.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: { ...remoteEvent.params, connectionId }\n });\n const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n if (eventTypeLowerCase !== "click") {\n const handlerAttributeName = "on" + eventTypeLowerCase;\n const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n if (handlerAttributeValue) {\n try {\n const fn = Function("event", handlerAttributeValue);\n fn.apply(realElement, [remoteEventObject]);\n } catch (e) {\n console.error("Error running event handler:", e);\n }\n }\n }\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log("WebBrowserDOMRunner.dispose");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n const dateBasedDocumentTime = Date.now() - documentLoadTime;\n if (document.timeline && document.timeline.currentTime) {\n const time = document.timeline.currentTime;\n if (dateBasedDocumentTime > time + 500) {\n return dateBasedDocumentTime;\n }\n return time;\n }\n return dateBasedDocumentTime;\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 {\n ...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_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 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\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", "../../../observable-dom-common/src/messages.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  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\n/**\n * The ObservableDOM class handles the running of an HTML document using a provided DOMRunnerFactory and converting the\n * mutations that are structured as references to live DOM elements into messages that refer to elements by nodeIds.\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  private loaded = false;\n  private preLoadLogMessages: Array<LogMessage> = [];\n\n  private documentTimeIntervalTimer: NodeJS.Timeout;\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.loaded = true;\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          for (const logMessage of this.preLoadLogMessages) {\n            this.callback(\n              {\n                logMessage,\n                documentTime: this.getDocumentTime(),\n              },\n              this,\n            );\n          }\n          this.preLoadLogMessages = [];\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          if (!this.loaded) {\n            this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n            return;\n          }\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 (https://github.com/mml-io/mml/issues/100) - walk back through the records to derive the intermediate\n      //  states (e.g. if an attribute is later added to an element created in an earlier record then it should not\n      //  have that attribute when the element is added. This is important as incorrect attribute sets can affect\n      //  visibility and expected client performance.\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      const targetNode = mutation.target as Element | Text;\n      const targetElement = this.realElementToVirtualElement.get(targetNode);\n      if (!targetElement) {\n        throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n      }\n\n      let firstNonIgnoredPreviousSibling: Element | Text | null = mutation.previousSibling as\n        | Element\n        | Text;\n      let insertionIndex = 0;\n      while (\n        firstNonIgnoredPreviousSibling &&\n        this.isIgnoredElement(firstNonIgnoredPreviousSibling as Element | Text)\n      ) {\n        firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling as\n          | Element\n          | Text\n          | null;\n      }\n      let previousSiblingElement: LiveVirtualDOMElement | undefined = undefined;\n      if (firstNonIgnoredPreviousSibling) {\n        previousSiblingElement = this.realElementToVirtualElement.get(\n          firstNonIgnoredPreviousSibling as Element | Text,\n        );\n        if (!previousSiblingElement) {\n          throw new Error(\"Unknown previous sibling\");\n        }\n        insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n        if (insertionIndex === -1) {\n          throw new Error(\"Previous sibling is not currently a child of the parent element\");\n        }\n        insertionIndex += 1;\n      }\n      const toAdd: Array<LiveVirtualDOMElement> = [];\n      const removedNodeIds: Array<number> = [];\n\n      if (mutation.type === \"childList\") {\n        mutation.removedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (this.isIgnoredElement(asElementOrText)) {\n            return;\n          }\n          const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n          if (!childDOMElement) {\n            /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            return;\n          } else {\n            const index = targetElement.childNodes.indexOf(childDOMElement);\n            if (index === -1) {\n              /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            } else {\n              this.removeVirtualDOMElement(childDOMElement);\n              removedNodeIds.push(childDOMElement.nodeId);\n              const removal = targetElement.childNodes.splice(index, 1);\n              if (removal.length !== 1) {\n                throw new Error(\"Removal length not 1\");\n              } else {\n                if (removal[0].nodeId !== childDOMElement.nodeId) {\n                  throw new Error(\"Removal node id mismatch\");\n                }\n              }\n            }\n          }\n        });\n\n        mutation.addedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (asElementOrText.parentNode !== targetNode) {\n            // Ignore this addition - it is likely overridden by an earlier addition of this element to its eventual node in this mutation batch\n          } else {\n            const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n              asElementOrText,\n              targetElement,\n            );\n            if (childVirtualDOMElement) {\n              toAdd.push(childVirtualDOMElement);\n            }\n          }\n        });\n        targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\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          const attributeValue = (targetNode as Element).getAttribute(attributeName);\n          if (attributeValue === null) {\n            delete targetElement.attributes[attributeName];\n          } else {\n            targetElement.attributes[attributeName] = attributeValue;\n          }\n        }\n      } else if (mutation.type === \"characterData\") {\n        targetElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n      }\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\n      const addedNodes: Array<StaticVirtualDOMElement> = toAdd.map(virtualDOMElementToStatic);\n\n      const mutationRecord: StaticVirtualDOMMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId: previousSiblingElement ? previousSiblingElement.nodeId : 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  }\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, existing] = this.createVirtualDOMElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if (existing) {\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, boolean] {\n    if (this.isIgnoredElement(node)) {\n      return [null, false];\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      /*\n       This is undesirable, but the batching of mutations from MutationObserver means that\n       this node could be being added in a mutation after a mutation of a parent that when\n       handled resulting in adding this node early.\n      */\n      return [existingValue, true];\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, false];\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\";\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 { 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(params, callback);\n};\n\nlet documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n  documentLoadTime = Date.now() - (document.timeline.currentTime as number);\n}\n\n/**\n * WebBrowserDOMRunner is a DOMRunnerInterface implementation that runs in a web browser. It is intended to be run in\n * an iframe and the parent window is expected to send messages to it to dispatch events and to receive mutation\n * messages.\n *\n * It is expected that the document contents is injected immediately after this class is instantiated.\n */\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(params: object, callback: (domRunnerMessage: DOMRunnerMessage) => void) {\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    const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n\n    // TODO - check if there are other events that automatically wire up similarly to click->onclick and avoid those too\n    if (eventTypeLowerCase !== \"click\") {\n      const handlerAttributeName = \"on\" + eventTypeLowerCase;\n      const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n      if (handlerAttributeValue) {\n        // This event is defined as an HTML event attribute.\n        try {\n          const fn = Function(\"event\", handlerAttributeValue);\n          fn.apply(realElement, [remoteEventObject]);\n        } catch (e) {\n          console.error(\"Error running event handler:\", e);\n        }\n      }\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    const dateBasedDocumentTime = Date.now() - documentLoadTime;\n    if (document.timeline && document.timeline.currentTime) {\n      const time = document.timeline.currentTime as number;\n      if (dateBasedDocumentTime > time + 500) {\n        // The timeline can be \"left behind\" if the tab is backgrounded for a while, so we use the date-based time\n        // instead. If/when the document is brought back into the foreground, the timeline will catch up.\n        return dateBasedDocumentTime;\n      }\n      // Ideal case - use the document.timeline as it's what is available to the document script\n      return time;\n    }\n    return dateBasedDocumentTime;\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 { 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/**\n * This is run in the iframe that will execute the document script to setup the listening for events messages and\n * observing of DOM mutation using the WebBrowserDOMRunner class.\n */\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 { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["args"]
}
\n';
5
+ var build_default = '// ../../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.loaded = false;\n this.preLoadLogMessages = [];\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.loaded = true;\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 for (const logMessage of this.preLoadLogMessages) {\n this.callback(\n {\n logMessage,\n documentTime: this.getDocumentTime()\n },\n this\n );\n }\n this.preLoadLogMessages = [];\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n if (!this.loaded) {\n this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n return;\n }\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 }\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 const targetNode = mutation.target;\n const targetElement = this.realElementToVirtualElement.get(targetNode);\n if (!targetElement) {\n throw new Error("Unknown node:" + targetNode + "," + mutation.type);\n }\n let previousSiblingElement = null;\n let insertionIndex = 0;\n const toAdd = [];\n const removedNodeIds = [];\n if (mutation.type === "childList") {\n mutation.removedNodes.forEach((node) => {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n return;\n }\n const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDOMElement) {\n return;\n } else {\n const index = targetElement.childNodes.indexOf(childDOMElement);\n if (index === -1) {\n } else {\n this.removeVirtualDOMElement(childDOMElement);\n removedNodeIds.push(childDOMElement.nodeId);\n const removal = targetElement.childNodes.splice(index, 1);\n if (removal.length !== 1) {\n throw new Error("Removal length not 1");\n } else {\n if (removal[0].nodeId !== childDOMElement.nodeId) {\n throw new Error("Removal node id mismatch");\n }\n }\n }\n }\n });\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n if (asElementOrText.parentNode !== targetNode) {\n } else {\n if (!previousSiblingElement) {\n let firstNonIgnoredPreviousSibling = asElementOrText.previousSibling;\n let virtualPreviousSibling;\n while (firstNonIgnoredPreviousSibling && !virtualPreviousSibling) {\n virtualPreviousSibling = this.realElementToVirtualElement.get(\n firstNonIgnoredPreviousSibling\n );\n if (virtualPreviousSibling && targetElement.childNodes.indexOf(virtualPreviousSibling) === -1) {\n virtualPreviousSibling = void 0;\n }\n firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling;\n }\n if (virtualPreviousSibling) {\n previousSiblingElement = virtualPreviousSibling;\n insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n if (insertionIndex === -1) {\n throw new Error(\n "Previous sibling is not currently a child of the parent element"\n );\n }\n insertionIndex += 1;\n }\n }\n const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n asElementOrText,\n targetElement\n );\n if (childVirtualDOMElement) {\n toAdd.push(childVirtualDOMElement);\n }\n }\n });\n targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\n } else if (mutation.type === "attributes") {\n const attributeName = mutation.attributeName;\n if (!this.isIgnoredAttribute(targetNode, attributeName)) {\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete targetElement.attributes[attributeName];\n } else {\n targetElement.attributes[attributeName] = attributeValue;\n }\n }\n } else if (mutation.type === "characterData") {\n targetElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n const addedNodes = toAdd.map(virtualDOMElementToStatic);\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: previousSiblingElement ? previousSiblingElement.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 }\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, existing] = this.createVirtualDOMElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (existing) {\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, false];\n }\n if (!node) {\n throw new Error("Cannot assign node id to null");\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n return [existingValue, true];\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, false];\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// ../../observable-dom-common/build/index.js\nvar ADD_CONNECTED_USER_ID_MESSAGE_TYPE = "addConnectedUserId";\nvar REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE = "removeConnectedUserId";\nvar DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE = "dispatchRemoteEventFromConnectionId";\nvar DOM_MESSAGE_TYPE = "dom";\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(params, callback);\n};\nvar documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n documentLoadTime = Date.now() - document.timeline.currentTime;\n}\nvar WebBrowserDOMRunner = class {\n constructor(params, callback) {\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?.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: { ...remoteEvent.params, connectionId }\n });\n const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n if (eventTypeLowerCase !== "click") {\n const handlerAttributeName = "on" + eventTypeLowerCase;\n const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n if (handlerAttributeValue) {\n try {\n const fn = Function("event", handlerAttributeValue);\n fn.apply(realElement, [remoteEventObject]);\n } catch (e) {\n console.error("Error running event handler:", e);\n }\n }\n }\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log("WebBrowserDOMRunner.dispose");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n const dateBasedDocumentTime = Date.now() - documentLoadTime;\n if (document.timeline && document.timeline.currentTime) {\n const time = document.timeline.currentTime;\n if (dateBasedDocumentTime > time + 500) {\n return dateBasedDocumentTime;\n }\n return time;\n }\n return dateBasedDocumentTime;\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 {\n ...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_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 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\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", "../../../observable-dom-common/src/messages.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  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\n/**\n * The ObservableDOM class handles the running of an HTML document using a provided DOMRunnerFactory and converting the\n * mutations that are structured as references to live DOM elements into messages that refer to elements by nodeIds.\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  private loaded = false;\n  private preLoadLogMessages: Array<LogMessage> = [];\n\n  private documentTimeIntervalTimer: NodeJS.Timeout;\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.loaded = true;\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          for (const logMessage of this.preLoadLogMessages) {\n            this.callback(\n              {\n                logMessage,\n                documentTime: this.getDocumentTime(),\n              },\n              this,\n            );\n          }\n          this.preLoadLogMessages = [];\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          if (!this.loaded) {\n            this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n            return;\n          }\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 (https://github.com/mml-io/mml/issues/100) - walk back through the records to derive the intermediate\n      //  states (e.g. if an attribute is later added to an element created in an earlier record then it should not\n      //  have that attribute when the element is added. This is important as incorrect attribute sets can affect\n      //  visibility and expected client performance.\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      const targetNode = mutation.target as Element | Text;\n      const targetElement = this.realElementToVirtualElement.get(targetNode);\n      if (!targetElement) {\n        throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n      }\n\n      let previousSiblingElement: LiveVirtualDOMElement | null = null;\n      let insertionIndex = 0;\n      const toAdd: Array<LiveVirtualDOMElement> = [];\n      const removedNodeIds: Array<number> = [];\n\n      if (mutation.type === \"childList\") {\n        mutation.removedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (this.isIgnoredElement(asElementOrText)) {\n            return;\n          }\n          const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n          if (!childDOMElement) {\n            /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            return;\n          } else {\n            const index = targetElement.childNodes.indexOf(childDOMElement);\n            if (index === -1) {\n              /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            } else {\n              this.removeVirtualDOMElement(childDOMElement);\n              removedNodeIds.push(childDOMElement.nodeId);\n              const removal = targetElement.childNodes.splice(index, 1);\n              if (removal.length !== 1) {\n                throw new Error(\"Removal length not 1\");\n              } else {\n                if (removal[0].nodeId !== childDOMElement.nodeId) {\n                  throw new Error(\"Removal node id mismatch\");\n                }\n              }\n            }\n          }\n        });\n\n        mutation.addedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (asElementOrText.parentNode !== targetNode) {\n            // Ignore this addition - it is likely overridden by an earlier addition of this element to its eventual node in this mutation batch\n          } else {\n            if (!previousSiblingElement) {\n              /*\n               Either there is no previous element (this is the first element)\n               or the previous element has not yet been determined.\n\n               Use the current previous sibling of this added node as the first\n               choice for the previous sibling, but only use previous siblings\n               that are not ignored (are tracked as virtual elements).\n              */\n              let firstNonIgnoredPreviousSibling: Element | Text | null =\n                asElementOrText.previousSibling as Element | Text;\n              let virtualPreviousSibling: LiveVirtualDOMElement | undefined;\n              while (firstNonIgnoredPreviousSibling && !virtualPreviousSibling) {\n                virtualPreviousSibling = this.realElementToVirtualElement.get(\n                  firstNonIgnoredPreviousSibling as Element | Text,\n                );\n                if (\n                  virtualPreviousSibling &&\n                  targetElement.childNodes.indexOf(virtualPreviousSibling) === -1\n                ) {\n                  // This element is not a child of the parent element - it is not a valid previous sibling\n                  virtualPreviousSibling = undefined;\n                }\n\n                firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling as\n                  | Element\n                  | Text\n                  | null;\n              }\n\n              if (virtualPreviousSibling) {\n                previousSiblingElement = virtualPreviousSibling;\n                insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n                if (insertionIndex === -1) {\n                  throw new Error(\n                    \"Previous sibling is not currently a child of the parent element\",\n                  );\n                }\n                insertionIndex += 1;\n              }\n            }\n            const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n              asElementOrText,\n              targetElement,\n            );\n            if (childVirtualDOMElement) {\n              toAdd.push(childVirtualDOMElement);\n            }\n          }\n        });\n        targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\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          const attributeValue = (targetNode as Element).getAttribute(attributeName);\n          if (attributeValue === null) {\n            delete targetElement.attributes[attributeName];\n          } else {\n            targetElement.attributes[attributeName] = attributeValue;\n          }\n        }\n      } else if (mutation.type === \"characterData\") {\n        targetElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n      }\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\n      const addedNodes: Array<StaticVirtualDOMElement> = toAdd.map(virtualDOMElementToStatic);\n\n      const mutationRecord: StaticVirtualDOMMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId: previousSiblingElement\n          ? (previousSiblingElement as LiveVirtualDOMElement).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  }\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, existing] = this.createVirtualDOMElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if (existing) {\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, boolean] {\n    if (this.isIgnoredElement(node)) {\n      return [null, false];\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      /*\n       This is undesirable, but the batching of mutations from MutationObserver means that\n       this node could be being added in a mutation after a mutation of a parent that when\n       handled resulting in adding this node early.\n      */\n      return [existingValue, true];\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, false];\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\";\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 { 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(params, callback);\n};\n\nlet documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n  documentLoadTime = Date.now() - (document.timeline.currentTime as number);\n}\n\n/**\n * WebBrowserDOMRunner is a DOMRunnerInterface implementation that runs in a web browser. It is intended to be run in\n * an iframe and the parent window is expected to send messages to it to dispatch events and to receive mutation\n * messages.\n *\n * It is expected that the document contents is injected immediately after this class is instantiated.\n */\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(params: object, callback: (domRunnerMessage: DOMRunnerMessage) => void) {\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    const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n\n    // TODO - check if there are other events that automatically wire up similarly to click->onclick and avoid those too\n    if (eventTypeLowerCase !== \"click\") {\n      const handlerAttributeName = \"on\" + eventTypeLowerCase;\n      const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n      if (handlerAttributeValue) {\n        // This event is defined as an HTML event attribute.\n        try {\n          const fn = Function(\"event\", handlerAttributeValue);\n          fn.apply(realElement, [remoteEventObject]);\n        } catch (e) {\n          console.error(\"Error running event handler:\", e);\n        }\n      }\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    const dateBasedDocumentTime = Date.now() - documentLoadTime;\n    if (document.timeline && document.timeline.currentTime) {\n      const time = document.timeline.currentTime as number;\n      if (dateBasedDocumentTime > time + 500) {\n        // The timeline can be \"left behind\" if the tab is backgrounded for a while, so we use the date-based time\n        // instead. If/when the document is brought back into the foreground, the timeline will catch up.\n        return dateBasedDocumentTime;\n      }\n      // Ideal case - use the document.timeline as it's what is available to the document script\n      return time;\n    }\n    return dateBasedDocumentTime;\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 { 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/**\n * This is run in the iframe that will execute the document script to setup the listening for events messages and\n * observing of DOM mutation using the WebBrowserDOMRunner class.\n */\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 { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["args"]
}
\n';
6
6
 
7
7
  // src/RunnerIframe.ts
8
8
  var RunnerIframe = class {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../src/index.ts", "runner-iframe-js-text-namespace:/home/runner/work/mml/mml/packages/networked-dom-web-runner/networked-dom-web-runner-iframe/build/index.js", "../src/RunnerIframe.ts", "../src/NetworkedDOMWebRunnerClient.ts", "../src/FakeWebsocket.ts", "../src/IframeObservableDOMFactory.ts"],
4
- "sourcesContent": ["export * from \"@mml-io/networked-dom-document\";\nexport * from \"./RunnerIframe\";\nexport * from \"./NetworkedDOMWebRunnerClient\";\nexport * from \"./FakeWebsocket\";\nexport * from \"./IframeObservableDOMFactory\";\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.loaded = false;\n this.preLoadLogMessages = [];\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.loaded = true;\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 for (const logMessage of this.preLoadLogMessages) {\n this.callback(\n {\n logMessage,\n documentTime: this.getDocumentTime()\n },\n this\n );\n }\n this.preLoadLogMessages = [];\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n if (!this.loaded) {\n this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n return;\n }\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 }\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 const targetNode = mutation.target;\n const targetElement = this.realElementToVirtualElement.get(targetNode);\n if (!targetElement) {\n throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n }\n let firstNonIgnoredPreviousSibling = mutation.previousSibling;\n let insertionIndex = 0;\n while (firstNonIgnoredPreviousSibling && this.isIgnoredElement(firstNonIgnoredPreviousSibling)) {\n firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling;\n }\n let previousSiblingElement = void 0;\n if (firstNonIgnoredPreviousSibling) {\n previousSiblingElement = this.realElementToVirtualElement.get(\n firstNonIgnoredPreviousSibling\n );\n if (!previousSiblingElement) {\n throw new Error(\"Unknown previous sibling\");\n }\n insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n if (insertionIndex === -1) {\n throw new Error(\"Previous sibling is not currently a child of the parent element\");\n }\n insertionIndex += 1;\n }\n const toAdd = [];\n const removedNodeIds = [];\n if (mutation.type === \"childList\") {\n mutation.removedNodes.forEach((node) => {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n return;\n }\n const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDOMElement) {\n return;\n } else {\n const index = targetElement.childNodes.indexOf(childDOMElement);\n if (index === -1) {\n } else {\n this.removeVirtualDOMElement(childDOMElement);\n removedNodeIds.push(childDOMElement.nodeId);\n const removal = targetElement.childNodes.splice(index, 1);\n if (removal.length !== 1) {\n throw new Error(\"Removal length not 1\");\n } else {\n if (removal[0].nodeId !== childDOMElement.nodeId) {\n throw new Error(\"Removal node id mismatch\");\n }\n }\n }\n }\n });\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n if (asElementOrText.parentNode !== targetNode) {\n } else {\n const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n asElementOrText,\n targetElement\n );\n if (childVirtualDOMElement) {\n toAdd.push(childVirtualDOMElement);\n }\n }\n });\n targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\n } else if (mutation.type === \"attributes\") {\n const attributeName = mutation.attributeName;\n if (!this.isIgnoredAttribute(targetNode, attributeName)) {\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete targetElement.attributes[attributeName];\n } else {\n targetElement.attributes[attributeName] = attributeValue;\n }\n }\n } else if (mutation.type === \"characterData\") {\n targetElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n const addedNodes = toAdd.map(virtualDOMElementToStatic);\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: previousSiblingElement ? previousSiblingElement.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 }\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, existing] = this.createVirtualDOMElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (existing) {\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, false];\n }\n if (!node) {\n throw new Error(\"Cannot assign node id to null\");\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n return [existingValue, true];\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, false];\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// ../../observable-dom-common/build/index.js\nvar ADD_CONNECTED_USER_ID_MESSAGE_TYPE = \"addConnectedUserId\";\nvar REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE = \"removeConnectedUserId\";\nvar DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE = \"dispatchRemoteEventFromConnectionId\";\nvar DOM_MESSAGE_TYPE = \"dom\";\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(params, callback);\n};\nvar documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n documentLoadTime = Date.now() - document.timeline.currentTime;\n}\nvar WebBrowserDOMRunner = class {\n constructor(params, callback) {\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?.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: { ...remoteEvent.params, connectionId }\n });\n const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n if (eventTypeLowerCase !== \"click\") {\n const handlerAttributeName = \"on\" + eventTypeLowerCase;\n const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n if (handlerAttributeValue) {\n try {\n const fn = Function(\"event\", handlerAttributeValue);\n fn.apply(realElement, [remoteEventObject]);\n } catch (e) {\n console.error(\"Error running event handler:\", e);\n }\n }\n }\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log(\"WebBrowserDOMRunner.dispose\");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n const dateBasedDocumentTime = Date.now() - documentLoadTime;\n if (document.timeline && document.timeline.currentTime) {\n const time = document.timeline.currentTime;\n if (dateBasedDocumentTime > time + 500) {\n return dateBasedDocumentTime;\n }\n return time;\n }\n return dateBasedDocumentTime;\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 {\n ...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_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 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\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", "../../../observable-dom-common/src/messages.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  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\n/**\n * The ObservableDOM class handles the running of an HTML document using a provided DOMRunnerFactory and converting the\n * mutations that are structured as references to live DOM elements into messages that refer to elements by nodeIds.\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  private loaded = false;\n  private preLoadLogMessages: Array<LogMessage> = [];\n\n  private documentTimeIntervalTimer: NodeJS.Timeout;\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.loaded = true;\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          for (const logMessage of this.preLoadLogMessages) {\n            this.callback(\n              {\n                logMessage,\n                documentTime: this.getDocumentTime(),\n              },\n              this,\n            );\n          }\n          this.preLoadLogMessages = [];\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          if (!this.loaded) {\n            this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n            return;\n          }\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 (https://github.com/mml-io/mml/issues/100) - walk back through the records to derive the intermediate\n      //  states (e.g. if an attribute is later added to an element created in an earlier record then it should not\n      //  have that attribute when the element is added. This is important as incorrect attribute sets can affect\n      //  visibility and expected client performance.\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      const targetNode = mutation.target as Element | Text;\n      const targetElement = this.realElementToVirtualElement.get(targetNode);\n      if (!targetElement) {\n        throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n      }\n\n      let firstNonIgnoredPreviousSibling: Element | Text | null = mutation.previousSibling as\n        | Element\n        | Text;\n      let insertionIndex = 0;\n      while (\n        firstNonIgnoredPreviousSibling &&\n        this.isIgnoredElement(firstNonIgnoredPreviousSibling as Element | Text)\n      ) {\n        firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling as\n          | Element\n          | Text\n          | null;\n      }\n      let previousSiblingElement: LiveVirtualDOMElement | undefined = undefined;\n      if (firstNonIgnoredPreviousSibling) {\n        previousSiblingElement = this.realElementToVirtualElement.get(\n          firstNonIgnoredPreviousSibling as Element | Text,\n        );\n        if (!previousSiblingElement) {\n          throw new Error(\"Unknown previous sibling\");\n        }\n        insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n        if (insertionIndex === -1) {\n          throw new Error(\"Previous sibling is not currently a child of the parent element\");\n        }\n        insertionIndex += 1;\n      }\n      const toAdd: Array<LiveVirtualDOMElement> = [];\n      const removedNodeIds: Array<number> = [];\n\n      if (mutation.type === \"childList\") {\n        mutation.removedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (this.isIgnoredElement(asElementOrText)) {\n            return;\n          }\n          const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n          if (!childDOMElement) {\n            /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            return;\n          } else {\n            const index = targetElement.childNodes.indexOf(childDOMElement);\n            if (index === -1) {\n              /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            } else {\n              this.removeVirtualDOMElement(childDOMElement);\n              removedNodeIds.push(childDOMElement.nodeId);\n              const removal = targetElement.childNodes.splice(index, 1);\n              if (removal.length !== 1) {\n                throw new Error(\"Removal length not 1\");\n              } else {\n                if (removal[0].nodeId !== childDOMElement.nodeId) {\n                  throw new Error(\"Removal node id mismatch\");\n                }\n              }\n            }\n          }\n        });\n\n        mutation.addedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (asElementOrText.parentNode !== targetNode) {\n            // Ignore this addition - it is likely overridden by an earlier addition of this element to its eventual node in this mutation batch\n          } else {\n            const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n              asElementOrText,\n              targetElement,\n            );\n            if (childVirtualDOMElement) {\n              toAdd.push(childVirtualDOMElement);\n            }\n          }\n        });\n        targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\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          const attributeValue = (targetNode as Element).getAttribute(attributeName);\n          if (attributeValue === null) {\n            delete targetElement.attributes[attributeName];\n          } else {\n            targetElement.attributes[attributeName] = attributeValue;\n          }\n        }\n      } else if (mutation.type === \"characterData\") {\n        targetElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n      }\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\n      const addedNodes: Array<StaticVirtualDOMElement> = toAdd.map(virtualDOMElementToStatic);\n\n      const mutationRecord: StaticVirtualDOMMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId: previousSiblingElement ? previousSiblingElement.nodeId : 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  }\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, existing] = this.createVirtualDOMElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if (existing) {\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, boolean] {\n    if (this.isIgnoredElement(node)) {\n      return [null, false];\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      /*\n       This is undesirable, but the batching of mutations from MutationObserver means that\n       this node could be being added in a mutation after a mutation of a parent that when\n       handled resulting in adding this node early.\n      */\n      return [existingValue, true];\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, false];\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\";\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 { 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(params, callback);\n};\n\nlet documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n  documentLoadTime = Date.now() - (document.timeline.currentTime as number);\n}\n\n/**\n * WebBrowserDOMRunner is a DOMRunnerInterface implementation that runs in a web browser. It is intended to be run in\n * an iframe and the parent window is expected to send messages to it to dispatch events and to receive mutation\n * messages.\n *\n * It is expected that the document contents is injected immediately after this class is instantiated.\n */\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(params: object, callback: (domRunnerMessage: DOMRunnerMessage) => void) {\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    const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n\n    // TODO - check if there are other events that automatically wire up similarly to click->onclick and avoid those too\n    if (eventTypeLowerCase !== \"click\") {\n      const handlerAttributeName = \"on\" + eventTypeLowerCase;\n      const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n      if (handlerAttributeValue) {\n        // This event is defined as an HTML event attribute.\n        try {\n          const fn = Function(\"event\", handlerAttributeValue);\n          fn.apply(realElement, [remoteEventObject]);\n        } catch (e) {\n          console.error(\"Error running event handler:\", e);\n        }\n      }\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    const dateBasedDocumentTime = Date.now() - documentLoadTime;\n    if (document.timeline && document.timeline.currentTime) {\n      const time = document.timeline.currentTime as number;\n      if (dateBasedDocumentTime > time + 500) {\n        // The timeline can be \"left behind\" if the tab is backgrounded for a while, so we use the date-based time\n        // instead. If/when the document is brought back into the foreground, the timeline will catch up.\n        return dateBasedDocumentTime;\n      }\n      // Ideal case - use the document.timeline as it's what is available to the document script\n      return time;\n    }\n    return dateBasedDocumentTime;\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 { 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/**\n * This is run in the iframe that will execute the document script to setup the listening for events messages and\n * observing of DOM mutation using the WebBrowserDOMRunner class.\n */\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 { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["args"]
}
\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\n/**\n * RunnerIframe is a class that creates an iframe that includes the networked-dom-web-runner-iframe package in it and\n * injects the provided HTML into it. This class then communicates with the iframe using postMessage.\n */\nexport class RunnerIframe {\n private iframe: HTMLIFrameElement;\n private postMessageListener: (messageEvent: MessageEvent) => void;\n\n constructor(\n private observableDOMParameters: ObservableDOMParameters,\n private 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 ...this.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.postMessageListener = (e: MessageEvent) => {\n if (e.source === this.iframe.contentWindow || (isJSDOM && e.source === null)) {\n const parsed = JSON.parse(e.data) as FromObservableDOMInstanceMessage;\n this.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\n/**\n * The NetworkedDOMWebRunnerClient class can be used to view and interact with a NetworkedDOM document instance that is\n * available directly in the browser (rather than exposed over the network). This is useful for usage modes where the\n * document does not need to be available to other clients, such as a single-user or an edit/preview mode.\n *\n * The class takes arguments for where the view of the document should be synchronized to in the DOM, and which window\n * instance to use to create any other elements (to allow for using iframes to isolate the document from the rest of\n * the page).\n */\nexport class NetworkedDOMWebRunnerClient {\n public readonly element: HTMLElement;\n\n public connectedState: {\n document: NetworkedDOM | EditableNetworkedDOM;\n domWebsocket: NetworkedDOMWebsocket;\n fakeWebsocket: FakeWebsocket;\n } | null = null;\n private enableEventHandling: boolean;\n\n constructor(enableEventHandling = true, element?: HTMLElement) {\n this.enableEventHandling = enableEventHandling;\n this.element = element || document.createElement(\"div\");\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.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.element.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.element,\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", "/**\n * WebsocketEnd is one end of a FakeWebsocket connection. It is used to simulate a websocket connection for testing or\n * matching the interface of a real websocket connection without doing any actual networking.\n */\nclass 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\n/**\n * FakeWebsocket is a pair of WebsocketEnds that are connected to each other. It is used to simulate a websocket\n * connection for testing or matching the interface of a real websocket connection without doing any actual networking.\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\n/**\n * Creates an ObservableDOMInterface that uses an iframe to run the document.\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"],
4
+ "sourcesContent": ["export * from \"@mml-io/networked-dom-document\";\nexport * from \"./RunnerIframe\";\nexport * from \"./NetworkedDOMWebRunnerClient\";\nexport * from \"./FakeWebsocket\";\nexport * from \"./IframeObservableDOMFactory\";\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.loaded = false;\n this.preLoadLogMessages = [];\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.loaded = true;\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 for (const logMessage of this.preLoadLogMessages) {\n this.callback(\n {\n logMessage,\n documentTime: this.getDocumentTime()\n },\n this\n );\n }\n this.preLoadLogMessages = [];\n } else if (domRunnerMessage.mutationList) {\n this.processModificationList(domRunnerMessage.mutationList);\n } else if (domRunnerMessage.logMessage) {\n if (!this.loaded) {\n this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n return;\n }\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 }\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 const targetNode = mutation.target;\n const targetElement = this.realElementToVirtualElement.get(targetNode);\n if (!targetElement) {\n throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n }\n let previousSiblingElement = null;\n let insertionIndex = 0;\n const toAdd = [];\n const removedNodeIds = [];\n if (mutation.type === \"childList\") {\n mutation.removedNodes.forEach((node) => {\n const asElementOrText = node;\n if (this.isIgnoredElement(asElementOrText)) {\n return;\n }\n const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n if (!childDOMElement) {\n return;\n } else {\n const index = targetElement.childNodes.indexOf(childDOMElement);\n if (index === -1) {\n } else {\n this.removeVirtualDOMElement(childDOMElement);\n removedNodeIds.push(childDOMElement.nodeId);\n const removal = targetElement.childNodes.splice(index, 1);\n if (removal.length !== 1) {\n throw new Error(\"Removal length not 1\");\n } else {\n if (removal[0].nodeId !== childDOMElement.nodeId) {\n throw new Error(\"Removal node id mismatch\");\n }\n }\n }\n }\n });\n mutation.addedNodes.forEach((node) => {\n const asElementOrText = node;\n if (asElementOrText.parentNode !== targetNode) {\n } else {\n if (!previousSiblingElement) {\n let firstNonIgnoredPreviousSibling = asElementOrText.previousSibling;\n let virtualPreviousSibling;\n while (firstNonIgnoredPreviousSibling && !virtualPreviousSibling) {\n virtualPreviousSibling = this.realElementToVirtualElement.get(\n firstNonIgnoredPreviousSibling\n );\n if (virtualPreviousSibling && targetElement.childNodes.indexOf(virtualPreviousSibling) === -1) {\n virtualPreviousSibling = void 0;\n }\n firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling;\n }\n if (virtualPreviousSibling) {\n previousSiblingElement = virtualPreviousSibling;\n insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n if (insertionIndex === -1) {\n throw new Error(\n \"Previous sibling is not currently a child of the parent element\"\n );\n }\n insertionIndex += 1;\n }\n }\n const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n asElementOrText,\n targetElement\n );\n if (childVirtualDOMElement) {\n toAdd.push(childVirtualDOMElement);\n }\n }\n });\n targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\n } else if (mutation.type === \"attributes\") {\n const attributeName = mutation.attributeName;\n if (!this.isIgnoredAttribute(targetNode, attributeName)) {\n const attributeValue = targetNode.getAttribute(attributeName);\n if (attributeValue === null) {\n delete targetElement.attributes[attributeName];\n } else {\n targetElement.attributes[attributeName] = attributeValue;\n }\n }\n } else if (mutation.type === \"characterData\") {\n targetElement.textContent = targetNode.textContent ? targetNode.textContent : void 0;\n }\n const addedNodes = toAdd.map(virtualDOMElementToStatic);\n const mutationRecord = {\n type: mutation.type,\n targetId: targetElement.nodeId,\n addedNodes,\n removedNodeIds,\n previousSiblingId: previousSiblingElement ? previousSiblingElement.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 }\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, existing] = this.createVirtualDOMElement(node, parent);\n if (!virtualElement) {\n return null;\n }\n if (existing) {\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, false];\n }\n if (!node) {\n throw new Error(\"Cannot assign node id to null\");\n }\n const existingValue = this.realElementToVirtualElement.get(node);\n if (existingValue !== void 0) {\n return [existingValue, true];\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, false];\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// ../../observable-dom-common/build/index.js\nvar ADD_CONNECTED_USER_ID_MESSAGE_TYPE = \"addConnectedUserId\";\nvar REMOVE_CONNECTED_USER_ID_MESSAGE_TYPE = \"removeConnectedUserId\";\nvar DISPATCH_REMOTE_EVENT_FROM_CONNECTION_ID_MESSAGE_TYPE = \"dispatchRemoteEventFromConnectionId\";\nvar DOM_MESSAGE_TYPE = \"dom\";\n\n// src/WebBrowserDOMRunner.ts\nvar WebBrowserDOMRunnerFactory = (htmlPath, htmlContents, params, callback) => {\n return new WebBrowserDOMRunner(params, callback);\n};\nvar documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n documentLoadTime = Date.now() - document.timeline.currentTime;\n}\nvar WebBrowserDOMRunner = class {\n constructor(params, callback) {\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?.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: { ...remoteEvent.params, connectionId }\n });\n const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n if (eventTypeLowerCase !== \"click\") {\n const handlerAttributeName = \"on\" + eventTypeLowerCase;\n const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n if (handlerAttributeValue) {\n try {\n const fn = Function(\"event\", handlerAttributeValue);\n fn.apply(realElement, [remoteEventObject]);\n } catch (e) {\n console.error(\"Error running event handler:\", e);\n }\n }\n }\n realElement.dispatchEvent(remoteEventObject);\n }\n dispose() {\n console.log(\"WebBrowserDOMRunner.dispose\");\n }\n getDocument() {\n return document;\n }\n getDocumentTime() {\n const dateBasedDocumentTime = Date.now() - documentLoadTime;\n if (document.timeline && document.timeline.currentTime) {\n const time = document.timeline.currentTime;\n if (dateBasedDocumentTime > time + 500) {\n return dateBasedDocumentTime;\n }\n return time;\n }\n return dateBasedDocumentTime;\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 {\n ...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_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 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\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", "../../../observable-dom-common/src/messages.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  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\n/**\n * The ObservableDOM class handles the running of an HTML document using a provided DOMRunnerFactory and converting the\n * mutations that are structured as references to live DOM elements into messages that refer to elements by nodeIds.\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  private loaded = false;\n  private preLoadLogMessages: Array<LogMessage> = [];\n\n  private documentTimeIntervalTimer: NodeJS.Timeout;\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.loaded = true;\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          for (const logMessage of this.preLoadLogMessages) {\n            this.callback(\n              {\n                logMessage,\n                documentTime: this.getDocumentTime(),\n              },\n              this,\n            );\n          }\n          this.preLoadLogMessages = [];\n        } else if (domRunnerMessage.mutationList) {\n          this.processModificationList(domRunnerMessage.mutationList);\n        } else if (domRunnerMessage.logMessage) {\n          if (!this.loaded) {\n            this.preLoadLogMessages.push(domRunnerMessage.logMessage);\n            return;\n          }\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 (https://github.com/mml-io/mml/issues/100) - walk back through the records to derive the intermediate\n      //  states (e.g. if an attribute is later added to an element created in an earlier record then it should not\n      //  have that attribute when the element is added. This is important as incorrect attribute sets can affect\n      //  visibility and expected client performance.\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      const targetNode = mutation.target as Element | Text;\n      const targetElement = this.realElementToVirtualElement.get(targetNode);\n      if (!targetElement) {\n        throw new Error(\"Unknown node:\" + targetNode + \",\" + mutation.type);\n      }\n\n      let previousSiblingElement: LiveVirtualDOMElement | null = null;\n      let insertionIndex = 0;\n      const toAdd: Array<LiveVirtualDOMElement> = [];\n      const removedNodeIds: Array<number> = [];\n\n      if (mutation.type === \"childList\") {\n        mutation.removedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (this.isIgnoredElement(asElementOrText)) {\n            return;\n          }\n          const childDOMElement = this.realElementToVirtualElement.get(asElementOrText);\n          if (!childDOMElement) {\n            /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            return;\n          } else {\n            const index = targetElement.childNodes.indexOf(childDOMElement);\n            if (index === -1) {\n              /*\n             This can happen if element was a child of a parent element, but was moved to a new parent in the same batch of mutations.\n             We can ignore this removal as the element will be in the correct place in the hierarchy already.\n            */\n            } else {\n              this.removeVirtualDOMElement(childDOMElement);\n              removedNodeIds.push(childDOMElement.nodeId);\n              const removal = targetElement.childNodes.splice(index, 1);\n              if (removal.length !== 1) {\n                throw new Error(\"Removal length not 1\");\n              } else {\n                if (removal[0].nodeId !== childDOMElement.nodeId) {\n                  throw new Error(\"Removal node id mismatch\");\n                }\n              }\n            }\n          }\n        });\n\n        mutation.addedNodes.forEach((node: Node) => {\n          const asElementOrText = node as Element | Text;\n          if (asElementOrText.parentNode !== targetNode) {\n            // Ignore this addition - it is likely overridden by an earlier addition of this element to its eventual node in this mutation batch\n          } else {\n            if (!previousSiblingElement) {\n              /*\n               Either there is no previous element (this is the first element)\n               or the previous element has not yet been determined.\n\n               Use the current previous sibling of this added node as the first\n               choice for the previous sibling, but only use previous siblings\n               that are not ignored (are tracked as virtual elements).\n              */\n              let firstNonIgnoredPreviousSibling: Element | Text | null =\n                asElementOrText.previousSibling as Element | Text;\n              let virtualPreviousSibling: LiveVirtualDOMElement | undefined;\n              while (firstNonIgnoredPreviousSibling && !virtualPreviousSibling) {\n                virtualPreviousSibling = this.realElementToVirtualElement.get(\n                  firstNonIgnoredPreviousSibling as Element | Text,\n                );\n                if (\n                  virtualPreviousSibling &&\n                  targetElement.childNodes.indexOf(virtualPreviousSibling) === -1\n                ) {\n                  // This element is not a child of the parent element - it is not a valid previous sibling\n                  virtualPreviousSibling = undefined;\n                }\n\n                firstNonIgnoredPreviousSibling = firstNonIgnoredPreviousSibling.previousSibling as\n                  | Element\n                  | Text\n                  | null;\n              }\n\n              if (virtualPreviousSibling) {\n                previousSiblingElement = virtualPreviousSibling;\n                insertionIndex = targetElement.childNodes.indexOf(previousSiblingElement);\n                if (insertionIndex === -1) {\n                  throw new Error(\n                    \"Previous sibling is not currently a child of the parent element\",\n                  );\n                }\n                insertionIndex += 1;\n              }\n            }\n            const childVirtualDOMElement = this.createVirtualDOMElementWithChildren(\n              asElementOrText,\n              targetElement,\n            );\n            if (childVirtualDOMElement) {\n              toAdd.push(childVirtualDOMElement);\n            }\n          }\n        });\n        targetElement.childNodes.splice(insertionIndex, 0, ...toAdd);\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          const attributeValue = (targetNode as Element).getAttribute(attributeName);\n          if (attributeValue === null) {\n            delete targetElement.attributes[attributeName];\n          } else {\n            targetElement.attributes[attributeName] = attributeValue;\n          }\n        }\n      } else if (mutation.type === \"characterData\") {\n        targetElement.textContent = targetNode.textContent ? targetNode.textContent : undefined;\n      }\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\n      const addedNodes: Array<StaticVirtualDOMElement> = toAdd.map(virtualDOMElementToStatic);\n\n      const mutationRecord: StaticVirtualDOMMutationIdsRecord = {\n        type: mutation.type,\n        targetId: targetElement.nodeId,\n        addedNodes,\n        removedNodeIds,\n        previousSiblingId: previousSiblingElement\n          ? (previousSiblingElement as LiveVirtualDOMElement).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  }\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, existing] = this.createVirtualDOMElement(node, parent);\n    if (!virtualElement) {\n      return null;\n    }\n    if (existing) {\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, boolean] {\n    if (this.isIgnoredElement(node)) {\n      return [null, false];\n    }\n    if (!node) {\n      throw new Error(\"Cannot assign node id to null\");\n    }\n\n    const existingValue = this.realElementToVirtualElement.get(node);\n    if (existingValue !== undefined) {\n      /*\n       This is undesirable, but the batching of mutations from MutationObserver means that\n       this node could be being added in a mutation after a mutation of a parent that when\n       handled resulting in adding this node early.\n      */\n      return [existingValue, true];\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, false];\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\";\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 { 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(params, callback);\n};\n\nlet documentLoadTime = Date.now();\nif (document.timeline && document.timeline.currentTime) {\n  documentLoadTime = Date.now() - (document.timeline.currentTime as number);\n}\n\n/**\n * WebBrowserDOMRunner is a DOMRunnerInterface implementation that runs in a web browser. It is intended to be run in\n * an iframe and the parent window is expected to send messages to it to dispatch events and to receive mutation\n * messages.\n *\n * It is expected that the document contents is injected immediately after this class is instantiated.\n */\nexport class WebBrowserDOMRunner implements DOMRunnerInterface {\n  private mutationObserver: MutationObserver;\n  private callback: (domRunnerMessage: DOMRunnerMessage) => void;\n\n  constructor(params: object, callback: (domRunnerMessage: DOMRunnerMessage) => void) {\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    const eventTypeLowerCase = remoteEvent.name.toLowerCase();\n\n    // TODO - check if there are other events that automatically wire up similarly to click->onclick and avoid those too\n    if (eventTypeLowerCase !== \"click\") {\n      const handlerAttributeName = \"on\" + eventTypeLowerCase;\n      const handlerAttributeValue = realElement.getAttribute(handlerAttributeName);\n      if (handlerAttributeValue) {\n        // This event is defined as an HTML event attribute.\n        try {\n          const fn = Function(\"event\", handlerAttributeValue);\n          fn.apply(realElement, [remoteEventObject]);\n        } catch (e) {\n          console.error(\"Error running event handler:\", e);\n        }\n      }\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    const dateBasedDocumentTime = Date.now() - documentLoadTime;\n    if (document.timeline && document.timeline.currentTime) {\n      const time = document.timeline.currentTime as number;\n      if (dateBasedDocumentTime > time + 500) {\n        // The timeline can be \"left behind\" if the tab is backgrounded for a while, so we use the date-based time\n        // instead. If/when the document is brought back into the foreground, the timeline will catch up.\n        return dateBasedDocumentTime;\n      }\n      // Ideal case - use the document.timeline as it's what is available to the document script\n      return time;\n    }\n    return dateBasedDocumentTime;\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 { 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/**\n * This is run in the iframe that will execute the document script to setup the listening for events messages and\n * observing of DOM mutation using the WebBrowserDOMRunner class.\n */\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 { setupIframeWebRunner } from \"./IframeWebRunner\";\n\nconst args = (window as any).args;\nsetupIframeWebRunner(args);\n"],
  "mappings": "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",
  "names": ["args"]
}
\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\n/**\n * RunnerIframe is a class that creates an iframe that includes the networked-dom-web-runner-iframe package in it and\n * injects the provided HTML into it. This class then communicates with the iframe using postMessage.\n */\nexport class RunnerIframe {\n private iframe: HTMLIFrameElement;\n private postMessageListener: (messageEvent: MessageEvent) => void;\n\n constructor(\n private observableDOMParameters: ObservableDOMParameters,\n private 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 ...this.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.postMessageListener = (e: MessageEvent) => {\n if (e.source === this.iframe.contentWindow || (isJSDOM && e.source === null)) {\n const parsed = JSON.parse(e.data) as FromObservableDOMInstanceMessage;\n this.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\n/**\n * The NetworkedDOMWebRunnerClient class can be used to view and interact with a NetworkedDOM document instance that is\n * available directly in the browser (rather than exposed over the network). This is useful for usage modes where the\n * document does not need to be available to other clients, such as a single-user or an edit/preview mode.\n *\n * The class takes arguments for where the view of the document should be synchronized to in the DOM, and which window\n * instance to use to create any other elements (to allow for using iframes to isolate the document from the rest of\n * the page).\n */\nexport class NetworkedDOMWebRunnerClient {\n public readonly element: HTMLElement;\n\n public connectedState: {\n document: NetworkedDOM | EditableNetworkedDOM;\n domWebsocket: NetworkedDOMWebsocket;\n fakeWebsocket: FakeWebsocket;\n } | null = null;\n private enableEventHandling: boolean;\n\n constructor(enableEventHandling = true, element?: HTMLElement) {\n this.enableEventHandling = enableEventHandling;\n this.element = element || document.createElement(\"div\");\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.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.element.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.element,\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", "/**\n * WebsocketEnd is one end of a FakeWebsocket connection. It is used to simulate a websocket connection for testing or\n * matching the interface of a real websocket connection without doing any actual networking.\n */\nclass 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\n/**\n * FakeWebsocket is a pair of WebsocketEnds that are connected to each other. It is used to simulate a websocket\n * connection for testing or matching the interface of a real websocket connection without doing any actual networking.\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\n/**\n * Creates an ObservableDOMInterface that uses an iframe to run the document.\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
5
  "mappings": ";AAAA,cAAc;;;ACAd;;;ACcO,IAAM,eAAN,MAAmB;AAAA,EAIxB,YACU,yBACA,mBACR;AAFQ;AACA;AAER,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,KAAK;AAAA,IACV;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,IAAI;AAClD,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,IAAI;AAAA,gBACjB,aAAU;AAAA,QAClB,wBAAwB,YAAY;AAAA;AAAA,MAEtC;AACA,eAAS,KAAK,OAAO,KAAK,MAAM;AAAA,IAClC;AAEA,SAAK,sBAAsB,CAAC,MAAoB;AAC9C,UAAI,EAAE,WAAW,KAAK,OAAO,iBAAkB,WAAW,EAAE,WAAW,MAAO;AAC5E,cAAM,SAAS,KAAK,MAAM,EAAE,IAAI;AAChC,aAAK,kBAAkB,MAAM;AAAA,MAC/B;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;;;AClFA,SAAS,6BAA6B;;;ACGtC,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;AAMO,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;;;ADjDO,IAAM,8BAAN,MAAkC;AAAA,EAUvC,YAAY,sBAAsB,MAAM,SAAuB;AAP/D,SAAO,iBAII;AAIT,SAAK,sBAAsB;AAC3B,SAAK,UAAU,WAAW,SAAS,cAAc,KAAK;AAAA,EACxD;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,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,QAAQ,iBAAiB,SAAS,CAAC,UAAiB;AACvD,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;;;AEpFA;AAAA,EACE;AAAA,EAGA;AAAA,OAIK;AAOA,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,sBAA8C;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.17.1",
3
+ "version": "0.18.0",
4
4
  "publishConfig": {
5
5
  "access": "public"
6
6
  },
@@ -20,8 +20,8 @@
20
20
  "test-iterate": "cross-env NODE_OPTIONS=--experimental-vm-modules jest --watch"
21
21
  },
22
22
  "dependencies": {
23
- "@mml-io/networked-dom-web": "^0.17.1",
24
- "@mml-io/observable-dom-common": "^0.17.1"
23
+ "@mml-io/networked-dom-web": "^0.18.0",
24
+ "@mml-io/observable-dom-common": "^0.18.0"
25
25
  },
26
26
  "devDependencies": {
27
27
  "@mml-io/networked-dom-web-runner-iframe": "file:../networked-dom-web-runner-iframe",
@@ -29,5 +29,5 @@
29
29
  "jest-environment-jsdom": "29.7.0",
30
30
  "jest-expect-message": "1.1.3"
31
31
  },
32
- "gitHead": "d1069c5541fadd1ae78700dbbd94d55a481d373a"
32
+ "gitHead": "b1a31e364d69287681b0afeecdca2b4bc576e3d3"
33
33
  }