Last active
August 29, 2015 14:26
-
-
Save datchley/0381a13e4d5782db9ef0 to your computer and use it in GitHub Desktop.
A decorator for modifying a function to lift functions to work with different types of arguments
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// | |
// Some common utility combinators and helpers | |
// | |
function flip(fn) { | |
return function() { | |
var args = [].slice.call(arguments); | |
return fn.apply(this, args.reverse()); | |
}; | |
} | |
function rightCurry(fn, n) { | |
var arity = n || fn.length, | |
fn = flip(fn); | |
return function curried() { | |
var args = [].slice.call(arguments), | |
context = this; | |
return args.length >= arity ? | |
fn.apply(context, args.slice(0, arity)) : | |
function () { | |
var rest = [].slice.call(arguments); | |
return curried.apply(context, args.concat(rest)); | |
}; | |
}; | |
} | |
function get(obj, prop) { | |
return obj[prop]; | |
} | |
function map(list, fn) { | |
return list.map(fn); | |
} | |
function pluck(obj, prop) { | |
return obj[prop]; | |
} | |
var getWith = rightCurry(get), | |
mapWith = rightCurry(map), | |
pluckWith = rightCurry(pluck); | |
// Similar to Ramda's useWith(fn,...), A function decorator which | |
// returns a function and takes a function to decorate, along with zero or | |
// more 'transform' functions. | |
// The returned function, when called, will called the decorated function | |
// with the parameters it receives being augmented by the corresponding | |
// transform function. | |
// If there are more arguments than transform functions, those arguments | |
// will be passed as is to the decorated function | |
// @return {Function} - a decorated version of the function passed as the first parameter | |
// @param {Function} - The function to decorate or wrap | |
// @param [{Function...}] - one or more transform functions, used to augment arguments | |
// passed to the decorated function when invoked. | |
// | |
function useWith(fn /*, txnformfn, ... */) { | |
var transforms = [].slice.call(arguments, 1), | |
_transform = function(args) { | |
return args.map(function(arg, i) { | |
return transforms[i](arg); | |
}); | |
}; | |
return function() { | |
var args = [].slice.call(arguments), | |
targs = args.slice(0,transforms.length), | |
remaining = args.slice(transforms.length); | |
return fn.apply(this, _transform(targs).concat(remaining)); | |
} | |
} | |
// A simple, resuable comparison for '>=' | |
function greaterThanOrEqual(a, b) { | |
return a >= b | |
} | |
// Right curried so we can create useful unary predicates | |
var greaterThanOrEqualTo = rightCurry(greaterThanOrEqual); | |
// Create a unary predicate to use with filter that lets us filter | |
// for values >= 30 days ago | |
var newerThan30Days = greaterThanOrEqualTo((new Date()).getTime() - (86400000 * 30)); | |
// Array of sample object data | |
var dates = [ | |
{ id: 1, date: (new Date('2015-07-29')).getTime() }, | |
{ id: 2, date: (new Date('2015-05-01')).getTime() } | |
]; | |
// Use our newerThan30Days predicate, modified using withProp | |
// to allow it to access object's `.date` property | |
dates.filter(useWith(newerThan30Days, pluckWith('date'))); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Updated and added a
useWith()
function which is a more generic version of my originalwithProp()
function, which matches the similarly named function in the Ramda.js functional library.useWith()
basically returns a function that modifies the original function passed in by passing any arguments to the called function through a set of transform functions provided as the 1...N arguments of the originaluseWith()
call.For instance, if we wanted to use our simple
greaterThanOrEqualTo()
comparison to allow using objects, we could call it as: