2015-03-24 23:31:14 +00:00
|
|
|
/*global define,describe,it,expect,beforeEach,jasmine*/
|
2014-11-25 01:01:37 +00:00
|
|
|
|
|
|
|
define(
|
|
|
|
["../../src/objects/EditableDomainObjectCache"],
|
|
|
|
function (EditableDomainObjectCache) {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
describe("Editable domain object cache", function () {
|
|
|
|
|
|
|
|
var captured,
|
|
|
|
completionCapability,
|
|
|
|
object,
|
2015-03-24 23:31:14 +00:00
|
|
|
mockQ,
|
2014-11-25 01:01:37 +00:00
|
|
|
cache;
|
|
|
|
|
|
|
|
|
|
|
|
// Constructors for test objects
|
|
|
|
function TestObject(id) {
|
|
|
|
return {
|
|
|
|
getId: function () { return id; },
|
|
|
|
getModel: function () { return {}; },
|
|
|
|
getCapability: function (name) {
|
|
|
|
return completionCapability;
|
2015-04-06 16:28:53 +00:00
|
|
|
},
|
|
|
|
hasCapability: function (name) {
|
|
|
|
return false;
|
2014-11-25 01:01:37 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2015-01-27 20:18:28 +00:00
|
|
|
function WrapObject(domainObject, model) {
|
2014-11-25 01:01:37 +00:00
|
|
|
var result = Object.create(domainObject);
|
|
|
|
result.wrapped = true;
|
2015-01-27 20:18:28 +00:00
|
|
|
result.wrappedModel = model;
|
2015-04-06 16:28:53 +00:00
|
|
|
result.hasCapability = function (name) {
|
|
|
|
return name === 'editor';
|
|
|
|
};
|
2014-11-25 01:01:37 +00:00
|
|
|
captured.wraps = (captured.wraps || 0) + 1;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
beforeEach(function () {
|
2015-03-24 23:31:14 +00:00
|
|
|
mockQ = jasmine.createSpyObj('$q', ['when', 'all']);
|
2014-11-25 01:01:37 +00:00
|
|
|
captured = {};
|
|
|
|
completionCapability = {
|
|
|
|
save: function () {
|
|
|
|
captured.saved = (captured.saved || 0) + 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-03-24 23:31:14 +00:00
|
|
|
cache = new EditableDomainObjectCache(WrapObject, mockQ);
|
2014-11-25 01:01:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("wraps objects using provided constructor", function () {
|
|
|
|
var domainObject = new TestObject('test-id'),
|
|
|
|
wrappedObject = cache.getEditableObject(domainObject);
|
|
|
|
expect(wrappedObject.wrapped).toBeTruthy();
|
|
|
|
expect(wrappedObject.getId()).toEqual(domainObject.getId());
|
|
|
|
});
|
|
|
|
|
2015-01-27 20:18:28 +00:00
|
|
|
it("wraps objects repeatedly, wraps models once", function () {
|
2014-11-25 01:01:37 +00:00
|
|
|
var domainObject = new TestObject('test-id'),
|
2015-01-27 20:18:28 +00:00
|
|
|
wrappedObjects = [];
|
2014-11-25 01:01:37 +00:00
|
|
|
|
|
|
|
// Verify precondition
|
|
|
|
expect(captured.wraps).toBeUndefined();
|
|
|
|
|
|
|
|
// Invoke a few more times; expect count not to increment
|
2015-01-27 20:18:28 +00:00
|
|
|
wrappedObjects.push(cache.getEditableObject(domainObject));
|
2014-11-25 01:01:37 +00:00
|
|
|
expect(captured.wraps).toEqual(1);
|
2015-01-27 20:18:28 +00:00
|
|
|
wrappedObjects.push(cache.getEditableObject(domainObject));
|
|
|
|
expect(captured.wraps).toEqual(2);
|
|
|
|
wrappedObjects.push(cache.getEditableObject(domainObject));
|
|
|
|
expect(captured.wraps).toEqual(3);
|
2014-11-25 01:01:37 +00:00
|
|
|
|
|
|
|
// Verify that the last call still gave us a wrapped object
|
2015-01-27 20:18:28 +00:00
|
|
|
expect(wrappedObjects[0].wrapped).toBeTruthy();
|
|
|
|
expect(wrappedObjects[0].getId()).toEqual(domainObject.getId());
|
|
|
|
|
|
|
|
// Verify that objects are distinct but models are identical
|
|
|
|
expect(wrappedObjects[0].wrappedModel)
|
|
|
|
.toBe(wrappedObjects[1].wrappedModel);
|
|
|
|
expect(wrappedObjects[0]).not
|
|
|
|
.toBe(wrappedObjects[1]);
|
2014-11-25 01:01:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("saves objects that have been marked dirty", function () {
|
|
|
|
var objects = ['a', 'b', 'c'].map(TestObject).map(cache.getEditableObject);
|
|
|
|
|
|
|
|
cache.markDirty(objects[0]);
|
|
|
|
cache.markDirty(objects[2]);
|
|
|
|
|
|
|
|
cache.saveAll();
|
|
|
|
|
|
|
|
expect(captured.saved).toEqual(2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not save objects that have been marked clean", function () {
|
|
|
|
var objects = ['a', 'b', 'c'].map(TestObject).map(cache.getEditableObject);
|
|
|
|
|
|
|
|
cache.markDirty(objects[0]);
|
|
|
|
cache.markDirty(objects[2]);
|
|
|
|
cache.markClean(objects[0]);
|
|
|
|
|
|
|
|
cache.saveAll();
|
|
|
|
|
|
|
|
expect(captured.saved).toEqual(1);
|
|
|
|
});
|
|
|
|
|
2015-03-18 19:36:51 +00:00
|
|
|
it("tracks the root object of the Edit mode subgraph", function () {
|
|
|
|
// Root object is the first object exposed to the cache
|
|
|
|
var domainObjects = ['a', 'b', 'c'].map(TestObject);
|
|
|
|
domainObjects.forEach(function (obj) {
|
|
|
|
cache.getEditableObject(obj);
|
|
|
|
});
|
|
|
|
expect(cache.isRoot(domainObjects[0])).toBeTruthy();
|
|
|
|
expect(cache.isRoot(domainObjects[1])).toBeFalsy();
|
|
|
|
expect(cache.isRoot(domainObjects[2])).toBeFalsy();
|
|
|
|
});
|
|
|
|
|
2015-04-06 16:28:53 +00:00
|
|
|
it("does not double-wrap objects", function () {
|
|
|
|
var domainObject = new TestObject('test-id'),
|
|
|
|
wrappedObject = cache.getEditableObject(domainObject);
|
|
|
|
|
|
|
|
// Same instance should be returned if you try to wrap
|
|
|
|
// twice. This is necessary, since it's possible to (e.g.)
|
|
|
|
// use a context capability on an object retrieved via
|
|
|
|
// composition, in which case a result will already be
|
|
|
|
// wrapped.
|
|
|
|
expect(cache.getEditableObject(wrappedObject))
|
|
|
|
.toBe(wrappedObject);
|
|
|
|
});
|
|
|
|
|
2014-11-25 01:01:37 +00:00
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
);
|