Commit 579ea905 authored by Pascal Hartig's avatar Pascal Hartig

CanJS: upgrade to 2.0.3

parent f119005f
/*! /*!
* CanJS - 2.0.2 * CanJS - 2.0.3
* http://canjs.us/ * http://canjs.us/
* Copyright (c) 2013 Bitovi * Copyright (c) 2013 Bitovi
* Thu, 14 Nov 2013 18:45:17 GMT * Tue, 26 Nov 2013 18:21:35 GMT
* Licensed MIT * Licensed MIT
* Includes: can/component,can/construct,can/observe,can/compute,can/model,can/view,can/control,can/route,can/control/route,can/view/mustache,can/view/bindings,can/view/live,can/view/scope,can/util/string * Includes: can/component,can/construct,can/map,can/list,can/observe,can/compute,can/model,can/view,can/control,can/route,can/control/route,can/view/mustache,can/view/bindings,can/view/live,can/view/scope,can/util/string
* Download from: http://canjs.com * Download from: http://canjs.com
*/ */
(function(undefined) { (function(undefined) {
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
return object._cid = (name || "") + (++cid) return object._cid = (name || "") + (++cid)
} }
} }
can.VERSION = '2.0.2'; can.VERSION = '2.0.3';
can.simpleExtend = function(d, s) { can.simpleExtend = function(d, s) {
for (var prop in s) { for (var prop in s) {
...@@ -384,24 +384,40 @@ ...@@ -384,24 +384,40 @@
oldClean(elems); oldClean(elems);
}; };
var oldDomManip = $.fn.domManip; var oldDomManip = $.fn.domManip,
cbIndex;
$.fn.domManip = function() { // feature detect which domManip we are using
var args = can.makeArray(arguments), $.fn.domManip = function(args, cb1, cb2) {
isNew$ = $.fn.jquery >= "2.0.0", for (var i = 1; i < arguments.length; i++) {
cbIndex = isNew$ ? 1 : 2, if (typeof arguments[i] === "function") {
callback = args[cbIndex]; cbIndex = i;
break;
args[cbIndex] = function(elem) { }
var isFragment = elem.nodeType === 11, //Node.DOCUMENT_FRAGMENT_NODE, }
targets = isFragment ? can.makeArray(elem.childNodes) : [elem], return oldDomManip.apply(this, arguments)
ret = callback.apply(this, arguments); }
can.inserted(targets); $(document.createElement("div")).append(document.createElement("div"))
return ret;
};
return oldDomManip.apply(this, args); $.fn.domManip = (cbIndex == 2 ? function(args, table, callback) {
}; return oldDomManip.call(this, args, table, function(elem) {
if (elem.nodeType === 11) {
var elems = can.makeArray(elem.childNodes);
}
var ret = callback.apply(this, arguments);
can.inserted(elems ? elems : [elem]);
return ret;
})
} : function(args, callback) {
return oldDomManip.call(this, args, function(elem) {
if (elem.nodeType === 11) {
var elems = can.makeArray(elem.childNodes);
}
var ret = callback.apply(this, arguments);
can.inserted(elems ? elems : [elem]);
return ret;
})
})
$.event.special.inserted = {}; $.event.special.inserted = {};
$.event.special.removed = {}; $.event.special.removed = {};
...@@ -1239,9 +1255,10 @@ ...@@ -1239,9 +1255,10 @@
madeMap[cid] = { madeMap[cid] = {
obj: obj, obj: obj,
instance: instance, instance: instance,
added: hasCid added: !hasCid
} }
} }
return teardown;
}; };
teardownMap = function() { teardownMap = function() {
for (var cid in madeMap) { for (var cid in madeMap) {
...@@ -1370,7 +1387,8 @@ ...@@ -1370,7 +1387,8 @@
// Sets all `attrs`. // Sets all `attrs`.
this._init = 1; this._init = 1;
this._setupComputes(); this._setupComputes();
var teardownMapping = obj && addToMap(obj, this) var teardownMapping = obj && addToMap(obj, this);
var data = can.extend(can.extend(true, {}, this.constructor.defaults || {}), obj) var data = can.extend(can.extend(true, {}, this.constructor.defaults || {}), obj)
this.attr(data); this.attr(data);
if (teardownMapping) { if (teardownMapping) {
...@@ -1380,6 +1398,7 @@ ...@@ -1380,6 +1398,7 @@
delete this._init; delete this._init;
}, },
_setupComputes: function() { _setupComputes: function() {
var prototype = this.constructor.prototype; var prototype = this.constructor.prototype;
this._computedBindings = {} this._computedBindings = {}
...@@ -1707,6 +1726,12 @@ ...@@ -1707,6 +1726,12 @@
list = Map( list = Map(
{
Map: Map
},
{ {
setup: function(instances, options) { setup: function(instances, options) {
this.length = 0; this.length = 0;
...@@ -1907,9 +1932,9 @@ ...@@ -1907,9 +1932,9 @@
can.extend(list.prototype, { can.extend(list.prototype, {
indexOf: function(item) { indexOf: function(item, fromIndex) {
this.attr('length') this.attr('length')
return can.inArray(item, this) return can.inArray(item, this, fromIndex)
}, },
...@@ -2796,6 +2821,8 @@ ...@@ -2796,6 +2821,8 @@
// ## view/scope/scope.js // ## view/scope/scope.js
var __m18 = (function(can) { var __m18 = (function(can) {
var isObserve = function(obj) { var isObserve = function(obj) {
return obj instanceof can.Map || (obj && obj.__get); return obj instanceof can.Map || (obj && obj.__get);
}, },
...@@ -2825,7 +2852,10 @@ ...@@ -2825,7 +2852,10 @@
var Scope = can.Construct.extend({ var Scope = can.Construct.extend(
{
// reads properties from a parent. A much more complex version of getObject. // reads properties from a parent. A much more complex version of getObject.
read: function(parent, reads, options) { read: function(parent, reads, options) {
...@@ -2909,28 +2939,31 @@ ...@@ -2909,28 +2939,31 @@
parent: prev parent: prev
}; };
} }
}, { },
init: function(data, parent) {
this._data = data; {
init: function(context, parent) {
this._context = context;
this._parent = parent; this._parent = parent;
}, },
attr: function(attr) {
return this.read(attr, { attr: function(key) {
return this.read(key, {
isArgument: true, isArgument: true,
returnObserveMethods: true, returnObserveMethods: true,
proxyMethods: false proxyMethods: false
}).value }).value
}, },
add: function(data) {
if (data !== this._data) { add: function(context) {
return new this.constructor(data, this); if (context !== this._context) {
return new this.constructor(context, this);
} else { } else {
return this; return this;
} }
}, },
computeData: function(attr, options) { computeData: function(key, options) {
options = options || { options = options || {
args: [] args: []
}; };
...@@ -2952,7 +2985,7 @@ ...@@ -2952,7 +2985,7 @@
return Scope.read(rootObserve, rootReads, options).value return Scope.read(rootObserve, rootReads, options).value
} }
// otherwise, go get the value // otherwise, go get the value
var data = self.read(attr, options); var data = self.read(key, options);
rootObserve = data.rootObserve; rootObserve = data.rootObserve;
rootReads = data.reads; rootReads = data.reads;
computeData.scope = data.scope; computeData.scope = data.scope;
...@@ -2972,11 +3005,11 @@ ...@@ -2972,11 +3005,11 @@
return this._parent.read(attr.substr(3), options) return this._parent.read(attr.substr(3), options)
} else if (attr == "..") { } else if (attr == "..") {
return { return {
value: this._parent._data value: this._parent._context
} }
} else if (attr == "." || attr == "this") { } else if (attr == "." || attr == "this") {
return { return {
value: this._data value: this._context
}; };
} }
...@@ -3023,7 +3056,7 @@ ...@@ -3023,7 +3056,7 @@
while (scope) { while (scope) {
// get the context // get the context
context = scope._data; context = scope._context;
if (context != null) { if (context != null) {
...@@ -3384,10 +3417,12 @@ ...@@ -3384,10 +3417,12 @@
// if this was an element like <foo-bar> that doesn't have a component, just render its content // if this was an element like <foo-bar> that doesn't have a component, just render its content
var res = tagCallback ? tagCallback(el, hookupOptions) : scope, var scope = hookupOptions.scope,
scope = hookupOptions.scope; res = tagCallback ? tagCallback(el, hookupOptions) : scope;
if (res) { // If the tagCallback gave us something to render with, and there is content within that element
// render it!
if (res && hookupOptions.subtemplate) {
if (scope !== res) { if (scope !== res) {
scope = scope.add(res) scope = scope.add(res)
...@@ -4016,12 +4051,12 @@ ...@@ -4016,12 +4051,12 @@
var teardown = function() { var teardown = function() {
unbind(data) unbind(data)
can.unbind.call(el, 'removed', teardown); can.unbind.call(el, 'removed', teardown);
return true
}, },
data = { data = {
// returns true if no parent
teardownCheck: function(parent) { teardownCheck: function(parent) {
if (!parent) { return parent ? false : teardown();
teardown();
}
} }
} }
...@@ -4071,6 +4106,12 @@ ...@@ -4071,6 +4106,12 @@
var nodesMap = [], var nodesMap = [],
// called when an item is added // called when an item is added
add = function(ev, items, index) { add = function(ev, items, index) {
// check that the placeholder textNode still has a parent.
// it's possible someone removed the contents of
// this element without removing the parent
if (data.teardownCheck(text.parentNode)) {
return
}
// Collect new html and mappings // Collect new html and mappings
var frag = document.createDocumentFragment(), var frag = document.createDocumentFragment(),
...@@ -4098,7 +4139,17 @@ ...@@ -4098,7 +4139,17 @@
}); });
[].splice.apply(nodesMap, [index, 0].concat(newMappings)); [].splice.apply(nodesMap, [index, 0].concat(newMappings));
}, },
remove = function(ev, items, index) { // Remove can be called during teardown or when items are
// removed from the element.
remove = function(ev, items, index, duringTeardown) {
// If this is because an element was removed, we should
// check to make sure the live elements are still in the page.
// If we did this during a teardown, it would cause an infinite loop.
if (!duringTeardown && data.teardownCheck(text.parentNode)) {
return
}
var removedMappings = nodesMap.splice(index, items.length), var removedMappings = nodesMap.splice(index, items.length),
itemsToRemove = []; itemsToRemove = [];
...@@ -4124,7 +4175,7 @@ ...@@ -4124,7 +4175,7 @@
// use remove to clean stuff up for us // use remove to clean stuff up for us
remove({}, { remove({}, {
length: nodesMap.length length: nodesMap.length
}, 0); }, 0, true);
} }
updateList = function(ev, newList, oldList) { updateList = function(ev, newList, oldList) {
...@@ -4140,11 +4191,7 @@ ...@@ -4140,11 +4191,7 @@
// Setup binding and teardown to add and remove events // Setup binding and teardown to add and remove events
var data = setup(parentNode, function() { var data = setup(parentNode, function() {
if (text.parentNode) {
can.isFunction(compute) && compute.bind("change", updateList) can.isFunction(compute) && compute.bind("change", updateList)
} else {
data.teardownCheck()
}
}, function() { }, function() {
can.isFunction(compute) && compute.unbind("change", updateList) can.isFunction(compute) && compute.unbind("change", updateList)
teardownList() teardownList()
...@@ -4201,7 +4248,7 @@ ...@@ -4201,7 +4248,7 @@
var parent = elements.getParentNode(el, parentNode); var parent = elements.getParentNode(el, parentNode);
// setup listening right away so we don't have to re-calculate value // setup listening right away so we don't have to re-calculate value
var data = listen(el.parentNode !== parent ? el.parentNode : parent, compute, function(ev, newVal, oldVal) { var data = listen(parent, compute, function(ev, newVal, oldVal) {
// Sometimes this is 'unknown' in IE and will throw an exception if it is // Sometimes this is 'unknown' in IE and will throw an exception if it is
if (typeof node.nodeValue != 'unknown') { if (typeof node.nodeValue != 'unknown') {
node.nodeValue = "" + newVal; node.nodeValue = "" + newVal;
...@@ -4569,10 +4616,10 @@ ...@@ -4569,10 +4616,10 @@
// used to make sure .fn and .inverse are always called with a Scope like object // used to make sure .fn and .inverse are always called with a Scope like object
makeConvertToScopes = function(orignal, scope, options) { makeConvertToScopes = function(orignal, scope, options) {
return function(updatedScope, updatedOptions) { return function(updatedScope, updatedOptions) {
if (updatedScope != null && !(updatedScope instanceof can.view.Scope)) { if (updatedScope !== undefined && !(updatedScope instanceof can.view.Scope)) {
updatedScope = scope.add(updatedScope) updatedScope = scope.add(updatedScope)
} }
if (updatedOptions != null && !(updatedOptions instanceof OptionsScope)) { if (updatedOptions !== undefined && !(updatedOptions instanceof OptionsScope)) {
updatedOptions = options.add(updatedOptions) updatedOptions = options.add(updatedOptions)
} }
return orignal(updatedScope, updatedOptions || options) return orignal(updatedScope, updatedOptions || options)
...@@ -5153,7 +5200,7 @@ ...@@ -5153,7 +5200,7 @@
// Add the reference to the list in the contexts. // Add the reference to the list in the contexts.
for (i = 0; i < name.length; i++) { for (i = 0; i < name.length; i++) {
result.push(helperOptions.fn(name[i] || '')); result.push(helperOptions.fn(name[i]));
// Ensure that live update works on observable lists // Ensure that live update works on observable lists
isObserveList && name.attr('' + i); isObserveList && name.attr('' + i);
...@@ -5174,7 +5221,7 @@ ...@@ -5174,7 +5221,7 @@
// This can cause issues if you are trying to // This can cause issues if you are trying to
// eval on the length but this is the more // eval on the length but this is the more
// common case. // common case.
return '' + (name !== undefined ? name : ''); return '' + (name != undefined ? name : '');
break; break;
} }
} }
...@@ -5354,12 +5401,14 @@ ...@@ -5354,12 +5401,14 @@
if (expr.isComputed || isObserveLike(expr) && typeof expr.attr('length') !== 'undefined') { if (expr.isComputed || isObserveLike(expr) && typeof expr.attr('length') !== 'undefined') {
return can.view.lists && can.view.lists(expr, function(item, key) { return can.view.lists && can.view.lists(expr, function(item, key) {
// Create a compute that listens to whenever the index of the item in our list changes. // Create a compute that listens to whenever the index of the item in our list changes.
var indexCompute = can.compute(function() { var index = function() {
var exprResolved = Mustache.resolve(expr); var exprResolved = Mustache.resolve(expr),
return (exprResolved).indexOf(item); fromIndex = key < (exprResolved).attr('length') ? key : undefined;
});
return (exprResolved).indexOf(item, fromIndex);
};
return options.fn(options.scope.add({ return options.fn(options.scope.add({
"@index": indexCompute "@index": index
}).add(item)); }).add(item));
}); });
} }
...@@ -5515,7 +5564,7 @@ ...@@ -5515,7 +5564,7 @@
returnObserveMethods: true, returnObserveMethods: true,
isArgument: true isArgument: true
}); });
return scopeData.value.call(scopeData.parent, data.scope._data, can.$(this), ev) return scopeData.value.call(scopeData.parent, data.scope._context, can.$(this), ev)
}; };
if (special[event]) { if (special[event]) {
...@@ -5612,7 +5661,7 @@ ...@@ -5612,7 +5661,7 @@
var self = this; var self = this;
this.on(this.scope, "change", function() { this.on(this.scope, "change", function() {
self.on(); self.on();
self.on(this.scope, "change", arguments.callee); self.on(self.scope, "change", arguments.callee);
}); });
return res; return res;
} }
...@@ -5627,10 +5676,18 @@ ...@@ -5627,10 +5676,18 @@
}) })
this.attributeScopeMappings = attributeScopeMappings; this.attributeScopeMappings = attributeScopeMappings;
// setup inheritance right away // If scope is an object,
if (!this.prototype.scope || typeof this.prototype.scope === "object") { if (!this.prototype.scope || typeof this.prototype.scope === "object") {
// use that object as the prototype of an extened Map constructor function.
// A new instance of that Map constructor function will be created and
// set as this.scope.
this.Map = can.Map.extend(this.prototype.scope || {}); this.Map = can.Map.extend(this.prototype.scope || {});
} }
// If scope is a can.Map constructor function,
else if (this.prototype.scope.prototype instanceof can.Map) {
// just use that.
this.Map = this.prototype.scope;
}
...@@ -5662,7 +5719,9 @@ ...@@ -5662,7 +5719,9 @@
component = this, component = this,
twoWayBindings = {}, twoWayBindings = {},
// what scope property is currently updating // what scope property is currently updating
scopePropertyUpdating; scopePropertyUpdating,
// the object added to the scope
componentScope;
// scope prototype properties marked with an "@" are added here // scope prototype properties marked with an "@" are added here
can.each(this.constructor.attributeScopeMappings, function(val, prop) { can.each(this.constructor.attributeScopeMappings, function(val, prop) {
...@@ -5713,18 +5772,19 @@ ...@@ -5713,18 +5772,19 @@
}) })
var componentScope
// save the scope
if (this.constructor.Map) { if (this.constructor.Map) {
componentScope = new this.constructor.Map(initalScopeData); componentScope = new this.constructor.Map(initalScopeData);
} else if (this.scope instanceof can.Map) { } else if (this.scope instanceof can.Map) {
componentScope = this.scope; componentScope = this.scope;
} else if (can.isFunction(this.scope)) { } else if (can.isFunction(this.scope)) {
var scopeResult = this.scope(initalScopeData, hookupOptions.scope, el); var scopeResult = this.scope(initalScopeData, hookupOptions.scope, el);
// if the function returns a can.Map, use that as the scope // if the function returns a can.Map, use that as the scope
if (scopeResult instanceof can.Map) { if (scopeResult instanceof can.Map) {
componentScope = scopeResult componentScope = scopeResult
} else if (typeof scopeResult == "function" && typeof scopeResult.extend == "function") { } else if (scopeResult.prototype instanceof can.Map) {
componentScope = new scopeResult(initalScopeData); componentScope = new scopeResult(initalScopeData);
} else { } else {
componentScope = new(can.Map.extend(scopeResult))(initalScopeData); componentScope = new(can.Map.extend(scopeResult))(initalScopeData);
...@@ -6099,6 +6159,7 @@ ...@@ -6099,6 +6159,7 @@
if (!can.Model) { if (!can.Model) {
return; return;
} }
this.List = ML({ this.List = ML({
Map: this Map: this
}, {}); }, {});
...@@ -6267,7 +6328,6 @@ ...@@ -6267,7 +6328,6 @@
// Model lists are just like `Map.List` except that when their items are // Model lists are just like `Map.List` except that when their items are
// destroyed, it automatically gets removed from the list. // destroyed, it automatically gets removed from the list.
var ML = can.Model.List = can.List({ var ML = can.Model.List = can.List({
setup: function(params) { setup: function(params) {
if (can.isPlainObject(params) && !can.isArray(params)) { if (can.isPlainObject(params) && !can.isArray(params)) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment