// ==UserScript==
// @name watcher
// @version 0.0.1
// @description Watch for added and removed elements and changes to attributes or text content
// ==/UserScript==
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
typeof define === 'function' && define.amd ? define(factory) :
(global.Watcher = factory());
}(this, (function () { 'use strict';
// ----------------------------------------------------
var Css;
(function (Css) {
Css.Inverse = 'color: white; background: black';
Css.Error = 'font-weight: bold; color: #f4f';
Css.Link = 'color: #05f; font-weight: normal; text-decoration: underline';
Css.Bold = 'font-weight: bold';
Css.Blue = 'color: #05f';
Css.Kw = 'color: #35b; font-weight: bold; font-style: normal; text-decoration: none';
Css.Attr = 'color: #563; font-weight: normal; font-style: italic; text-decoration: none';
Css.Val = 'color: #c36; font-weight: normal; font-style: normal; text-decoration: none';
})(Css || (Css = {}));
//# sourceMappingURL=interfaces.js.map
// ----------------------------------------------------------
function _log(collapsed, title, objs) {
if (collapsed) {
console.groupCollapsed(`%c${title}`, Css.Kw);
}
else {
console.group(`%c${title}`, Css.Kw);
}
for (const obj of objs) {
console.dir(obj);
}
console.groupEnd();
}
function log(title, ...objs) {
_log(false, title, objs);
}
function logc(title, ...objs) {
_log(true, title, objs);
}
//# sourceMappingURL=l.js.map
// ----------------------------------------------------
// ----------------------------------------------------
var WatchEvents;
(function (WatchEvents) {
WatchEvents[WatchEvents["ElementsAdded"] = 1] = "ElementsAdded";
WatchEvents[WatchEvents["ElementsRemoved"] = 2] = "ElementsRemoved";
WatchEvents[WatchEvents["AttributesChanged"] = 4] = "AttributesChanged";
WatchEvents[WatchEvents["TextChanged"] = 8] = "TextChanged";
WatchEvents[WatchEvents["ElementsChanged"] = 3] = "ElementsChanged";
WatchEvents[WatchEvents["AllChanges"] = 15] = "AllChanges";
})(WatchEvents || (WatchEvents = {}));
//# sourceMappingURL=watch-options.js.map
// ----------------------------------------------------
class AttributeChange {
constructor(element, name, value, oldValue) {
this.element = element;
this.name = name;
this.value = value;
this.oldValue = oldValue;
}
}
// ----------------------------------------------------
class TextChange {
constructor(element, value, oldValue) {
this.element = element;
this.value = value;
this.oldValue = oldValue;
}
}
// ----------------------------------------------------
class WatchResult {
constructor(parent, added, removed = [], attributeChanges = [], textChanges = []) {
this.parent = parent;
this.added = added;
this.removed = removed;
this.attributeChanges = attributeChanges;
this.textChanges = textChanges;
}
}
//# sourceMappingURL=watch-result.js.map
class ElementSet extends Set {
// get [Symbol.toStringTag]: string () {
// return 'ElementSet'
// }
// ----------------------------------------------------
addAll(elements) {
for (const element of elements) {
super.add(element);
}
return this;
}
// ----------------------------------------------------
toArray() {
return Array.from(this);
}
}
//# sourceMappingURL=element-set.js.map
// ----------------------------------------------------
class Matcher {
constructor(root, selector = '*') {
this.root = root;
this.selector = selector;
}
// ----------------------------------------------------
matchesElement(element) {
return element.matches(this.selector);
}
// ----------------------------------------------------
findAllMatchesInSubTree(element) {
const matches = Array.from(element.querySelectorAll(this.selector));
if (this.matchesElement(element)) {
matches.unshift(element);
}
return matches;
}
}
//# sourceMappingURL=matcher.js.map
// ----------------------------------------------------
// ----------------------------------------------------
function getElementNodesFromNodeList(nodes) {
return getNodesByType(nodes, 1);
}
// ----------------------------------------------------
function getTextNodesFromNodeList(nodes) {
return getNodesByType(nodes, 3);
}
// ----------------------------------------------------
function getNodesByType(nodes, nodeType) {
return Array.from(nodes).filter(node => node.nodeType === nodeType);
}
//# sourceMappingURL=dom.js.map
// ----------------------------------------------------------
class Watch {
// ----------------------------------------------------
constructor(parent, options, callback) {
this.parent = parent;
this.options = options;
this.callback = callback;
this.attributes = new Set();
this.allAttributes = false;
this.addedElementSet = new ElementSet();
this.removedElementSet = new ElementSet();
this.attributeChanges = [];
this.textChanges = [];
this.findExisting = typeof options.findExisting === 'boolean'
? options.findExisting
: true;
this.events = options.events || WatchEvents.ElementsChanged;
if (options.attributes) {
this.attributes = new Set(options.attributes);
}
else if (options.attribute) {
this.attributes.add(options.attribute);
}
else {
this.allAttributes = true;
}
this.matcher = new Matcher(this.parent.root, this.options.selector);
}
// ----------------------------------------------------
get [Symbol.toStringTag]() {
return 'Watch';
}
// ----------------------------------------------------
get selector() {
return this.matcher.selector;
}
// ----------------------------------------------------
processExistingElements() {
if (this.findExisting && this.events & WatchEvents.ElementsAdded) {
const matchingElements = this.matcher.findAllMatchesInSubTree(this.parent.root);
if (matchingElements.length > 0) {
this.addedElementSet.addAll(matchingElements);
this.doResultCallback();
}
}
}
// ----------------------------------------------------
initialise() {
this.addedElementSet.clear();
this.removedElementSet.clear();
this.attributeChanges = [];
this.textChanges = [];
if (this.parent.debug) {
logc(`Watch.initialise()`, this);
}
}
// ----------------------------------------------------
doResultCallback() {
if (this.parent.debug) {
logc(`Watch.processResult(): addedElementSet, removedElementSet, attributeChanges, textChanges`, this.addedElementSet, this.removedElementSet, this.attributeChanges, this.textChanges);
}
if (this.addedElementSet.size > 0 ||
this.removedElementSet.size > 0 ||
this.attributeChanges.length > 0 ||
this.textChanges.length > 0) {
const result = new WatchResult(this.parent, [...this.addedElementSet], [...this.removedElementSet], [...this.attributeChanges], [...this.textChanges]);
this.callback(result);
this.initialise();
}
}
// ----------------------------------------------------
processRecords(records) {
for (const [idx, record] of records.entries()) {
if (this.parent.debug) {
log(`Watch.processRecords(${idx}, type: ${record.type})`, record);
}
switch (record.type) {
case 'childList':
this.onNodeMutation(record);
break;
case 'attributes':
if (this.events & WatchEvents.AttributesChanged) {
this.onAttrMutation(record);
}
break;
case 'characterData':
if (this.events & WatchEvents.TextChanged) {
this.onTextMutation(record);
}
break;
default:
throw new Error('Unknown mutation type "${record.type}"');
}
}
this.doResultCallback();
}
// ----------------------------------------------------
onNodeMutation(summary) {
if (this.events & WatchEvents.ElementsAdded && summary.addedNodes.length > 0) {
for (const element of getElementNodesFromNodeList(summary.addedNodes)) {
this.addedElementSet.addAll(this.matcher.findAllMatchesInSubTree(element));
}
}
if (this.events & WatchEvents.ElementsRemoved && summary.removedNodes.length > 0) {
for (const element of getElementNodesFromNodeList(summary.removedNodes)) {
this.removedElementSet.addAll(this.matcher.findAllMatchesInSubTree(element));
}
}
if (this.events & WatchEvents.TextChanged) {
const addedTextNodes = getTextNodesFromNodeList(summary.addedNodes);
if (addedTextNodes.length > 0) {
const removedTextNodes = getTextNodesFromNodeList(summary.removedNodes);
const oldValue = removedTextNodes.length > 0
? removedTextNodes[0].textContent
: null;
const value = addedTextNodes[addedTextNodes.length - 1].textContent;
const change = new TextChange(summary.target, value, oldValue);
this.textChanges.push(change);
}
}
}
// ----------------------------------------------------
onAttrMutation(summary) {
const { target, attributeName, oldValue } = summary;
if (this.allAttributes || this.attributes.has(attributeName)) {
const element = target;
const value = element.getAttribute(attributeName);
const change = new AttributeChange(element, attributeName, value, oldValue);
this.attributeChanges.push(change);
}
}
// ----------------------------------------------------
onTextMutation(summary) {
const { target, oldValue } = summary;
const element = target.parentElement;
const change = new TextChange(element, element.textContent, oldValue);
this.textChanges.push(change);
}
// ----------------------------------------------------
dump() {
console.groupCollapsed(`%cWatch(%cselector: %c"${this.options.selector}"%c)`, Css.Kw, Css.Attr, Css.Link, Css.Kw);
console.dir(this.options);
console.log(this.callback.toString());
console.groupEnd();
}
}
//# sourceMappingURL=watch.js.map
// ----------------------------------------------------------
class Watcher {
// ----------------------------------------------------
constructor(root = document.body, debug = false) {
this.root = root;
this.debug = debug;
this.observer = null;
this.watches = [];
if (!(root instanceof HTMLElement)) {
throw new TypeError('Watch root is not a valid HTML element!');
}
}
// ----------------------------------------------------
get [Symbol.toStringTag]() {
return 'Watcher';
}
add(options, callback) {
if (typeof options === 'string') {
options = {
selector: options
};
}
else if (typeof options === 'function') {
callback = options;
options = {};
}
if (!callback) {
throw new Error('No callback function specified when calling Watcher.add()');
}
if (this.debug) {
console.groupCollapsed(`%cWatcher.add(selector: %c${options.selector}%c, %c${this.watchCount} watches%c)`, Css.Kw, Css.Link, Css.Kw, Css.Val, Css.Kw);
console.log(callback.toString());
if (options) {
console.dir(options);
}
console.groupEnd();
}
const watch = new Watch(this, options, callback);
if (this.observing) {
watch.processExistingElements();
}
this.watches.push(watch);
return watch;
}
// ----------------------------------------------------
get observing() {
return !!this.observer;
}
// ----------------------------------------------------
get watchCount() {
return this.watches.length;
}
// ----------------------------------------------------
start() {
if (!this.watchCount) {
throw new Error('Cannot start Watcher without any watches!');
}
if (this.debug) {
console.info(`%cWatcher.start(%cenabled = %c${this.observing ? 'true' : 'false'}%c, %c${this.watchCount} watches%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw, Css.Val, Css.Kw);
}
if (!this.observer) {
// Check for existing elements, pass to callback
for (const watch of this.watches) {
watch.processExistingElements();
}
this.observer = new MutationObserver(summaries => {
for (const watch of this.watches) {
watch.processRecords(summaries);
}
});
this.observer.observe(this.root, {
childList: true,
subtree: true,
attributes: true,
attributeOldValue: true,
characterData: true,
characterDataOldValue: true
});
}
return this;
}
// ----------------------------------------------------
stop() {
if (this.observer) {
const records = this.observer.takeRecords();
this.observer.disconnect();
for (const watch of this.watches) {
watch.processRecords(records);
}
this.observer = null;
}
return this;
}
}
//# sourceMappingURL=index.js.map
return Watcher;
})));
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"watcher.umd.js","sources":["../src/interfaces.ts","../src/utils/l.ts","../src/watch-options.ts","../src/watch-result.ts","../src/element-set.ts","../src/matcher.ts","../src/utils/dom.ts","../src/watch.ts","../src/watcher.ts"],"sourcesContent":["\ninterface Console {\n  group (groupTitle?: string, ...optionalParams: any[]): void\n}\n\n// ----------------------------------------------------\n\nexport namespace Css {\n  export const Inverse = 'color: white; background: black'\n  export const Error = 'font-weight: bold; color: #f4f'\n  export const Link = 'color: #05f; font-weight: normal; text-decoration: underline'\n  export const Bold = 'font-weight: bold'\n  export const Blue = 'color: #05f'\n  export const Kw = 'color: #35b; font-weight: bold; font-style: normal; text-decoration: none'\n  export const Attr = 'color: #563; font-weight: normal; font-style: italic; text-decoration: none'\n  export const Val = 'color: #c36; font-weight: normal; font-style: normal; text-decoration: none'\n}\n\n// ----------------------------------------------------\n\nexport interface INode {\n  nodes?: INode[]\n}\n\n// ----------------------------------------------------\n\nexport interface NodeMutationRecord {\n  type: 'childList'\n  target: Node\n  addedNodes: NodeList\n  removedNodes: NodeList\n  previousSibling: Node | null\n  nextSibling: Node | null\n}\n\nexport interface AttrMutationRecord {\n  type: 'attributes'\n  target: Node\n  attributeName: string\n  attributeNamespace: string\n  oldValue: string\n}\n\nexport interface TextMutationRecord {\n  type: 'characterData'\n  target: Node\n  oldValue: string\n}\n\nexport type MutationRecords = NodeMutationRecord | AttrMutationRecord | TextMutationRecord\n","import { Css } from '../interfaces'\n\n// ----------------------------------------------------------\n\nexport function logElement (tag: string, attrs: any, children: any[]) {\n  console.group(`%c${tag}`, Css.Bold)\n  console.info(`%cattrs = ${JSON.stringify(attrs, null, 2)}`, Css.Inverse)\n  // console.info(`%cdata = ${JSON.stringify(data, null, 2)}`, Css.Inverse)\n  if (children) {\n    console.table(children)\n  }\n  console.groupEnd()\n}\n\n// ----------------------------------------------------------\n\nfunction _log (collapsed: boolean, title: string, objs: any[]) {\n  if (collapsed) {\n    console.groupCollapsed(`%c${title}`, Css.Kw)\n  } else {\n    console.group(`%c${title}`, Css.Kw)\n  }\n\n  for (const obj of objs) {\n    console.dir(obj)\n  }\n\n  console.groupEnd()\n}\n\nexport function log (title: string, ...objs: any[]) {\n  _log(false, title, objs)\n}\n\nexport function logc (title: string, ...objs: any[]) {\n  _log(true, title, objs)\n}\n","\n// ----------------------------------------------------\n\nexport interface WatchOptions {\n  selector?: string\n  findExisting?: boolean\n  events?: WatchEvents\n  attribute?: string\n  attributes?: string[]\n}\n\n// ----------------------------------------------------\n\nexport enum WatchEvents {\n  ElementsAdded = 1,\n  ElementsRemoved = 2,\n  AttributesChanged = 4,\n  TextChanged = 8,\n\n  ElementsChanged = ElementsAdded | ElementsRemoved,\n  AllChanges = ElementsAdded | ElementsRemoved | AttributesChanged | TextChanged\n}\n","import Watcher from './watcher'\n\n// ----------------------------------------------------\n\nexport class AttributeChange {\n  constructor (\n    public element: Element,\n    public name: string,\n    public value: string | null,\n    public oldValue: string | null\n  ) {}\n}\n\n// ----------------------------------------------------\n\nexport class TextChange {\n  constructor (\n    public element: Element,\n    public value: string | null,\n    public oldValue: string | null\n  ) {}\n}\n\n// ----------------------------------------------------\n\nexport class WatchResult {\n  constructor (\n    private parent: Watcher,\n    public added: Element[],\n    public removed: Element[] = [],\n    public attributeChanges: AttributeChange[] = [],\n    public textChanges: TextChange[] = []\n  ) {}\n}\n","\nexport class ElementSet extends Set<Element> {\n  // get [Symbol.toStringTag]: string () {\n  //   return 'ElementSet'\n  // }\n\n  // ----------------------------------------------------\n\n  addAll (elements: Element[] | ElementSet): this {\n    for (const element of elements) {\n      super.add(element)\n    }\n\n    return this\n  }\n\n  // ----------------------------------------------------\n\n  toArray (): Element[] {\n    return Array.from(this)\n  }\n}\n","\n// ----------------------------------------------------\n\nexport class Matcher {\n  constructor (\n    public readonly root: Element,\n    public readonly selector: string = '*'\n  ) {}\n\n  // ----------------------------------------------------\n\n  matchesElement (element: Element): boolean {\n    return element.matches(this.selector)\n  }\n\n  // ----------------------------------------------------\n\n  findAllMatchesInSubTree (element: Element): Element[] {\n    const matches: Element[] = Array.from(element.querySelectorAll(this.selector))\n\n    if (this.matchesElement(element)) {\n      matches.unshift(element)\n    }\n\n    return matches\n  }\n}\n","import { SelectorFunc } from '../watch'\n\n// ----------------------------------------------------\n\nexport function getSelectorFunction (selector: string): SelectorFunc {\n  return function (element: HTMLElement): HTMLElement[] {\n    const matches: HTMLElement[] = []\n\n    if (element.matches(selector)) {\n      matches.push(element)\n    }\n\n    return matches.concat(Array.from(element.querySelectorAll(selector)))\n  }\n}\n\n// ----------------------------------------------------\n\nexport function getElementNodesFromNodeList (nodes: NodeList): HTMLElement[] {\n  return getNodesByType<HTMLElement>(nodes, 1)\n}\n\n// ----------------------------------------------------\n\nexport function getTextNodesFromNodeList (nodes: NodeList): Text[] {\n  return getNodesByType<Text>(nodes, 3)\n}\n\n// ----------------------------------------------------\n\nexport function getNodesByType<T extends Node> (nodes: NodeList, nodeType: number): T[] {\n  return Array.from(nodes).filter(node => node.nodeType === nodeType) as T[]\n}\n","import { Css, MutationRecords, NodeMutationRecord, AttrMutationRecord, TextMutationRecord } from './interfaces'\nimport { log, logc } from './utils/l'\n\nimport Watcher from './watcher'\nimport { WatchOptions, WatchEvents } from './watch-options'\nimport { WatchResult, AttributeChange, TextChange } from './watch-result'\nimport { ElementSet } from './element-set'\nimport { Matcher } from './matcher'\n\nimport { getElementNodesFromNodeList, getTextNodesFromNodeList } from './utils/dom'\n\n// ----------------------------------------------------\n\nexport type SelectorFunc = (element: HTMLElement) => HTMLElement[]\n\nexport type WatchCallback = (result: WatchResult) => void\n\n// ----------------------------------------------------------\n\nexport class Watch {\n  public findExisting: boolean\n\n  public events: WatchEvents\n\n  public attributes: Set<string> = new Set()\n  public allAttributes: boolean = false\n\n  matcher: Matcher\n\n  addedElementSet: ElementSet = new ElementSet()\n  removedElementSet: ElementSet = new ElementSet()\n\n  attributeChanges: AttributeChange[] = []\n  textChanges: TextChange[] = []\n\n  // ----------------------------------------------------\n\n  get [Symbol.toStringTag] () {\n    return 'Watch'\n  }\n\n  // ----------------------------------------------------\n\n  constructor (\n    private parent: Watcher,\n    public readonly options: WatchOptions,\n    public readonly callback: WatchCallback\n  ) {\n    this.findExisting = typeof options.findExisting === 'boolean'\n      ? options.findExisting\n      : true\n\n    this.events = options.events || WatchEvents.ElementsChanged\n\n    if (options.attributes) {\n      this.attributes = new Set(options.attributes)\n    } else if (options.attribute) {\n      this.attributes.add(options.attribute)\n    } else {\n      this.allAttributes = true\n    }\n\n    this.matcher = new Matcher(this.parent.root, this.options.selector)\n  }\n\n  // ----------------------------------------------------\n\n  get selector () {\n    return this.matcher.selector\n  }\n\n  // ----------------------------------------------------\n\n  processExistingElements (): void {\n    if (this.findExisting && this.events & WatchEvents.ElementsAdded) {\n      const matchingElements = this.matcher.findAllMatchesInSubTree(this.parent.root)\n\n      if (matchingElements.length > 0) {\n        this.addedElementSet.addAll(matchingElements)\n\n        this.doResultCallback()\n      }\n    }\n  }\n\n  // ----------------------------------------------------\n\n  initialise () {\n    this.addedElementSet.clear()\n    this.removedElementSet.clear()\n\n    this.attributeChanges = []\n    this.textChanges = []\n\n    if (this.parent.debug) {\n      logc(`Watch.initialise()`, this)\n    }\n  }\n\n  // ----------------------------------------------------\n\n  doResultCallback () {\n    if (this.parent.debug) {\n      logc(`Watch.processResult(): addedElementSet, removedElementSet, attributeChanges, textChanges`, this.addedElementSet, this.removedElementSet, this.attributeChanges, this.textChanges)\n    }\n\n    if (this.addedElementSet.size > 0 ||\n        this.removedElementSet.size > 0 ||\n        this.attributeChanges.length > 0 ||\n        this.textChanges.length > 0) {\n      const result = new WatchResult(\n        this.parent,\n        [ ...this.addedElementSet ],\n        [ ...this.removedElementSet ],\n        [ ...this.attributeChanges ],\n        [ ...this.textChanges ]\n      )\n\n      this.callback(result)\n\n      this.initialise()\n    }\n  }\n\n  // ----------------------------------------------------\n\n  processRecords (records: MutationRecord[]): void {\n    for (const [ idx, record ] of records.entries()) {\n      if (this.parent.debug) {\n        log(`Watch.processRecords(${idx}, type: ${record.type})`, record)\n      }\n\n      switch (record.type) {\n        case 'childList':\n          this.onNodeMutation(record as NodeMutationRecord)\n          break\n\n        case 'attributes':\n          if (this.events & WatchEvents.AttributesChanged) {\n            this.onAttrMutation(record as AttrMutationRecord)\n          }\n          break\n\n        case 'characterData':\n          if (this.events & WatchEvents.TextChanged) {\n            this.onTextMutation(record as TextMutationRecord)\n          }\n          break\n\n        default:\n          throw new Error('Unknown mutation type \"${record.type}\"')\n      }\n    }\n\n    this.doResultCallback()\n  }\n\n  // ----------------------------------------------------\n\n  onNodeMutation (summary: NodeMutationRecord) {\n    if (this.events & WatchEvents.ElementsAdded && summary.addedNodes.length > 0) {\n      for (const element of getElementNodesFromNodeList(summary.addedNodes)) {\n        this.addedElementSet.addAll(this.matcher.findAllMatchesInSubTree(element))\n      }\n    }\n\n    if (this.events & WatchEvents.ElementsRemoved && summary.removedNodes.length > 0) {\n      for (const element of getElementNodesFromNodeList(summary.removedNodes)) {\n        this.removedElementSet.addAll(this.matcher.findAllMatchesInSubTree(element))\n      }\n    }\n\n    if (this.events & WatchEvents.TextChanged) {\n      const addedTextNodes = getTextNodesFromNodeList(summary.addedNodes)\n\n      if (addedTextNodes.length > 0) {\n        const removedTextNodes = getTextNodesFromNodeList(summary.removedNodes)\n\n        const oldValue = removedTextNodes.length > 0\n          ? removedTextNodes[ 0 ].textContent\n          : null\n\n        const value = addedTextNodes[ addedTextNodes.length - 1 ].textContent\n\n        const change = new TextChange(summary.target as Element, value, oldValue)\n        this.textChanges.push(change)\n      }\n    }\n  }\n\n  // ----------------------------------------------------\n\n  onAttrMutation (summary: AttrMutationRecord) {\n    const { target, attributeName, oldValue } = summary\n\n    if (this.allAttributes || this.attributes.has(attributeName)) {\n      const element = target as Element\n      const value = element.getAttribute(attributeName)\n\n      const change = new AttributeChange(element, attributeName, value, oldValue)\n      this.attributeChanges.push(change)\n    }\n  }\n\n  // ----------------------------------------------------\n\n  onTextMutation (summary: TextMutationRecord) {\n    const { target, oldValue } = summary\n    const element = target.parentElement as Element\n\n    const change = new TextChange(element, element.textContent, oldValue)\n    this.textChanges.push(change)\n  }\n\n  // ----------------------------------------------------\n\n  dump () {\n    console.groupCollapsed(`%cWatch(%cselector: %c\"${this.options.selector}\"%c)`, Css.Kw, Css.Attr, Css.Link, Css.Kw)\n    console.dir(this.options)\n    console.log(this.callback.toString())\n    console.groupEnd()\n  }\n}\n","import { Css } from './interfaces'\n\nimport { WatchCallback, Watch } from './watch'\nimport { WatchOptions } from './watch-options'\n\n// ----------------------------------------------------------\n\nexport default class Watcher {\n\n  observer: MutationObserver | null = null\n\n  readonly watches: Watch[] = []\n\n  // ----------------------------------------------------\n\n  get [Symbol.toStringTag] () {\n    return 'Watcher'\n  }\n\n  // ----------------------------------------------------\n\n  constructor (\n    public readonly root: HTMLElement = document.body,\n    public readonly debug: boolean = false\n  ) {\n    if (!(root instanceof HTMLElement)) {\n      throw new TypeError('Watch root is not a valid HTML element!')\n    }\n  }\n\n  // ----------------------------------------------------\n\n  add (callback: WatchCallback): Watch\n  add (options: string | WatchOptions, callback: WatchCallback): Watch\n\n  add (options: string | WatchOptions | WatchCallback, callback?: WatchCallback): Watch {\n    if (typeof options === 'string') {\n      options = {\n        selector: options\n      }\n    } else if (typeof options === 'function') {\n      callback = options\n      options = {}\n    }\n\n    if (!callback) {\n      throw new Error('No callback function specified when calling Watcher.add()')\n    }\n\n    if (this.debug) {\n      console.groupCollapsed(`%cWatcher.add(selector: %c${options.selector}%c, %c${this.watchCount} watches%c)`, Css.Kw, Css.Link, Css.Kw, Css.Val, Css.Kw)\n      console.log(callback.toString())\n      if (options) {\n        console.dir(options)\n      }\n      console.groupEnd()\n    }\n\n    const watch = new Watch(this, options, callback)\n\n    if (this.observing) {\n      watch.processExistingElements()\n    }\n\n    this.watches.push(watch)\n\n    return watch\n  }\n\n  // ----------------------------------------------------\n\n  get observing (): boolean {\n    return !!this.observer\n  }\n\n  // ----------------------------------------------------\n\n  get watchCount (): number {\n    return this.watches.length\n  }\n\n  // ----------------------------------------------------\n\n  start (): this {\n    if (!this.watchCount) {\n      throw new Error('Cannot start Watcher without any watches!')\n    }\n\n    if (this.debug) {\n      console.info(`%cWatcher.start(%cenabled = %c${this.observing ? 'true' : 'false'}%c, %c${this.watchCount} watches%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw, Css.Val, Css.Kw)\n    }\n\n    if (!this.observer) {\n      // Check for existing elements, pass to callback\n      for (const watch of this.watches) {\n        watch.processExistingElements()\n      }\n\n      this.observer = new MutationObserver(summaries => {\n        for (const watch of this.watches) {\n          watch.processRecords(summaries)\n        }\n      })\n\n      this.observer.observe(this.root, {\n        childList: true,\n        subtree: true,\n        attributes: true,\n        attributeOldValue: true,\n        characterData: true,\n        characterDataOldValue: true\n      })\n    }\n\n    return this\n  }\n\n  // ----------------------------------------------------\n\n  stop (): this {\n    if (this.observer) {\n      const records = this.observer.takeRecords()\n      this.observer.disconnect()\n\n      for (const watch of this.watches) {\n        watch.processRecords(records)\n      }\n\n      this.observer = null\n    }\n\n    return this\n  }\n}\n"],"names":[],"mappings":";;;;;;AAKA;AAEA,IAAiB,GAAG,CASnB;AATD,WAAiB,GAAG;IACL,WAAO,GAAG,iCAAiC,CAAA;IAC3C,SAAK,GAAG,gCAAgC,CAAA;IACxC,QAAI,GAAG,8DAA8D,CAAA;IACrE,QAAI,GAAG,mBAAmB,CAAA;IAC1B,QAAI,GAAG,aAAa,CAAA;IACpB,MAAE,GAAG,2EAA2E,CAAA;IAChF,QAAI,GAAG,6EAA6E,CAAA;IACpF,OAAG,GAAG,6EAA6E,CAAA;CACjG,EATgB,GAAG,KAAH,GAAG,QASnB;;;ACFD;AAEA,cAAe,SAAkB,EAAE,KAAa,EAAE,IAAW;IAC3D,IAAI,SAAS,EAAE;QACb,OAAO,CAAC,cAAc,CAAC,KAAK,KAAK,EAAE,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;KAC7C;SAAM;QACL,OAAO,CAAC,KAAK,CAAC,KAAK,KAAK,EAAE,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;KACpC;IAED,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;QACtB,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAA;KACjB;IAED,OAAO,CAAC,QAAQ,EAAE,CAAA;CACnB;AAED,aAAqB,KAAa,EAAE,GAAG,IAAW;IAChD,IAAI,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,CAAC,CAAA;CACzB;AAED,cAAsB,KAAa,EAAE,GAAG,IAAW;IACjD,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,CAAA;CACxB;;;ACnCD;;AAYA,AAAA,IAAY,WAQX;AARD,WAAY,WAAW;IACrB,+DAAiB,CAAA;IACjB,mEAAmB,CAAA;IACnB,uEAAqB,CAAA;IACrB,2DAAe,CAAA;IAEf,mEAAiD,CAAA;IACjD,0DAA8E,CAAA;CAC/E,EARW,WAAW,KAAX,WAAW,QAQtB;;;ACnBD;AAEA;IACE,YACS,OAAgB,EAChB,IAAY,EACZ,KAAoB,EACpB,QAAuB;QAHvB,YAAO,GAAP,OAAO,CAAS;QAChB,SAAI,GAAJ,IAAI,CAAQ;QACZ,UAAK,GAAL,KAAK,CAAe;QACpB,aAAQ,GAAR,QAAQ,CAAe;KAC5B;CACL;;AAID;IACE,YACS,OAAgB,EAChB,KAAoB,EACpB,QAAuB;QAFvB,YAAO,GAAP,OAAO,CAAS;QAChB,UAAK,GAAL,KAAK,CAAe;QACpB,aAAQ,GAAR,QAAQ,CAAe;KAC5B;CACL;;AAID;IACE,YACU,MAAe,EAChB,KAAgB,EAChB,UAAqB,EAAE,EACvB,mBAAsC,EAAE,EACxC,cAA4B,EAAE;QAJ7B,WAAM,GAAN,MAAM,CAAS;QAChB,UAAK,GAAL,KAAK,CAAW;QAChB,YAAO,GAAP,OAAO,CAAgB;QACvB,qBAAgB,GAAhB,gBAAgB,CAAwB;QACxC,gBAAW,GAAX,WAAW,CAAmB;KACnC;CACL;;;gBChCuB,SAAQ,GAAY;;;;;IAO1C,MAAM,CAAE,QAAgC;QACtC,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE;YAC9B,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;SACnB;QAED,OAAO,IAAI,CAAA;KACZ;;IAID,OAAO;QACL,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;KACxB;CACF;;;ACpBD;AAEA;IACE,YACkB,IAAa,EACb,WAAmB,GAAG;QADtB,SAAI,GAAJ,IAAI,CAAS;QACb,aAAQ,GAAR,QAAQ,CAAc;KACpC;;IAIJ,cAAc,CAAE,OAAgB;QAC9B,OAAO,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;KACtC;;IAID,uBAAuB,CAAE,OAAgB;QACvC,MAAM,OAAO,GAAc,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAA;QAE9E,IAAI,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,EAAE;YAChC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAA;SACzB;QAED,OAAO,OAAO,CAAA;KACf;CACF;;;ACxBD;AAEA,AAYA;AAEA,qCAA6C,KAAe;IAC1D,OAAO,cAAc,CAAc,KAAK,EAAE,CAAC,CAAC,CAAA;CAC7C;;AAID,kCAA0C,KAAe;IACvD,OAAO,cAAc,CAAO,KAAK,EAAE,CAAC,CAAC,CAAA;CACtC;;AAID,wBAAgD,KAAe,EAAE,QAAgB;IAC/E,OAAO,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,IAAI,IAAI,CAAC,QAAQ,KAAK,QAAQ,CAAQ,CAAA;CAC3E;;;ACfD;AAEA;;IAwBE,YACU,MAAe,EACP,OAAqB,EACrB,QAAuB;QAF/B,WAAM,GAAN,MAAM,CAAS;QACP,YAAO,GAAP,OAAO,CAAc;QACrB,aAAQ,GAAR,QAAQ,CAAe;QAtBlC,eAAU,GAAgB,IAAI,GAAG,EAAE,CAAA;QACnC,kBAAa,GAAY,KAAK,CAAA;QAIrC,oBAAe,GAAe,IAAI,UAAU,EAAE,CAAA;QAC9C,sBAAiB,GAAe,IAAI,UAAU,EAAE,CAAA;QAEhD,qBAAgB,GAAsB,EAAE,CAAA;QACxC,gBAAW,GAAiB,EAAE,CAAA;QAe5B,IAAI,CAAC,YAAY,GAAG,OAAO,OAAO,CAAC,YAAY,KAAK,SAAS;cACzD,OAAO,CAAC,YAAY;cACpB,IAAI,CAAA;QAER,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,WAAW,CAAC,eAAe,CAAA;QAE3D,IAAI,OAAO,CAAC,UAAU,EAAE;YACtB,IAAI,CAAC,UAAU,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,UAAU,CAAC,CAAA;SAC9C;aAAM,IAAI,OAAO,CAAC,SAAS,EAAE;YAC5B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAA;SACvC;aAAM;YACL,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;SAC1B;QAED,IAAI,CAAC,OAAO,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAA;KACpE;;IA1BD,KAAK,MAAM,CAAC,WAAW,CAAC;QACtB,OAAO,OAAO,CAAA;KACf;;IA4BD,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAA;KAC7B;;IAID,uBAAuB;QACrB,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,aAAa,EAAE;YAChE,MAAM,gBAAgB,GAAG,IAAI,CAAC,OAAO,CAAC,uBAAuB,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAA;YAE/E,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE;gBAC/B,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAA;gBAE7C,IAAI,CAAC,gBAAgB,EAAE,CAAA;aACxB;SACF;KACF;;IAID,UAAU;QACR,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAA;QAC5B,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE,CAAA;QAE9B,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAA;QAC1B,IAAI,CAAC,WAAW,GAAG,EAAE,CAAA;QAErB,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE;YACrB,IAAI,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAA;SACjC;KACF;;IAID,gBAAgB;QACd,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE;YACrB,IAAI,CAAC,0FAA0F,EAAE,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,gBAAgB,EAAE,IAAI,CAAC,WAAW,CAAC,CAAA;SACxL;QAED,IAAI,IAAI,CAAC,eAAe,CAAC,IAAI,GAAG,CAAC;YAC7B,IAAI,CAAC,iBAAiB,CAAC,IAAI,GAAG,CAAC;YAC/B,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC;YAChC,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/B,MAAM,MAAM,GAAG,IAAI,WAAW,CAC5B,IAAI,CAAC,MAAM,EACX,CAAE,GAAG,IAAI,CAAC,eAAe,CAAE,EAC3B,CAAE,GAAG,IAAI,CAAC,iBAAiB,CAAE,EAC7B,CAAE,GAAG,IAAI,CAAC,gBAAgB,CAAE,EAC5B,CAAE,GAAG,IAAI,CAAC,WAAW,CAAE,CACxB,CAAA;YAED,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;YAErB,IAAI,CAAC,UAAU,EAAE,CAAA;SAClB;KACF;;IAID,cAAc,CAAE,OAAyB;QACvC,KAAK,MAAM,CAAE,GAAG,EAAE,MAAM,CAAE,IAAI,OAAO,CAAC,OAAO,EAAE,EAAE;YAC/C,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE;gBACrB,GAAG,CAAC,wBAAwB,GAAG,WAAW,MAAM,CAAC,IAAI,GAAG,EAAE,MAAM,CAAC,CAAA;aAClE;YAED,QAAQ,MAAM,CAAC,IAAI;gBACjB,KAAK,WAAW;oBACd,IAAI,CAAC,cAAc,CAAC,MAA4B,CAAC,CAAA;oBACjD,MAAK;gBAEP,KAAK,YAAY;oBACf,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,iBAAiB,EAAE;wBAC/C,IAAI,CAAC,cAAc,CAAC,MAA4B,CAAC,CAAA;qBAClD;oBACD,MAAK;gBAEP,KAAK,eAAe;oBAClB,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,WAAW,EAAE;wBACzC,IAAI,CAAC,cAAc,CAAC,MAA4B,CAAC,CAAA;qBAClD;oBACD,MAAK;gBAEP;oBACE,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAA;aAC5D;SACF;QAED,IAAI,CAAC,gBAAgB,EAAE,CAAA;KACxB;;IAID,cAAc,CAAE,OAA2B;QACzC,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,aAAa,IAAI,OAAO,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE;YAC5E,KAAK,MAAM,OAAO,IAAI,2BAA2B,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE;gBACrE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC,CAAA;aAC3E;SACF;QAED,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,eAAe,IAAI,OAAO,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YAChF,KAAK,MAAM,OAAO,IAAI,2BAA2B,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE;gBACvE,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC,CAAA;aAC7E;SACF;QAED,IAAI,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,WAAW,EAAE;YACzC,MAAM,cAAc,GAAG,wBAAwB,CAAC,OAAO,CAAC,UAAU,CAAC,CAAA;YAEnE,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;gBAC7B,MAAM,gBAAgB,GAAG,wBAAwB,CAAC,OAAO,CAAC,YAAY,CAAC,CAAA;gBAEvE,MAAM,QAAQ,GAAG,gBAAgB,CAAC,MAAM,GAAG,CAAC;sBACxC,gBAAgB,CAAE,CAAC,CAAE,CAAC,WAAW;sBACjC,IAAI,CAAA;gBAER,MAAM,KAAK,GAAG,cAAc,CAAE,cAAc,CAAC,MAAM,GAAG,CAAC,CAAE,CAAC,WAAW,CAAA;gBAErE,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,OAAO,CAAC,MAAiB,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAA;gBACzE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;aAC9B;SACF;KACF;;IAID,cAAc,CAAE,OAA2B;QACzC,MAAM,EAAE,MAAM,EAAE,aAAa,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAA;QAEnD,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE;YAC5D,MAAM,OAAO,GAAG,MAAiB,CAAA;YACjC,MAAM,KAAK,GAAG,OAAO,CAAC,YAAY,CAAC,aAAa,CAAC,CAAA;YAEjD,MAAM,MAAM,GAAG,IAAI,eAAe,CAAC,OAAO,EAAE,aAAa,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAA;YAC3E,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;SACnC;KACF;;IAID,cAAc,CAAE,OAA2B;QACzC,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAA;QACpC,MAAM,OAAO,GAAG,MAAM,CAAC,aAAwB,CAAA;QAE/C,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,OAAO,EAAE,OAAO,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAA;QACrE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;KAC9B;;IAID,IAAI;QACF,OAAO,CAAC,cAAc,CAAC,0BAA0B,IAAI,CAAC,OAAO,CAAC,QAAQ,MAAM,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;QACjH,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;QACzB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAA;QACrC,OAAO,CAAC,QAAQ,EAAE,CAAA;KACnB;CACF;;;ACzND;AAEA;;IAcE,YACkB,OAAoB,QAAQ,CAAC,IAAI,EACjC,QAAiB,KAAK;QADtB,SAAI,GAAJ,IAAI,CAA6B;QACjC,UAAK,GAAL,KAAK,CAAiB;QAdxC,aAAQ,GAA4B,IAAI,CAAA;QAE/B,YAAO,GAAY,EAAE,CAAA;QAc5B,IAAI,EAAE,IAAI,YAAY,WAAW,CAAC,EAAE;YAClC,MAAM,IAAI,SAAS,CAAC,yCAAyC,CAAC,CAAA;SAC/D;KACF;;IAbD,KAAK,MAAM,CAAC,WAAW,CAAC;QACtB,OAAO,SAAS,CAAA;KACjB;IAkBD,GAAG,CAAE,OAA8C,EAAE,QAAwB;QAC3E,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE;YAC/B,OAAO,GAAG;gBACR,QAAQ,EAAE,OAAO;aAClB,CAAA;SACF;aAAM,IAAI,OAAO,OAAO,KAAK,UAAU,EAAE;YACxC,QAAQ,GAAG,OAAO,CAAA;YAClB,OAAO,GAAG,EAAE,CAAA;SACb;QAED,IAAI,CAAC,QAAQ,EAAE;YACb,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAA;SAC7E;QAED,IAAI,IAAI,CAAC,KAAK,EAAE;YACd,OAAO,CAAC,cAAc,CAAC,6BAA6B,OAAO,CAAC,QAAQ,SAAS,IAAI,CAAC,UAAU,aAAa,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;YACrJ,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAA;YAChC,IAAI,OAAO,EAAE;gBACX,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;aACrB;YACD,OAAO,CAAC,QAAQ,EAAE,CAAA;SACnB;QAED,MAAM,KAAK,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAA;QAEhD,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,KAAK,CAAC,uBAAuB,EAAE,CAAA;SAChC;QAED,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QAExB,OAAO,KAAK,CAAA;KACb;;IAID,IAAI,SAAS;QACX,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAA;KACvB;;IAID,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAA;KAC3B;;IAID,KAAK;QACH,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YACpB,MAAM,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAA;SAC7D;QAED,IAAI,IAAI,CAAC,KAAK,EAAE;YACd,OAAO,CAAC,IAAI,CAAC,iCAAiC,IAAI,CAAC,SAAS,GAAG,MAAM,GAAG,OAAO,SAAS,IAAI,CAAC,UAAU,aAAa,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;SAC1K;QAED,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;;YAElB,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,OAAO,EAAE;gBAChC,KAAK,CAAC,uBAAuB,EAAE,CAAA;aAChC;YAED,IAAI,CAAC,QAAQ,GAAG,IAAI,gBAAgB,CAAC,SAAS;gBAC5C,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,OAAO,EAAE;oBAChC,KAAK,CAAC,cAAc,CAAC,SAAS,CAAC,CAAA;iBAChC;aACF,CAAC,CAAA;YAEF,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE;gBAC/B,SAAS,EAAE,IAAI;gBACf,OAAO,EAAE,IAAI;gBACb,UAAU,EAAE,IAAI;gBAChB,iBAAiB,EAAE,IAAI;gBACvB,aAAa,EAAE,IAAI;gBACnB,qBAAqB,EAAE,IAAI;aAC5B,CAAC,CAAA;SACH;QAED,OAAO,IAAI,CAAA;KACZ;;IAID,IAAI;QACF,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAA;YAC3C,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAA;YAE1B,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,OAAO,EAAE;gBAChC,KAAK,CAAC,cAAc,CAAC,OAAO,CAAC,CAAA;aAC9B;YAED,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAA;SACrB;QAED,OAAO,IAAI,CAAA;KACZ;CACF;;;;;;;;;;"}