multitude
Version:
Comprehensive streams for push and pull
105 lines (91 loc) • 11 kB
JavaScript
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.useObservable = useObservable;
exports.usePropsMulticast = usePropsMulticast;
var _from = require("../creators/from");
var _Multicast = require("../classes/Multicast");
var _typeCore = require("type-core");
function useObservable(React, observable, projection) {
const state = React.useState(0);
const store = React.useMemo(() => {
const members = {
subscription: null,
result: null,
response: null
};
let i = 0;
let open = false;
let updated = false;
function update(value) {
updated = true;
const previous = members.response;
members.result = value;
members.response = projection ? projection(value) : value;
if (open && members.response !== previous) state[1](i = i + 1);
}
(0, _from.from)(_typeCore.TypeGuard.isFunction(observable) ? observable() : observable).subscribe({
start(subscription) {
members.subscription = subscription;
},
next(item) {
update({
success: true,
done: false,
data: item
});
},
error(reason) {
update({
success: false,
done: true,
data: reason
});
},
complete() {
update(members.result && members.result.success ? {
success: true,
done: true,
data: members.result.data
} : {
success: false,
done: true,
data: Error(`Observable completed before pushing any value`)
});
}
});
open = true;
if (!updated && projection) members.response = projection(null);
return members;
}, []);
React.useEffect(() => () => {
if (store.subscription) store.subscription.unsubscribe();
}, []);
return store.response;
}
function usePropsMulticast(React, props, projection) {
const store = React.useMemo(() => {
let observer;
const multicast = new _Multicast.Multicast(obs => {
observer = obs;
}, {
replay: true
}, {
onCreate: connect => connect()
});
observer.next(projection ? projection(props) : props);
return {
props,
observer,
multicast
};
}, []);
if (props !== store.props) {
store.props = props;
const value = projection ? projection(props) : props;
if (value !== store.multicast.value) store.observer.next(value);
}
return store.multicast;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/push/utils/react.ts"],"names":["useObservable","React","observable","projection","state","useState","store","useMemo","members","subscription","result","response","i","open","updated","update","value","previous","TypeGuard","isFunction","subscribe","start","next","item","success","done","data","error","reason","complete","Error","useEffect","unsubscribe","usePropsMulticast","props","observer","multicast","Multicast","obs","replay","onCreate","connect"],"mappings":"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","sourcesContent":["import { Push } from '@definitions';\nimport { from } from '../creators/from';\nimport { Multicast } from '../classes/Multicast';\nimport { NullaryFn, TypeGuard, UnaryFn } from 'type-core';\n\nexport interface ReactHooksDependency {\n  useState<S>(initial: S): [S, UnaryFn<S>];\n  useMemo<T>(factory: () => T, deps?: any[]): T;\n  useEffect(effect: NullaryFn<void | NullaryFn>, deps?: any[]): void;\n}\n\nexport type ObservableHookResult<T> =\n  | { success: true; done: boolean; data: T }\n  | { success: false; done: true; data: Error };\n\nexport function useObservable<T, U = ObservableHookResult<T> | null>(\n  React: ReactHooksDependency,\n  observable: Push.Convertible<T> | NullaryFn<Push.Convertible<T>>,\n  projection?: UnaryFn<ObservableHookResult<T> | null, U>\n): U {\n  const state = React.useState(0);\n\n  const store = React.useMemo(() => {\n    const members = {\n      subscription: null as null | Push.Subscription,\n      result: null as ObservableHookResult<T> | null,\n      response: null as U | null\n    };\n\n    let i = 0;\n    let open = false;\n    let updated = false;\n    function update(value: ObservableHookResult<T> | null): void {\n      updated = true;\n      const previous = members.response;\n      members.result = value;\n      members.response = projection ? projection(value) : (value as any);\n      if (open && members.response !== previous) state[1]((i = i + 1));\n    }\n\n    from(\n      TypeGuard.isFunction(observable) ? observable() : observable\n    ).subscribe({\n      start(subscription) {\n        members.subscription = subscription;\n      },\n      next(item) {\n        update({ success: true, done: false, data: item });\n      },\n      error(reason) {\n        update({ success: false, done: true, data: reason });\n      },\n      complete() {\n        update(\n          members.result && members.result.success\n            ? { success: true, done: true, data: members.result.data }\n            : {\n                success: false,\n                done: true,\n                data: Error(`Observable completed before pushing any value`)\n              }\n        );\n      }\n    });\n\n    open = true;\n    if (!updated && projection) members.response = projection(null);\n    return members;\n  }, []);\n\n  React.useEffect(\n    () => () => {\n      if (store.subscription) store.subscription.unsubscribe();\n    },\n    []\n  );\n\n  return store.response as U;\n}\n\nexport function usePropsMulticast<P, R = P>(\n  React: ReactHooksDependency,\n  props: P,\n  projection?: UnaryFn<P, R>\n): Push.Multicast<R> {\n  const store = React.useMemo(() => {\n    let observer: any;\n\n    const multicast = new Multicast(\n      (obs) => {\n        observer = obs;\n      },\n      { replay: true },\n      { onCreate: (connect) => connect() }\n    );\n\n    observer.next(projection ? projection(props) : props);\n\n    return { props, observer, multicast };\n  }, []);\n\n  if (props !== store.props) {\n    store.props = props;\n    const value = projection ? projection(props) : props;\n    if (value !== store.multicast.value) store.observer.next(value);\n  }\n\n  return store.multicast;\n}\n"]}
;