Skip to content

Commit

Permalink
feat: refactor Condition code
Browse files Browse the repository at this point in the history
  • Loading branch information
vonagam committed Feb 7, 2019
1 parent 64ec24c commit d5f16b7
Showing 1 changed file with 39 additions and 33 deletions.
72 changes: 39 additions & 33 deletions src/Condition.js
Original file line number Diff line number Diff line change
@@ -1,52 +1,58 @@
import has from 'lodash/has';
import isSchema from './util/isSchema';

function callOrConcat(schema) {
if (typeof schema === 'function') return schema;
if (!schema) return base => base;
return (base, options) => base.concat(schema.resolve(options));
function wrapCusomFn(fn) {
return function(...args) {
args.pop();
return fn.apply(this, args);
};
}

class Conditional {
constructor(refs, options) {
let { is, then, otherwise } = options;
function makeFn(options) {
if (typeof options === 'function') return wrapCusomFn(options);

this.refs = [].concat(refs);
if (!has(options, 'is'))
throw new TypeError('`is:` is required for `when()` conditions');

then = callOrConcat(then);
otherwise = callOrConcat(otherwise);
if (!options.then && !options.otherwise)
throw new TypeError(
'either `then:` or `otherwise:` is required for `when()` conditions',
);

if (typeof options === 'function') this.fn = options;
else {
if (!has(options, 'is'))
throw new TypeError('`is:` is required for `when()` conditions');
let { is, then, otherwise } = options;

if (!options.then && !options.otherwise)
throw new TypeError(
'either `then:` or `otherwise:` is required for `when()` conditions',
);
let check;
if (typeof is === 'function') {
check = is;
} else {
check = (...values) => values.every(value => value === is);
}

let isFn =
typeof is === 'function'
? is
: (...values) => values.every(value => value === is);
let fn = function(...args) {
let options = args.pop();
let schema = args.pop();
let branch = check(...args) ? then : otherwise;

this.fn = function(...values) {
let options = values.pop();
let schema = values.pop();
let option = isFn(...values) ? then : otherwise;
if (!branch) return undefined;
if (typeof branch === 'function') return branch(schema);
return schema.concat(branch.resolve(options));
};

return option(schema, options);
};
}
return fn;
}

class Condition {
constructor(refs, options) {
this.refs = refs;
this.fn = makeFn(options);
}

resolve(ctx, options) {
resolve(base, options) {
let values = this.refs.map(ref => ref.getValue(options));

let schema = this.fn.apply(ctx, values.concat(ctx, options));
let schema = this.fn.apply(base, values.concat(base, options));

if (schema === undefined) return ctx;
if (schema === undefined || schema === base) return base;

if (!isSchema(schema))
throw new TypeError('conditions must return a schema object');
Expand All @@ -55,4 +61,4 @@ class Conditional {
}
}

export default Conditional;
export default Condition;

0 comments on commit d5f16b7

Please sign in to comment.