chain-able
Version:
interfaces that describe their intentions.
247 lines (233 loc) • 16.2 kB
JavaScript
/**
* @since 2.0.0
*/
var dot = require('../deps/dot')
var isDot = require('../deps/is/dot')
/**
* @desc checks if this.meta.dot != false & isDot(key) - scoped
*
* @private
* @since 3.0.1
*
* @param {string} key key in .get/.has/.delete/set
* @param {DotProp} thisArg Chain
* @return {boolean} shouldDot
*
* @see DotProp.dot
* @see deps/is/dot
* @see deps/meta
* @see https://lodash.com/docs/#get
* @see https://github.com/sindresorhus/dot-prop
*
* @example
*
* const chain = new DotProp()
* shouldDot('me.me', chain)
* //=> true
*
* const chain = new DotProp()
* shouldDot('me', chain)
* //=> false
*
* const chain = new DotProp()
* chain.dot(false)
* shouldDot('me.me', chain)
* //=> false
*
*/
var shouldDot = function (key, thisArg) { return thisArg.meta.dot !== false && isDot(key); }
/**
* @class DotProp
* @member Observe
* @extends {ChainedMap}
* @memberOf compose
* @category Chainable
*
* @param {Class | Composable} SuperClass composable class
* @return {DotProp} class
*
* @tests DotProp
* @types DotProp
*
* @see deps/dot
*
* @example
*
* const {compose} = require('chain-able')
* const {DotProp} = compose
* new DotProp()
* //=> DotProp
*
* @example
*
* const chain = new Chain()
*
* chain.set('moose.simple', 1)
* //=> Chain
*
* chain.get('moose.simple')
* //=>1
*
* chain.get('moose')
* //=> {simple: 1}
*
* chain.set('moose.canada.eh', true).set('moose.canada.igloo', true)
* //=> Chain
*
* //set, has, get, delete :-)
* chain.delete('moose.canada.eh')
* //=> Chain
*
* //also works with an array (moose.canada.igloo)
* chain.get(['moose', 'canada', 'igloo'])
* //=> true
*
*/
module.exports = function (SuperClass) {
return (function (SuperClass) {
function DotProp () {
SuperClass.apply(this, arguments);
}
if ( SuperClass ) DotProp.__proto__ = SuperClass;
DotProp.prototype = Object.create( SuperClass && SuperClass.prototype );
DotProp.prototype.constructor = DotProp;
DotProp.prototype.dot = function dot (useDot) {
this.meta.dot = useDot
return this
};
/**
* @desc since we have a map,
* we need to ensure the first property is available
* otherwise we have an empty map.entries obj
* which does nothing by reference
* @since 3.0.1
* @memberOf DotProp
*
* @override
* @inheritdoc
*
* @see ChainedMap.set
* @see .dot
*
* @example
* const chain = new Chain()
*
* chain.set('moose.simple', 1)
* //=> Chain store:Map: { moose: { simple: 1 } }
*/
DotProp.prototype.set = function set (key, val) {
if (shouldDot(key, this)) {
// first accessor
// @example: `canada` in `canada.eh`
var prop = key.split('.').shift()
// we already know it is .dot, call super instead
// if (!super.has(prop)) super.set(prop, {})
// spread
var data = SuperClass.prototype.entries.call(this)
// set on the spread data
dot.set(data, key, val)
// is already by ref, but be extra safe, + observables
return SuperClass.prototype.set.call(this, prop, data[prop], key)
}
return SuperClass.prototype.set.call(this, key, val)
};
/**
* @desc dot-prop enabled get
* @method get
* @memberOf DotProp
*
* @since 3.0.1
* @override
* @inheritdoc
*
* @param {Primitive} key dot prop key, or any primitive key
* @param {any} [fallback=undefined] fallback value, if it cannot find value with key path
* @return {any} value for path, or fallback value if provided
*
* @see ChainedMap.get
* @see deps/dot
* @see deps/is/dot
*
* @TODO dot-prop on non-store instance.property when using nested chains...
*
* @example
*
* chain.set('moose.simple', 1)
* //=> Chain
*
* chain.get('moose.simple')
* //=>1
*
* chain.get('moose')
* //=> {simple: 1}
*
* @example
*
* //also works with an array (moose.simple)
* chain.get(['moose', 'simple'])
* //=> 1
*
*/
DotProp.prototype.get = function get (key, fallback) {
return shouldDot(key, this)
? dot.get(SuperClass.prototype.entries.call(this), key, fallback)
: SuperClass.prototype.get.call(this, key)
};
/**
* @method has
* @methodOf DotProp
* @since 3.0.1
* @override
* @inheritdoc
*
* @see deps/dot
* @see deps/is/dot
*
* @example
*
* chain.set('one.two', 3)
* chain.has('one.two')
* //=> true
*
*/
DotProp.prototype.has = function has (key) {
return shouldDot(key, this)
? dot.has(SuperClass.prototype.entries.call(this), key)
: SuperClass.prototype.has.call(this, key)
};
/**
* @method delete
* @methodOf DotProp
* @since 3.0.1
*
* @override
* @inheritdoc
* @see deps/dot
* @see deps/is/dot
*
* @example
*
* chain.set('moose.canada.eh', true)
* chain.set('moose.canada.igloo', true)
* //=> Chain
*
* chain.delete('moose.canada.eh')
* //=> Chain
*
* chain.has('moose.canada.eh')
* //=> true
*
* //still has moose.canada.igloo
* chain.has('moose.canada')
* //=> true
*
*/
DotProp.prototype.delete = function delete$1 (key) {
return shouldDot(key, this)
? dot.delete(SuperClass.prototype.entries.call(this), key)
: SuperClass.prototype.delete.call(this, key)
};
return DotProp;
}(SuperClass))
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"DotProp.js","sources":["DotProp.js"],"sourcesContent":["/**\n * @since 2.0.0\n */\nconst dot = require('../deps/dot')\nconst isDot = require('../deps/is/dot')\n\n/**\n * @desc checks if this.meta.dot != false & isDot(key) - scoped\n *\n * @private\n * @since 3.0.1\n *\n * @param  {string} key key in .get/.has/.delete/set\n * @param  {DotProp} thisArg Chain\n * @return {boolean} shouldDot\n *\n * @see DotProp.dot\n * @see deps/is/dot\n * @see deps/meta\n * @see https://lodash.com/docs/#get\n * @see https://github.com/sindresorhus/dot-prop\n *\n * @example\n *\n *  const chain = new DotProp()\n *  shouldDot('me.me', chain)\n *  //=> true\n *\n *  const chain = new DotProp()\n *  shouldDot('me', chain)\n *  //=> false\n *\n *  const chain = new DotProp()\n *  chain.dot(false)\n *  shouldDot('me.me', chain)\n *  //=> false\n *\n */\nconst shouldDot = (key, thisArg) => thisArg.meta.dot !== false && isDot(key)\n\n/**\n * @class DotProp\n * @member Observe\n * @extends {ChainedMap}\n * @memberOf compose\n * @category Chainable\n *\n * @param  {Class | Composable} SuperClass composable class\n * @return {DotProp} class\n *\n * @tests DotProp\n * @types DotProp\n *\n * @see deps/dot\n *\n * @example\n *\n *    const {compose} = require('chain-able')\n *    const {DotProp} = compose\n *    new DotProp()\n *    //=> DotProp\n *\n * @example\n *\n *    const chain = new Chain()\n *\n *    chain.set('moose.simple', 1)\n *    //=> Chain\n *\n *    chain.get('moose.simple')\n *    //=>1\n *\n *    chain.get('moose')\n *    //=> {simple: 1}\n *\n *    chain.set('moose.canada.eh', true).set('moose.canada.igloo', true)\n *    //=> Chain\n *\n *    //set, has, get, delete :-)\n *    chain.delete('moose.canada.eh')\n *    //=> Chain\n *\n *    //also works with an array (moose.canada.igloo)\n *    chain.get(['moose', 'canada', 'igloo'])\n *    //=> true\n *\n */\nmodule.exports = SuperClass => {\n  return class DotProp extends SuperClass {\n    /**\n     * @method dot\n     * @methodChain DotProp\n     * @since 3.0.1\n     *\n     * @param  {boolean} [useDot=undefined] use dot prop or not\n     * @return {DotProp} @chainable\n     *\n     * @see deps/meta\n     *\n     * @example\n     *\n     *     const chain = new Chain()\n     *     chain.dot(false)\n     *     chain.set('moose.simple', 1)\n     *\n     *     toArr(chain.store.keys())\n     *     //=> ['moose.simple']\n     *\n     */\n    dot(useDot) {\n      this.meta.dot = useDot\n      return this\n    }\n\n    /**\n     * @desc since we have a map,\n     *       we need to ensure the first property is available\n     *       otherwise we have an empty map.entries obj\n     *       which does nothing by reference\n     * @since 3.0.1\n     * @memberOf DotProp\n     *\n     * @override\n     * @inheritdoc\n     *\n     * @see ChainedMap.set\n     * @see .dot\n     *\n     * @example\n     *    const chain = new Chain()\n     *\n     *    chain.set('moose.simple', 1)\n     *    //=> Chain store:Map:  { moose: { simple: 1 } }\n     */\n    set(key, val) {\n      if (shouldDot(key, this)) {\n        // first accessor\n        // @example: `canada` in `canada.eh`\n        const prop = key.split('.').shift()\n\n        // we already know it is .dot, call super instead\n        // if (!super.has(prop)) super.set(prop, {})\n\n        // spread\n        const data = super.entries()\n\n        // set on the spread data\n        dot.set(data, key, val)\n\n        // is already by ref, but be extra safe, + observables\n        return super.set(prop, data[prop], key)\n      }\n      return super.set(key, val)\n    }\n\n    /**\n     * @desc dot-prop enabled get\n     * @method get\n     * @memberOf DotProp\n     *\n     * @since 3.0.1\n     * @override\n     * @inheritdoc\n     *\n     * @param {Primitive} key dot prop key, or any primitive key\n     * @param {any} [fallback=undefined] fallback value, if it cannot find value with key path\n     * @return {any} value for path, or fallback value if provided\n     *\n     * @see ChainedMap.get\n     * @see deps/dot\n     * @see deps/is/dot\n     *\n     * @TODO dot-prop on non-store instance.property when using nested chains...\n     *\n     * @example\n     *\n     *    chain.set('moose.simple', 1)\n     *    //=> Chain\n     *\n     *    chain.get('moose.simple')\n     *    //=>1\n     *\n     *    chain.get('moose')\n     *    //=> {simple: 1}\n     *\n     * @example\n     *\n     *    //also works with an array (moose.simple)\n     *    chain.get(['moose', 'simple'])\n     *    //=> 1\n     *\n     */\n    get(key, fallback) {\n      return shouldDot(key, this)\n        ? dot.get(super.entries(), key, fallback)\n        : super.get(key)\n    }\n\n    /**\n     * @method has\n     * @methodOf DotProp\n     * @since 3.0.1\n     * @override\n     * @inheritdoc\n     *\n     * @see deps/dot\n     * @see deps/is/dot\n     *\n     * @example\n     *\n     *  chain.set('one.two', 3)\n     *  chain.has('one.two')\n     *  //=> true\n     *\n     */\n    has(key) {\n      return shouldDot(key, this)\n        ? dot.has(super.entries(), key)\n        : super.has(key)\n    }\n\n    /**\n     * @method delete\n     * @methodOf DotProp\n     * @since 3.0.1\n     *\n     * @override\n     * @inheritdoc\n     * @see deps/dot\n     * @see deps/is/dot\n     *\n     * @example\n     *\n     *    chain.set('moose.canada.eh', true)\n     *    chain.set('moose.canada.igloo', true)\n     *    //=> Chain\n     *\n     *    chain.delete('moose.canada.eh')\n     *    //=> Chain\n     *\n     *    chain.has('moose.canada.eh')\n     *    //=> true\n     *\n     *    //still has moose.canada.igloo\n     *    chain.has('moose.canada')\n     *    //=> true\n     *\n     */\n    delete(key) {\n      return shouldDot(key, this)\n        ? dot.delete(super.entries(), key)\n        : super.delete(key)\n    }\n  }\n}\n"],"names":["const","super"],"mappings":"AAAA;;;AAGAA,GAAK,CAAC,GAAG,GAAG,OAAO,CAAC,aAAa,CAAC;AAClCA,GAAK,CAAC,KAAK,GAAG,OAAO,CAAC,gBAAgB,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAkCvCA,GAAK,CAAC,SAAS,GAAG,SAAA,CAAC,GAAG,EAAE,OAAO,EAAE,AAAG,SAAA,OAAO,CAAC,IAAI,CAAC,GAAG,KAAK,KAAK,IAAI,KAAK,CAAC,GAAG,CAAC,GAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAiD5E,MAAM,CAAC,OAAO,GAAG,UAAA,UAAU,CAAA,CAAC,AAAG;EAC7B,OAAO;IAAA,AAAgC,AAAC;;;;;;;;IAAA,AAqBtC,kBAAA,GAAG,gBAAA,CAAC,MAAM,EAAE;MACV,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,MAAM;MACtB,OAAO,IAAI;KACZ,CAAA;;;;;;;;;;;;;;;;;;;;;;IAsBD,kBAAA,GAAG,gBAAA,CAAC,GAAG,EAAE,GAAG,EAAE;MACZ,IAAI,SAAS,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE;;;QAGxBA,GAAK,CAAC,IAAI,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE;;;;;;QAMnCA,GAAK,CAAC,IAAI,GAAGC,oBAAK,CAAC,OAAO,KAAA,CAAC,IAAA,CAAC;;;QAG5B,GAAG,CAAC,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE,GAAG,CAAC;;;QAGvB,OAAOA,oBAAK,CAAC,GAAG,KAAA,CAAC,MAAA,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE,GAAG,CAAC;OACxC;MACD,OAAOA,oBAAK,CAAC,GAAG,KAAA,CAAC,MAAA,GAAG,EAAE,GAAG,CAAC;KAC3B,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAuCD,kBAAA,GAAG,gBAAA,CAAC,GAAG,EAAE,QAAQ,EAAE;MACjB,OAAO,SAAS,CAAC,GAAG,EAAE,IAAI,CAAC;UACvB,GAAG,CAAC,GAAG,CAACA,oBAAK,CAAC,OAAO,KAAA,CAAC,IAAA,CAAC,EAAE,GAAG,EAAE,QAAQ,CAAC;UACvCA,oBAAK,CAAC,GAAG,KAAA,CAAC,MAAA,GAAG,CAAC;KACnB,CAAA;;;;;;;;;;;;;;;;;;;IAmBD,kBAAA,GAAG,gBAAA,CAAC,GAAG,EAAE;MACP,OAAO,SAAS,CAAC,GAAG,EAAE,IAAI,CAAC;UACvB,GAAG,CAAC,GAAG,CAACA,oBAAK,CAAC,OAAO,KAAA,CAAC,IAAA,CAAC,EAAE,GAAG,CAAC;UAC7BA,oBAAK,CAAC,GAAG,KAAA,CAAC,MAAA,GAAG,CAAC;KACnB,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IA6BD,kBAAA,MAAM,qBAAA,CAAC,GAAG,EAAE;MACV,OAAO,SAAS,CAAC,GAAG,EAAE,IAAI,CAAC;UACvB,GAAG,CAAC,MAAM,CAACA,oBAAK,CAAC,OAAO,KAAA,CAAC,IAAA,CAAC,EAAE,GAAG,CAAC;UAChCA,oBAAK,CAAC,MAAM,KAAA,CAAC,MAAA,GAAG,CAAC;KACtB,CAAA,AACF;;;IArK4B,UAqK5B,EAAA;CACF;"}