Skip to content

Instantly share code, notes, and snippets.

@spiralx
Created July 4, 2018 16:23
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save spiralx/1326b73f02cc9c732a3153812694c749 to your computer and use it in GitHub Desktop.
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
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":"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"}
@spiralx
Copy link
Author

spiralx commented Jul 4, 2018

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment