extend natives with non-enumerable properties, other clean up

This commit is contained in:
Sami Samhuri 2011-05-29 23:44:55 -07:00
parent 154dead245
commit d177c83f5e
6 changed files with 232 additions and 196 deletions

View file

@ -1,30 +1,39 @@
// batteries
// Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
var ArrayExt =
{ at: at
, compact: compact
, first: first
, flatten: flatten
, grep: grep
, last: last
, max: max
, min: min
, pluck: pluck
, sortBy: sortBy
, unique: unique
};
exports.extendNative = function() { exports.extendNative = function() {
Object.keys(ArrayExt).forEach(function(k) { require('./ext').extend(Array, ArrayExt);
if (Array.prototype[k]) return; // don't overwrite existing members
Array.prototype[k] = function() {
var fn = ArrayExt[k]
, args = [].slice.call(arguments)
;
args.unshift(this);
return fn.apply(ArrayExt, args);
};
});
}; };
var ArrayToString = [].toString(); require('./object-ext').extend(exports, ArrayExt);
var ArrayExt = exports.ArrayExt = { // TODO
// - abbrev
// abbrev // - pack
// - partition
// [1,2,3,4,5].at(-1) => 5 // [1,2,3,4,5].at(-1) => 5
at: function(a, i) { function at(a, i) {
if (i >= 0) return a[i]; if (i >= 0) return a[i];
return a[a.length + i]; return a[a.length + i];
} }
// TODO make this work for non-array objects // TODO make this work for non-array objects
, compact: function(a) { function compact(a) {
var b = [] var b = []
, i = a.length , i = a.length
; ;
@ -34,10 +43,10 @@ var ArrayExt = exports.ArrayExt = {
return b; return b;
} }
, first: function(a) { return a[0]; } function first(a) { return a[0]; }
// Based on underscore.js's flatten // Based on underscore.js's flatten
, flatten: function(a) { function flatten(a) {
return a.reduce(function(initial, elem) { return a.reduce(function(initial, elem) {
if (elem && elem.flatten) initial = initial.concat(elem.flatten()); if (elem && elem.flatten) initial = initial.concat(elem.flatten());
else initial.push(elem); else initial.push(elem);
@ -45,30 +54,27 @@ var ArrayExt = exports.ArrayExt = {
}); });
} }
, grep: function(a, regex) { function grep(a, regex) {
return a.filter(function(v) { return regex.test(v); }); return a.filter(function(v) { return regex.test(v); });
} }
, last: function(a) { return a[a.length-1]; } function last(a) { return a[a.length-1]; }
, max: function(a) { function max(a) {
return a.reduce(function(max, v) { return v > max ? v : max }); return a.reduce(function(max, v) { return v > max ? v : max });
} }
, min: function(a) { function min(a) {
return a.reduce(function(min, v) { return v < min ? v : min }); return a.reduce(function(min, v) { return v < min ? v : min });
} }
// pack function pluck(a /* , key paths, ... */) {
// partition
, pluck: function(a /* , key paths, ... */) {
var args = [].slice.call(arguments, 1); var args = [].slice.call(arguments, 1);
args.unshift(a); args.unshift(a);
return pluck.apply(null, args); return pluck.apply(null, args);
} }
, sortBy: function(arr, keyPath) { function sortBy(arr, keyPath) {
return arr.slice().sort(function(a, b) { return arr.slice().sort(function(a, b) {
var propA = drillInto(a) var propA = drillInto(a)
, propB = drillInto(b); , propB = drillInto(b);
@ -78,11 +84,7 @@ var ArrayExt = exports.ArrayExt = {
}); });
} }
, toString: function(a) { function unique(a) {
return '[' + ArrayToString.call(a) + ']';
}
, unique: function(a) {
var b = [] var b = []
, i = 0 , i = 0
, n = a.length , n = a.length
@ -92,7 +94,6 @@ var ArrayExt = exports.ArrayExt = {
} }
return b; return b;
} }
};
// pluck // pluck

View file

@ -1,16 +1,14 @@
// batteries
// Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
var DateExt = { format: format };
exports.extendNative = function() { exports.extendNative = function() {
Object.keys(DateExt).forEach(function(k) { require('./ext').extend(Date, DateExt);
if (Date.prototype[k]) return; // don't overwrite existing members
Date.prototype[k] = function() {
var fn = DateExt[k]
, args = [].slice.call(arguments)
;
args.shift(this);
fn.apply(DateExt, args);
};
});
}; };
require('./object-ext').extend(exports, DateExt);
var Weekdays = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', var Weekdays = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday',
'Friday', 'Saturday']; 'Friday', 'Saturday'];
@ -22,14 +20,7 @@ var Months = ['January', 'February', 'March', 'April', 'May', 'June', 'July',
var MonthsShort = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', var MonthsShort = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug',
'Sep', 'Oct', 'Nov', 'Dec']; 'Sep', 'Oct', 'Nov', 'Dec'];
function pad(n, padding) { function format(d, fmt) {
padding = padding || '0';
return n < 10 ? (padding + n) : n;
}
var DateExt = {
// FIXME write a c extension that uses strftime to do the heavy lifting
format: function(d, fmt) {
return fmt.replace(/%(.)/, function(_, c) { return fmt.replace(/%(.)/, function(_, c) {
switch (c) { switch (c) {
case 'A': return Weekdays[d.getDay()]; case 'A': return Weekdays[d.getDay()];
@ -37,10 +28,10 @@ var DateExt = {
case 'B': return Months[d.getMonth()]; case 'B': return Months[d.getMonth()];
case 'b': // fall through case 'b': // fall through
case 'h': return MonthsShort[d.getMonth()]; case 'h': return MonthsShort[d.getMonth()];
case 'D': return DateExt.format(d, '%m/%d/%y'); case 'D': return format(d, '%m/%d/%y');
case 'd': return pad(d.getDate()); case 'd': return pad(d.getDate());
case 'e': return d.getDate(); case 'e': return d.getDate();
case 'F': return DateExt.format(d, '%Y-%m-%d'); case 'F': return format(d, '%Y-%m-%d');
case 'H': return pad(d.getHours()); case 'H': return pad(d.getHours());
case 'I': case 'I':
var hour = d.getHours(); var hour = d.getHours();
@ -57,16 +48,16 @@ var DateExt = {
case 'm': return pad(d.getMonth() + 1); case 'm': return pad(d.getMonth() + 1);
case 'n': return '\n'; case 'n': return '\n';
case 'p': return d.getHours() < 12 ? 'AM' : 'PM'; case 'p': return d.getHours() < 12 ? 'AM' : 'PM';
case 'R': return DateExt.format(d, '%H:%M'); case 'R': return format(d, '%H:%M');
case 'r': return DateExt.format(d, '%I:%M:%S %p'); case 'r': return format(d, '%I:%M:%S %p');
case 'S': return pad(d.getSeconds()); case 'S': return pad(d.getSeconds());
case 's': return d.getTime(); case 's': return d.getTime();
case 'T': return DateExt.format(d, '%H:%M:%S'); case 'T': return format(d, '%H:%M:%S');
case 't': return '\t'; case 't': return '\t';
case 'u': case 'u':
var day = d.getDay(); var day = d.getDay();
return day == 0 ? 7 : day; // 1 - 7, Monday is first day of the week return day == 0 ? 7 : day; // 1 - 7, Monday is first day of the week
case 'v': return DateExt.format(d, '%e-%b-%Y'); case 'v': return format(d, '%e-%b-%Y');
case 'w': return d.getDay(); // 0 - 6, Sunday is first day of the week case 'w': return d.getDay(); // 0 - 6, Sunday is first day of the week
case 'Y': return d.getFullYear(); case 'Y': return d.getFullYear();
case 'y': case 'y':
@ -81,18 +72,25 @@ var DateExt = {
default: return c; default: return c;
} }
}); });
}, }
month: function(d) {
function pad(n, padding) {
padding = padding || '0';
return n < 10 ? (padding + n) : n;
}
function month(d) {
return Months(d.getMonth()); return Months(d.getMonth());
}, }
shortMonth: function(d) {
function shortMonth(d) {
return MonthsShort(d.getMonth()); return MonthsShort(d.getMonth());
}, }
weekday: function(d) {
function weekday(d) {
return Weekdays(d.getDay()); return Weekdays(d.getDay());
}, }
shortWeekday: function(d) {
function shortWeekday(d) {
return WeekdaysShort(d.getDay()); return WeekdaysShort(d.getDay());
} }
};
exports.DateExt = DateExt;

View file

@ -1,13 +1,18 @@
// batteries
// Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
exports.extend = function(obj, ext) { exports.extend = function(obj, ext) {
// FIXME why doesn't this work when the caller supplies
// a native type's proto for obj? e.g. Array.prototype
Object.keys(ext).forEach(function(k) { Object.keys(ext).forEach(function(k) {
if (obj[k]) return; // don't overwrite existing members if (k in obj.prototype) return; // don't overwrite existing members
obj[k] = function() { var wrapper = function() {
var fn = ext[k] var fn = ext[k]
, args = [].slice.call(arguments) , args = [].slice.call(arguments)
; ;
fn.apply(null, args); args.unshift(this);
return fn.apply(this, args);
}; };
Object.defineProperty(obj.prototype, k, {
get: function() { return wrapper; }
});
}); });
}; };

View file

@ -1,9 +1,16 @@
exports.extendNative = function() { // batteries
Math.sum = sum; // Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
Math.avg = avg;
var MathExt =
{ avg: avg
, sum: sum
};
exports.extendNative = function() {
Math.avg = avg;
Math.sum = sum;
}; };
exports.sum = sum;
function sum() { function sum() {
var nums = Array.isArray(arguments[0]) ? arguments[0] : arguments var nums = Array.isArray(arguments[0]) ? arguments[0] : arguments
, i = nums.length , i = nums.length
@ -13,7 +20,6 @@ function sum() {
return sum; return sum;
} }
exports.avg = avg;
function avg() { function avg() {
var nums = Array.isArray(arguments[0]) ? arguments[0] : arguments; var nums = Array.isArray(arguments[0]) ? arguments[0] : arguments;
return sum.call(this, nums) / nums.length; return sum.call(this, nums) / nums.length;

View file

@ -1,10 +1,26 @@
var ext = exports.ObjectExt = { // batteries
// Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
cmp: function(a, b) { var ObjectExt =
{ cmp: cmp
, extend: extend
}
exports.extendNative = function() {
require('./ext').extend(Object, ObjectExt);
};
extend(module.exports, ObjectExt);
function cmp(a, b) {
if (a === b) return 0; if (a === b) return 0;
if (a < b) return -1; if (a < b) return -1;
if (a > b) return 1; if (a > b) return 1;
throw new Error('cannot effectively compare values'); throw new Error('cannot effectively compare values');
} }
}; function extend(a, b) {
Object.getOwnPropertyNames(b).forEach(function(k) {
a[k] = b[k];
});
}

View file

@ -1,11 +1,21 @@
var StringExt = exports.StringExt = { // batteries
// Copyright 2010 - 2011 Sami Samhuri <sami@samhuri.net>
cmp: function(a, b) { // TODO
// - reverse
// - unpack
// - sha1
var StringExt = { cmp: cmp };
exports.extendNative = function() {
require('./ext').extend(String, StringExt);
};
require('./object-ext').extend(exports, StringExt);
function cmp(a, b) {
if (a === b) return 0; if (a === b) return 0;
if (a < b) return -1; if (a < b) return -1;
return 1; // a > b return 1; // a > b
} }
// unpack
};