angular.js/src/Angular.js

415 lines
10 KiB
JavaScript
Raw Normal View History

2010-04-01 00:56:16 +00:00
//////////////////////////////
//UrlWatcher
//////////////////////////////
function UrlWatcher(location) {
this.location = location;
this.delay = 25;
this.setTimeout = function(fn, delay) {
window.setTimeout(fn, delay);
};
this.expectedUrl = location.href;
this.listeners = [];
}
UrlWatcher.prototype = {
watch: function(fn){
this.listeners.push(fn);
},
start: function() {
var self = this;
(function pull () {
if (self.expectedUrl !== self.location.href) {
foreach(self.listeners, function(listener){
listener(self.location.href);
});
self.expectedUrl = self.location.href;
}
self.setTimeout(pull, self.delay);
})();
},
set: function(url) {
var existingURL = this.location.href;
if (!existingURL.match(/#/))
existingURL += '#';
if (existingURL != url)
this.location.href = url;
this.existingURL = url;
},
get: function() {
return this.location.href;
}
};
2010-04-01 21:10:28 +00:00
////////////////////////////////////
2010-04-01 00:56:16 +00:00
2010-01-06 00:36:58 +00:00
if (typeof document.getAttribute == 'undefined')
2010-01-12 00:15:12 +00:00
document.getAttribute = function() {};
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
if (!window['console']) window['console']={'log':noop, 'error':noop};
var consoleNode,
2010-04-01 00:56:16 +00:00
PRIORITY_FIRST = -99999;
PRIORITY_WATCH = -1000;
PRIORITY_LAST = 99999;
NOOP = 'noop',
2010-03-30 22:39:51 +00:00
NG_ERROR = 'ng-error',
2010-03-31 20:57:25 +00:00
NG_EXCEPTION = 'ng-exception',
2010-03-30 22:39:51 +00:00
NG_VALIDATION_ERROR = 'ng-validation-error',
2010-03-23 21:57:11 +00:00
jQuery = window['jQuery'] || window['$'], // weirdness to make IE happy
_ = window['_'],
2010-03-31 20:57:25 +00:00
msie = !!/(msie) ([\w.]+)/.exec(lowercase(navigator.userAgent)),
jqLite = jQuery || jqLiteWrap,
2010-03-23 21:57:11 +00:00
slice = Array.prototype.slice,
2010-04-01 00:56:16 +00:00
angular = window['angular'] || (window['angular'] = {}),
2010-03-25 21:43:05 +00:00
angularTextMarkup = extensionMap(angular, 'textMarkup'),
angularAttrMarkup = extensionMap(angular, 'attrMarkup'),
2010-03-23 21:57:11 +00:00
angularDirective = extensionMap(angular, 'directive'),
angularWidget = extensionMap(angular, 'widget'),
angularValidator = extensionMap(angular, 'validator'),
angularFilter = extensionMap(angular, 'filter'),
angularFormatter = extensionMap(angular, 'formatter'),
2010-04-01 00:56:16 +00:00
angularService = extensionMap(angular, 'service'),
2010-03-23 21:57:11 +00:00
angularCallbacks = extensionMap(angular, 'callbacks'),
2010-04-01 00:56:16 +00:00
urlWatcher = new UrlWatcher(window.location);
function angularAlert(){
log(arguments); window.alert.apply(window, arguments);
};
extend(angular, {
'compile': compile,
'startUrlWatch': bind(urlWatcher, urlWatcher.start),
'copy': copy,
'extend': extend,
'foreach': foreach,
'noop':noop,
'identity':identity,
'isUndefined': isUndefined,
'isDefined': isDefined,
'isString': isString,
'isFunction': isFunction,
'isNumber': isNumber,
'isArray': isArray,
'alert': angularAlert
});
2010-03-23 21:57:11 +00:00
function foreach(obj, iterator, context) {
var key;
if (obj) {
if (obj.forEach) {
obj.forEach(iterator, context);
} else if (isObject(obj) && isNumber(obj.length)) {
2010-03-23 21:57:11 +00:00
for (key = 0; key < obj.length; key++)
iterator.call(context, obj[key], key);
} else {
for (key in obj)
iterator.call(context, obj[key], key);
}
}
return obj;
}
2010-04-01 00:56:16 +00:00
function foreachSorted(obj, iterator, context) {
var keys = [];
for (var key in obj) keys.push(key);
keys.sort();
for ( var i = 0; i < keys.length; i++) {
iterator.call(context, obj[keys[i]], keys[i]);
}
return keys;
}
function extend(dst) {
foreach(arguments, function(obj){
if (obj !== dst) {
foreach(obj, function(value, key){
dst[key] = value;
});
}
2010-03-23 21:57:11 +00:00
});
return dst;
}
function noop() {}
function identity($) {return $;}
function extensionMap(angular, name) {
var extPoint;
return angular[name] || (extPoint = angular[name] = function (name, fn, prop){
if (isDefined(fn)) {
extPoint[name] = extend(fn, prop || {});
}
return extPoint[name];
});
}
function jqLiteWrap(element) {
if (typeof element == 'string') {
var div = document.createElement('div');
div.innerHTML = element;
element = div.childNodes[0];
}
return element instanceof JQLite ? element : new JQLite(element);
}
function isUndefined(value){ return typeof value == 'undefined'; }
2010-03-23 04:29:57 +00:00
function isDefined(value){ return typeof value != 'undefined'; }
function isObject(value){ return typeof value == 'object';}
function isString(value){ return typeof value == 'string';}
function isNumber(value){ return typeof value == 'number';}
2010-03-23 04:29:57 +00:00
function isArray(value) { return value instanceof Array; }
function isFunction(value){ return typeof value == 'function';}
2010-03-30 21:55:04 +00:00
function isTextNode(node) { return nodeName(node) == '#text'; }
function lowercase(value){ return isString(value) ? value.toLowerCase() : value; }
function uppercase(value){ return isString(value) ? value.toUpperCase() : value; }
function trim(value) { return isString(value) ? value.replace(/^\s*/, '').replace(/\s*$/, '') : value; };
2010-03-30 21:55:04 +00:00
function nodeName(element) { return (element[0] || element || {}).nodeName; }
function map(obj, iterator, context) {
var results = [];
foreach(obj, function(value, index, list) {
results.push(iterator.call(context, value, index, list));
});
return results;
};
function size(obj) {
var size = 0;
if (obj) {
if (isNumber(obj.length)) {
return obj.length;
} else if (isObject(obj)){
for (key in obj)
size++;
}
}
return size;
}
function includes(array, obj) {
for ( var i = 0; i < array.length; i++) {
if (obj === array[i]) return true;
}
return false;
}
2010-03-20 05:18:39 +00:00
function indexOf(array, obj) {
for ( var i = 0; i < array.length; i++) {
if (obj === array[i]) return i;
}
return -1;
}
function log(a, b, c){
var console = window['console'];
switch(arguments.length) {
case 1:
console['log'](a);
break;
case 2:
console['log'](a, b);
break;
default:
console['log'](a, b, c);
break;
}
}
function error(a, b, c){
var console = window['console'];
switch(arguments.length) {
case 1:
console['error'](a);
break;
case 2:
console['error'](a, b);
break;
default:
console['error'](a, b, c);
break;
}
}
2010-01-12 00:15:12 +00:00
function consoleLog(level, objs) {
2010-01-06 00:36:58 +00:00
var log = document.createElement("div");
log.className = level;
var msg = "";
var sep = "";
for ( var i = 0; i < objs.length; i++) {
var obj = objs[i];
2010-01-09 23:02:43 +00:00
msg += sep + (typeof obj == 'string' ? obj : toJson(obj));
2010-01-06 00:36:58 +00:00
sep = " ";
}
log.appendChild(document.createTextNode(msg));
2010-01-09 23:02:43 +00:00
consoleNode.appendChild(log);
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function isNode(inp) {
2010-01-06 00:36:58 +00:00
return inp &&
inp.tagName &&
inp.nodeName &&
inp.ownerDocument &&
inp.removeAttribute;
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function isLeafNode (node) {
2010-03-23 21:57:11 +00:00
if (node) {
switch (node.nodeName) {
case "OPTION":
case "PRE":
case "TITLE":
return true;
}
2010-01-06 00:36:58 +00:00
}
2010-03-23 21:57:11 +00:00
return false;
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-03-15 21:36:50 +00:00
function copy(source, destination){
if (!destination) {
if (!source) {
return source;
} else if (isArray(source)) {
2010-03-15 21:36:50 +00:00
return copy(source, []);
} else {
return copy(source, {});
}
} else {
if (isArray(source)) {
2010-03-15 21:36:50 +00:00
while(destination.length) {
destination.pop();
}
} else {
foreach(destination, function(value, key){
2010-03-15 21:36:50 +00:00
delete destination[key];
});
}
foreach(source, function(value, key){
destination[key] = isArray(value) ? copy(value, []) : (isObject(value) ? copy(value, {}) : value);
});
return destination;
2010-03-15 21:36:50 +00:00
}
};
2010-01-12 00:15:12 +00:00
function setHtml(node, html) {
2010-01-09 23:02:43 +00:00
if (isLeafNode(node)) {
if (msie) {
2010-01-06 00:36:58 +00:00
node.innerText = html;
} else {
node.textContent = html;
}
} else {
node.innerHTML = html;
}
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function escapeHtml(html) {
2010-01-06 00:36:58 +00:00
if (!html || !html.replace)
return html;
return html.
replace(/&/g, '&amp;').
replace(/</g, '&lt;').
replace(/>/g, '&gt;');
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-03-30 22:39:51 +00:00
function elementError(element, type, error) {
2010-03-30 04:36:34 +00:00
if (error) {
2010-03-30 22:39:51 +00:00
element.addClass(type);
2010-03-30 04:36:34 +00:00
element.attr(NG_ERROR, error);
} else {
2010-03-30 22:39:51 +00:00
element.removeClass(type);
2010-03-30 04:36:34 +00:00
element.removeAttr(NG_ERROR);
}
}
2010-01-12 00:15:12 +00:00
function escapeAttr(html) {
2010-01-06 00:36:58 +00:00
if (!html || !html.replace)
return html;
return html.replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g,
'&quot;');
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function bind(_this, _function) {
var curryArgs = slice.call(arguments, 2, arguments.length);
2010-01-06 00:36:58 +00:00
return function() {
return _function.apply(_this, curryArgs.concat(slice.call(arguments, 0, arguments.length)));
2010-01-06 00:36:58 +00:00
};
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function outerHTML(node) {
2010-01-06 00:36:58 +00:00
var temp = document.createElement('div');
temp.appendChild(node);
var outerHTML = temp.innerHTML;
temp.removeChild(node);
return outerHTML;
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function toBoolean(value) {
2010-03-30 21:55:04 +00:00
if (value && value.length !== 0) {
var v = lowercase("" + value);
value = !(v == 'f' || v == '0' || v == 'false' || v == 'no' || v == '[]');
} else {
2010-01-06 00:36:58 +00:00
value = false;
2010-03-30 21:55:04 +00:00
}
return value;
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-01-12 00:15:12 +00:00
function merge(src, dst) {
2010-01-06 00:36:58 +00:00
for ( var key in src) {
var value = dst[key];
var type = typeof value;
if (type == 'undefined') {
2010-01-09 23:02:43 +00:00
dst[key] = fromJson(toJson(src[key]));
} else if (type == 'object' && value.constructor != array &&
2010-01-06 00:36:58 +00:00
key.substring(0, 1) != "$") {
2010-01-09 23:02:43 +00:00
merge(src[key], value);
2010-01-06 00:36:58 +00:00
}
}
2010-01-12 00:15:12 +00:00
}
2010-01-06 00:36:58 +00:00
2010-04-01 00:56:16 +00:00
function compile(element, config) {
var compiler = new Compiler(angularTextMarkup, angularAttrMarkup, angularDirective, angularWidget);
$element = jqLite(element),
2010-04-01 00:56:16 +00:00
rootScope = createScope({
$element: $element,
$config: extend({
'onUpdateView': noop,
'server': "",
'location': {
'get':bind(urlWatcher, urlWatcher.get),
'set':bind(urlWatcher, urlWatcher.set),
'watch':bind(urlWatcher, urlWatcher.watch)
}
}, config || {})
}, serviceAdapter(angularService));
return compiler.compile($element)($element, rootScope);
}
/////////////////////////////////////////////////
2010-04-01 21:10:28 +00:00
function parseKeyValue(keyValue) {
var obj = {}, key_value, key;
foreach((keyValue || "").split('&'), function(keyValue){
if (keyValue) {
key_value = keyValue.split('=');
key = decodeURIComponent(key_value[0]);
obj[key] = key_value[1] ? decodeURIComponent(key_value[1]) : true;
}
});
return obj;
}
2010-04-02 18:10:36 +00:00
function toKeyValue(obj) {
var parts = [];
foreach(obj, function(value, key){
parts.push(encodeURIComponent(key) + '=' + encodeURIComponent(value));
});
return parts.length ? parts.join('&') : '';
};
2010-04-01 21:10:28 +00:00
function angularInit(config){
if (config.autobind) {
compile(window.document, config).$init();
}
}
2010-04-02 18:16:49 +00:00