UNPKG

chain-able

Version:

interfaces that describe their intentions.

122 lines (108 loc) 11.7 kB
var toarr = require('../deps/to-arr') var traverse = require('../deps/traverse') var eq = require('../deps/traversers/eq') var match = require('../deps/matcher') var getPathSegments = require('../deps/dot/segments') var dot = require('../deps/dot') var OBSERVERS_KEY = require('../deps/meta/observers') /** * scoped clones * @private * @type {Map} */ var objs = new Map() /** * @desc > subscribe to changes * ❗ called only on **change** * observers are only called when data they subscribe to changes * * @since 3.0.1 * @class Observe * @member Observe * @extends {ChainedMap} * @extends {DotProp} * @memberOf compose * @category Chainable * * @param {Class | Composable} SuperClass composable class * @return {Observe} class * * @tests Observe * @types Observe * * @see ChainedMap * @see DotProp * @see deps/matcher * @see deps/traversers/eq * @see deps/traverse * @see DotProp * * {@link https://github.com/iluwatar/java-design-patterns/tree/master/observer observer-pattern} * {@link https://github.com/ReactiveX/rxjs/blob/master/src/Subscriber.ts reactivex} * {@link https://github.com/sindresorhus/awesome-observables awesome-observables} * {@link https://medium.com/@benlesh/learning-observable-by-building-observable-d5da57405d87 building-observables} * {@link https://github.com/addyosmani/essential-js-design-patterns/blob/master/diagrams/observer.png js-observer-png} * {@link https://github.com/addyosmani/essential-js-design-patterns/blob/master/diagrams/publishsubscribe.png pubsub-png} * @see {@link reactivex} * @see {@link awesome-observables} * @see {@link building-observables} * @see {@link observer-pattern} * * @example * * const {compose} = require('chain-able') * const {DotProp} = compose * new DotProp() * //=> DotProp * */ module.exports = function (SuperClass) { return (function (SuperClass) { function Observe () { SuperClass.apply(this, arguments); } if ( SuperClass ) Observe.__proto__ = SuperClass; Observe.prototype = Object.create( SuperClass && SuperClass.prototype ); Observe.prototype.constructor = Observe; Observe.prototype.observe = function observe (properties, fn) { var this$1 = this; var props = toarr(properties) var hashKey = props.join('_') var data = {} /* prettier-ignore */ return this.meta(OBSERVERS_KEY, function (changed) { /** * match the keys, make the data out of it */ var m = match(changed.key, props) // @@debugger for (var i = 0; i < m.length; i++) { var segments = getPathSegments(m[i]) dot.set(data, segments, this$1.get(segments)) } /** * if we have called it at least once... * and it has not changed, leave it * else * clone it * call the observer */ if (objs.has(hashKey) && eq(objs.get(hashKey), data)) { // @@debugger return } // @@debugger /** * it did change - clone it for next deepEquals check */ objs.set(hashKey, traverse(data).clone()) /** * call the observer - it matched & data changed */ fn.call(this$1, data, this$1) }) }; return Observe; }(SuperClass)) } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"Observe.js","sources":["Observe.js"],"sourcesContent":["const toarr = require('../deps/to-arr')\nconst traverse = require('../deps/traverse')\nconst eq = require('../deps/traversers/eq')\nconst match = require('../deps/matcher')\nconst getPathSegments = require('../deps/dot/segments')\nconst dot = require('../deps/dot')\nconst OBSERVERS_KEY = require('../deps/meta/observers')\n\n/**\n * scoped clones\n * @private\n * @type {Map}\n */\nlet objs = new Map()\n\n/**\n * @desc > subscribe to changes\n *       ❗ called only on **change**\n *       observers are only called when data they subscribe to changes\n *\n * @since 3.0.1\n * @class Observe\n * @member Observe\n * @extends {ChainedMap}\n * @extends {DotProp}\n * @memberOf compose\n * @category Chainable\n *\n * @param  {Class | Composable} SuperClass composable class\n * @return {Observe} class\n *\n * @tests Observe\n * @types Observe\n *\n * @see ChainedMap\n * @see DotProp\n * @see deps/matcher\n * @see deps/traversers/eq\n * @see deps/traverse\n * @see DotProp\n *\n * {@link https://github.com/iluwatar/java-design-patterns/tree/master/observer observer-pattern}\n * {@link https://github.com/ReactiveX/rxjs/blob/master/src/Subscriber.ts reactivex}\n * {@link https://github.com/sindresorhus/awesome-observables awesome-observables}\n * {@link https://medium.com/@benlesh/learning-observable-by-building-observable-d5da57405d87 building-observables}\n * {@link https://github.com/addyosmani/essential-js-design-patterns/blob/master/diagrams/observer.png js-observer-png}\n * {@link https://github.com/addyosmani/essential-js-design-patterns/blob/master/diagrams/publishsubscribe.png pubsub-png}\n * @see {@link reactivex}\n * @see {@link awesome-observables}\n * @see {@link building-observables}\n * @see {@link observer-pattern}\n *\n * @example\n *\n *    const {compose} = require('chain-able')\n *    const {DotProp} = compose\n *    new DotProp()\n *    //=> DotProp\n *\n */\nmodule.exports = SuperClass => {\n  return class Observe extends SuperClass {\n    /**\n     * @desc observe properties when they change\n     *\n     * @method\n     * @memberOf Observe\n     * @since 4.0.0 <- refactored with dot-prop\n     * @since 1.0.0\n     *\n     * @param  {Matchable} properties Matchable properties to observe\n     * @param  {Function} fn onChanged\n     * @return {Chain} @chainable\n     *\n     * @see traversers/eq\n     * @see toarr\n     * @see matcher\n     *\n     * @see {@link https://jsfiddle.net/wqxuags2/28/} for a Demo Clock with observable\n     * @see {@link examples/playground/TodoStore} TodoStore\n     *\n     * @TODO gotta update `data` if `deleting` too...\n     * @TODO un-observe\n     * @TODO should hash these callback properties\n     * @TODO just throttle the `.set` to allow easier version of .commit\n     *\n     * @example\n     *\n     *   const Chain = require('chain-able')\n     *\n     *   const chain = new Chain()\n     *   const log = arg => console.log(arg)\n     *\n     *   chain\n     *     .extend(['eh'])\n     *     .observe('eh', data => log(data))\n     *     .eh(true)\n     *   //=> {eh: true}\n     *\n     * @example\n     *\n     *    chain\n     *      .extend(['canada', 'timbuck'])\n     *      .observe(['canad*'], data => console.log(data.canada))\n     *      .canada(true)\n     *      .canada(true)\n     *      .timbuck(false)\n     *\n     *    //=> true\n     *    //=> false\n     *\n     *    // only called when changed,\n     *    // otherwise it would be 2 `true` & 1 `false`\n     */\n    observe(properties, fn) {\n      const props = toarr(properties)\n      const hashKey = props.join('_')\n      let data = {}\n\n      /* prettier-ignore */\n      return this.meta(OBSERVERS_KEY, changed => {\n        /**\n         * match the keys, make the data out of it\n         */\n        const m = match(changed.key, props)\n\n        // @@debugger\n\n        for (let i = 0; i < m.length; i++) {\n          const segments = getPathSegments(m[i])\n          dot.set(data, segments, this.get(segments))\n        }\n\n        /**\n         * if we have called it at least once...\n         *    and it has not changed, leave it\n         * else\n         *    clone it\n         *    call the observer\n         */\n        if (objs.has(hashKey) && eq(objs.get(hashKey), data)) {\n          // @@debugger\n          return\n        }\n\n        // @@debugger\n\n        /**\n         * it did change - clone it for next deepEquals check\n         */\n        objs.set(hashKey, traverse(data).clone())\n\n        /**\n         * call the observer - it matched & data changed\n         */\n        fn.call(this, data, this)\n      })\n    }\n  }\n}\n"],"names":["const","let","this"],"mappings":"AAAAA,GAAK,CAAC,KAAK,GAAG,OAAO,CAAC,gBAAgB,CAAC;AACvCA,GAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,kBAAkB,CAAC;AAC5CA,GAAK,CAAC,EAAE,GAAG,OAAO,CAAC,uBAAuB,CAAC;AAC3CA,GAAK,CAAC,KAAK,GAAG,OAAO,CAAC,iBAAiB,CAAC;AACxCA,GAAK,CAAC,eAAe,GAAG,OAAO,CAAC,sBAAsB,CAAC;AACvDA,GAAK,CAAC,GAAG,GAAG,OAAO,CAAC,aAAa,CAAC;AAClCA,GAAK,CAAC,aAAa,GAAG,OAAO,CAAC,wBAAwB,CAAC;;;;;;;AAOvDC,GAAG,CAAC,IAAI,GAAG,IAAI,GAAG,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA+CpB,MAAM,CAAC,OAAO,GAAG,UAAA,UAAU,CAAA,CAAC,AAAG;EAC7B,OAAO;IAAA,AAAgC,AAAC;;;;;;;;IAAA,AAqDtC,kBAAA,OAAO,oBAAA,CAAC,UAAU,EAAE,EAAE,EAAE,CAAC;;AAAA;MACvBD,GAAK,CAAC,KAAK,GAAG,KAAK,CAAC,UAAU,CAAC;MAC/BA,GAAK,CAAC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC;MAC/BC,GAAG,CAAC,IAAI,GAAG,EAAE;;;MAGb,OAAO,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,UAAA,OAAO,CAAA,CAAC,AAAG;;;;QAIzCD,GAAK,CAAC,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,KAAK,CAAC;;;;QAInC,KAAKC,GAAG,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;UACjCD,GAAK,CAAC,QAAQ,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;UACtC,GAAG,CAAC,GAAG,CAAC,IAAI,EAAE,QAAQ,EAAEE,MAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;SAC5C;;;;;;;;;QASD,IAAI,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,IAAI,CAAC,EAAE;;UAEpD,MAAM;SACP;;;;;;;QAOD,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC;;;;;QAKzC,EAAE,CAAC,IAAI,CAACA,MAAI,EAAE,IAAI,EAAEA,MAAI,CAAC;OAC1B,CAAC;KACH,CAAA,AACF;;;IAjG4B,UAiG5B,EAAA;CACF;"}