UNPKG

multitude

Version:

Comprehensive streams for push and pull

181 lines (130 loc) 16.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Multicast = void 0; var _helpers = require("../helpers"); var _from = require("../creators/from"); var _helpers2 = require("./helpers"); var _Observable = require("./Observable"); var _typeCore = require("type-core"); let _Symbol$observable; function _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); } function _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError("Cannot initialize the same private elements twice on an object"); } } function _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, "set"); _classApplyDescriptorSet(receiver, descriptor, value); return value; } function _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError("attempted to set read only private field"); } descriptor.value = value; } } function _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, "get"); return _classApplyDescriptorGet(receiver, descriptor); } function _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError("attempted to " + action + " private field on non-instance"); } return privateMap.get(receiver); } function _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; } var _value = /*#__PURE__*/new WeakMap(); var _termination = /*#__PURE__*/new WeakMap(); _Symbol$observable = Symbol.observable; class Multicast extends _Observable.Observable { static of(item, ...args) { const options = args[0]; const hooks = args[1]; return new this(obs => { obs.next(item); obs.complete(); }, options, hooks); } static from(item, options, hooks) { if (item.constructor === this) return item; const observable = (0, _from.from)(item); return new this(obs => observable.subscribe(obs), options, hooks); } constructor(subscriber, options, hooks) { super(obs => { const termination = _classPrivateFieldGet(this, _termination); if (termination) { return typeof termination === 'boolean' ? obs.complete() : obs.error(termination[0]); } for (const value of values) { obs.next(value); } items.add(obs); if (hooks) { const onSubscribe = hooks.onSubscribe; if (!_typeCore.TypeGuard.isEmpty(onSubscribe)) { onSubscribe.call(hooks, connect); } } return () => { items.delete(obs); if (hooks) { const onUnsubscribe = hooks.onUnsubscribe; if (!_typeCore.TypeGuard.isEmpty(onUnsubscribe)) { onUnsubscribe.call(hooks, connect); } } }; }); _classPrivateFieldInitSpec(this, _value, { writable: true, value: void 0 }); _classPrivateFieldInitSpec(this, _termination, { writable: true, value: void 0 }); const opts = options || {}; const items = new Set(); const values = []; const replay = Math.max(0, Number(opts.replay)); const observable = new _Observable.Observable(subscriber); _classPrivateFieldSet(this, _value, opts.value); _classPrivateFieldSet(this, _termination, false); let subscription; const connect = () => { if (subscription && (!subscription.closed || this.closed)) { return subscription; } return observable.subscribe({ start: subs => subscription = subs, next: value => { if (this.closed) return; _classPrivateFieldSet(this, _value, value); if (replay) { values.push(value); if (values.length > replay) values.shift(); } _helpers2.Invoke.subscriptionObservers('next', value, items); }, error: error => { if (this.closed) { const hooks = _helpers.HooksManager.get(); hooks.onUnhandledError(error, subscription); } _classPrivateFieldSet(this, _termination, [error]); _helpers2.Invoke.subscriptionObservers('error', error, items); }, complete: () => { if (this.closed) return; _classPrivateFieldSet(this, _termination, true); _helpers2.Invoke.subscriptionObservers('complete', undefined, items); } }); }; if (hooks) { try { const onCreate = hooks.onCreate; if (!_typeCore.TypeGuard.isEmpty(onCreate)) { onCreate.call(hooks, connect); } } catch (err) { _classPrivateFieldSet(this, _termination, [err]); } } } get value() { return _classPrivateFieldGet(this, _value); } get closed() { return Boolean(_classPrivateFieldGet(this, _termination)); } [_Symbol$observable]() { return _Observable.Observable.from(this); } } exports.Multicast = Multicast; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/push/classes/Multicast.ts"],"names":["Symbol","observable","Multicast","Observable","of","item","args","options","hooks","obs","next","complete","from","constructor","subscribe","subscriber","termination","error","value","values","items","add","onSubscribe","TypeGuard","isEmpty","call","connect","delete","onUnsubscribe","opts","Set","replay","Math","max","Number","subscription","closed","start","subs","push","length","shift","Invoke","subscriptionObservers","HooksManager","get","onUnhandledError","undefined","onCreate","err","Boolean"],"mappings":";;;;;;;AAEA;;AACA;;AACA;;AACA;;AACA;;;;;;;;;;;;;;;;;;;;;;qBAwJUA,MAAM,CAACC,U;;AAxIV,MAAMC,SAAN,SACGC,sBADH,CAGP;AAOkB,SAAFC,EAAE,CAAIC,IAAJ,EAAa,GAAGC,IAAhB,EAA8C;AAC5D,UAAMC,OAAO,GAAGD,IAAI,CAAC,CAAD,CAApB;AACA,UAAME,KAAK,GAAGF,IAAI,CAAC,CAAD,CAAlB;AAEA,WAAO,IAAI,IAAJ,CACJG,GAAD,IAAS;AACPA,MAAAA,GAAG,CAACC,IAAJ,CAASL,IAAT;AACAI,MAAAA,GAAG,CAACE,QAAJ;AACD,KAJI,EAKLJ,OALK,EAMLC,KANK,CAAP;AAQD;;AACiB,SAAJI,IAAI,CAChBP,IADgB,EAEhBE,OAFgB,EAGhBC,KAHgB,EAIC;AACjB,QAAIH,IAAI,CAACQ,WAAL,KAAqB,IAAzB,EAA+B,OAAOR,IAAP;AAE/B,UAAMJ,UAAU,GAAG,gBAAKI,IAAL,CAAnB;AACA,WAAO,IAAI,IAAJ,CAAgBI,GAAD,IAASR,UAAU,CAACa,SAAX,CAAqBL,GAArB,CAAxB,EAAmDF,OAAnD,EAA4DC,KAA5D,CAAP;AACD;;AAGMK,EAAAA,WAAW,CAChBE,UADgB,EAEhBR,OAFgB,EAGhBC,KAHgB,EAIhB;AACA,UAAOC,GAAD,IAAS;AACb,YAAMO,WAAW,yBAAG,IAAH,eAAjB;;AACA,UAAIA,WAAJ,EAAiB;AACf,eAAO,OAAOA,WAAP,KAAuB,SAAvB,GACHP,GAAG,CAACE,QAAJ,EADG,GAEHF,GAAG,CAACQ,KAAJ,CAAUD,WAAW,CAAC,CAAD,CAArB,CAFJ;AAGD;;AAED,WAAK,MAAME,KAAX,IAAoBC,MAApB,EAA4B;AAC1BV,QAAAA,GAAG,CAACC,IAAJ,CAASQ,KAAT;AACD;;AAEDE,MAAAA,KAAK,CAACC,GAAN,CAAUZ,GAAV;;AAEA,UAAID,KAAJ,EAAW;AACT,cAAMc,WAAW,GAAGd,KAAK,CAACc,WAA1B;;AACA,YAAI,CAACC,oBAAUC,OAAV,CAAkBF,WAAlB,CAAL,EAAqC;AACnCA,UAAAA,WAAW,CAACG,IAAZ,CAAiBjB,KAAjB,EAAwBkB,OAAxB;AACD;AACF;;AAED,aAAO,MAAM;AACXN,QAAAA,KAAK,CAACO,MAAN,CAAalB,GAAb;;AAEA,YAAID,KAAJ,EAAW;AACT,gBAAMoB,aAAa,GAAGpB,KAAK,CAACoB,aAA5B;;AACA,cAAI,CAACL,oBAAUC,OAAV,CAAkBI,aAAlB,CAAL,EAAuC;AACrCA,YAAAA,aAAa,CAACH,IAAd,CAAmBjB,KAAnB,EAA0BkB,OAA1B;AACD;AACF;AACF,OATD;AAUD,KA/BD;;AADA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAkCA,UAAMG,IAAI,GAAGtB,OAAO,IAAI,EAAxB;AACA,UAAMa,KAAK,GAAG,IAAIU,GAAJ,EAAd;AACA,UAAMX,MAAW,GAAG,EAApB;AACA,UAAMY,MAAM,GAAGC,IAAI,CAACC,GAAL,CAAS,CAAT,EAAYC,MAAM,CAACL,IAAI,CAACE,MAAN,CAAlB,CAAf;AACA,UAAM9B,UAAU,GAAG,IAAIE,sBAAJ,CAAeY,UAAf,CAAnB;;AACA,wCAAcc,IAAI,CAACX,KAAnB;;AACA,8CAAoB,KAApB;;AAEA,QAAIiB,YAAJ;;AACA,UAAMT,OAAO,GAAG,MAAyB;AACvC,UAAIS,YAAY,KAAK,CAACA,YAAY,CAACC,MAAd,IAAwB,KAAKA,MAAlC,CAAhB,EAA2D;AACzD,eAAOD,YAAP;AACD;;AAED,aAAOlC,UAAU,CAACa,SAAX,CAAqB;AAC1BuB,QAAAA,KAAK,EAAGC,IAAD,IAAWH,YAAY,GAAGG,IADP;AAE1B5B,QAAAA,IAAI,EAAGQ,KAAD,IAAW;AACf,cAAI,KAAKkB,MAAT,EAAiB;;AAEjB,8CAAclB,KAAd;;AAEA,cAAIa,MAAJ,EAAY;AACVZ,YAAAA,MAAM,CAACoB,IAAP,CAAYrB,KAAZ;AACA,gBAAIC,MAAM,CAACqB,MAAP,GAAgBT,MAApB,EAA4BZ,MAAM,CAACsB,KAAP;AAC7B;;AAEDC,2BAAOC,qBAAP,CAA6B,MAA7B,EAAqCzB,KAArC,EAA4CE,KAA5C;AACD,SAbyB;AAc1BH,QAAAA,KAAK,EAAGA,KAAD,IAAW;AAChB,cAAI,KAAKmB,MAAT,EAAiB;AACf,kBAAM5B,KAAK,GAAGoC,sBAAaC,GAAb,EAAd;;AACArC,YAAAA,KAAK,CAACsC,gBAAN,CAAuB7B,KAAvB,EAA8BkB,YAA9B;AACD;;AAED,oDAAoB,CAAClB,KAAD,CAApB;;AACAyB,2BAAOC,qBAAP,CAA6B,OAA7B,EAAsC1B,KAAtC,EAA6CG,KAA7C;AACD,SAtByB;AAuB1BT,QAAAA,QAAQ,EAAE,MAAM;AACd,cAAI,KAAKyB,MAAT,EAAiB;;AAEjB,oDAAoB,IAApB;;AACAM,2BAAOC,qBAAP,CAA6B,UAA7B,EAAyCI,SAAzC,EAAoD3B,KAApD;AACD;AA5ByB,OAArB,CAAP;AA8BD,KAnCD;;AAqCA,QAAIZ,KAAJ,EAAW;AACT,UAAI;AACF,cAAMwC,QAAQ,GAAGxC,KAAK,CAACwC,QAAvB;;AACA,YAAI,CAACzB,oBAAUC,OAAV,CAAkBwB,QAAlB,CAAL,EAAkC;AAChCA,UAAAA,QAAQ,CAACvB,IAAT,CAAcjB,KAAd,EAAqBkB,OAArB;AACD;AACF,OALD,CAKE,OAAOuB,GAAP,EAAY;AACZ,kDAAoB,CAACA,GAAD,CAApB;AACD;AACF;AACF;;AACe,MAAL/B,KAAK,GAAU;AACxB,iCAAO,IAAP;AACD;;AACgB,MAANkB,MAAM,GAAY;AAC3B,WAAOc,OAAO,uBAAC,IAAD,gBAAd;AACD;;AACD,yBAA4C;AAC1C,WAAO/C,uBAAWS,IAAX,CAAgB,IAAhB,CAAP;AACD;;AAvIH","sourcesContent":["/* eslint-disable @typescript-eslint/no-use-before-define */\nimport { Push } from '@definitions';\nimport { HooksManager } from '../helpers';\nimport { from } from '../creators/from';\nimport { Invoke } from './helpers';\nimport { Observable } from './Observable';\nimport { NullaryFn, TypeGuard, UnaryFn } from 'type-core';\n\nexport declare namespace Multicast {\n  export interface Options<U> {\n    /** Sets initial `multicast.value`; it won't be emitted. */\n    value?: U;\n    replay?: boolean | number;\n  }\n  export interface Hooks {\n    onCreate?: UnaryFn<Multicast.Connect>;\n    onSubscribe?: UnaryFn<Multicast.Connect>;\n    onUnsubscribe?: UnaryFn<Multicast.Connect>;\n  }\n  export type Connect = NullaryFn<Push.Subscription>;\n}\n\nexport class Multicast<T = any, U extends T | void = T | void>\n  extends Observable<T>\n  implements Push.Multicast<T, U>\n{\n  public static of<T>(item: T): Multicast<T, T>;\n  public static of<T>(\n    item: T,\n    options?: Multicast.Options<T>,\n    hooks?: Multicast.Hooks\n  ): Multicast<T, T>;\n  public static of<T>(item: T, ...args: any[]): Multicast<T, T> {\n    const options = args[0];\n    const hooks = args[1];\n\n    return new this<T, T>(\n      (obs) => {\n        obs.next(item);\n        obs.complete();\n      },\n      options,\n      hooks\n    );\n  }\n  public static from<T, U extends T | void = T | void>(\n    item: Push.Convertible<T>,\n    options?: Multicast.Options<U>,\n    hooks?: Multicast.Hooks\n  ): Multicast<T, U> {\n    if (item.constructor === this) return item;\n\n    const observable = from(item);\n    return new this<T, U>((obs) => observable.subscribe(obs), options, hooks);\n  }\n  #value: T | U;\n  #termination: boolean | [Error];\n  public constructor(\n    subscriber: Push.Subscriber<T>,\n    options?: Multicast.Options<U>,\n    hooks?: Multicast.Hooks\n  ) {\n    super((obs) => {\n      const termination = this.#termination;\n      if (termination) {\n        return typeof termination === 'boolean'\n          ? obs.complete()\n          : obs.error(termination[0]);\n      }\n\n      for (const value of values) {\n        obs.next(value);\n      }\n\n      items.add(obs);\n\n      if (hooks) {\n        const onSubscribe = hooks.onSubscribe;\n        if (!TypeGuard.isEmpty(onSubscribe)) {\n          onSubscribe.call(hooks, connect);\n        }\n      }\n\n      return () => {\n        items.delete(obs);\n\n        if (hooks) {\n          const onUnsubscribe = hooks.onUnsubscribe;\n          if (!TypeGuard.isEmpty(onUnsubscribe)) {\n            onUnsubscribe.call(hooks, connect);\n          }\n        }\n      };\n    });\n\n    const opts = options || {};\n    const items = new Set<Push.SubscriptionObserver<T>>();\n    const values: T[] = [];\n    const replay = Math.max(0, Number(opts.replay));\n    const observable = new Observable(subscriber);\n    this.#value = opts.value as U;\n    this.#termination = false;\n\n    let subscription: any;\n    const connect = (): Push.Subscription => {\n      if (subscription && (!subscription.closed || this.closed)) {\n        return subscription;\n      }\n\n      return observable.subscribe({\n        start: (subs) => (subscription = subs),\n        next: (value) => {\n          if (this.closed) return;\n\n          this.#value = value;\n\n          if (replay) {\n            values.push(value);\n            if (values.length > replay) values.shift();\n          }\n\n          Invoke.subscriptionObservers('next', value, items);\n        },\n        error: (error) => {\n          if (this.closed) {\n            const hooks = HooksManager.get();\n            hooks.onUnhandledError(error, subscription);\n          }\n\n          this.#termination = [error];\n          Invoke.subscriptionObservers('error', error, items);\n        },\n        complete: () => {\n          if (this.closed) return;\n\n          this.#termination = true;\n          Invoke.subscriptionObservers('complete', undefined, items);\n        }\n      });\n    };\n\n    if (hooks) {\n      try {\n        const onCreate = hooks.onCreate;\n        if (!TypeGuard.isEmpty(onCreate)) {\n          onCreate.call(hooks, connect);\n        }\n      } catch (err) {\n        this.#termination = [err as Error];\n      }\n    }\n  }\n  public get value(): T | U {\n    return this.#value;\n  }\n  public get closed(): boolean {\n    return Boolean(this.#termination);\n  }\n  public [Symbol.observable](): Observable<T> {\n    return Observable.from(this);\n  }\n}\n"]}