paper.js/lib/bootstrap.js

267 lines
7.2 KiB
JavaScript
Raw Normal View History

/**
* Bootstrap JavaScript Library
2011-03-04 17:22:22 -05:00
* (c) 2006 - 2011 Juerg Lehni, http://lehni.org/
*
* Bootstrap is released under the MIT license
2011-03-04 17:22:22 -05:00
* http://bootstrapjs.org/
*
* Inspirations:
* http://dean.edwards.name/weblog/2006/03/base/
* http://dev.helma.org/Wiki/JavaScript+Inheritance+Sugar/
* http://prototypejs.org/
* http://mootools.net/
*/
2011-03-06 19:50:44 -05:00
var Base = this.Base = new function() {
2011-03-04 16:46:50 -05:00
var fix = !this.__proto__,
hidden = /^(statics|generics|preserve|beans|enumerable|prototype|__proto__|toString|valueOf)$/,
proto = Object.prototype,
has = fix
? function(name) {
return name !== '__proto__' && this.hasOwnProperty(name);
}
: proto.hasOwnProperty,
toString = proto.toString,
isArray = Array.isArray = Array.isArray || function(obj) {
return toString.call(obj) === '[object Array]';
},
proto = Array.prototype,
slice = proto.slice,
forEach = proto.forEach = proto.forEach || function(iter, bind) {
for (var i = 0, l = this.length; i < l; i++)
iter.call(bind, this[i], i, this);
},
forIn = function(iter, bind) {
for (var i in this)
if (has.call(this, i))
iter.call(bind, this[i], i, this);
},
2011-03-04 16:46:50 -05:00
_define = Object.defineProperty,
_describe = Object.getOwnPropertyDescriptor;
function define(obj, name, desc) {
2011-03-04 17:22:22 -05:00
if (_define) {
try {
delete obj[name];
return _define(obj, name, desc);
} catch (e) {}
}
if ((desc.get || desc.set) && obj.__defineGetter__) {
2011-03-04 17:22:22 -05:00
desc.get && obj.__defineGetter__(name, desc.get);
desc.set && obj.__defineSetter__(name, desc.set);
} else {
obj[name] = desc.value;
2011-02-07 13:28:09 -05:00
}
return obj;
}
function describe(obj, name) {
2011-03-04 17:22:22 -05:00
if (_describe) {
try {
return _describe(obj, name);
} catch (e) {}
}
var get = obj.__lookupGetter__ && obj.__lookupGetter__(name);
return get
2011-03-04 17:22:22 -05:00
? { get: get, set: obj.__lookupSetter__(name), enumerable: true,
configurable: true }
2011-03-04 16:46:50 -05:00
: has.call(obj, name)
2011-03-04 17:22:22 -05:00
? { value: obj[name], enumerable: true, configurable: true,
writable: true }
: null;
}
2011-02-07 13:28:09 -05:00
function inject(dest, src, enumerable, base, preserve, generics) {
var beans, bean;
function field(name, val, dontCheck, generics) {
2011-03-04 17:22:22 -05:00
var val = val || (val = describe(src, name))
&& (val.get ? val : val.value),
func = typeof val === 'function', res = val,
2011-03-04 17:22:22 -05:00
prev = preserve || func
? (val && val.get ? name in dest : dest[name]) : null;
2011-03-04 17:22:22 -05:00
if (generics && func && (!preserve || !generics[name])) {
generics[name] = function(bind) {
return bind && dest[name].apply(bind,
slice.call(arguments, 1));
}
2011-02-07 13:28:09 -05:00
}
2011-03-04 17:22:22 -05:00
if ((dontCheck || val !== undefined && has.call(src, name))
&& (!preserve || !prev)) {
2011-02-07 13:28:09 -05:00
if (func) {
if (prev && /\bthis\.base\b/.test(val)) {
var fromBase = base && base[name] == prev;
res = function() {
var tmp = describe(this, 'base');
2011-03-04 17:22:22 -05:00
define(this, 'base', { value: fromBase
? base[name] : prev, configurable: true });
2011-03-04 17:22:22 -05:00
try {
return val.apply(this, arguments);
} finally {
tmp ? define(this, 'base', tmp)
: delete this.base;
}
};
// Make wrapping closure pretend to be the original
// function on inspection
res.toString = function() {
return val.toString();
}
res.valueOf = function() {
return val.valueOf();
}
2011-02-07 13:28:09 -05:00
}
// Only add getter beans if they do not expect arguments
// Functions that should function both with optional
// arguments and as beans should not declare the parameters
// and use the arguments array internally instead.
if (beans && val.length == 0
&& (bean = name.match(/^(get|is)(([A-Z])(.*))$/)))
beans.push([ bean[3].toLowerCase() + bean[4], bean[2] ]);
2011-02-07 13:28:09 -05:00
}
if (!res || func || !res.get && !res.set)
res = { value: res, writable: true };
2011-03-04 17:22:22 -05:00
if ((describe(dest, name)
|| { configurable: true }).configurable) {
res.configurable = true;
res.enumerable = enumerable;
}
define(dest, name, res);
2011-02-07 13:28:09 -05:00
}
}
if (src) {
beans = src.beans && [];
2011-02-07 13:28:09 -05:00
for (var name in src)
if (has.call(src, name) && !hidden.test(name))
field(name, null, true, generics);
2011-02-07 13:28:09 -05:00
field('toString');
field('valueOf');
for (var i = 0, l = beans && beans.length; i < l; i++)
try {
var bean = beans[i], part = bean[1];
field(bean[0], {
get: dest['get' + part] || dest['is' + part],
set: dest['set' + part]
}, true);
} catch (e) {}
2011-02-07 13:28:09 -05:00
}
return dest;
2011-02-07 13:28:09 -05:00
}
function extend(obj) {
function ctor(dont) {
if (fix) define(this, '__proto__', { value: obj });
2011-02-07 13:28:09 -05:00
if (this.initialize && dont !== ctor.dont)
return this.initialize.apply(this, arguments);
}
ctor.prototype = obj;
ctor.toString = function() {
return (this.prototype.initialize || function() {}).toString();
}
return ctor;
}
function iterator(iter) {
return !iter
? function(val) { return val }
: typeof iter !== 'function'
? function(val) { return val == iter }
2011-03-05 15:43:00 -05:00
: iter;
}
function each(iter, bind, asArray) {
try {
2011-03-07 06:52:04 -05:00
(asArray || asArray === undefined && isArray(this) ? forEach : forIn)
.call(this, iterator(iter), bind = bind || this);
} catch (e) {
if (e !== Base.stop) throw e;
}
return bind;
}
function clone(obj) {
return each.call(obj, function(val, i) {
this[i] = val;
}, new obj.constructor());
2011-02-07 13:28:09 -05:00
}
// Inject into new ctor object that's passed to inject(), and then returned
return inject(function() {}, {
inject: function(src) {
if (src) {
var proto = this.prototype,
base = proto.__proto__ && proto.__proto__.constructor,
statics = src.statics == true ? src : src.statics;
if (statics != src)
inject(proto, src, src.enumerable, base && base.prototype,
src.preserve, src.generics && this);
inject(this, statics, true, base, src.preserve);
}
for (var i = 1, l = arguments.length; i < l; i++)
this.inject(arguments[i]);
return this;
},
extend: function(src) {
var proto = new this(this.dont), ctor = extend(proto);
define(proto, 'constructor',
{ value: ctor, writable: true, configurable: true });
ctor.dont = {};
inject(ctor, this, true);
return arguments.length ? this.inject.apply(ctor, arguments) : ctor;
}
// Pass true for enumerable, so inject() and extend() can be passed on
// to subclasses of Base through Base.inject() / extend().
}, true).inject({
2011-03-04 16:46:50 -05:00
has: has,
each: each,
2011-02-07 13:28:09 -05:00
inject: function() {
2011-02-07 13:28:09 -05:00
for (var i = 0, l = arguments.length; i < l; i++)
inject(this, arguments[i]);
return this;
},
extend: function() {
2011-02-07 13:28:09 -05:00
var res = new (extend(this));
return res.inject.apply(res, arguments);
},
clone: function() {
return clone(this);
},
2011-02-07 13:28:09 -05:00
statics: {
clone: clone,
define: define,
describe: describe,
iterator: iterator,
2011-03-04 16:46:50 -05:00
has: function(obj, name) {
return has.call(obj, name);
},
each: function(obj, iter, bind) {
return each.call(obj, iter, bind);
},
2011-03-04 16:46:50 -05:00
2011-02-07 13:28:09 -05:00
type: function(obj) {
return (obj || obj === 0) && (obj._type || typeof obj) || null;
2011-02-07 13:28:09 -05:00
},
check: function(obj) {
return !!(obj || obj === 0);
},
pick: function() {
for (var i = 0, l = arguments.length; i < l; i++)
if (arguments[i] !== undefined)
return arguments[i];
return null;
},
stop: {}
}
});
}