/* * Quick-n-dirty algebraic datatypes. * * These let us handle the possibility of failure without having to constantly write code to check for it. * We can apply all of the transformations we need as if the data is present using `map`. * If there's a None, or a FetchError, or a Pending, those are left untouched. * * I've used perhaps an odd bit of terminology from scalaz in `fold`. This is basically a `switch` statement: * You pass it a set of functions to handle the various different states of the datatype, and if it finds the * function it'll call it on its value. * * It's handy to have this in functional style when dealing with React as we can dispatch different ways of rendering * really simply: * ``` * bundleFetchStatus.fold({ * some: (fetchStatus) => <ProgressBar fetchsStatus={fetchStatus} />, * }), * ``` */ class Optional { static from(value) { return (value && Some.of(value)) || None; } map(f) { return this; } flatMap(f) { return this; } fold({ none }) { return none && none(); } } class Some extends Optional { constructor(value) { super(); this.value = value; } map(f) { return Some.of(f(this.value)); } flatMap(f) { return f(this.value); } fold({ some }) { return some && some(this.value); } static of(value) { return new Some(value); } } const None = new Optional(); class FetchStatus { constructor(opt = {}) { this.opt = { at: Date.now(), ...opt }; } map(f) { return this; } flatMap(f) { return this; } } class Success extends FetchStatus { static of(value) { return new Success(value); } constructor(value, opt) { super(opt); this.value = value; } map(f) { return new Success(f(this.value), this.opt); } flatMap(f) { return f(this.value, this.opt); } fold({ success }) { return success instanceof Function ? success(this.value, this.opt) : undefined; } } class Pending extends FetchStatus { static of(opt) { return new Pending(opt); } constructor(opt) { super(opt); } fold({ pending }) { return pending instanceof Function ? pending(this.opt) : undefined; } } class FetchError extends FetchStatus { static of(reason, opt) { return new FetchError(reason, opt); } constructor(reason, opt) { super(opt); this.reason = reason; } fold({ error }) { return error instanceof Function ? error(this.reason, this.opt) : undefined; } }