chain-able
Version:
interfaces that describe their intentions.
196 lines (178 loc) • 15.9 kB
JavaScript
var TraverseChain = require('../TraverseChain')
var isObj = require('../deps/is/obj')
var isTrue = require('../deps/is/true')
var isFalse = require('../deps/is/false')
var isUndefined = require('../deps/is/undefined')
var ObjectKeys = require('../deps/util/keys')
var dotPropPaths = require('../deps/dot/paths')
var TRANSFORMERS_KEY = require('../deps/meta/transformers')
var OBSERVERS_KEY = require('../deps/meta/observers')
/**
* @param {Class | Composable} SuperClass composable class
* @return {TransformChain} class
* @example
* compose(class {})
* //=> TransformChain
*/
module.exports = function (SuperClass) {
/**
* @class TransformChain
* @member TransformChain
* @extends {ChainedMap}
* @memberOf compose
* @category Chainable
*
* @tests TransformChain
* @types TransformChain
*
* @symb 🤖
* @type {Map}
*
* @see deps/traverse
* @see TraverseChain
*
* {@link https://github.com/iluwatar/java-design-patterns/tree/master/state state-pattern}
* {@link https://github.com/iluwatar/java-design-patterns/tree/master/strategy strategy-pattern}
*/
return (function (SuperClass) {
function Transform () {
SuperClass.apply(this, arguments);
}
if ( SuperClass ) Transform.__proto__ = SuperClass;
Transform.prototype = Object.create( SuperClass && SuperClass.prototype );
Transform.prototype.constructor = Transform;
Transform.prototype.traverse = function traverse (useThis) {
if ( useThis === void 0 ) useThis = false;
/* prettier-ignore */
return new TraverseChain(this)
.obj(isFalse(useThis)
? this.entries(true)
: isTrue(useThis)
? this
: useThis
)
};
/**
* @since 1.0.2
* @memberOf TransformChain
*
* @param {string | Function} key currently just string
* @param {Function} value callback accepting the value as only arg to transform with
* @return {TransformChain} @chainable
*
* @TODO dot-prop here
*
* @example
*
* // coerce values with .id into the value they hold
* chain
* .transform('dis', val => (typeof val === 'string' ? val : val.id))
*
* chain.set('dis', 'eh')
* chain.get('dis')
* //=> 'eh'
*
* chain.set('dis', {id: 'eh'})
* chain.get('dis')
* //=> 'eh'
*
*
* @example
*
* import {format} from 'date-fns/esm'
* import {Chain} from 'chain-able'
*
* const chain = new Chain()
* chain.transform('created_at', date => format(date, 'MM/DD/YYYY'))
* chain.set('created_at', new Date())
*
* // is formatted human-readable pretty!
* const {created_at} = chain.entries()
* //=> '02/11/2014'
*
*/
Transform.prototype.transform = function transform (key, value) {
return this.meta(TRANSFORMERS_KEY, key, value)
};
/**
* @memberOf TransformChain
*
* @override
* @inheritdoc
* @since 1.0.0
*
* @param {Primitive} key key to set with
* @param {any} val value to set for key
* @param {undefined | string | Array<string>} dotPropKey special key used for initializing dot prop values in an optimized way to keep reference
* @return {Chainable} @chainable
*
* @see this.observe, this.transform
*/
Transform.prototype.set = function set (key, val, dotPropKey) {
var this$1 = this;
var value = val
// get
var transformers = this.meta(TRANSFORMERS_KEY, key)
for (var t = 0; t < transformers.length; t++) {
value = transformers[t].call(this$1, value, this$1)
}
SuperClass.prototype.set.call(this, key, value)
var data = {key: dotPropKey, value: value}
if (isUndefined(dotPropKey)) {
data.key = isObj(value) ? dotPropPaths(key, value) : key
}
// get
var observers = this.meta(OBSERVERS_KEY)
for (var o = 0; o < observers.length; o++) {
observers[o](data)
}
return this
};
// --- remap ---
/**
* @desc remap properties from 1 to another, for example, apis with inconsistent naming
* @memberOf TransformChain
* @since 1.0.0
* @symb 🗺
*
* @param {string | Object} from property name string, or {[from]: to}
* @param {string} [to=undefined] property name to change key to
* @return {Chain} @chainable
*
* @see TransformChain.transform
* @IDEA could also be a function, but then might as well use .transform
*
* @example
*
* chain
* .remap('dis', 'dat')
* .from({dis: true})
*
* chain.entries()
* //=> {dat: true}
*
* @example
*
* chain
* .remap({dis: 'dat'})
* .from({dis: 1, other: true}}
*
* chain.entries()
* //=> {dist: 1, other: true}
*
*/
Transform.prototype.remap = function remap (from, to) {
var this$1 = this;
var remap = from
if (!isObj(from)) { remap = {[from]: to} }
/* prettier-ignore */
ObjectKeys(remap).forEach(function (key) { return this$1.transform(key, function (val) {
this$1.set(remap[key], val)
return val
}); })
return this
};
return Transform;
}(SuperClass))
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"Transform.js","sources":["Transform.js"],"sourcesContent":["const TraverseChain = require('../TraverseChain')\nconst isObj = require('../deps/is/obj')\nconst isTrue = require('../deps/is/true')\nconst isFalse = require('../deps/is/false')\nconst isUndefined = require('../deps/is/undefined')\nconst ObjectKeys = require('../deps/util/keys')\nconst dotPropPaths = require('../deps/dot/paths')\nconst TRANSFORMERS_KEY = require('../deps/meta/transformers')\nconst OBSERVERS_KEY = require('../deps/meta/observers')\n\n/**\n * @param  {Class | Composable} SuperClass composable class\n * @return {TransformChain} class\n * @example\n *    compose(class {})\n *    //=> TransformChain\n */\nmodule.exports = SuperClass => {\n  /**\n   * @class TransformChain\n   * @member TransformChain\n   * @extends {ChainedMap}\n   * @memberOf compose\n   * @category Chainable\n   *\n   * @tests TransformChain\n   * @types TransformChain\n   *\n   * @symb 🤖\n   * @type {Map}\n   *\n   * @see deps/traverse\n   * @see TraverseChain\n   *\n   * {@link https://github.com/iluwatar/java-design-patterns/tree/master/state state-pattern}\n   * {@link https://github.com/iluwatar/java-design-patterns/tree/master/strategy strategy-pattern}\n   */\n  return class Transform extends SuperClass {\n    // -------------------------------------------\n\n    /**\n     * @desc traverse `this`, or `this.entries`\n     * @since 1.0.2\n     *\n     * @param  {boolean | traversable} [useThis=false] use the instance properties that are `mapish` as well\n     * @return {TraverseChain} @chainable\n     *\n     * @see TraverseChain\n     * @see js-traverse\n     *\n     * @example\n     *  TAKE FROM TRAVERSECHAIN\n     */\n    traverse(useThis = false) {\n      /* prettier-ignore */\n      return new TraverseChain(this)\n        .obj(isFalse(useThis)\n          ? this.entries(true)\n          : isTrue(useThis)\n            ? this\n            : useThis\n        )\n    }\n\n    /**\n     * @since 1.0.2\n     * @memberOf TransformChain\n     *\n     * @param  {string | Function} key currently just string\n     * @param  {Function} value callback accepting the value as only arg to transform with\n     * @return {TransformChain} @chainable\n     *\n     * @TODO dot-prop here\n     *\n     * @example\n     *\n     *   // coerce values with .id into the value they hold\n     *   chain\n     *     .transform('dis', val => (typeof val === 'string' ? val : val.id))\n     *\n     *   chain.set('dis', 'eh')\n     *   chain.get('dis')\n     *   //=> 'eh'\n     *\n     *   chain.set('dis', {id: 'eh'})\n     *   chain.get('dis')\n     *   //=> 'eh'\n     *\n     *\n     * @example\n     *\n     *    import {format} from 'date-fns/esm'\n     *    import {Chain} from 'chain-able'\n     *\n     *    const chain = new Chain()\n     *    chain.transform('created_at', date => format(date, 'MM/DD/YYYY'))\n     *    chain.set('created_at', new Date())\n     *\n     *    // is formatted human-readable pretty!\n     *    const {created_at} = chain.entries()\n     *    //=> '02/11/2014'\n     *\n     */\n    transform(key, value) {\n      return this.meta(TRANSFORMERS_KEY, key, value)\n    }\n\n    /**\n     * @memberOf TransformChain\n     *\n     * @override\n     * @inheritdoc\n     * @since 1.0.0\n     *\n     * @param {Primitive} key key to set with\n     * @param {any} val value to set for key\n     * @param {undefined | string | Array<string>} dotPropKey special key used for initializing dot prop values in an optimized way to keep reference\n     * @return {Chainable} @chainable\n     *\n     * @see this.observe, this.transform\n     */\n    set(key, val, dotPropKey) {\n      let value = val\n\n      // get\n      const transformers = this.meta(TRANSFORMERS_KEY, key)\n      for (let t = 0; t < transformers.length; t++) {\n        value = transformers[t].call(this, value, this)\n      }\n\n      super.set(key, value)\n\n      const data = {key: dotPropKey, value}\n      if (isUndefined(dotPropKey)) {\n        data.key = isObj(value) ? dotPropPaths(key, value) : key\n      }\n\n      // get\n      const observers = this.meta(OBSERVERS_KEY)\n      for (let o = 0; o < observers.length; o++) {\n        observers[o](data)\n      }\n\n      return this\n    }\n\n    // --- remap ---\n    /**\n     * @desc remap properties from 1 to another, for example, apis with inconsistent naming\n     * @memberOf TransformChain\n     * @since 1.0.0\n     * @symb 🗺\n     *\n     * @param  {string | Object} from property name string, or {[from]: to}\n     * @param  {string} [to=undefined] property name to change key to\n     * @return {Chain} @chainable\n     *\n     * @see TransformChain.transform\n     * @IDEA could also be a function, but then might as well use .transform\n     *\n     * @example\n     *\n     *  chain\n     *    .remap('dis', 'dat')\n     *    .from({dis: true})\n     *\n     *  chain.entries()\n     *  //=> {dat: true}\n     *\n     * @example\n     *\n     *  chain\n     *    .remap({dis: 'dat'})\n     *    .from({dis: 1, other: true}}\n     *\n     *  chain.entries()\n     *  //=> {dist: 1, other: true}\n     *\n     */\n    remap(from, to) {\n      let remap = from\n      if (!isObj(from)) remap = {[from]: to}\n\n      /* prettier-ignore */\n      ObjectKeys(remap).forEach(key => this.transform(key, val => {\n        this.set(remap[key], val)\n        return val\n      }))\n\n      return this\n    }\n  }\n}\n"],"names":["const","let","this","super"],"mappings":"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"}