Commit 7e3db802 authored by Sindre Sorhus's avatar Sindre Sorhus

Merge pull request #886 from bantic/update-ember-1.5.0

Update to Ember 1.5.0, updates ember-data also
parents 32003552 6251179c
...@@ -5,8 +5,8 @@ ...@@ -5,8 +5,8 @@
"todomvc-common": "~0.1.4", "todomvc-common": "~0.1.4",
"jquery": "~2.1.0", "jquery": "~2.1.0",
"handlebars": "~1.3.0", "handlebars": "~1.3.0",
"ember": "~1.3.1", "ember": "~1.5.0",
"ember-data": "1.0.0-beta.6", "ember-data": "1.0.0-beta.7",
"ember-localstorage-adapter": "latest" "ember-localstorage-adapter": "latest"
} }
} }
This source diff could not be displayed because it is too large. You can view the blob instead.
/*global Ember*/ /*global Ember*/
/*global DS*/ /*global DS*/
(function () { 'use strict';
'use strict';
DS.LSAdapter = DS.Adapter.extend(Ember.Evented, { DS.LSSerializer = DS.JSONSerializer.extend({
init: function () { serializeHasMany: function(record, json, relationship) {
this._loadData(); var key = relationship.key,
}, relationshipType = DS.RelationshipChange.determineRelationshipType(record.constructor, relationship);
generateIdForRecord: function () { if (relationshipType === 'manyToNone' ||
return Math.random().toString(32).slice(2).substr(0, 5); relationshipType === 'manyToMany' ||
relationshipType === 'manyToOne') {
json[key] = record.get(key).mapBy('id');
// TODO support for polymorphic manyToNone and manyToMany relationships
}
}, },
find: function (store, type, id) { extractSingle: function(store, type, payload) {
if (payload && payload._embedded) {
for (var relation in payload._embedded) {
var typeName = Ember.String.singularize(relation),
embeddedPayload = payload._embedded[relation];
if (embeddedPayload) {
if (Ember.isArray(embeddedPayload)) {
store.pushMany(typeName, embeddedPayload);
} else {
store.push(typeName, embeddedPayload);
}
}
}
delete payload._embedded;
}
return this.normalize(type, payload);
}
});
DS.LSAdapter = DS.Adapter.extend(Ember.Evented, {
/**
This is the main entry point into finding records. The first parameter to
this method is the model's name as a string.
@method find
@param {DS.Model} type
@param {Object|String|Integer|null} id
*/
find: function(store, type, id, opts) {
var adapter = this;
var allowRecursive = true;
var namespace = this._namespaceForType(type); var namespace = this._namespaceForType(type);
return Ember.RSVP.resolve(Ember.copy(namespace.records[id]));
/**
* In the case where there are relationships, this method is called again
* for each relation. Given the relations have references to the main
* object, we use allowRecursive to avoid going further into infinite
* recursiveness.
*
* Concept from ember-indexdb-adapter
*/
if (opts && typeof opts.allowRecursive !== 'undefined') {
allowRecursive = opts.allowRecursive;
}
return new Ember.RSVP.Promise(function(resolve, reject) {
var record = Ember.A(namespace.records[id]);
if (allowRecursive && record) {
adapter.loadRelationships(type, record).then(function(finalRecord) {
resolve(finalRecord);
});
} else {
if (!record) {
reject();
} else {
resolve(record);
}
}
});
resolve(record);
}, },
findMany: function (store, type, ids) { findMany: function (store, type, ids) {
var adapter = this;
var namespace = this._namespaceForType(type); var namespace = this._namespaceForType(type);
return new Ember.RSVP.Promise(function(resolve, reject) {
var results = []; var results = [];
for (var i = 0; i < ids.length; i++) { for (var i = 0; i < ids.length; i++) {
results.push(Ember.copy(namespace.records[ids[i]])); results.push(Ember.copy(namespace.records[ids[i]]));
} }
return Ember.RSVP.resolve(results);
resolve(results);
});
}, },
// Supports queries that look like this: // Supports queries that look like this:
...@@ -42,14 +114,15 @@ ...@@ -42,14 +114,15 @@
// //
// { complete: true, name: /foo|bar/ } // { complete: true, name: /foo|bar/ }
findQuery: function (store, type, query, recordArray) { findQuery: function (store, type, query, recordArray) {
var namespace = this._namespaceForType(type); var namespace = this._namespaceForType(type),
var results = this.query(namespace.records, query); results = this.query(namespace.records, query);
return Ember.RSVP.resolve(results); return Ember.RSVP.resolve(results);
}, },
query: function (records, query) { query: function (records, query) {
var results = []; var results = [],
var id, record, property, test, push; id, record, property, test, push;
for (id in records) { for (id in records) {
record = records[id]; record = records[id];
for (property in query) { for (property in query) {
...@@ -69,8 +142,9 @@ ...@@ -69,8 +142,9 @@
}, },
findAll: function (store, type) { findAll: function (store, type) {
var namespace = this._namespaceForType(type); var namespace = this._namespaceForType(type),
var results = []; results = [];
for (var id in namespace.records) { for (var id in namespace.records) {
results.push(Ember.copy(namespace.records[id])); results.push(Ember.copy(namespace.records[id]));
} }
...@@ -78,53 +152,260 @@ ...@@ -78,53 +152,260 @@
}, },
createRecord: function (store, type, record) { createRecord: function (store, type, record) {
var namespace = this._namespaceForType(type); var namespaceRecords = this._namespaceForType(type),
this._addRecordToNamespace(namespace, record); recordHash = record.serialize({includeId: true});
this._saveData();
namespaceRecords.records[recordHash.id] = recordHash;
this.persistData(type, namespaceRecords);
return Ember.RSVP.resolve(); return Ember.RSVP.resolve();
}, },
updateRecord: function (store, type, record) { updateRecord: function (store, type, record) {
var namespace = this._namespaceForType(type); var namespaceRecords = this._namespaceForType(type),
var id = record.get('id'); id = record.get('id');
namespace.records[id] = record.toJSON({ includeId: true });
this._saveData(); namespaceRecords.records[id] = record.serialize({ includeId: true });
this.persistData(type, namespaceRecords);
return Ember.RSVP.resolve(); return Ember.RSVP.resolve();
}, },
deleteRecord: function (store, type, record) { deleteRecord: function (store, type, record) {
var namespace = this._namespaceForType(type); var namespaceRecords = this._namespaceForType(type),
var id = record.get('id'); id = record.get('id');
delete namespace.records[id];
this._saveData(); delete namespaceRecords.records[id];
this.persistData(type, namespaceRecords);
return Ember.RSVP.resolve(); return Ember.RSVP.resolve();
}, },
generateIdForRecord: function () {
return Math.random().toString(32).slice(2).substr(0, 5);
},
// private // private
_getNamespace: function () { adapterNamespace: function () {
return this.namespace || 'DS.LSAdapter'; return this.namespace || 'DS.LSAdapter';
}, },
_loadData: function () { loadData: function () {
var storage = localStorage.getItem(this._getNamespace()); var storage = localStorage.getItem(this.adapterNamespace());
this._data = storage ? JSON.parse(storage) : {}; return storage ? JSON.parse(storage) : {};
}, },
_saveData: function () { persistData: function(type, data) {
localStorage.setItem(this._getNamespace(), JSON.stringify(this._data)); var modelNamespace = this.modelNamespace(type),
localStorageData = this.loadData();
localStorageData[modelNamespace] = data;
localStorage.setItem(this.adapterNamespace(), JSON.stringify(localStorageData));
}, },
_namespaceForType: function (type) { _namespaceForType: function (type) {
var namespace = type.url || type.toString(); var namespace = this.modelNamespace(type),
return this._data[namespace] || ( storage = localStorage.getItem(this.adapterNamespace());
this._data[namespace] = {records: {}}
); return storage ? JSON.parse(storage)[namespace] || {records: {}} : {records: {}};
},
modelNamespace: function(type) {
return type.url || type.toString();
},
/**
* This takes a record, then analyzes the model relationships and replaces
* ids with the actual values.
*
* Stolen from ember-indexdb-adapter
*
* Consider the following JSON is entered:
*
* ```js
* {
* "id": 1,
* "title": "Rails Rambo",
* "comments": [1, 2]
* }
*
* This will return:
*
* ```js
* {
* "id": 1,
* "title": "Rails Rambo",
* "comments": [1, 2]
*
* "_embedded": {
* "comment": [{
* "_id": 1,
* "comment_title": "FIRST"
* }, {
* "_id": 2,
* "comment_title": "Rails is unagi"
* }]
* }
* }
*
* This way, whenever a resource returned, its relationships will be also
* returned.
*
* @method loadRelationships
* @private
* @param {DS.Model} type
* @param {Object} record
*/
loadRelationships: function(type, record) {
var adapter = this;
return new Ember.RSVP.Promise(function(resolve, reject) {
var resultJSON = {},
typeKey = type.typeKey,
relationshipNames, relationships,
relationshipPromises = [];
relationshipNames = Ember.get(type, 'relationshipNames');
relationships = relationshipNames.belongsTo;
relationships = relationships.concat(relationshipNames.hasMany);
relationships.forEach(function(relationName) {
var relationModel = type.typeForRelationship(relationName),
relationEmbeddedId = record[relationName],
relationProp = adapter.relationshipProperties(type, relationName),
relationType = relationProp.kind,
/**
* This is the relationship field.
*/
promise, embedPromise;
var opts = {allowRecursive: false};
/**
* embeddedIds are ids of relations that are included in the main
* payload, such as:
*
* {
* cart: {
* id: "s85fb",
* customer: "rld9u"
* }
* }
*
* In this case, cart belongsTo customer and its id is present in the
* main payload. We find each of these records and add them to _embedded.
*/
if (relationEmbeddedId) {
if (relationType == 'belongsTo' || relationType == 'hasOne') {
promise = adapter.find(null, relationModel, relationEmbeddedId, opts)
} else if (relationType == 'hasMany') {
promise = adapter.findMany(null, relationModel, relationEmbeddedId, opts)
}
embedPromise = new Ember.RSVP.Promise(function(resolve, reject) {
promise.then(function(relationRecord) {
var finalPayload = adapter.addEmbeddedPayload(record, relationName, relationRecord)
resolve(finalPayload);
});
});
relationshipPromises.push(embedPromise);
}
});
Ember.RSVP.all(relationshipPromises).then(function() {
resolve(record);
});
});
}, },
_addRecordToNamespace: function (namespace, record) {
var data = record.serialize({includeId: true}); /**
namespace.records[data.id] = data; * Given the following payload,
*
* {
* cart: {
* id: "1",
* customer: "2"
* }
* }
*
* With `relationshipName` being `customer` and `relationshipRecord`
*
* {id: "2", name: "Rambo"}
*
* This method returns the following payload:
*
* {
* cart: {
* id: "1",
* customer: "2"
* },
* _embedded: {
* customer: {
* id: "2",
* name: "Rambo"
* }
* }
* }
*
* which is then treated by the serializer later.
*
* @method addEmbeddedPayload
* @private
* @param {Object} payload
* @param {String} relationshipName
* @param {Object} relationshipRecord
*/
addEmbeddedPayload: function(payload, relationshipName, relationshipRecord) {
var objectHasId = (relationshipRecord && relationshipRecord.id),
arrayHasIds = (relationshipRecord.length && relationshipRecord.everyBy("id")),
isValidRelationship = (objectHasId || arrayHasIds);
if (isValidRelationship) {
if (!payload['_embedded']) {
payload['_embedded'] = {}
}
payload['_embedded'][relationshipName] = relationshipRecord;
if (relationshipRecord.length) {
payload[relationshipName] = relationshipRecord.mapBy('id');
} else {
payload[relationshipName] = relationshipRecord.id;
}
} }
if (this.isArray(payload[relationshipName])) {
payload[relationshipName] = payload[relationshipName].filter(function(id) {
return id;
}); });
})(); }
return payload;
},
isArray: function(value) {
return Object.prototype.toString.call(value) === '[object Array]';
},
/**
*
* @method relationshipProperties
* @private
* @param {DS.Model} type
* @param {String} relationName
*/
relationshipProperties: function(type, relationName) {
var relationships = Ember.get(type, 'relationshipsByName');
if (relationName) {
return relationships.get(relationName);
} else {
return relationships;
}
}
});
This source diff could not be displayed because it is too large. You can view the blob instead.
...@@ -57,11 +57,15 @@ ...@@ -57,11 +57,15 @@
<a href="http://github.com/tomdale">Tom Dale</a>, <a href="http://github.com/tomdale">Tom Dale</a>,
<a href="http://github.com/addyosmani">Addy Osmani</a> <a href="http://github.com/addyosmani">Addy Osmani</a>
</p> </p>
<p>
Updated by
<a href="http://github.com/bantic">Cory Forsyth</a>
</p>
<p>Part of <a href="http://todomvc.com">TodoMVC</a></p> <p>Part of <a href="http://todomvc.com">TodoMVC</a></p>
</footer> </footer>
</script> </script>
<script src="bower_components/todomvc-common/base.js"></script> <script src="bower_components/todomvc-common/base.js"></script>
<script src="bower_components/jquery/jquery.js"></script> <script src="bower_components/jquery/dist/jquery.js"></script>
<script src="bower_components/handlebars/handlebars.js"></script> <script src="bower_components/handlebars/handlebars.js"></script>
<script src="bower_components/ember/ember.js"></script> <script src="bower_components/ember/ember.js"></script>
<script src="bower_components/ember-data/ember-data.js"></script> <script src="bower_components/ember-data/ember-data.js"></script>
......
/*global Todos, Ember */ /*global Ember */
(function () { (function () {
'use strict'; 'use strict';
Ember.Handlebars.helper('pluralize', function (singular, count) { Ember.Handlebars.helper('pluralize', function (singular, count) {
/* From Ember-Data */ /* From Ember-Data */
var inflector = new Ember.Inflector(Ember.Inflector.defaultRules); var inflector = Ember.Inflector.inflector;
return count === 1 ? singular : inflector.pluralize(singular); return count === 1 ? singular : inflector.pluralize(singular);
}); });
......
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