Created
July 4, 2018 16:23
-
-
Save spiralx/1326b73f02cc9c732a3153812694c749 to your computer and use it in GitHub Desktop.
A small module I've written to let you subscribe to DOM changes that match specified criteria
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var Watcher = (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 = {})); | |
// ---------------------------------------------------- | |
// ---------------------------------------------------- | |
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 = {})); | |
// ---------------------------------------------------- | |
// ---------------------------------------------------- | |
class WatchResult { | |
constructor() { | |
this.added = new Array(); | |
this.removed = new Array(); | |
this.attributeChanges = new Array(); | |
this.textChanges = new Array(); | |
} | |
} | |
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); | |
} | |
} | |
// ---------------------------------------------------- | |
function getSelectorFunction(selector) { | |
return function (element) { | |
const matches = []; | |
if (element.matches(selector)) { | |
matches.push(element); | |
} | |
return matches.concat(Array.from(element.querySelectorAll(selector))); | |
}; | |
} | |
// ---------------------------------------------------- | |
function getElementNodesFromNodeList(nodes) { | |
return getElementNodes(Array.from(nodes)); | |
} | |
// ---------------------------------------------------- | |
function getElementNodes(nodes) { | |
return nodes.filter(node => node instanceof HTMLElement); | |
} | |
// ---------------------------------------------------------- | |
class Watch { | |
// ---------------------------------------------------- | |
constructor(options, callback) { | |
this.options = options; | |
this.callback = callback; | |
this.attributes = new Set(); | |
this.selector = this.options.selector || '*'; | |
this.selectorFunction = getSelectorFunction(this.selector); | |
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); | |
} | |
} | |
// ---------------------------------------------------- | |
get [Symbol.toStringTag]() { | |
return 'Watch'; | |
} | |
// ---------------------------------------------------- | |
processSummary(summary, debug = false) { | |
const addedElements = getElementNodesFromNodeList(summary.addedNodes); | |
const removedElements = getElementNodesFromNodeList(summary.removedNodes); | |
const matchingAddedElements = this.processElements(addedElements); | |
const matchingRemovedElements = this.processElements(removedElements); | |
if (debug) { | |
console.groupCollapsed(`%cWatch.processSummary(%ctype=%c${summary.type}%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw); | |
if (addedElements.length) { | |
console.group(`Added elements`); | |
console.dir(addedElements); | |
console.dir(matchingAddedElements); | |
console.groupEnd(); | |
} | |
if (removedElements.length) { | |
console.group(`Removed elements`); | |
console.dir(removedElements); | |
console.dir(matchingRemovedElements); | |
console.groupEnd(); | |
} | |
console.groupEnd(); | |
} | |
this.invoke(matchingAddedElements.toArray(), matchingRemovedElements.toArray(), debug); | |
} | |
// ---------------------------------------------------- | |
processElement(element) { | |
this.invoke(this.selectorFunction(element), []); | |
} | |
// ---------------------------------------------------- | |
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(); | |
} | |
// ---------------------------------------------------- | |
processElements(elements) { | |
return elements.reduce((matches, element) => matches.addAll(this.selectorFunction(element)), new ElementSet()); | |
} | |
// ---------------------------------------------------- | |
invoke(added, removed, debug = false) { | |
if (added.length > 0 || removed.length > 0) { | |
const result = new WatchResult(); | |
result.added = added; | |
result.removed = removed; | |
if (debug) { | |
console.groupCollapsed(`%cWatch.invoke()`, Css.Kw); | |
console.dir(result); | |
console.groupEnd(); | |
} | |
this.callback(result); | |
// this.callback.call(this.context, result) | |
} | |
} | |
} | |
// ---------------------------------------------------------- | |
class Watcher { | |
// ---------------------------------------------------- | |
constructor(root = document.body, debug = false) { | |
this.root = root; | |
this.debug = debug; | |
this.observer = null; | |
// readonly watcheMap: Map<string, Watch> = new Map() | |
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(options, callback); | |
this.watches.push(watch); | |
return watch; | |
} | |
// ---------------------------------------------------- | |
get observing() { | |
return !!this.observer; | |
} | |
// ---------------------------------------------------- | |
get watchCount() { | |
return this.watches.length; | |
} | |
// ---------------------------------------------------- | |
// get watches (): Watch[] { | |
// return [ ...this.watchMap.values() ] | |
// } | |
// ---------------------------------------------------- | |
processSummary(summary) { | |
if (this.debug) { | |
console.groupCollapsed(`%cWatcher.processSummary(%ctype=%c${summary.type}%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw); | |
console.dir(summary); | |
console.groupEnd(); | |
} | |
for (const watch of this.watches) { | |
watch.processSummary(summary, this.debug); | |
} | |
} | |
// ---------------------------------------------------- | |
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) { | |
if (watch.findExisting && watch.events & WatchEvents.ElementsAdded) { | |
watch.processElement(this.root); | |
} | |
} | |
this.observer = new MutationObserver(summaries => { | |
summaries.forEach(summary => this.processSummary(summary)); | |
}); | |
this.observer.observe(this.root, { | |
childList: true, | |
// attributes: true, | |
subtree: true | |
}); | |
} | |
return this; | |
} | |
// ---------------------------------------------------- | |
stop() { | |
if (this.observer) { | |
this.observer.takeRecords().forEach(summary => this.processSummary(summary)); | |
this.observer.disconnect(); | |
this.observer = null; | |
} | |
return this; | |
} | |
} | |
return Watcher; | |
}()); | |
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"watcher.js","sources":["../src/interfaces.ts","../src/watch-options.ts","../src/watch-result.ts","../src/element-set.ts","../src/utils/dom.ts","../src/watch.ts","../src/watcher.ts"],"sourcesContent":["\r\ninterface Console {\r\n  group (groupTitle?: string, ...optionalParams: any[]): void\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport namespace Css {\r\n  export const Inverse = 'color: white; background: black'\r\n  export const Error = 'font-weight: bold; color: #f4f'\r\n  export const Link = 'color: #05f; font-weight: normal; text-decoration: underline'\r\n  export const Bold = 'font-weight: bold'\r\n  export const Blue = 'color: #05f'\r\n  export const Kw = 'color: #35b; font-weight: bold; font-style: normal; text-decoration: none'\r\n  export const Attr = 'color: #563; font-weight: normal; font-style: italic; text-decoration: none'\r\n  export const Val = 'color: #c36; font-weight: normal; font-style: normal; text-decoration: none'\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport interface INode {\r\n  nodes?: INode[]\r\n}\r\n","\r\n// ----------------------------------------------------\r\n\r\nexport interface WatchOptions {\r\n  selector?: string\r\n  findExisting?: boolean\r\n  events?: WatchEvents\r\n  attribute?: string\r\n  attributes?: string[]\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport enum WatchEvents {\r\n  ElementsAdded = 1,\r\n  ElementsRemoved = 2,\r\n  AttributesChanged = 4,\r\n  TextChanged = 8,\r\n\r\n  ElementsChanged = ElementsAdded | ElementsRemoved,\r\n  AllChanges = ElementsAdded | ElementsRemoved | AttributesChanged | TextChanged\r\n}\r\n","\r\n// ----------------------------------------------------\r\n\r\nexport interface AttributeChange {\r\n  target: HTMLElement\r\n  attribute: string\r\n  value: string | SVGNumberList\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport interface TextChange {\r\n  target: HTMLElement\r\n  text: string\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport class WatchResult {\r\n  added: Array<HTMLElement> = new Array<HTMLElement>()\r\n  removed: Array<HTMLElement> = new Array<HTMLElement>()\r\n  attributeChanges: Array<AttributeChange> = new Array<AttributeChange>()\r\n  textChanges: Array<TextChange> = new Array<TextChange>()\r\n}\r\n","\r\nexport class ElementSet extends Set<HTMLElement> {\r\n  // get [Symbol.toStringTag]: string () {\r\n  //   return 'ElementSet'\r\n  // }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  addAll (elements: HTMLElement[] | ElementSet): this {\r\n    for (const element of elements) {\r\n      super.add(element)\r\n    }\r\n\r\n    return this\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  toArray (): HTMLElement[] {\r\n    return Array.from(this)\r\n  }\r\n}\r\n","import { SelectorFunc } from '../watch'\r\n\r\n// ----------------------------------------------------\r\n\r\nexport function getSelectorFunction (selector: string): SelectorFunc {\r\n  return function (element: HTMLElement): HTMLElement[] {\r\n    const matches: HTMLElement[] = []\r\n\r\n    if (element.matches(selector)) {\r\n      matches.push(element)\r\n    }\r\n\r\n    return matches.concat(Array.from(element.querySelectorAll(selector)))\r\n  }\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport function getElementNodesFromNodeList (nodes: NodeList): HTMLElement[] {\r\n  return getElementNodes(Array.from(nodes))\r\n}\r\n\r\n// ----------------------------------------------------\r\n\r\nexport function getElementNodes (nodes: Node[]): HTMLElement[] {\r\n  return nodes.filter(node => node instanceof HTMLElement) as HTMLElement[]\r\n}\r\n","import { Css } from './interfaces'\r\n\r\nimport { WatchOptions, WatchEvents } from './watch-options'\r\nimport { WatchResult } from './watch-result'\r\nimport { ElementSet } from './element-set'\r\n\r\nimport { getSelectorFunction, getElementNodesFromNodeList } from './utils/dom'\r\n\r\n// ----------------------------------------------------\r\n\r\nexport type SelectorFunc = (element: HTMLElement) => HTMLElement[]\r\n\r\nexport type WatchCallback = (result: WatchResult) => void\r\n\r\n// ----------------------------------------------------------\r\n\r\nexport class Watch {\r\n  public selector: string\r\n  public selectorFunction: SelectorFunc\r\n\r\n  public findExisting: boolean\r\n\r\n  public events: WatchEvents\r\n  public attributes: Set<string> = new Set()\r\n\r\n  // ----------------------------------------------------\r\n\r\n  get [Symbol.toStringTag] () {\r\n    return 'Watch'\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  constructor (\r\n    public readonly options: WatchOptions,\r\n    public readonly callback: WatchCallback\r\n  ) {\r\n    this.selector = this.options.selector || '*'\r\n    this.selectorFunction = getSelectorFunction(this.selector)\r\n\r\n    this.findExisting = typeof options.findExisting === 'boolean'\r\n      ? options.findExisting\r\n      : true\r\n\r\n    this.events = options.events || WatchEvents.ElementsChanged\r\n\r\n    if (options.attributes) {\r\n      this.attributes = new Set(options.attributes)\r\n    } else if (options.attribute) {\r\n      this.attributes.add(options.attribute)\r\n    }\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  processSummary (summary: MutationRecord, debug: boolean = false): void {\r\n    const addedElements = getElementNodesFromNodeList(summary.addedNodes)\r\n    const removedElements = getElementNodesFromNodeList(summary.removedNodes)\r\n\r\n    const matchingAddedElements: ElementSet = this.processElements(addedElements)\r\n    const matchingRemovedElements: ElementSet = this.processElements(removedElements)\r\n\r\n    if (debug) {\r\n      console.groupCollapsed(`%cWatch.processSummary(%ctype=%c${summary.type}%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw)\r\n\r\n      if (addedElements.length) {\r\n        console.group(`Added elements`)\r\n        console.dir(addedElements)\r\n        console.dir(matchingAddedElements)\r\n        console.groupEnd()\r\n      }\r\n\r\n      if (removedElements.length) {\r\n        console.group(`Removed elements`)\r\n        console.dir(removedElements)\r\n        console.dir(matchingRemovedElements)\r\n        console.groupEnd()\r\n      }\r\n\r\n      console.groupEnd()\r\n    }\r\n\r\n    this.invoke(matchingAddedElements.toArray(), matchingRemovedElements.toArray(), debug)\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  processElement (element: HTMLElement): void {\r\n    this.invoke(this.selectorFunction(element), [])\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  dump () {\r\n    console.groupCollapsed(`%cWatch(%cselector: %c\"${this.options.selector}\"%c)`, Css.Kw, Css.Attr, Css.Link, Css.Kw)\r\n    console.dir(this.options)\r\n    console.log(this.callback.toString())\r\n    console.groupEnd()\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  private processElements (elements: HTMLElement[]): ElementSet {\r\n    return elements.reduce((matches: ElementSet, element: HTMLElement) => matches.addAll(this.selectorFunction(element)), new ElementSet())\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  private invoke (added: HTMLElement[], removed: HTMLElement[], debug: boolean = false) {\r\n    if (added.length > 0 || removed.length > 0) {\r\n      const result = new WatchResult()\r\n      result.added = added\r\n      result.removed = removed\r\n\r\n      if (debug) {\r\n        console.groupCollapsed(`%cWatch.invoke()`, Css.Kw)\r\n        console.dir(result)\r\n        console.groupEnd()\r\n      }\r\n\r\n      this.callback(result)\r\n      // this.callback.call(this.context, result)\r\n    }\r\n  }\r\n}\r\n","import { Css } from './interfaces'\r\n\r\nimport { WatchCallback, Watch } from './watch'\r\nimport { WatchOptions, WatchEvents } from './watch-options'\r\n\r\n// ----------------------------------------------------------\r\n\r\nexport default class Watcher {\r\n\r\n  observer: MutationObserver | null = null\r\n\r\n  // readonly watcheMap: Map<string, Watch> = new Map()\r\n  readonly watches: Watch[] = []\r\n\r\n  // ----------------------------------------------------\r\n\r\n  get [Symbol.toStringTag] () {\r\n    return 'Watcher'\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  constructor (\r\n    public readonly root: HTMLElement = document.body,\r\n    public readonly debug: boolean = false\r\n  ) {\r\n    if (!(root instanceof HTMLElement)) {\r\n      throw new TypeError('Watch root is not a valid HTML element!')\r\n    }\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  add (callback: WatchCallback): Watch\r\n  add (options: string | WatchOptions, callback: WatchCallback): Watch\r\n\r\n  add (options: string | WatchOptions | WatchCallback, callback?: WatchCallback): Watch {\r\n    if (typeof options === 'string') {\r\n      options = {\r\n        selector: options\r\n      }\r\n    } else if (typeof options === 'function') {\r\n      callback = options\r\n      options = {}\r\n    }\r\n\r\n    if (!callback) {\r\n      throw new Error('No callback function specified when calling Watcher.add()')\r\n    }\r\n\r\n    if (this.debug) {\r\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)\r\n      console.log(callback.toString())\r\n      if (options) {\r\n        console.dir(options)\r\n      }\r\n      console.groupEnd()\r\n    }\r\n\r\n    const watch = new Watch(options, callback)\r\n\r\n    this.watches.push(watch)\r\n\r\n    return watch\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  get observing (): boolean {\r\n    return !!this.observer\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  get watchCount (): number {\r\n    return this.watches.length\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  // get watches (): Watch[] {\r\n  //   return [ ...this.watchMap.values() ]\r\n  // }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  processSummary (summary: MutationRecord): void {\r\n    if (this.debug) {\r\n      console.groupCollapsed(`%cWatcher.processSummary(%ctype=%c${summary.type}%c)`, Css.Kw, Css.Attr, Css.Val, Css.Kw)\r\n      console.dir(summary)\r\n      console.groupEnd()\r\n    }\r\n\r\n    for (const watch of this.watches) {\r\n      watch.processSummary(summary, this.debug)\r\n    }\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  start (): this {\r\n    if (!this.watchCount) {\r\n      throw new Error('Cannot start Watcher without any watches!')\r\n    }\r\n\r\n    if (this.debug) {\r\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)\r\n    }\r\n\r\n    if (!this.observer) {\r\n      // Check for existing elements, pass to callback\r\n      for (const watch of this.watches) {\r\n        if (watch.findExisting && watch.events & WatchEvents.ElementsAdded) {\r\n          watch.processElement(this.root)\r\n        }\r\n      }\r\n\r\n      this.observer = new MutationObserver(summaries => {\r\n        summaries.forEach(summary => this.processSummary(summary))\r\n      })\r\n\r\n      this.observer.observe(this.root, {\r\n        childList: true,\r\n        // attributes: true,\r\n        subtree: true\r\n      })\r\n    }\r\n\r\n    return this\r\n  }\r\n\r\n  // ----------------------------------------------------\r\n\r\n  stop (): this {\r\n    if (this.observer) {\r\n      this.observer.takeRecords().forEach(summary => this.processSummary(summary))\r\n\r\n      this.observer.disconnect()\r\n      this.observer = null\r\n    }\r\n\r\n    return this\r\n  }\r\n}\r\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;;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;;ACpBD;;AAiBA;IAAA;QACE,UAAK,GAAuB,IAAI,KAAK,EAAe,CAAA;QACpD,YAAO,GAAuB,IAAI,KAAK,EAAe,CAAA;QACtD,qBAAgB,GAA2B,IAAI,KAAK,EAAmB,CAAA;QACvE,gBAAW,GAAsB,IAAI,KAAK,EAAc,CAAA;KACzD;CAAA;;gBCtBuB,SAAQ,GAAgB;;;;;IAO9C,MAAM,CAAE,QAAoC;QAC1C,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;;ACnBD;AAEA,6BAAqC,QAAgB;IACnD,OAAO,UAAU,OAAoB;QACnC,MAAM,OAAO,GAAkB,EAAE,CAAA;QAEjC,IAAI,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;YAC7B,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;SACtB;QAED,OAAO,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAA;KACtE,CAAA;CACF;;AAID,qCAA6C,KAAe;IAC1D,OAAO,eAAe,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAA;CAC1C;;AAID,yBAAiC,KAAa;IAC5C,OAAO,KAAK,CAAC,MAAM,CAAC,IAAI,IAAI,IAAI,YAAY,WAAW,CAAkB,CAAA;CAC1E;;ACZD;AAEA;;IAiBE,YACkB,OAAqB,EACrB,QAAuB;QADvB,YAAO,GAAP,OAAO,CAAc;QACrB,aAAQ,GAAR,QAAQ,CAAe;QAZlC,eAAU,GAAgB,IAAI,GAAG,EAAE,CAAA;QAcxC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,GAAG,CAAA;QAC5C,IAAI,CAAC,gBAAgB,GAAG,mBAAmB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QAE1D,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;KACF;;IAxBD,KAAK,MAAM,CAAC,WAAW,CAAC;QACtB,OAAO,OAAO,CAAA;KACf;;IA0BD,cAAc,CAAE,OAAuB,EAAE,QAAiB,KAAK;QAC7D,MAAM,aAAa,GAAG,2BAA2B,CAAC,OAAO,CAAC,UAAU,CAAC,CAAA;QACrE,MAAM,eAAe,GAAG,2BAA2B,CAAC,OAAO,CAAC,YAAY,CAAC,CAAA;QAEzE,MAAM,qBAAqB,GAAe,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC,CAAA;QAC7E,MAAM,uBAAuB,GAAe,IAAI,CAAC,eAAe,CAAC,eAAe,CAAC,CAAA;QAEjF,IAAI,KAAK,EAAE;YACT,OAAO,CAAC,cAAc,CAAC,mCAAmC,OAAO,CAAC,IAAI,KAAK,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;YAE/G,IAAI,aAAa,CAAC,MAAM,EAAE;gBACxB,OAAO,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAA;gBAC/B,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,CAAA;gBAC1B,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAA;gBAClC,OAAO,CAAC,QAAQ,EAAE,CAAA;aACnB;YAED,IAAI,eAAe,CAAC,MAAM,EAAE;gBAC1B,OAAO,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAA;gBACjC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,CAAA;gBAC5B,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAA;gBACpC,OAAO,CAAC,QAAQ,EAAE,CAAA;aACnB;YAED,OAAO,CAAC,QAAQ,EAAE,CAAA;SACnB;QAED,IAAI,CAAC,MAAM,CAAC,qBAAqB,CAAC,OAAO,EAAE,EAAE,uBAAuB,CAAC,OAAO,EAAE,EAAE,KAAK,CAAC,CAAA;KACvF;;IAID,cAAc,CAAE,OAAoB;QAClC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,EAAE,EAAE,CAAC,CAAA;KAChD;;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;;IAIO,eAAe,CAAE,QAAuB;QAC9C,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC,OAAmB,EAAE,OAAoB,KAAK,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,EAAE,IAAI,UAAU,EAAE,CAAC,CAAA;KACxI;;IAIO,MAAM,CAAE,KAAoB,EAAE,OAAsB,EAAE,QAAiB,KAAK;QAClF,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;YAC1C,MAAM,MAAM,GAAG,IAAI,WAAW,EAAE,CAAA;YAChC,MAAM,CAAC,KAAK,GAAG,KAAK,CAAA;YACpB,MAAM,CAAC,OAAO,GAAG,OAAO,CAAA;YAExB,IAAI,KAAK,EAAE;gBACT,OAAO,CAAC,cAAc,CAAC,kBAAkB,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;gBAClD,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAA;gBACnB,OAAO,CAAC,QAAQ,EAAE,CAAA;aACnB;YAED,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;;SAEtB;KACF;CACF;;ACvHD;AAEA;;IAeE,YACkB,OAAoB,QAAQ,CAAC,IAAI,EACjC,QAAiB,KAAK;QADtB,SAAI,GAAJ,IAAI,CAA6B;QACjC,UAAK,GAAL,KAAK,CAAiB;QAfxC,aAAQ,GAA4B,IAAI,CAAA;;QAG/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,OAAO,EAAE,QAAQ,CAAC,CAAA;QAE1C,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;;;;;;IAUD,cAAc,CAAE,OAAuB;QACrC,IAAI,IAAI,CAAC,KAAK,EAAE;YACd,OAAO,CAAC,cAAc,CAAC,qCAAqC,OAAO,CAAC,IAAI,KAAK,EAAE,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,CAAA;YACjH,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;YACpB,OAAO,CAAC,QAAQ,EAAE,CAAA;SACnB;QAED,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,OAAO,EAAE;YAChC,KAAK,CAAC,cAAc,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,CAAA;SAC1C;KACF;;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,IAAI,KAAK,CAAC,YAAY,IAAI,KAAK,CAAC,MAAM,GAAG,WAAW,CAAC,aAAa,EAAE;oBAClE,KAAK,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;iBAChC;aACF;YAED,IAAI,CAAC,QAAQ,GAAG,IAAI,gBAAgB,CAAC,SAAS;gBAC5C,SAAS,CAAC,OAAO,CAAC,OAAO,IAAI,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAA;aAC3D,CAAC,CAAA;YAEF,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE;gBAC/B,SAAS,EAAE,IAAI;;gBAEf,OAAO,EAAE,IAAI;aACd,CAAC,CAAA;SACH;QAED,OAAO,IAAI,CAAA;KACZ;;IAID,IAAI;QACF,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,OAAO,IAAI,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAA;YAE5E,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAA;YAC1B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAA;SACrB;QAED,OAAO,IAAI,CAAA;KACZ;CACF;;;;;;;;"} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Gist - Raw
RawGit - Development
RawGit - Production