From 6354cee838ae49f08d66f12ce55746000d7bef6d Mon Sep 17 00:00:00 2001 From: Ricardo Villarreal Date: Wed, 24 Aug 2016 14:41:59 -0700 Subject: [PATCH 01/16] EPICENTER-2129: update jQuery dependency to version 3 --- bower.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bower.json b/bower.json index c6028c03..24e2773d 100644 --- a/bower.json +++ b/bower.json @@ -16,7 +16,7 @@ "tests" ], "dependencies": { - "jquery": "~2.1.1", + "jquery": "~3.1.0", "cometd-jquery": "2.9.0" }, "devDependencies": {} From 6e26b102d07d99c0a1e35090ec7a7affc75cd062 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Wed, 7 Sep 2016 14:27:41 -0700 Subject: [PATCH 02/16] gitattributes --- .gitattributes | 1 + .gitignore | 1 + 2 files changed, 2 insertions(+) create mode 100644 .gitattributes diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 00000000..967a087a --- /dev/null +++ b/.gitattributes @@ -0,0 +1 @@ +dist/* binary \ No newline at end of file diff --git a/.gitignore b/.gitignore index fa5e5267..367bd67d 100644 --- a/.gitignore +++ b/.gitignore @@ -12,6 +12,7 @@ pids !.bowerrc !.gitignore !.jscsrc +!.gitattributes # sass .sass-cache/ From 235a561096ba7860d5b84715701d0ec00745a9d6 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 13:12:25 -0700 Subject: [PATCH 03/16] fix context issue --- src/managers/run-strategies/multiplayer-strategy.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/managers/run-strategies/multiplayer-strategy.js b/src/managers/run-strategies/multiplayer-strategy.js index cb118a6c..40c42420 100644 --- a/src/managers/run-strategies/multiplayer-strategy.js +++ b/src/managers/run-strategies/multiplayer-strategy.js @@ -60,7 +60,7 @@ var Strategy = classFrom(IdentityStrategy, { var serverError = function (error) { // is this possible? - dtd.reject(error, session, this.options); + dtd.reject(error, session, _this.options); }; this.worldApi From 3dad06e4f6afb2dcc67cba463ef789f45652e7dc Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 13:32:00 -0700 Subject: [PATCH 04/16] fix error by checking with callback --- tests/spec/test-variables-api-service.js | 77 ++++++++++++------------ 1 file changed, 40 insertions(+), 37 deletions(-) diff --git a/tests/spec/test-variables-api-service.js b/tests/spec/test-variables-api-service.js index 23d69893..fc1b172c 100644 --- a/tests/spec/test-variables-api-service.js +++ b/tests/spec/test-variables-api-service.js @@ -47,7 +47,7 @@ xhr.respond(200, { 'Content-Type': 'application/json' }, JSON.stringify(variablesCD)); return true; }); - server.autoRespond = true; + server.respondImmediately = true; rs = new RunService({ account: account, project: project }); vs = rs.variables(); @@ -68,14 +68,14 @@ }); it('should use the right url', function () { vs.load('price'); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';/variables/price/'); }); it('should not add the autorestore run flag', function () { vs.load('price'); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.not.have.property('X-AutoRestore'); @@ -84,7 +84,7 @@ var rs = new RunService({ account: account, project: 'js-libs', filter: 'myfancyrunid' }); var vs = rs.variables(); vs.load('price'); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.have.property('X-AutoRestore', true); @@ -93,7 +93,7 @@ var rs = new RunService({ account: account, project: 'js-libs', filter: 'myfancyrunid', autoRestore: false }); var vs = rs.variables(); vs.load('price'); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.not.have.property('X-AutoRestore'); @@ -103,28 +103,28 @@ describe('#query()', function () { it('should do a GET', function () { vs.query(['price', 'sales']); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.method.toUpperCase().should.equal('GET'); }); it('should convert includes', function () { vs.query({ include: ['price', 'sales'] }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';/variables/?include=price,sales'); }); it('should convert sets', function () { vs.query({ set: 'a' }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';/variables/?set=a'); }); it('should convert sets & includes', function () { vs.query({ set: ['a', 'b'], include: 'price' }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';/variables/?set=a,b&include=price'); @@ -134,7 +134,7 @@ var include = createLargeInclude(); vs.query(include); - server.respond(); + // server.respond(); server.requests.length.should.be.above(1); server.requests.forEach(function (xhr) { xhr.url.length.should.be.below(2048); @@ -142,24 +142,27 @@ server.requests = []; }); it('should aggregate the response from the multiple GETs from the variables API', function () { - server.requests = []; - var done = sinon.spy(); - var fail = sinon.spy(); + var success = sinon.spy(); + var fail = sinon.spy(function () { + console.log('failll'); + }); var rs = new RunService({ account: account, project: project }); var include = createLargeInclude(); include.push('variables_c_d'); include = ['variables_a_b'].concat(include); - rs.query({}, { include: include }).done(done).fail(fail); - server.respond(); - done.should.have.been.calledWith({ - 'varA': 'Value A', - 'varB': 0.0001, - 'varC': 'Another string for run1', - 'varD': '2015-11-16 10:10:10' - }); - fail.should.not.have.been.called; - server.requests = []; + return rs.query({}, { include: include }) + .then(success) + .fail(fail) + .then(function () { + success.should.have.been.calledWith({ + varA: 'Value A', + varB: 0.0001, + varC: 'Another string for run1', + varD: '2015-11-16 10:10:10' + }); + fail.should.not.have.been.called; + }); }); it('the multiple GETs encoded urls length should not be larger than 2048', function () { server.requests = []; @@ -174,7 +177,7 @@ }); it('should not add the autorestore run flag', function () { vs.query({ set: ['a', 'b'], include: 'price' }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.not.have.property('X-AutoRestore'); @@ -183,7 +186,7 @@ var rs = new RunService({ account: account, project: 'js-libs', filter: 'myfancyrunid' }); var vs = rs.variables(); vs.query({ set: ['a', 'b'], include: 'price' }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.have.property('X-AutoRestore', true); @@ -192,7 +195,7 @@ var rs = new RunService({ account: account, project: 'js-libs', filter: 'myfancyrunid', autoRestore: false }); var vs = rs.variables(); vs.query({ set: ['a', 'b'], include: 'price' }); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.requestHeaders.should.not.have.property('X-AutoRestore'); @@ -204,7 +207,7 @@ // Temporarily using PATCH to mean PUT // it('should do a PUT', function () { // vs.save({ a: 1, b: 2 }); - // server.respond(); + // server.respond(); // var req = server.requests.pop(); @@ -213,7 +216,7 @@ it('should do a PATCH', function () { vs.save({ a: 1, b: 2 }); - server.respond(); + // server.respond(); var req = server.requests.pop(); @@ -223,7 +226,7 @@ it('should send requests in the body', function () { var params = { a: 1, b: 2 }; vs.save(params); - server.respond(); + // server.respond(); var req = server.requests.pop(); @@ -232,7 +235,7 @@ }); it('should support setting key, value syntax', function () { vs.save('a', 1); - server.respond(); + // server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';/variables/'); @@ -243,7 +246,7 @@ // describe('#merge()', function () { // it('should do a PATCH', function () { // vs.merge({ a: 1, b: 2 }); - // server.respond(); + // server.respond(); // var req = server.requests.pop(); // req.method.toUpperCase().should.equal('PATCH'); @@ -252,7 +255,7 @@ // it('should send requests in the body', function () { // var params = { a: 1, b: 2 }; // vs.merge(params); - // server.respond(); + // server.respond(); // var req = server.requests.pop(); // req.url.should.equal(baseURL + ';/variables/'); @@ -261,7 +264,7 @@ // it('should support setting key, value syntax', function () { // vs.merge('a', 1); - // server.respond(); + // server.respond(); // var req = server.requests.pop(); // req.url.should.equal(baseURL + ';/variables/'); @@ -275,7 +278,7 @@ var cb1 = sinon.spy(); vs.load('sales', null, { success: cb1 }); - server.respond(); + // server.respond(); cb1.called.should.equal(true); }); }); @@ -284,7 +287,7 @@ var cb1 = sinon.spy(); vs.query({ include: ['price', 'sales'] }, null, { success: cb1 }); - server.respond(); + // server.respond(); cb1.called.should.equal(true); }); }); @@ -295,7 +298,7 @@ vs.save({ a: 1, b: 2 }, { success: cb1 }); vs.save('a', 1, { success: cb2 }); - server.respond(); + // server.respond(); cb1.called.should.equal(true); cb2.called.should.equal(true); }); @@ -307,7 +310,7 @@ // vs.merge({ a: 1, b: 2 }, { success: cb1 }); // vs.merge('a', 1, { success: cb2 }); - // server.respond(); + // server.respond(); // cb1.called.should.equal(true); // cb2.called.should.equal(true); // }); From faf53c7fd6ae8d31a06c9838aee0f2680656a6d2 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 13:41:31 -0700 Subject: [PATCH 05/16] passing tests --- tests/spec/test-asset-api-adapter.js | 36 +++++++++++++--------------- 1 file changed, 17 insertions(+), 19 deletions(-) diff --git a/tests/spec/test-asset-api-adapter.js b/tests/spec/test-asset-api-adapter.js index fe85d04f..b5bc9020 100644 --- a/tests/spec/test-asset-api-adapter.js +++ b/tests/spec/test-asset-api-adapter.js @@ -25,7 +25,7 @@ server.respondWith('POST', /(.*)\/asset\/(.*)\/(.*)/, function (xhr, id) { xhr.respond(204); }); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { @@ -177,29 +177,27 @@ it('should get the list of the assets for the user', function () { var callback = sinon.spy(); var aa = new F.service.Asset(defaults); - aa.list({ scope: 'user', fullUrl: false }).done(callback); - - server.respond(); - var req = server.requests.pop(); - req.url.should.equal(baseURL + 'user/forio/js-libs/asset-group/myUserId'); - req.method.should.equal('GET'); - callback.should.have.been.called; - callback.should.have.been.calledWith(['file.txt', 'file2.txt']); + return aa.list({ scope: 'user', fullUrl: false }).then(callback).then(function () { + var req = server.requests.pop(); + req.url.should.equal(baseURL + 'user/forio/js-libs/asset-group/myUserId'); + req.method.should.equal('GET'); + callback.should.have.been.called; + callback.should.have.been.calledWith(['file.txt', 'file2.txt']); + }); }); it('should get the list of the assets for the user with the full URL', function () { var callback = sinon.spy(); var aa = new F.service.Asset(defaults); - aa.list({ scope: 'user' }).done(callback); - - server.respond(); - var req = server.requests.pop(); - req.url.should.equal(baseURL + 'user/forio/js-libs/asset-group/myUserId'); - req.method.should.equal('GET'); - callback.should.have.been.called; - callback.should.have.been.calledWith([ - baseURL + 'user/forio/js-libs/asset-group/myUserId/file.txt', - baseURL + 'user/forio/js-libs/asset-group/myUserId/file2.txt']); + return aa.list({ scope: 'user' }).then(callback).then(function () { + var req = server.requests.pop(); + req.url.should.equal(baseURL + 'user/forio/js-libs/asset-group/myUserId'); + req.method.should.equal('GET'); + callback.should.have.been.called; + callback.should.have.been.calledWith([ + baseURL + 'user/forio/js-libs/asset-group/myUserId/file.txt', + baseURL + 'user/forio/js-libs/asset-group/myUserId/file2.txt']); + }); }); }); From e631d6b09f7e487c56e5baeef308fa0949e76e64 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 14:06:03 -0700 Subject: [PATCH 06/16] passing tests --- tests/spec/test-run-api-service-callbacks.js | 159 +++++++++---------- 1 file changed, 71 insertions(+), 88 deletions(-) diff --git a/tests/spec/test-run-api-service-callbacks.js b/tests/spec/test-run-api-service-callbacks.js index 1d83c8cb..5a40684b 100644 --- a/tests/spec/test-run-api-service-callbacks.js +++ b/tests/spec/test-run-api-service-callbacks.js @@ -30,7 +30,7 @@ xhr.respond(400, { 'Content-Type': 'application/json' }, JSON.stringify({ url: xhr.url })); }); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { @@ -43,11 +43,10 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); - rs.do('add', [1,2], { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.do('add', [1,2], { success: cb1 }).then(cb2).then( function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -56,12 +55,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs', filter: { saved: true } }); - rs.do('add', [1,2], { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.do('add', [1,2], { error: cb1 }).then(null, cb3).then(null, function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#serial', function () { @@ -70,12 +68,10 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); - rs.serial([{ first: [1,2] }, { second: [2,3] }], null, { success: cb1 }).then(cb2); - server.respond(); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.serial([{ first: [1,2] }, { second: [2,3] }], null, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -83,12 +79,11 @@ var cb3 = sinon.spy(); var rs = new RunService({ account: 'failure', project: 'js-libs', filter: { saved: true } }); - rs.serial([{ first: [1,2] }, { second: [2,3] }], null, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.serial([{ first: [1,2] }, { second: [2,3] }], null, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#parallel', function () { @@ -98,11 +93,10 @@ var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); - rs.parallel([{ first: [1,2] }, { second: [2,3] }], null, { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.parallel([{ first: [1,2] }, { second: [2,3] }], null, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -111,12 +105,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs', filter: { saved: true } }); - rs.parallel([{ first: [1,2] }, { second: [2,3] }], null, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.parallel([{ first: [1,2] }, { second: [2,3] }], null, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); @@ -126,11 +119,10 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs' }); - rs.create('model.jl', { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.create('model.jl', { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -139,12 +131,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs' }); - rs.create('model.jl', { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.create('model.jl', { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#query', function () { @@ -153,10 +144,10 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs' }); - rs.query({ saved: true, '.price': '>1' }, { page: 1 }, { success: cb1 }).then(cb2); - server.respond(); + return rs.query({ saved: true, '.price': '>1' }, { page: 1 }, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; - cb1.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -165,12 +156,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs' }); - rs.query({ saved: true, '.price': '>1' }, { page: 1 }, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.query({ saved: true, '.price': '>1' }, { page: 1 }, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#filter', function () { @@ -179,10 +169,9 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs' }); - rs.filter({ saved: true, '.price': '>1' }, { page: 1 }, { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; + return rs.filter({ saved: true, '.price': '>1' }, { page: 1 }, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -191,12 +180,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs' }); - rs.filter({ saved: true, '.price': '>1' }, { page: 1 }, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.filter({ saved: true, '.price': '>1' }, { page: 1 }, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#load', function () { @@ -204,11 +192,10 @@ var cb1 = sinon.spy(); var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs' }); - rs.load('myfancyrunid', { include: 'score' }, { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.load('myfancyrunid', { include: 'score' }, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -217,12 +204,11 @@ var rs = new RunService({ account: 'failure', project: 'js-libs' }); - rs.load('myfancyrunid', { include: 'score' }, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.load('myfancyrunid', { include: 'score' }, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); describe('#save', function () { @@ -231,11 +217,10 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); - rs.save({ completed: true }, { success: cb1 }).then(cb2); - server.respond(); - - cb1.should.have.been.called; - cb2.should.have.been.called; + return rs.save({ completed: true }, { success: cb1 }).then(cb2).then(function () { + cb1.should.have.been.called; + cb2.should.have.been.called; + }); }); it('passes error callbacks', function () { var cb1 = sinon.spy(); @@ -244,14 +229,12 @@ var rs = new RunService({ account: 'failure', project: 'js-libs', filter: { saved: true } }); - rs.save({ completed: true }, { error: cb1 }).fail(cb3); - server.respond(); - - cb1.should.have.been.called; - cb2.should.not.have.been.called; - cb3.should.have.been.called; + return rs.save({ completed: true }, { error: cb1 }).then(null, cb3).then(function () { + cb1.should.have.been.called; + cb2.should.not.have.been.called; + cb3.should.have.been.called; + }); }); }); - }); })(); From b5c0cd46be3b721e5e99db908a1ccbf890dceebd Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 14:11:25 -0700 Subject: [PATCH 07/16] passing tests --- tests/spec/test-run-api-service.js | 191 ++++++++++++----------------- 1 file changed, 81 insertions(+), 110 deletions(-) diff --git a/tests/spec/test-run-api-service.js b/tests/spec/test-run-api-service.js index c3813fb6..efad2073 100644 --- a/tests/spec/test-run-api-service.js +++ b/tests/spec/test-run-api-service.js @@ -18,7 +18,7 @@ return include; }; - describe('Run API Service', function () { + describe.only('Run API Service', function () { var server; before(function () { server = sinon.fakeServer.create(); @@ -133,9 +133,12 @@ xhr.respond(500, { 'Content-Type': 'application/json' }, JSON.stringify({ message: 'Internal server error' })); return true; }); - server.autoRespond = true; + server.respondImmediately = true; }); + afterEach(function () { + server.requests = []; + }); after(function () { server.restore(); }); @@ -165,24 +168,22 @@ it('should return promiseables', function () { var callback = sinon.spy(); var rs = new RunService({ account: account, project: project }); - rs + return rs .create('model.jl') - .then(callback); - - server.respond(); - - callback.should.have.been.called; - callback.should.have.been.calledWith({ - 'id': '065dfe50-d29d-4b55-a0fd-30868d7dd26c', - 'model': 'model.vmf', - 'account': 'mit', - 'project': 'afv', - 'saved': false, - 'lastModified': '2014-06-20T04:09:45.738Z', - 'created': '2014-06-20T04:09:45.738Z' + .then(callback) + .then(function () { + callback.should.have.been.called; + callback.should.have.been.calledWith({ + 'id': '065dfe50-d29d-4b55-a0fd-30868d7dd26c', + 'model': 'model.vmf', + 'account': 'mit', + 'project': 'afv', + 'saved': false, + 'lastModified': '2014-06-20T04:09:45.738Z', + 'created': '2014-06-20T04:09:45.738Z' + }); }); // callback.should.have.been.calledOn(rs); - }); describe('transport.options', function () { @@ -192,7 +193,6 @@ var rs = new RunService({ account: account, project: 'js-libs', transport: { beforeSend: beforeSend, complete: complete } }); rs.create('model.jl'); - server.respond(); beforeSend.should.have.been.called; complete.should.have.been.called; }); @@ -203,7 +203,6 @@ var rs = new RunService({ account: account, project: 'js-libs', transport: { complete: originalComplete } }); rs.create('model.jl', { complete: complete }); - server.respond(); originalComplete.should.not.have.been.called; complete.should.have.been.called; }); @@ -214,7 +213,6 @@ var rs = new RunService({ account: account, project: 'js-libs', transport: { complete: originalComplete } }); rs.create('model.jl', { complete: complete }); - server.respond(); originalComplete.should.not.have.been.called; complete.should.have.been.called; }); @@ -225,7 +223,6 @@ var rs = new RunService({ account: account, project: 'js-libs', success: originalSuccess, transport: { complete: transportSuccess } }); rs.create('model.jl'); - server.respond(); originalSuccess.should.have.been.called; transportSuccess.should.have.been.called; }); @@ -300,20 +297,13 @@ }); it('should be idempotent across multiple queries', function () { - server.requests = []; var rs = new RunService({ account: account, project: project }); - rs.query({ saved: true, '.price': '>1' }); - server.respond(); - - server.requests[0].url.should.equal(baseURL + ';saved=true;.price>1/'); - - - rs.query({ saved: false, '.sales': '<4' }); - server.respond(); - - server.requests[1].url.should.equal(baseURL + ';saved=false;.sales<4/'); - server.requests = []; - + return rs.query({ saved: true, '.price': '>1' }).then(function () { + return rs.query({ saved: false, '.sales': '<4' }).then(function () { + server.requests[0].url.should.equal(baseURL + ';saved=true;.price>1/'); + server.requests[1].url.should.equal(baseURL + ';saved=false;.sales<4/'); + }); + }); }); it('should convert op modifiers to query strings', function () { var rs = new RunService({ account: account, project: project }); @@ -323,94 +313,86 @@ req.url.should.equal(baseURL + ';/?page=1&limit=2'); }); it('should split the get in multiple GETs', function () { - server.requests = []; var rs = new RunService({ account: account, project: project }); var include = createLargeInclude(); rs.query({}, { include: include }); - //server.respond(); + server.respond(); server.requests.length.should.be.above(1); server.requests.forEach(function (xhr) { xhr.url.length.should.be.below(2049); }); - server.requests = []; }); it('should fail if one or more of the multiple GETs fail', function () { - server.requests = []; - var done = sinon.spy(); + var success = sinon.spy(); var fail = sinon.spy(); var rs = new RunService({ account: account, project: project }); var include = createLargeInclude(); include.push('internal_server_error'); - rs.query({}, { include: include }).then(done, fail); - server.respond(); - fail.should.have.been.called; - done.should.not.have.been.called; - server.requests = []; + return rs.query({}, { include: include }).then(success, fail).then(function () { + fail.should.have.been.called; + success.should.not.have.been.called; + }); }); it('should aggregate the response from the multiple GETs for a single run', function () { - server.requests = []; - var done = sinon.spy(); + var success = sinon.spy(); var fail = sinon.spy(); var rs = new RunService({ account: account, project: project }); var include = createLargeInclude(); include.push('single_variables_c_d'); include = ['single_variables_a_b'].concat(include); - rs.query({}, { include: include }).done(done).fail(fail); - server.respond(); - done.should.have.been.calledWith({ - 'id': '065dfe50-d29d-4b55-a0fd-30868d7dd26c', - 'model': 'model.vmf', - 'account': account, - 'project': 'js-libs', - 'saved': false, - 'lastModified': '2014-06-20T04:09:45.738Z', - 'created': '2014-06-20T04:09:45.738Z', - 'variables': { - 'varA': 9999.99, - 'varB': 'A string', - 'varC': 'Another string', - 'varD': 10.22, - } + return rs.query({}, { include: include }).then(success, fail).then(function () { + success.should.have.been.calledWith({ + 'id': '065dfe50-d29d-4b55-a0fd-30868d7dd26c', + 'model': 'model.vmf', + 'account': account, + 'project': 'js-libs', + 'saved': false, + 'lastModified': '2014-06-20T04:09:45.738Z', + 'created': '2014-06-20T04:09:45.738Z', + 'variables': { + 'varA': 9999.99, + 'varB': 'A string', + 'varC': 'Another string', + 'varD': 10.22, + } + }); + fail.should.not.have.been.called; }); - fail.should.not.have.been.called; - server.requests = []; }); it('should aggregate the reponse from the multiple GETs for a multiple runs', function () { - server.requests = []; - var done = sinon.spy(); + var success = sinon.spy(); var fail = sinon.spy(); var rs = new RunService({ account: account, project: project }); var include = createLargeInclude(); include.push('multiple_variables_c_d'); include = ['multiple_variables_a_b'].concat(include); - rs.query({}, { include: include }).done(done).fail(fail); - server.respond(); - done.should.have.been.calledWith([ - { - 'id': 'run1', - 'variables': { - 'varA': 1111.11, - 'varB': 'A string for run1', - 'varC': 'Another string for run1', - 'varD': '2015-11-16 10:10:10' - } - }, - { - 'id': 'run2', - 'variables': { - 'varA': 2222.22, - 'varB': 'A string for run2', - 'varC': 'Another string for run2', - 'varD': '2015-11-16 20:20:20' - } - }, - ]); - fail.should.not.have.been.called; - server.requests = []; + return rs.query({}, { include: include }).then(success, fail).then(function () { + success.should.have.been.calledWith([ + { + 'id': 'run1', + 'variables': { + 'varA': 1111.11, + 'varB': 'A string for run1', + 'varC': 'Another string for run1', + 'varD': '2015-11-16 10:10:10' + } + }, + { + 'id': 'run2', + 'variables': { + 'varA': 2222.22, + 'varB': 'A string for run2', + 'varC': 'Another string for run2', + 'varD': '2015-11-16 20:20:20' + } + }, + ]); + fail.should.not.have.been.called; + }); }); }); @@ -437,20 +419,16 @@ req.url.should.equal(baseURL + ';/?page=1&limit=2'); }); it('should pass through options across multiple queries', function () { - server.requests = []; var rs = new RunService({ account: account, project: project }); rs.filter({ saved: true, '.price': '>1' }); - server.respond(); server.requests[0].url.should.equal(baseURL + ';saved=true;.price>1/'); rs.filter({ saved: false, '.sales': '<4' }); - server.respond(); server.requests[1].url.should.equal(baseURL + ';saved=false;.price>1;.sales<4/'); - server.requests = []; }); it('should not include the AutoRestore header', function () { @@ -597,7 +575,6 @@ var rs = new RunService({ account: account, project: 'js-libs', filter: { saved: true } }); rs.do('init'); - server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';saved=true/operations/init/'); req.requestBody.should.equal(JSON.stringify({ arguments: [] })); @@ -611,25 +588,21 @@ ret.should.throw(Error); }); - it('should send multiple operations calls once by one', function () { - server.requests = []; - + it.only('should send multiple operations calls one by one', function () { var rs = new RunService({ account: account, project: 'js-libs', filter: { saved: true } }); - rs.serial([{ first: [1,2] }, { second: [2,3] }]); - server.respond(); - - server.requests.length.should.equal(2); - server.requests[0].url.should.equal(baseURL + ';saved=true/operations/first/'); - server.requests[0].requestBody.should.equal(JSON.stringify({ arguments: [1,2] })); - - server.requests[1].url.should.equal(baseURL + ';saved=true/operations/second/'); - server.requests[1].requestBody.should.equal(JSON.stringify({ arguments: [2,3] })); + return rs.serial([{ first: [1, 2] }, { second: [2, 3] }]).then(function () { + server.requests.length.should.equal(2); + server.requests[0].url.should.equal(baseURL + ';saved=true/operations/first/'); + server.requests[0].requestBody.should.equal(JSON.stringify({ arguments: [1,2] })); + + server.requests[1].url.should.equal(baseURL + ';saved=true/operations/second/'); + server.requests[1].requestBody.should.equal(JSON.stringify({ arguments: [2,3] })); + }); }); it('should send operations without any parameters', function () { var rs = new RunService({ account: account, project: 'js-libs', filter: { saved: true } }); rs.serial(['init']); - server.respond(); var req = server.requests.pop(); req.url.should.equal(baseURL + ';saved=true/operations/init/'); @@ -645,11 +618,9 @@ }); it('should send multiple operations calls once by one', function () { - server.requests = []; var rs = new RunService({ account: account, project: 'js-libs', filter: { saved: true } }); rs.parallel([{ first: [1,2] }, { second: [2,3] }]); - server.respond(); server.requests.length.should.equal(2); }); From 175d5146ce1295b01bc7a19310479cb2b5b22ce7 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 14:17:09 -0700 Subject: [PATCH 08/16] then => done --- tests/spec/test-auth-manager.js | 26 +++++++++++++------------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/tests/spec/test-auth-manager.js b/tests/spec/test-auth-manager.js index 9fbf5684..f34d0122 100644 --- a/tests/spec/test-auth-manager.js +++ b/tests/spec/test-auth-manager.js @@ -110,7 +110,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { //jshint camelcase: false //jscs:disable response.auth.access_token.should.equal(token); @@ -126,7 +126,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { var session = am.getCurrentUserSessionInfo(); session.groupName.should.equal('rv-test'); session.groupId.should.equal('111efcc9-726c-47b8-ba94-2895f110bd39'); @@ -147,7 +147,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { multipleGroupsResponse = false; done(new Error('Login should not work')); }).fail(function (data) { @@ -163,7 +163,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test', groupId: '111efcc9-726c-47b8-ba94-2895f110bd32' }).done(function (response) { + am.login({ userName: 'test', password: 'test', groupId: '111efcc9-726c-47b8-ba94-2895f110bd32' }).then(function (response) { var session = am.getCurrentUserSessionInfo(); session.groupName.should.equal('rv-test2'); multipleGroupsResponse = false; @@ -181,7 +181,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test', groupId: 'wrong-id' }).done(function (response) { + am.login({ userName: 'test', password: 'test', groupId: 'wrong-id' }).then(function (response) { multipleGroupsResponse = false; done(new Error('Login should not work')); }).fail(function (data) { @@ -197,7 +197,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { var req = server.requests.pop(); req.method.toUpperCase().should.equal('GET'); req.url.should.match(/https:\/\/api\.forio\.com\/group\/local\/?/); @@ -219,7 +219,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { multipleGroupsResponse = false; teamMemberResponse = false; done(new Error('Login should not work')); @@ -248,7 +248,7 @@ domain: '.forio.com' } }); - am.logout().done(function (response) { + am.logout().then(function (response) { var spyCall = cookie.set.getCall(0); spyCall.args[0].should.match(/epicenterjs\.session=; expires=Thu, 01 Jan 1970 00:00:00 GMT; domain=\.forio\.com; path=\/app\/accountName\/projectName/); spyCall = cookie.set.getCall(1); @@ -292,7 +292,7 @@ project: 'projectName', userName: 'test', password: 'test', - }).done(function (response) { + }).then(function (response) { var pathIdx = cookie.get().indexOf('path=/app/accountName/projectName'); pathIdx.should.not.equal(-1); done(); @@ -310,7 +310,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { var session = am.getCurrentUserSessionInfo(); Object.keys(session.groups).should.have.lengthOf(1); am.sessionManager.removeSession(); @@ -325,7 +325,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { am.addGroups({ project: 'test-project', groupName: 'rv-test2', @@ -348,7 +348,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { am.addGroups([{ project: 'test-project', groupName: 'rv-test2', @@ -376,7 +376,7 @@ account: 'accountName', project: 'projectName', }); - am.login({ userName: 'test', password: 'test' }).done(function (response) { + am.login({ userName: 'test', password: 'test' }).then(function (response) { am.addGroups([{ project: 'projectName', groupName: 'rv-test2', From 69fa7f8986c4eba44c073c048716ef5bc31a8b29 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 14:18:51 -0700 Subject: [PATCH 09/16] done => then --- dist/components/assignment/assignment.js | 4 +- .../epicenter-multiplayer-dependencies.min.js | 2 +- ...center-multiplayer-dependencies.min.js.map | 2 +- src/components/assignment/js/assignment.js | 2 +- src/components/login/login.js | 2 - src/env-load.js | 4 +- src/managers/auth-manager.js | 2 +- src/service/run-api-service.js | 2 +- tests/integration/assets/assets-test.js | 8 +-- tests/integration/multiplayer/test-script.js | 2 +- tests/spec/test-multiplayer-strategy.js | 58 +++++++------------ tests/spec/test-run-api-service.js | 4 +- 12 files changed, 38 insertions(+), 54 deletions(-) diff --git a/dist/components/assignment/assignment.js b/dist/components/assignment/assignment.js index ddf61fed..369e2736 100644 --- a/dist/components/assignment/assignment.js +++ b/dist/components/assignment/assignment.js @@ -176,7 +176,7 @@ Assignment.prototype = { this._showUpdating(); var maxUsers = +this.$('#max-users').val(); return this.worlds.autoAssignAll({ maxUsers: maxUsers }) - .done(this._hideUpdating) + .then(this._hideUpdating) .fail(this._hideUpdating) .then(function () { this.worlds.joinUsers(); @@ -1231,4 +1231,4 @@ module.exports = function (base, props, staticProps) { }; },{}]},{},[6]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/components/assignment/js/assignment-row.js","src/components/assignment/js/assignment.js","src/components/assignment/js/base-collection.js","src/components/assignment/js/base-model.js","src/components/assignment/js/defaults.js","src/components/assignment/js/index.js","src/components/assignment/js/project-model.js","src/components/assignment/js/service-locator.js","src/components/assignment/js/templates.js","src/components/assignment/js/user-model.js","src/components/assignment/js/users-collection.js","src/components/assignment/js/world-model.js","src/components/assignment/js/worlds-collection.js","src/util/ajax-queue.js","src/util/inherit.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nvar templates = require('./templates');\n\nvar AssignmentRow = function (options) {\n    this.$el = $('<tr>');\n    this.el = this.$el[0];\n    this.$ = _.partialRight($, this.$el);\n\n    this.model = options.model;\n    this.options = options;\n    this.worlds = options.worlds;\n    this.project = options.project;\n\n    _.bindAll(this, ['setEditMode', 'removeEditMode', 'saveEdit', 'cancelEdit', 'updateData']);\n\n    this.bindEvents();\n\n};\n\n_.extend(AssignmentRow.prototype, {\n\n    template: templates['user-row'],\n\n    editTemplate: templates['edit-user-row'],\n\n    bindEvents: function () {\n        this.$el.on('click', 'button.edit', this.setEditMode);\n        this.$el.on('click', 'button.save', this.saveEdit);\n        this.$el.on('click', 'button.cancel', this.cancelEdit);\n    },\n\n    remove: function () {\n        this.$el.off('click', null, null);\n        // this only gives a delay to remove the tr\n        // animation of height of the tr does not work\n        this.$(':checkbox').attr('checked', false);\n        this.$el\n            .css({ opacity: 0.3 })\n            .animate({ height: 0 }, {\n                duration: 300,\n                complete: function () {\n                    this.remove();\n                }\n            });\n    },\n\n    makeInactive: function () {\n        return this.model.makeInactive();\n    },\n\n    setEditMode: function () {\n        this.model.set('edit-mode', true);\n        this.render();\n    },\n\n    removeEditMode: function () {\n        this.model.set('edit-mode', false);\n        this.render();\n    },\n\n    saveEdit: function () {\n        var _this = this;\n        this.updateData();\n        this.worlds\n            .updateUser(this.model)\n            .then(function () {\n                _this.removeEditMode();\n                _this.$el.trigger('update', _this);\n            });\n    },\n\n    cancelEdit: function () {\n        this.removeEditMode();\n    },\n\n    render: function () {\n        var templ = this.model.get('edit-mode') ? this.editTemplate : this.template;\n        var vm = _.extend({\n            roles: this.project.get('roles'),\n            optionalRoles: this.project.get('optionalRoles'),\n            worlds: this.worlds.getWorldNames(),\n            newWorld: this.worlds.getNextWorldName()\n        }, this.model.toJSON());\n\n        this.$el.html(templ(vm));\n\n        return this;\n    },\n\n    updateData: function () {\n        var _this = this;\n        this.$('[data-field]').each(function () {\n            var el = $(this);\n            var field = el.data('field');\n            var val = el.val();\n\n            _this.model.set(field, val);\n        });\n    }\n});\n\n\nmodule.exports = AssignmentRow;","'use strict';\n\nvar UsersCollection = require('./users-collection');\nvar WorldsCollection = require('./worlds-collection');\nvar ProjectModel = require('./project-model');\nvar AssignemntRow = require('./assignment-row');\nvar env = require('./defaults');\nvar AjaxQueue = require('../../../util/ajax-queue');\n\nfunction setEnvironment(options) {\n    env.set(_.omit(options, 'el'));\n}\n\nvar Assignment = function (options) {\n    setEnvironment(options);\n    this.initialize(options);\n};\n\nAssignment.prototype = {\n\n    initialize: function (options) {\n        this.el = typeof options.el === 'string' ? $(options.el)[0] : options.el;\n        this.$el = $(this.el);\n        this.$ = _.partialRight($, this.el);\n\n        this.users = new UsersCollection();\n        this.worlds = new WorldsCollection();\n        this.project = new ProjectModel();\n\n        _.bindAll(this, ['render', 'renderTable', 'toggleControlls', 'saveEdit', 'selectAll', 'usassignSelected', '_showUpdating', '_hideUpdating', 'autoAssignAll', 'makeUserInactive']);\n\n        this.bindEvents();\n    },\n\n    bindEvents: function () {\n        this.$el.on('update', 'tr', this.saveEdit);\n        this.$el.on('click', 'input:checkbox:not(#select-all)', this.toggleControlls);\n        this.$el.on('click', '#select-all', this.selectAll);\n        this.$el.on('click', '.unassign-user', this.usassignSelected);\n        this.$el.on('click', '.auto-assign-all', this.autoAssignAll);\n        this.$el.on('click', '.make-user-inactive', this.makeUserInactive);\n    },\n\n    load: function () {\n\n        var join = function () {\n            this.worlds.setUsersCollection(this.users);\n            this.worlds.joinUsers();\n            this.render();\n        }.bind(this);\n\n        return $.when(\n            this.worlds.fetch(),\n            this.users.fetch(),\n            this.project.fetch()\n        ).then(join);\n\n    },\n\n    saveEdit: function () {\n        this.worlds.fetch()\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n                this.updateControls();\n            }.bind(this));\n    },\n\n    autoAssignAll: function () {\n        this._showUpdating();\n        var maxUsers = +this.$('#max-users').val();\n        return this.worlds.autoAssignAll({ maxUsers: maxUsers })\n            .done(this._hideUpdating)\n            .fail(this._hideUpdating)\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n            }.bind(this));\n    },\n\n    getSelectedIds: function () {\n        return this.$('tbody :checkbox:checked').map(function () {\n            return $(this).data('id');\n        });\n    },\n\n    findRowViews: function (ids) {\n        return _.map(ids, function (id) {\n            return this.rowViews[id];\n        }, this);\n    },\n\n    unassignUsers: function (ids) {\n        var dtd = $.Deferred();\n        var done = function () {\n            dtd.resolve();\n        };\n\n        // for now we need to sequence the calls to unassign users from worlds\n        var queue = new AjaxQueue();\n\n        _.each(ids, function (userId) {\n            var user = this.users.getById(userId);\n            user.set('world', '');\n            user.set('role', '');\n            queue.add(_.partial(_.bind(this.worlds.updateUser, this.worlds), user));\n        }, this);\n\n        queue.execute(this).then(done);\n\n        return dtd.promise();\n    },\n\n    usassignSelected: function (e) {\n        e.preventDefault();\n\n        var ids = this.getSelectedIds();\n\n        var done = function () {\n            this.worlds.fetch().then(function () {\n                this.worlds.joinUsers();\n                this._hideUpdating();\n                this.render();\n\n            }.bind(this));\n        }.bind(this);\n\n        this._showUpdating();\n\n        return this.unassignUsers(ids).then(done);\n    },\n\n    makeUserInactive: function (e) {\n        e.preventDefault();\n        var ids = this.getSelectedIds();\n        var done = function () {\n            this.toggleControlls();\n        }.bind(this);\n\n        var makeUsersInactive = function () {\n            var rows = this.findRowViews(ids);\n            // for now we need to sequence the calls to patch the users\n            // since the API can only operate on one call per group at a time\n            var queue = new AjaxQueue();\n            _.each(rows, function (view) {\n                var user = view.model;\n                queue.add(function () {\n                    return view.makeInactive()\n                        .then(function () {\n                            user.remove();\n                            view.remove();\n                        });\n                    });\n\n            }, this);\n\n            queue.execute(this).then(done);\n        }.bind(this);\n\n        return this.unassignUsers(ids)\n            .then(makeUsersInactive);\n\n\n    },\n\n    render: function () {\n        this.$('table tbody').empty();\n        this.renderTable();\n        this.toggleControlls();\n    },\n\n    renderTable: function () {\n        this.rowViews = {};\n        var rows = [];\n        this.users.each(function (u) {\n            var view = new AssignemntRow({ model: u, worlds: this.worlds, project: this.project });\n            this.rowViews[u.get('id')] = view;\n            rows.push(view.render().el);\n        }, this);\n\n        this.$('table tbody').append(rows);\n    },\n\n\n    updateControls: function () {\n        this.updateControlsForSelection();\n        this.updateAutoAssignButton();\n        this.updateStatus();\n    },\n\n    updateStatus: function () {\n        var incolpleteWorlds = this.worlds.getIncompleteWorldsCount();\n        var unassignedUsers = this.users.getUnassignedUsersCount();\n        var totalWorlds = this.worlds.size();\n\n        var usersText = unassignedUsers ? unassignedUsers === 1 ? '1 user needs assignment.' : unassignedUsers + ' users need assignment.' : 'All users have been assigned.';\n        var worldsText = !totalWorlds ? 'No worlds have been created.' : !incolpleteWorlds ? 'All worlds are complete.' : incolpleteWorlds === 1 ? '1 incomplete world needs attention.' : incolpleteWorlds + ' incomplete worlds need attention.';\n\n        this.$('#users-status .text').text(usersText);\n        this.$('#worlds-status .text').text(worldsText);\n\n        if (unassignedUsers) {\n            this.$('#users-status').addClass('incomplete');\n        } else {\n            this.$('#users-status').removeClass('incomplete');\n        }\n\n        if (incolpleteWorlds || !totalWorlds) {\n            this.$('#worlds-status').addClass('incomplete');\n        } else {\n            this.$('#worlds-status').removeClass('incomplete');\n        }\n\n        this.$('.status-widget').addClass('visible');\n    },\n\n    updateControlsForSelection: function () {\n        var numSelected = this.$('tbody :checkbox:checked').length;\n        this.$('.component.controls')[numSelected ? 'addClass' : 'removeClass']('visible');\n    },\n\n    updateAutoAssignButton: function () {\n\n        if (this.project.isDynamicAssignment()) {\n            var hasRoles = this.project.hasRoles();\n            this.$('.table-controls .single').hide();\n            this.$('.table-controls .dynamic').show();\n            this.$('.table-controls .dynamic-no-roles-text')[hasRoles ? 'hide' : 'show']();\n            this.$('.table-controls .no-roles')[hasRoles ? 'hide' : 'show']();\n        } else {\n            this.$('.table-controls .dynamic').hide();\n            this.$('.table-controls .dynamic-no-roles-text').hide();\n            this.$('.table-controls .single').show();\n            this.$('.table-controls .no-roles').show();\n\n        }\n\n        if (this.users.allUsersAssigned()) {\n            this.$('.table-controls').removeClass('visible');\n        } else {\n            this.$('.table-controls').addClass('visible');\n        }\n    },\n\n    selectAll: function (e) {\n        this.$('tbody :checkbox').prop('checked', e.target.checked);\n        this.updateControls();\n    },\n\n    toggleControlls: function (e) {\n        var total = this.$('tbody :checkbox');\n        var checked = this.$('tbody :checkbox:checked');\n\n        if (total.length === checked.length) {\n            this.$('#select-all').attr('checked', 'checked');\n        } else {\n            this.$('#select-all').removeAttr('checked');\n        }\n\n        this.updateControls();\n    },\n\n    _showUpdating: function () {\n        this.$el.css({ opacity: 0.4 });\n    },\n\n    _hideUpdating: function () {\n        this.$el.css({ opacity: 1 });\n    }\n\n};\n\nmodule.exports = Assignment;","'use strict';\n\nvar BaseCollection = function (models, options) {\n    this._models = [];\n    this.options = options;\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseCollection.prototype, {\n    idAttribute: 'id',\n\n    initialize: function (models, options) {\n    },\n\n    create: function (attr, options) {\n        var m = new this.model(attr, options);\n        this.set(m);\n        return m;\n    },\n\n    reset: function (models, options) {\n        this._models.length = 0;\n        this.set(models);\n    },\n\n    remove: function (model) {\n        _.remove(this._models, function (m) {\n            return m === model;\n        });\n\n        delete model.collection;\n\n        return model;\n    },\n\n    set: function (models) {\n        if (!models) {\n            return;\n        }\n\n        models = [].concat(models);\n\n        if (!models.length) {\n            return;\n        }\n\n        _.each(models, function (m) {\n            if (!(m instanceof this.model)) {\n                m = new this.model(m);\n            }\n\n            m.collection = this;\n\n            this._models.push(m);\n        }, this);\n\n        this.sort.call(this);\n\n        return models;\n    },\n\n    sortFn: function (a, b) {\n        return b._data[this.idAttribute] - a._data[this.idAttribute];\n    },\n\n    sort: function () {\n        this._models = this._models.sort(this.sortFn.bind(this));\n\n        return this._models;\n    },\n\n    getById: function (id) {\n        return _.find(this._models, function (m) {\n            return m.get(this.idAttribute) === id;\n        }, this);\n    },\n\n    each: function (cb, ctx) {\n        return _.each(this._models, cb, ctx || this);\n    },\n\n    all: function (cb, ctx) {\n        return _.all(this._models, cb, ctx || this);\n    },\n\n    toJSON: function () {\n        return _.invoke(this._models, 'toJSON');\n    },\n\n    find: function (fn) {\n        return _.find(this._models, fn);\n    },\n\n    filter: function (fn) {\n        return _.filter(this._models, fn);\n    },\n\n    size: function () {\n        return this._models.length;\n    },\n\n    map: function (fn, ctx) {\n        return _.map(this._models, function (model) {\n            return fn.call(ctx, model.toJSON());\n        });\n    },\n\n    pluck: function (field) {\n        return this.map(function (m) {\n            return m[field];\n        });\n    }\n\n});\n\nmodule.exports = BaseCollection;","'use strict';\n\n\nvar BaseModel = function (attr, options) {\n    attr = _.defaults({}, attr, _.result(this, 'defaults'));\n    this._data = {};\n    this.set(attr, options);\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseModel.prototype, {\n    initialize: function (attr, options) {\n\n    },\n\n    set: function (key, val, options) {\n\n        if (key == null) {\n            return this;\n        }\n\n        var attrs;\n        if (typeof key === 'object') {\n            attrs = key;\n            options = val;\n        } else {\n            (attrs = {})[key] = val;\n        }\n\n        options = options || {};\n\n        _.extend(this._data, attrs);\n\n        return this;\n    },\n\n    get: function (key, options) {\n        return this._data[key];\n    },\n\n    remove: function () {\n        if (this.collection) {\n            this.collection.remove(this);\n        }\n\n        return this;\n    },\n\n    toJSON: function () {\n        return this._data;\n    },\n\n    pick: function (keys) {\n        return _.pick(this._data, keys);\n    }\n\n});\n\nmodule.exports = BaseModel;","'use strict';\n\nvar env = {\n    account: '',\n    project: '',\n    group: '',\n    groupId: '',\n    token: '',\n    server: {\n        host: 'api.forio.com',\n        protocol: 'https'\n    }\n};\n\nmodule.exports = {\n    set: function (options) {\n        env = _.merge(env, options);\n    },\n\n    get: function () {\n        return env;\n    }\n};","(function () {\n    'use strict';\n    var App = require('./assignment.js');\n\n    window.forio = window.forio || {};\n    window.forio.MultiplayerAssignmentComponent = App;\n})();\n","'use strict';\n\nvar serviceLocator = require('./service-locator');\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\n// var __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    isDynamicAssignment: function () {\n        return this.get('worlds') === 'dynamic';\n    },\n\n    hasRoles: function () {\n        var roles = this.get('roles');\n        return roles && !!roles.length;\n    },\n\n    fetch: function () {\n        var api = serviceLocator.worldApi();\n\n        return api.getProjectSettings().then(function (settings) {\n            this.set(settings);\n        }.bind(this));\n    }\n});","'use strict';\n\nvar env = require('./defaults.js');\n\nvar cache = {};\n\nmodule.exports = {\n    worldApi: function () {\n        if (!cache.worldApi) {\n            cache.worldApi = new F.service.World(env.get());\n        }\n\n        return cache.worldApi;\n    },\n\n    memberApi: function () {\n        if (!cache.memberApi) {\n            cache.memberApi = new F.service.Member(_.pick(env.get(), ['groupId', 'server']));\n        }\n\n        return cache.memberApi;\n    },\n\n    userApi: function () {\n        if (!cache.userApi) {\n            cache.userApi = new F.service.User(_.pick(env.get(), ['account', 'server']));\n        }\n\n        return cache.userApi;\n    }\n};","exports[\"edit-user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape, __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id )) == null ? '' : __t) +\n'\"</td>\\n<td></td>\\n<td>\\n    <select name=\"worlds\" class=\"form-control\" data-field=\"world\">\\n\\n    ';\n _.each(worlds, function (w) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( w )) == null ? '' : __t) +\n'\" ' +\n((__t = ( w === world ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( w )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n        <option value=\"' +\n((__t = ( newWorld )) == null ? '' : __t) +\n'\" class=\"new-world-text\"><i>' +\n((__t = ( newWorld )) == null ? '' : __t) +\n' - New -</i></option>\\n    </select>\\n</td>\\n<td>\\n    <select name=\"roles\" class=\"form-control\" data-field=\"role\">\\n    ';\n _.each(roles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n\\n    ';\n _.each(optionalRoles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n' <i>(Optional)</i></option>\\n    ';\n }); ;\n__p += '\\n    </select>\\n</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\">\\n    <button class=\"btn btn-primary btn-tools btn-save save\">Save</button>\\n    <button class=\"btn btn-tools btn-cancel cancel\">Cancel</button>\\n</td>';\n\n}\nreturn __p\n};\nexports[\"user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape;\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id)) == null ? '' : __t) +\n'\"</td>\\n<td>' +\n((__t = ( !isWorldComplete ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( world )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( role )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\"><button class=\"btn edit btn-edit btn-tools auto-hide\">Edit</button></td>';\n\n}\nreturn __p\n};","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar serviceLocator = require('./service-locator');\n\n\nmodule.exports = classFrom(Base, {\n    defaults: {\n        world: '',\n        role: '',\n        active: true,\n        isWorldComplete: true,\n        firstName: '',\n        lastName: ''\n    },\n\n    makeActive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', true);\n\n        return memberApi.makeUserActive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    },\n\n    makeInactive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', false);\n\n        return memberApi.makeUserInactive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    }\n\n});\n","'use strict';\n\nvar classFrom = require('../../../util/inherit');\n\nvar Model = require('./user-model');\nvar Base = require('./base-collection');\nvar env = require('./defaults');\nvar serviceLocator = require('./service-locator');\n\n\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    sortFn: function (a, b) {\n        var aw = a.get('world').toLowerCase();\n        var bw = b.get('world').toLowerCase();\n        if (aw !== bw) {\n            return aw < bw ? -1 : 1;\n        }\n\n        return b.get('userName') > a.get('userName') ? -1 : 1;\n    },\n\n    initialize: function () {\n        $.ajaxSetup({\n            headers: {\n                Authorization: 'Bearer ' + env.get().token\n            }\n        });\n    },\n\n    allUsersAssigned: function () {\n        return this.all(function (u) {\n            return !!u.get('world');\n        });\n    },\n\n    getUnassignedUsersCount: function () {\n        return this.filter(function (u) {\n            return !u.get('world');\n        }).length;\n    },\n\n    fetch: function () {\n        var dtd = $.Deferred();\n        var _this = this;\n        var groupId = env.get().groupId;\n\n        var getGroupUsers = function () {\n            var memberApi = serviceLocator.memberApi();\n            var userApi = serviceLocator.userApi();\n\n            var loadGroupMembers = function () {\n                return memberApi.getGroupDetails();\n            };\n\n            var loadUsersInfo = function (group) {\n                var nonFacAndActive = function (u) { return u.active && u.role !== 'facilitator'; };\n                var users = _.pluck(_.filter(group.members, nonFacAndActive), 'userId');\n                return userApi.get({ id: users });\n            };\n\n            return loadGroupMembers()\n                .then(loadUsersInfo)\n                .fail(dtd.reject);\n        };\n\n        getGroupUsers()\n            .then(function (users) {\n                users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); });\n                _this.set(users);\n                dtd.resolve(users, _this);\n            });\n\n        return dtd.promise();\n    }\n\n});\n","'use strict';\nvar serviceLocator = require('./service-locator');\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    defaults: {\n        users: null,\n        model: 'model.eqn'\n    },\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n\n        this._data.users = this._data.users || [];\n\n        this._worldApi = serviceLocator.worldApi();\n\n        var id = this.get('id');\n        if (id) {\n            this._worldApi.updateConfig({ filter: id });\n        }\n    },\n\n    addUser: function (user) {\n        var users = this.get('users');\n        users.push(user);\n\n        return this.save();\n    },\n\n    removeUser: function (user) {\n        var id = this.get('id');\n        var checkWorld = function () {\n            if (!this.get('users').length) {\n                this.remove();\n                return this._worldApi.updateConfig({ filter: id }).delete();\n            }\n        }.bind(this);\n\n        _.remove(this.get('users'), function (u) {\n            return u.get('id') === user.get('id');\n        });\n\n        return this._worldApi\n            .updateConfig({ filter: id })\n            .removeUser({ userId: user.get('id') })\n            .then(checkWorld);\n    },\n\n    save: function () {\n        var _this = this;\n        var mapUsers = function () {\n            return _.map(this.get('users'), function (u) {\n                var res = { userId: u.get('id') };\n                var role = u.get('role');\n\n                if (role) {\n                    res.role = role;\n                }\n\n                return res;\n            });\n        }.bind(this);\n\n        var createWorld = _.partial(this._worldApi.create, this.pick(['model', 'name', 'minUsers']));\n        var addUsers = _.partial(_this._worldApi.addUsers, mapUsers(), { filter: _this.get('id') });\n        var savedUsers = this.get('users');\n        if (this.isNew()) {\n            // we need to create the world in the API and then add the users\n            return createWorld()\n                .then(function (world) {\n                    _this.set(world);\n                    _this._worldApi.updateConfig({ filter: world.id });\n                })\n                .then(addUsers)\n                .then(function (users) {\n                    // since we re-set the world, re-set the users\n                    _this.set('users', savedUsers);\n                });\n        } else {\n            // the world is already created just add the users\n            return addUsers();\n        }\n    },\n\n    isNew: function () {\n        return !this.get('lastModified');\n    }\n\n});","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Model = require('./world-model');\nvar UserModel = require('./user-model');\nvar serviceLocator = require('./service-locator');\n\nvar Base = require('./base-collection');\nvar __super = Base.prototype;\n\nvar doneFn = function (dtd, after) {\n    return _.after(after, dtd.resolve);\n};\n\nvar worldApi;\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n        worldApi = serviceLocator.worldApi();\n    },\n\n    autoAssignAll: function (options) {\n        return worldApi.autoAssign(options)\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    getIncompleteWorldsCount: function () {\n        return this.filter(function (w) {\n            return !w.get('complete');\n        }).length;\n    },\n\n    updateUser: function (user) {\n        var worldName = user.get('world');\n        var dtd = $.Deferred();\n        var prevWorld = this.getWorldByUser(user);\n        var curWorld = this.getOrCreateWorld(worldName);\n        var done = doneFn(dtd, 1);\n\n        // check if there's anything to do\n        if (!prevWorld && !curWorld) {\n            return dtd.resolve().promise();\n        }\n\n        if (prevWorld) {\n            prevWorld.removeUser(user)\n                .then(function () {\n                    if (curWorld) {\n                        return curWorld.addUser(user);\n                    }\n                })\n                .then(done);\n        } else if (curWorld) {\n            curWorld.addUser(user)\n                .then(done);\n        }\n\n        return dtd.promise();\n    },\n\n    getOrCreateWorld: function (worldName) {\n        if (!worldName) {\n            return;\n        }\n\n        var world = this.getWordByName(worldName);\n\n        if (!world) {\n            world = this.create({ name: worldName });\n        }\n\n        return world;\n    },\n\n    getWordByName: function (worldName) {\n        return this.find(function (world) {\n            return world.get('name') === worldName;\n        });\n    },\n\n    getWorldByUser: function (user) {\n        if (!user.get) {\n            throw new Error('getWorldByUser expectes a model (' + user + ')');\n        }\n\n        var id = user.get('id');\n        return this.getWorldByUserId(id);\n    },\n\n    getWorldByUserId: function (userId) {\n        return this.find(function (world) {\n            return _.find(world.get('users'), function (u) {\n                return u.get('id') === userId;\n            });\n        });\n    },\n\n    getWorldNames: function () {\n        return this.pluck('name');\n    },\n\n    getNextWorldName: function () {\n        var pad = function (num, places) {\n            var zeros = '000000000000000000';\n            var digits = num.toString().length;\n            var needed = places - digits;\n            return zeros.substr(0, needed) + num;\n        };\n\n        var worlds = this.getWorldNames();\n\n        if (!worlds.length) {\n            return 'World001';\n        }\n\n        var properNames = _.filter(worlds, function (w) { return /World\\d\\d\\d/.test(w); }).sort();\n        var lastWorld = properNames[properNames.length - 1];\n        var numWorld = +lastWorld.match(/World(\\d\\d\\d)/)[1];\n        return 'World' + pad(numWorld + 1, 3);\n    },\n\n    setUsersCollection: function (usersCollection) {\n        this.usersCollection = usersCollection;\n    },\n\n    joinUsers: function () {\n        var usersHash = {};\n        var usersCollection = this.usersCollection;\n        usersCollection.each(function (u) {\n            u.set({ isWorldComplete: true });\n            return (usersHash[u.get('id')] = u);\n        });\n\n        this.each(function (w, i) {\n            var name = w.get('name');\n            var isComplete = w.get('complete');\n            w.set({ index: i, name: name || (i + 1) + '' });\n            _.each(w.get('users'), function (u) {\n                if (usersHash[u.get('userId')]) {\n                    usersHash[u.get('userId')].set({ world: name, role: u.get('role'), isWorldComplete: isComplete });\n                }\n            });\n        }, this);\n\n        usersCollection.sort();\n    },\n\n    fetch: function () {\n        return worldApi.list()\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    parse: function (worlds) {\n        if (worlds.length) {\n            worlds = _.map(worlds, function (w) {\n                var users = _.map(w.users, function (u) {\n                    // in the world api users Ids comes as userId\n                    // make sure we add it as id so we can use the\n                    // same code to access models that come from the\n                    // member/local api as with the world api\n                    u.id = u.userId;\n                    return new UserModel(u);\n                });\n\n                w.users = users;\n\n                return w;\n            });\n        }\n\n        return worlds;\n    }\n});\n","'use strict';\n\n/**\n* Utility class to make ajax calls sequencial\n*/\nfunction AjaxQueue () {\n    this.queue = [];\n}\n\n$.extend(AjaxQueue.prototype, {\n    add: function (fn) {\n        return this.queue.push(fn);\n    },\n\n    execute: function (context) {\n        var dtd = $.Deferred();\n        var _this = this;\n        context = context || this;\n\n        function next() {\n            if (_this.queue.length) {\n                var fn = _this.queue.shift();\n\n                fn.call(context)\n                    .then(next)\n                    .fail(dtd.reject);\n            } else {\n                dtd.resolve();\n            }\n        }\n\n        next();\n\n        return dtd.promise();\n    }\n});\n\n\nmodule.exports = AjaxQueue;","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j<obj.length; j++) {\n        if (!(current = obj[j])) {\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        /*jshint -W089 */\n        for (var key in current) {\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n"]} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/components/assignment/js/assignment-row.js","src/components/assignment/js/assignment.js","src/components/assignment/js/base-collection.js","src/components/assignment/js/base-model.js","src/components/assignment/js/defaults.js","src/components/assignment/js/index.js","src/components/assignment/js/project-model.js","src/components/assignment/js/service-locator.js","src/components/assignment/js/templates.js","src/components/assignment/js/user-model.js","src/components/assignment/js/users-collection.js","src/components/assignment/js/world-model.js","src/components/assignment/js/worlds-collection.js","src/util/ajax-queue.js","src/util/inherit.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nvar templates = require('./templates');\n\nvar AssignmentRow = function (options) {\n    this.$el = $('<tr>');\n    this.el = this.$el[0];\n    this.$ = _.partialRight($, this.$el);\n\n    this.model = options.model;\n    this.options = options;\n    this.worlds = options.worlds;\n    this.project = options.project;\n\n    _.bindAll(this, ['setEditMode', 'removeEditMode', 'saveEdit', 'cancelEdit', 'updateData']);\n\n    this.bindEvents();\n\n};\n\n_.extend(AssignmentRow.prototype, {\n\n    template: templates['user-row'],\n\n    editTemplate: templates['edit-user-row'],\n\n    bindEvents: function () {\n        this.$el.on('click', 'button.edit', this.setEditMode);\n        this.$el.on('click', 'button.save', this.saveEdit);\n        this.$el.on('click', 'button.cancel', this.cancelEdit);\n    },\n\n    remove: function () {\n        this.$el.off('click', null, null);\n        // this only gives a delay to remove the tr\n        // animation of height of the tr does not work\n        this.$(':checkbox').attr('checked', false);\n        this.$el\n            .css({ opacity: 0.3 })\n            .animate({ height: 0 }, {\n                duration: 300,\n                complete: function () {\n                    this.remove();\n                }\n            });\n    },\n\n    makeInactive: function () {\n        return this.model.makeInactive();\n    },\n\n    setEditMode: function () {\n        this.model.set('edit-mode', true);\n        this.render();\n    },\n\n    removeEditMode: function () {\n        this.model.set('edit-mode', false);\n        this.render();\n    },\n\n    saveEdit: function () {\n        var _this = this;\n        this.updateData();\n        this.worlds\n            .updateUser(this.model)\n            .then(function () {\n                _this.removeEditMode();\n                _this.$el.trigger('update', _this);\n            });\n    },\n\n    cancelEdit: function () {\n        this.removeEditMode();\n    },\n\n    render: function () {\n        var templ = this.model.get('edit-mode') ? this.editTemplate : this.template;\n        var vm = _.extend({\n            roles: this.project.get('roles'),\n            optionalRoles: this.project.get('optionalRoles'),\n            worlds: this.worlds.getWorldNames(),\n            newWorld: this.worlds.getNextWorldName()\n        }, this.model.toJSON());\n\n        this.$el.html(templ(vm));\n\n        return this;\n    },\n\n    updateData: function () {\n        var _this = this;\n        this.$('[data-field]').each(function () {\n            var el = $(this);\n            var field = el.data('field');\n            var val = el.val();\n\n            _this.model.set(field, val);\n        });\n    }\n});\n\n\nmodule.exports = AssignmentRow;","'use strict';\n\nvar UsersCollection = require('./users-collection');\nvar WorldsCollection = require('./worlds-collection');\nvar ProjectModel = require('./project-model');\nvar AssignemntRow = require('./assignment-row');\nvar env = require('./defaults');\nvar AjaxQueue = require('../../../util/ajax-queue');\n\nfunction setEnvironment(options) {\n    env.set(_.omit(options, 'el'));\n}\n\nvar Assignment = function (options) {\n    setEnvironment(options);\n    this.initialize(options);\n};\n\nAssignment.prototype = {\n\n    initialize: function (options) {\n        this.el = typeof options.el === 'string' ? $(options.el)[0] : options.el;\n        this.$el = $(this.el);\n        this.$ = _.partialRight($, this.el);\n\n        this.users = new UsersCollection();\n        this.worlds = new WorldsCollection();\n        this.project = new ProjectModel();\n\n        _.bindAll(this, ['render', 'renderTable', 'toggleControlls', 'saveEdit', 'selectAll', 'usassignSelected', '_showUpdating', '_hideUpdating', 'autoAssignAll', 'makeUserInactive']);\n\n        this.bindEvents();\n    },\n\n    bindEvents: function () {\n        this.$el.on('update', 'tr', this.saveEdit);\n        this.$el.on('click', 'input:checkbox:not(#select-all)', this.toggleControlls);\n        this.$el.on('click', '#select-all', this.selectAll);\n        this.$el.on('click', '.unassign-user', this.usassignSelected);\n        this.$el.on('click', '.auto-assign-all', this.autoAssignAll);\n        this.$el.on('click', '.make-user-inactive', this.makeUserInactive);\n    },\n\n    load: function () {\n\n        var join = function () {\n            this.worlds.setUsersCollection(this.users);\n            this.worlds.joinUsers();\n            this.render();\n        }.bind(this);\n\n        return $.when(\n            this.worlds.fetch(),\n            this.users.fetch(),\n            this.project.fetch()\n        ).then(join);\n\n    },\n\n    saveEdit: function () {\n        this.worlds.fetch()\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n                this.updateControls();\n            }.bind(this));\n    },\n\n    autoAssignAll: function () {\n        this._showUpdating();\n        var maxUsers = +this.$('#max-users').val();\n        return this.worlds.autoAssignAll({ maxUsers: maxUsers })\n            .then(this._hideUpdating)\n            .fail(this._hideUpdating)\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n            }.bind(this));\n    },\n\n    getSelectedIds: function () {\n        return this.$('tbody :checkbox:checked').map(function () {\n            return $(this).data('id');\n        });\n    },\n\n    findRowViews: function (ids) {\n        return _.map(ids, function (id) {\n            return this.rowViews[id];\n        }, this);\n    },\n\n    unassignUsers: function (ids) {\n        var dtd = $.Deferred();\n        var done = function () {\n            dtd.resolve();\n        };\n\n        // for now we need to sequence the calls to unassign users from worlds\n        var queue = new AjaxQueue();\n\n        _.each(ids, function (userId) {\n            var user = this.users.getById(userId);\n            user.set('world', '');\n            user.set('role', '');\n            queue.add(_.partial(_.bind(this.worlds.updateUser, this.worlds), user));\n        }, this);\n\n        queue.execute(this).then(done);\n\n        return dtd.promise();\n    },\n\n    usassignSelected: function (e) {\n        e.preventDefault();\n\n        var ids = this.getSelectedIds();\n\n        var done = function () {\n            this.worlds.fetch().then(function () {\n                this.worlds.joinUsers();\n                this._hideUpdating();\n                this.render();\n\n            }.bind(this));\n        }.bind(this);\n\n        this._showUpdating();\n\n        return this.unassignUsers(ids).then(done);\n    },\n\n    makeUserInactive: function (e) {\n        e.preventDefault();\n        var ids = this.getSelectedIds();\n        var done = function () {\n            this.toggleControlls();\n        }.bind(this);\n\n        var makeUsersInactive = function () {\n            var rows = this.findRowViews(ids);\n            // for now we need to sequence the calls to patch the users\n            // since the API can only operate on one call per group at a time\n            var queue = new AjaxQueue();\n            _.each(rows, function (view) {\n                var user = view.model;\n                queue.add(function () {\n                    return view.makeInactive()\n                        .then(function () {\n                            user.remove();\n                            view.remove();\n                        });\n                    });\n\n            }, this);\n\n            queue.execute(this).then(done);\n        }.bind(this);\n\n        return this.unassignUsers(ids)\n            .then(makeUsersInactive);\n\n\n    },\n\n    render: function () {\n        this.$('table tbody').empty();\n        this.renderTable();\n        this.toggleControlls();\n    },\n\n    renderTable: function () {\n        this.rowViews = {};\n        var rows = [];\n        this.users.each(function (u) {\n            var view = new AssignemntRow({ model: u, worlds: this.worlds, project: this.project });\n            this.rowViews[u.get('id')] = view;\n            rows.push(view.render().el);\n        }, this);\n\n        this.$('table tbody').append(rows);\n    },\n\n\n    updateControls: function () {\n        this.updateControlsForSelection();\n        this.updateAutoAssignButton();\n        this.updateStatus();\n    },\n\n    updateStatus: function () {\n        var incolpleteWorlds = this.worlds.getIncompleteWorldsCount();\n        var unassignedUsers = this.users.getUnassignedUsersCount();\n        var totalWorlds = this.worlds.size();\n\n        var usersText = unassignedUsers ? unassignedUsers === 1 ? '1 user needs assignment.' : unassignedUsers + ' users need assignment.' : 'All users have been assigned.';\n        var worldsText = !totalWorlds ? 'No worlds have been created.' : !incolpleteWorlds ? 'All worlds are complete.' : incolpleteWorlds === 1 ? '1 incomplete world needs attention.' : incolpleteWorlds + ' incomplete worlds need attention.';\n\n        this.$('#users-status .text').text(usersText);\n        this.$('#worlds-status .text').text(worldsText);\n\n        if (unassignedUsers) {\n            this.$('#users-status').addClass('incomplete');\n        } else {\n            this.$('#users-status').removeClass('incomplete');\n        }\n\n        if (incolpleteWorlds || !totalWorlds) {\n            this.$('#worlds-status').addClass('incomplete');\n        } else {\n            this.$('#worlds-status').removeClass('incomplete');\n        }\n\n        this.$('.status-widget').addClass('visible');\n    },\n\n    updateControlsForSelection: function () {\n        var numSelected = this.$('tbody :checkbox:checked').length;\n        this.$('.component.controls')[numSelected ? 'addClass' : 'removeClass']('visible');\n    },\n\n    updateAutoAssignButton: function () {\n\n        if (this.project.isDynamicAssignment()) {\n            var hasRoles = this.project.hasRoles();\n            this.$('.table-controls .single').hide();\n            this.$('.table-controls .dynamic').show();\n            this.$('.table-controls .dynamic-no-roles-text')[hasRoles ? 'hide' : 'show']();\n            this.$('.table-controls .no-roles')[hasRoles ? 'hide' : 'show']();\n        } else {\n            this.$('.table-controls .dynamic').hide();\n            this.$('.table-controls .dynamic-no-roles-text').hide();\n            this.$('.table-controls .single').show();\n            this.$('.table-controls .no-roles').show();\n\n        }\n\n        if (this.users.allUsersAssigned()) {\n            this.$('.table-controls').removeClass('visible');\n        } else {\n            this.$('.table-controls').addClass('visible');\n        }\n    },\n\n    selectAll: function (e) {\n        this.$('tbody :checkbox').prop('checked', e.target.checked);\n        this.updateControls();\n    },\n\n    toggleControlls: function (e) {\n        var total = this.$('tbody :checkbox');\n        var checked = this.$('tbody :checkbox:checked');\n\n        if (total.length === checked.length) {\n            this.$('#select-all').attr('checked', 'checked');\n        } else {\n            this.$('#select-all').removeAttr('checked');\n        }\n\n        this.updateControls();\n    },\n\n    _showUpdating: function () {\n        this.$el.css({ opacity: 0.4 });\n    },\n\n    _hideUpdating: function () {\n        this.$el.css({ opacity: 1 });\n    }\n\n};\n\nmodule.exports = Assignment;","'use strict';\n\nvar BaseCollection = function (models, options) {\n    this._models = [];\n    this.options = options;\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseCollection.prototype, {\n    idAttribute: 'id',\n\n    initialize: function (models, options) {\n    },\n\n    create: function (attr, options) {\n        var m = new this.model(attr, options);\n        this.set(m);\n        return m;\n    },\n\n    reset: function (models, options) {\n        this._models.length = 0;\n        this.set(models);\n    },\n\n    remove: function (model) {\n        _.remove(this._models, function (m) {\n            return m === model;\n        });\n\n        delete model.collection;\n\n        return model;\n    },\n\n    set: function (models) {\n        if (!models) {\n            return;\n        }\n\n        models = [].concat(models);\n\n        if (!models.length) {\n            return;\n        }\n\n        _.each(models, function (m) {\n            if (!(m instanceof this.model)) {\n                m = new this.model(m);\n            }\n\n            m.collection = this;\n\n            this._models.push(m);\n        }, this);\n\n        this.sort.call(this);\n\n        return models;\n    },\n\n    sortFn: function (a, b) {\n        return b._data[this.idAttribute] - a._data[this.idAttribute];\n    },\n\n    sort: function () {\n        this._models = this._models.sort(this.sortFn.bind(this));\n\n        return this._models;\n    },\n\n    getById: function (id) {\n        return _.find(this._models, function (m) {\n            return m.get(this.idAttribute) === id;\n        }, this);\n    },\n\n    each: function (cb, ctx) {\n        return _.each(this._models, cb, ctx || this);\n    },\n\n    all: function (cb, ctx) {\n        return _.all(this._models, cb, ctx || this);\n    },\n\n    toJSON: function () {\n        return _.invoke(this._models, 'toJSON');\n    },\n\n    find: function (fn) {\n        return _.find(this._models, fn);\n    },\n\n    filter: function (fn) {\n        return _.filter(this._models, fn);\n    },\n\n    size: function () {\n        return this._models.length;\n    },\n\n    map: function (fn, ctx) {\n        return _.map(this._models, function (model) {\n            return fn.call(ctx, model.toJSON());\n        });\n    },\n\n    pluck: function (field) {\n        return this.map(function (m) {\n            return m[field];\n        });\n    }\n\n});\n\nmodule.exports = BaseCollection;","'use strict';\n\n\nvar BaseModel = function (attr, options) {\n    attr = _.defaults({}, attr, _.result(this, 'defaults'));\n    this._data = {};\n    this.set(attr, options);\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseModel.prototype, {\n    initialize: function (attr, options) {\n\n    },\n\n    set: function (key, val, options) {\n\n        if (key == null) {\n            return this;\n        }\n\n        var attrs;\n        if (typeof key === 'object') {\n            attrs = key;\n            options = val;\n        } else {\n            (attrs = {})[key] = val;\n        }\n\n        options = options || {};\n\n        _.extend(this._data, attrs);\n\n        return this;\n    },\n\n    get: function (key, options) {\n        return this._data[key];\n    },\n\n    remove: function () {\n        if (this.collection) {\n            this.collection.remove(this);\n        }\n\n        return this;\n    },\n\n    toJSON: function () {\n        return this._data;\n    },\n\n    pick: function (keys) {\n        return _.pick(this._data, keys);\n    }\n\n});\n\nmodule.exports = BaseModel;","'use strict';\n\nvar env = {\n    account: '',\n    project: '',\n    group: '',\n    groupId: '',\n    token: '',\n    server: {\n        host: 'api.forio.com',\n        protocol: 'https'\n    }\n};\n\nmodule.exports = {\n    set: function (options) {\n        env = _.merge(env, options);\n    },\n\n    get: function () {\n        return env;\n    }\n};","(function () {\n    'use strict';\n    var App = require('./assignment.js');\n\n    window.forio = window.forio || {};\n    window.forio.MultiplayerAssignmentComponent = App;\n})();\n","'use strict';\n\nvar serviceLocator = require('./service-locator');\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\n// var __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    isDynamicAssignment: function () {\n        return this.get('worlds') === 'dynamic';\n    },\n\n    hasRoles: function () {\n        var roles = this.get('roles');\n        return roles && !!roles.length;\n    },\n\n    fetch: function () {\n        var api = serviceLocator.worldApi();\n\n        return api.getProjectSettings().then(function (settings) {\n            this.set(settings);\n        }.bind(this));\n    }\n});","'use strict';\n\nvar env = require('./defaults.js');\n\nvar cache = {};\n\nmodule.exports = {\n    worldApi: function () {\n        if (!cache.worldApi) {\n            cache.worldApi = new F.service.World(env.get());\n        }\n\n        return cache.worldApi;\n    },\n\n    memberApi: function () {\n        if (!cache.memberApi) {\n            cache.memberApi = new F.service.Member(_.pick(env.get(), ['groupId', 'server']));\n        }\n\n        return cache.memberApi;\n    },\n\n    userApi: function () {\n        if (!cache.userApi) {\n            cache.userApi = new F.service.User(_.pick(env.get(), ['account', 'server']));\n        }\n\n        return cache.userApi;\n    }\n};","exports[\"edit-user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape, __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id )) == null ? '' : __t) +\n'\"</td>\\n<td></td>\\n<td>\\n    <select name=\"worlds\" class=\"form-control\" data-field=\"world\">\\n\\n    ';\n _.each(worlds, function (w) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( w )) == null ? '' : __t) +\n'\" ' +\n((__t = ( w === world ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( w )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n        <option value=\"' +\n((__t = ( newWorld )) == null ? '' : __t) +\n'\" class=\"new-world-text\"><i>' +\n((__t = ( newWorld )) == null ? '' : __t) +\n' - New -</i></option>\\n    </select>\\n</td>\\n<td>\\n    <select name=\"roles\" class=\"form-control\" data-field=\"role\">\\n    ';\n _.each(roles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n\\n    ';\n _.each(optionalRoles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n' <i>(Optional)</i></option>\\n    ';\n }); ;\n__p += '\\n    </select>\\n</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\">\\n    <button class=\"btn btn-primary btn-tools btn-save save\">Save</button>\\n    <button class=\"btn btn-tools btn-cancel cancel\">Cancel</button>\\n</td>';\n\n}\nreturn __p\n};\nexports[\"user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape;\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id)) == null ? '' : __t) +\n'\"</td>\\n<td>' +\n((__t = ( !isWorldComplete ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( world )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( role )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\"><button class=\"btn edit btn-edit btn-tools auto-hide\">Edit</button></td>';\n\n}\nreturn __p\n};","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar serviceLocator = require('./service-locator');\n\n\nmodule.exports = classFrom(Base, {\n    defaults: {\n        world: '',\n        role: '',\n        active: true,\n        isWorldComplete: true,\n        firstName: '',\n        lastName: ''\n    },\n\n    makeActive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', true);\n\n        return memberApi.makeUserActive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    },\n\n    makeInactive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', false);\n\n        return memberApi.makeUserInactive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    }\n\n});\n","'use strict';\n\nvar classFrom = require('../../../util/inherit');\n\nvar Model = require('./user-model');\nvar Base = require('./base-collection');\nvar env = require('./defaults');\nvar serviceLocator = require('./service-locator');\n\n\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    sortFn: function (a, b) {\n        var aw = a.get('world').toLowerCase();\n        var bw = b.get('world').toLowerCase();\n        if (aw !== bw) {\n            return aw < bw ? -1 : 1;\n        }\n\n        return b.get('userName') > a.get('userName') ? -1 : 1;\n    },\n\n    initialize: function () {\n        $.ajaxSetup({\n            headers: {\n                Authorization: 'Bearer ' + env.get().token\n            }\n        });\n    },\n\n    allUsersAssigned: function () {\n        return this.all(function (u) {\n            return !!u.get('world');\n        });\n    },\n\n    getUnassignedUsersCount: function () {\n        return this.filter(function (u) {\n            return !u.get('world');\n        }).length;\n    },\n\n    fetch: function () {\n        var dtd = $.Deferred();\n        var _this = this;\n        var groupId = env.get().groupId;\n\n        var getGroupUsers = function () {\n            var memberApi = serviceLocator.memberApi();\n            var userApi = serviceLocator.userApi();\n\n            var loadGroupMembers = function () {\n                return memberApi.getGroupDetails();\n            };\n\n            var loadUsersInfo = function (group) {\n                var nonFacAndActive = function (u) { return u.active && u.role !== 'facilitator'; };\n                var users = _.pluck(_.filter(group.members, nonFacAndActive), 'userId');\n                return userApi.get({ id: users });\n            };\n\n            return loadGroupMembers()\n                .then(loadUsersInfo)\n                .fail(dtd.reject);\n        };\n\n        getGroupUsers()\n            .then(function (users) {\n                users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); });\n                _this.set(users);\n                dtd.resolve(users, _this);\n            });\n\n        return dtd.promise();\n    }\n\n});\n","'use strict';\nvar serviceLocator = require('./service-locator');\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    defaults: {\n        users: null,\n        model: 'model.eqn'\n    },\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n\n        this._data.users = this._data.users || [];\n\n        this._worldApi = serviceLocator.worldApi();\n\n        var id = this.get('id');\n        if (id) {\n            this._worldApi.updateConfig({ filter: id });\n        }\n    },\n\n    addUser: function (user) {\n        var users = this.get('users');\n        users.push(user);\n\n        return this.save();\n    },\n\n    removeUser: function (user) {\n        var id = this.get('id');\n        var checkWorld = function () {\n            if (!this.get('users').length) {\n                this.remove();\n                return this._worldApi.updateConfig({ filter: id }).delete();\n            }\n        }.bind(this);\n\n        _.remove(this.get('users'), function (u) {\n            return u.get('id') === user.get('id');\n        });\n\n        return this._worldApi\n            .updateConfig({ filter: id })\n            .removeUser({ userId: user.get('id') })\n            .then(checkWorld);\n    },\n\n    save: function () {\n        var _this = this;\n        var mapUsers = function () {\n            return _.map(this.get('users'), function (u) {\n                var res = { userId: u.get('id') };\n                var role = u.get('role');\n\n                if (role) {\n                    res.role = role;\n                }\n\n                return res;\n            });\n        }.bind(this);\n\n        var createWorld = _.partial(this._worldApi.create, this.pick(['model', 'name', 'minUsers']));\n        var addUsers = _.partial(_this._worldApi.addUsers, mapUsers(), { filter: _this.get('id') });\n        var savedUsers = this.get('users');\n        if (this.isNew()) {\n            // we need to create the world in the API and then add the users\n            return createWorld()\n                .then(function (world) {\n                    _this.set(world);\n                    _this._worldApi.updateConfig({ filter: world.id });\n                })\n                .then(addUsers)\n                .then(function (users) {\n                    // since we re-set the world, re-set the users\n                    _this.set('users', savedUsers);\n                });\n        } else {\n            // the world is already created just add the users\n            return addUsers();\n        }\n    },\n\n    isNew: function () {\n        return !this.get('lastModified');\n    }\n\n});","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Model = require('./world-model');\nvar UserModel = require('./user-model');\nvar serviceLocator = require('./service-locator');\n\nvar Base = require('./base-collection');\nvar __super = Base.prototype;\n\nvar doneFn = function (dtd, after) {\n    return _.after(after, dtd.resolve);\n};\n\nvar worldApi;\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n        worldApi = serviceLocator.worldApi();\n    },\n\n    autoAssignAll: function (options) {\n        return worldApi.autoAssign(options)\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    getIncompleteWorldsCount: function () {\n        return this.filter(function (w) {\n            return !w.get('complete');\n        }).length;\n    },\n\n    updateUser: function (user) {\n        var worldName = user.get('world');\n        var dtd = $.Deferred();\n        var prevWorld = this.getWorldByUser(user);\n        var curWorld = this.getOrCreateWorld(worldName);\n        var done = doneFn(dtd, 1);\n\n        // check if there's anything to do\n        if (!prevWorld && !curWorld) {\n            return dtd.resolve().promise();\n        }\n\n        if (prevWorld) {\n            prevWorld.removeUser(user)\n                .then(function () {\n                    if (curWorld) {\n                        return curWorld.addUser(user);\n                    }\n                })\n                .then(done);\n        } else if (curWorld) {\n            curWorld.addUser(user)\n                .then(done);\n        }\n\n        return dtd.promise();\n    },\n\n    getOrCreateWorld: function (worldName) {\n        if (!worldName) {\n            return;\n        }\n\n        var world = this.getWordByName(worldName);\n\n        if (!world) {\n            world = this.create({ name: worldName });\n        }\n\n        return world;\n    },\n\n    getWordByName: function (worldName) {\n        return this.find(function (world) {\n            return world.get('name') === worldName;\n        });\n    },\n\n    getWorldByUser: function (user) {\n        if (!user.get) {\n            throw new Error('getWorldByUser expectes a model (' + user + ')');\n        }\n\n        var id = user.get('id');\n        return this.getWorldByUserId(id);\n    },\n\n    getWorldByUserId: function (userId) {\n        return this.find(function (world) {\n            return _.find(world.get('users'), function (u) {\n                return u.get('id') === userId;\n            });\n        });\n    },\n\n    getWorldNames: function () {\n        return this.pluck('name');\n    },\n\n    getNextWorldName: function () {\n        var pad = function (num, places) {\n            var zeros = '000000000000000000';\n            var digits = num.toString().length;\n            var needed = places - digits;\n            return zeros.substr(0, needed) + num;\n        };\n\n        var worlds = this.getWorldNames();\n\n        if (!worlds.length) {\n            return 'World001';\n        }\n\n        var properNames = _.filter(worlds, function (w) { return /World\\d\\d\\d/.test(w); }).sort();\n        var lastWorld = properNames[properNames.length - 1];\n        var numWorld = +lastWorld.match(/World(\\d\\d\\d)/)[1];\n        return 'World' + pad(numWorld + 1, 3);\n    },\n\n    setUsersCollection: function (usersCollection) {\n        this.usersCollection = usersCollection;\n    },\n\n    joinUsers: function () {\n        var usersHash = {};\n        var usersCollection = this.usersCollection;\n        usersCollection.each(function (u) {\n            u.set({ isWorldComplete: true });\n            return (usersHash[u.get('id')] = u);\n        });\n\n        this.each(function (w, i) {\n            var name = w.get('name');\n            var isComplete = w.get('complete');\n            w.set({ index: i, name: name || (i + 1) + '' });\n            _.each(w.get('users'), function (u) {\n                if (usersHash[u.get('userId')]) {\n                    usersHash[u.get('userId')].set({ world: name, role: u.get('role'), isWorldComplete: isComplete });\n                }\n            });\n        }, this);\n\n        usersCollection.sort();\n    },\n\n    fetch: function () {\n        return worldApi.list()\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    parse: function (worlds) {\n        if (worlds.length) {\n            worlds = _.map(worlds, function (w) {\n                var users = _.map(w.users, function (u) {\n                    // in the world api users Ids comes as userId\n                    // make sure we add it as id so we can use the\n                    // same code to access models that come from the\n                    // member/local api as with the world api\n                    u.id = u.userId;\n                    return new UserModel(u);\n                });\n\n                w.users = users;\n\n                return w;\n            });\n        }\n\n        return worlds;\n    }\n});\n","'use strict';\n\n/**\n* Utility class to make ajax calls sequencial\n*/\nfunction AjaxQueue () {\n    this.queue = [];\n}\n\n$.extend(AjaxQueue.prototype, {\n    add: function (fn) {\n        return this.queue.push(fn);\n    },\n\n    execute: function (context) {\n        var dtd = $.Deferred();\n        var _this = this;\n        context = context || this;\n\n        function next() {\n            if (_this.queue.length) {\n                var fn = _this.queue.shift();\n\n                fn.call(context)\n                    .then(next)\n                    .fail(dtd.reject);\n            } else {\n                dtd.resolve();\n            }\n        }\n\n        next();\n\n        return dtd.promise();\n    }\n});\n\n\nmodule.exports = AjaxQueue;","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j<obj.length; j++) {\n        if (!(current = obj[j])) {\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        /*jshint -W089 */\n        for (var key in current) {\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n"]} diff --git a/dist/epicenter-multiplayer-dependencies.min.js b/dist/epicenter-multiplayer-dependencies.min.js index 15a6be71..d403fa82 100644 --- a/dist/epicenter-multiplayer-dependencies.min.js +++ b/dist/epicenter-multiplayer-dependencies.min.js @@ -1,2 +1,2 @@ -this.org=this.org||{},org.cometd={},org.cometd.JSON={},org.cometd.JSON.toJSON=org.cometd.JSON.fromJSON=function(a){throw"Abstract"},org.cometd.Utils={},org.cometd.Utils.isString=function(a){return void 0!==a&&null!==a&&("string"==typeof a||a instanceof String)},org.cometd.Utils.isArray=function(a){return void 0!==a&&null!==a&&a instanceof Array},org.cometd.Utils.inArray=function(a,b){for(var c=0;c0;){var b=l[0],c=b[0],d=b[1];if(c.url!==a.url||c.sync!==a.sync)break;l.shift(),a.messages=a.messages.concat(c.messages),this._debug("Coalesced",c.messages.length,"messages from request",d.id)}}function b(a,b){if(this.transportSend(a,b),b.expired=!1,!a.sync){var c=this.getConfiguration().maxNetworkDelay,d=c;b.metaConnect===!0&&(d+=this.getAdvice().timeout),this._debug("Transport",this.getType(),"waiting at most",d,"ms for the response, maxNetworkDelay",c);var e=this;b.timeout=this.setTimeout(function(){b.expired=!0;var c="Request "+b.id+" of transport "+e.getType()+" exceeded "+d+" ms max network delay",f={reason:c},g=b.xhr;f.httpCode=e.xhrStatus(g),e.abortXHR(g),e._debug(c),e.complete(b,!1,b.metaConnect),a.onFailure(g,a.messages,f)},d)}}function c(a){var c=++i,d={id:c,metaConnect:!1};k.length=0&&k.splice(e,1),l.length>0){var f=l.shift(),g=f[0],h=f[1];if(this._debug("Transport dequeued request",h.id),d)this.getConfiguration().autoBatch&&a.call(this,g),c.call(this,g),this._debug("Transport completed request",b.id,g);else{var i=this;this.setTimeout(function(){i.complete(h,!1,h.metaConnect);var a={reason:"Previous request failed"},b=h.xhr;a.httpCode=i.xhrStatus(b),g.onFailure(b,g.messages,a)},0)}}}function f(a){if(null!==j)throw"Concurrent metaConnect requests not allowed, request id="+j.id+" not yet completed";var c=++i;this._debug("Transport",this.getType(),"metaConnect send, request",c,"envelope",a);var d={id:c,metaConnect:!0};b.call(this,a,d),j=d}var g=new org.cometd.Transport,h=org.cometd.Transport.derive(g),i=0,j=null,k=[],l=[];return h.complete=function(a,b,c){c?d.call(this,a):e.call(this,a,b)},h.transportSend=function(a,b){throw"Abstract"},h.transportSuccess=function(a,b,c){b.expired||(this.clearTimeout(b.timeout),this.complete(b,!0,b.metaConnect),c&&c.length>0?a.onSuccess(c):a.onFailure(b.xhr,a.messages,{httpCode:204}))},h.transportFailure=function(a,b,c){b.expired||(this.clearTimeout(b.timeout),this.complete(b,!1,b.metaConnect),a.onFailure(b.xhr,a.messages,c))},h.send=function(a,b){b?f.call(this,a):c.call(this,a)},h.abort=function(){g.abort();for(var a=0;a0;){var h=org.cometd.JSON.toJSON(a.messages.slice(e,e+f)),i=a.url.length+encodeURI(h).length;if(i>c){if(1===f)return void this.setTimeout(function(){d.transportFailure(a,b,{reason:"Bayeux message too big, max is "+c})},0);--f}else g.push(f),e+=f,f=a.messages.length-e}var j=a;if(g.length>1){var k=0,l=g[0];this._debug("Transport",this.getType(),"split",a.messages.length,"messages into",g.join(" + ")),j=this._mixin(!1,{},a),j.messages=a.messages.slice(k,l),j.onSuccess=a.onSuccess,j.onFailure=a.onFailure;for(var m=1;m0&&(h=this.setTimeout(function(){h=null,f._debug("Transport",f.getType(),"timed out while connecting to URL",a,":",j,"ms");var b={code:1e3,reason:"Connect Timeout"};f.webSocketClose(c,b.code,b.reason),f.onClose(c,b)},j));var k=function(){f._debug("WebSocket opened",c),l=!1,h&&(f.clearTimeout(h),h=null),m?(d._warn("Closing Extra WebSocket Connections",c,m),f.webSocketClose(c,1e3,"Extra Connection")):f.onOpen(c)},n=function(a){a=a||{code:1e3},f._debug("WebSocket closing",c,a),l=!1,h&&(f.clearTimeout(h),h=null),null!==m&&c!==m?f._debug("Closed Extra WebSocket Connection",c):f.onClose(c,a)},o=function(a){f._debug("WebSocket message",a,c),c!==m&&d._warn("Extra WebSocket Connections",c,m),f.onMessage(c,a)};c.onopen=k,c.onclose=n,c.onerror=function(){n({code:1002,reason:"Error"})},c.onmessage=o,this._debug("Transport",this.getType(),"configured callbacks on",c)}}function b(a,b,c){var d=org.cometd.JSON.toJSON(b.messages);a.send(d),this._debug("Transport",this.getType(),"sent",b,"metaConnect =",c);var e=this.getConfiguration().maxNetworkDelay,f=e;c&&(f+=this.getAdvice().timeout,n=!0);for(var g=this,h=[],i=0;i=0){i=!0,q.splice(r,1);var s=j[p][0],t=j[p][1];delete j[p],q.length>0&&(j[q.join(",")]=[s,t]);break}}}i&&this._debug("Transport",this.getType(),"removed envelope, envelopes",j),o.call(this,d),c&&this.webSocketClose(a,1e3,"Disconnect")},f.onClose=function(a,b){this._debug("Transport",this.getType(),"closed",a,b),g=i&&h;for(var c in k)this.clearTimeout(k[c]);k={};for(var d in j){var e=j[d][0],f=j[d][1];f&&(n=!1),e.onFailure(a,e.messages,{websocketCode:b.code,reason:b.reason})}j={},m=null},f.registered=function(a,b){e.registered(a,b),d=b},f.accept=function(a,b,c){return g&&!!org.cometd.WebSocket&&d.websocketEnabled!==!1},f.send=function(a,b){this._debug("Transport",this.getType(),"sending",a,"metaConnect =",b);for(var d=[],e=0;e0)fa._info("Appending message type to URI "+f+g+" is not supported, disabling 'appendMessageTypeToURL' configuration"),xa.appendMessageTypeToURL=!1;else{var h=f.split("/"),i=h.length-1;f.match(/\/$/)&&(i-=1),h[i].indexOf(".")>=0&&(fa._info("Appending message type to URI "+f+" is not supported, disabling 'appendMessageTypeToURL' configuration"),xa.appendMessageTypeToURL=!1)}}function g(a){if(a){var b=pa[a.channel];b&&b[a.id]&&(delete b[a.id],fa._debug("Removed",a.listener?"listener":"subscription",a))}}function h(a){a&&!a.listener&&g(a)}function i(){for(var a in pa){var b=pa[a];if(b)for(var c=0;c",a),ja=a)}function k(){return"disconnecting"===ja||"disconnected"===ja}function l(){return++ka}function m(a,b,c,e,f){try{return b.call(a,e)}catch(g){fa._debug("Exception during execution of extension",c,g);var h=fa.onExtensionException;if(d(h)){fa._debug("Invoking extension exception callback",c,g);try{h.call(fa,g,c,f,e)}catch(i){fa._info("Exception during execution of exception callback in extension",c,i)}}return e}}function n(a){for(var b=0;b0)for(var e=0;e0;--e){var f=c.slice(0,e).join("/")+"/*";e===d&&p(f,b),f+="*",p(f,b)}}function r(){null!==ra&&org.cometd.Utils.clearTimeout(ra),ra=null}function s(a){r();var b=ta.interval+qa;fa._debug("Function scheduled in",b,"ms, interval =",ta.interval,"backoff =",qa,a),ra=org.cometd.Utils.setTimeout(fa,a,b)}function t(a,b,c,e){for(var f=0;f0||oa===!0?na.push(a):t(!1,[a],!1)}function v(){qa=0}function w(){qa0&&t(!1,a,!1)}function z(){if(--ma,ma<0)throw"Calls to startBatch() and endBatch() are not paired";0!==ma||k()||oa||y()}function A(){if(!k()){var a={channel:"/meta/connect",connectionType:ca.getType()};wa||(a.advice={timeout:0}),j("connecting"),fa._debug("Connect sent",a),t(!1,[a],!0,"connect"),j("connected")}}function B(){j("connecting"),s(function(){A()})}function C(a){a&&(ta=fa._mixin(!1,{},xa.advice,a),fa._debug("New advice",ta))}function D(a){if(r(),a&&ca&&ca.abort(),la=null,j("disconnected"),ma=0,v(),ca=null,na.length>0){var b=na;na=[],za.call(fa,void 0,b,{reason:"Disconnected"})}}function E(a,b,c){var e=fa.onTransportFailure;if(d(e)){fa._debug("Invoking transport failure callback",a,b,c);try{e.call(fa,a,b,c)}catch(f){fa._info("Exception during execution of transport failure callback",f)}}}function F(a,b){d(a)&&(b=a,a=void 0),la=null,i(),k()?(ia.reset(),C(xa.advice)):C(fa._mixin(!1,ta,{reconnect:"retry"})),ma=0,oa=!0,da=a,ea=b;var c="1.0",e=fa.getURL(),f=ia.findTransportTypes(c,ha,e),g={version:c,minimumVersion:c,channel:"/meta/handshake",supportedConnectionTypes:f,_callback:b,advice:{timeout:ta.timeout,interval:ta.interval}},h=fa._mixin(!1,{},da,g);if(!ca&&(ca=ia.negotiateTransport(f,c,ha,e),!ca)){var l="Could not find initial transport among: "+ia.getTransportTypes();throw fa._warn(l),l}fa._debug("Initial transport is",ca.getType()),j("handshaking"),fa._debug("Handshake sent",h),t(!1,[h],!1,"handshake")}function G(){j("handshaking"),oa=!0,s(function(){F(da,ea)})}function H(a){var b=ua[a.id];d(b)&&(delete ua[a.id],b.call(fa,a))}function I(a){H(a),q("/meta/handshake",a),q("/meta/unsuccessful",a);var b=!k()&&"none"!==ta.reconnect;b?(w(),G()):D(!1)}function J(a){if(a.successful){la=a.clientId;var b=fa.getURL(),c=ia.negotiateTransport(a.supportedConnectionTypes,a.version,ha,b);if(null===c){var d="Could not negotiate transport with server; client=["+ia.findTransportTypes(a.version,ha,b)+"], server=["+a.supportedConnectionTypes+"]",e=fa.getTransport();return E(e.getType(),null,{reason:d,connectionType:e.getType(),transport:e}),fa._warn(d),void D(!0)}ca!==c&&(fa._debug("Transport",ca.getType(),"->",c.getType()),ca=c),oa=!1,y(),a.reestablish=va,va=!0,H(a),q("/meta/handshake",a);var f=k()?"none":ta.reconnect;switch(f){case"retry":v(),B();break;case"none":D(!1);break;default:throw"Unrecognized advice action "+f}}else I(a)}function K(a){var b="1.0",c=fa.getURL(),d=fa.getTransport(),e=ia.findTransportTypes(b,ha,c),f=ia.negotiateTransport(e,b,ha,c);f?(fa._debug("Transport",d.getType(),"->",f.getType()),E(d.getType(),f.getType(),a.failure),I(a),ca=f):(E(d.getType(),null,a.failure),fa._warn("Could not negotiate transport; client=["+e+"]"),D(!0),I(a))}function L(a){q("/meta/connect",a),q("/meta/unsuccessful",a);var b=k()?"none":ta.reconnect;switch(b){case"retry":B(),w();break;case"handshake":ia.reset(),v(),G();break;case"none":D(!1);break;default:throw"Unrecognized advice action"+b}}function M(a){if(wa=a.successful){q("/meta/connect",a);var b=k()?"none":ta.reconnect;switch(b){case"retry":v(),B();break;case"none":D(!1);break;default:throw"Unrecognized advice action "+b}}else L(a)}function N(a){wa=!1,L(a)}function O(a){D(!0),H(a),q("/meta/disconnect",a),q("/meta/unsuccessful",a)}function P(a){a.successful?(D(!1),H(a),q("/meta/disconnect",a)):O(a)}function Q(a){O(a)}function R(a){var b=pa[a.subscription];if(b)for(var c=b.length-1;c>=0;--c){var d=b[c];if(d&&!d.listener){delete b[c],fa._debug("Removed failed subscription",d);break}}H(a),q("/meta/subscribe",a),q("/meta/unsuccessful",a)}function S(a){a.successful?(H(a),q("/meta/subscribe",a)):R(a)}function T(a){R(a)}function U(a){H(a),q("/meta/unsubscribe",a),q("/meta/unsuccessful",a)}function V(a){a.successful?(H(a),q("/meta/unsubscribe",a)):U(a)}function W(a){U(a)}function X(a){H(a),q("/meta/publish",a),q("/meta/unsuccessful",a)}function Y(a){void 0===a.successful?void 0!==a.data?q(a.channel,a):fa._warn("Unknown Bayeux Message",a):a.successful?(H(a),q("/meta/publish",a)):X(a)}function Z(a){X(a)}function $(a){if(a=n(a),void 0!==a&&null!==a){C(a.advice);var b=a.channel;switch(b){case"/meta/handshake":J(a);break;case"/meta/connect":M(a);break;case"/meta/disconnect":P(a);break;case"/meta/subscribe":S(a);break;case"/meta/unsubscribe":V(a);break;default:Y(a)}}}function _(a){var b=pa[a];if(b)for(var c=0;c0;){var b=l[0],c=b[0],d=b[1];if(c.url!==a.url||c.sync!==a.sync)break;l.shift(),a.messages=a.messages.concat(c.messages),this._debug("Coalesced",c.messages.length,"messages from request",d.id)}}function b(a,b){if(this.transportSend(a,b),b.expired=!1,!a.sync){var c=this.getConfiguration().maxNetworkDelay,d=c;b.metaConnect===!0&&(d+=this.getAdvice().timeout),this._debug("Transport",this.getType(),"waiting at most",d,"ms for the response, maxNetworkDelay",c);var e=this;b.timeout=this.setTimeout(function(){b.expired=!0;var c="Request "+b.id+" of transport "+e.getType()+" exceeded "+d+" ms max network delay",f={reason:c},g=b.xhr;f.httpCode=e.xhrStatus(g),e.abortXHR(g),e._debug(c),e.complete(b,!1,b.metaConnect),a.onFailure(g,a.messages,f)},d)}}function c(a){var c=++i,d={id:c,metaConnect:!1};k.length=0&&k.splice(e,1),l.length>0){var f=l.shift(),g=f[0],h=f[1];if(this._debug("Transport dequeued request",h.id),d)this.getConfiguration().autoBatch&&a.call(this,g),c.call(this,g),this._debug("Transport completed request",b.id,g);else{var i=this;this.setTimeout(function(){i.complete(h,!1,h.metaConnect);var a={reason:"Previous request failed"},b=h.xhr;a.httpCode=i.xhrStatus(b),g.onFailure(b,g.messages,a)},0)}}}function f(a){if(null!==j)throw"Concurrent metaConnect requests not allowed, request id="+j.id+" not yet completed";var c=++i;this._debug("Transport",this.getType(),"metaConnect send, request",c,"envelope",a);var d={id:c,metaConnect:!0};b.call(this,a,d),j=d}var g=new org.cometd.Transport,h=org.cometd.Transport.derive(g),i=0,j=null,k=[],l=[];return h.complete=function(a,b,c){c?d.call(this,a):e.call(this,a,b)},h.transportSend=function(a,b){throw"Abstract"},h.transportSuccess=function(a,b,c){b.expired||(this.clearTimeout(b.timeout),this.complete(b,!0,b.metaConnect),c&&c.length>0?a.onSuccess(c):a.onFailure(b.xhr,a.messages,{httpCode:204}))},h.transportFailure=function(a,b,c){b.expired||(this.clearTimeout(b.timeout),this.complete(b,!1,b.metaConnect),a.onFailure(b.xhr,a.messages,c))},h.send=function(a,b){b?f.call(this,a):c.call(this,a)},h.abort=function(){g.abort();for(var a=0;a0;){var h=org.cometd.JSON.toJSON(a.messages.slice(e,e+f)),i=a.url.length+encodeURI(h).length;if(i>c){if(1===f)return void this.setTimeout(function(){d.transportFailure(a,b,{reason:"Bayeux message too big, max is "+c})},0);--f}else g.push(f),e+=f,f=a.messages.length-e}var j=a;if(g.length>1){var k=0,l=g[0];this._debug("Transport",this.getType(),"split",a.messages.length,"messages into",g.join(" + ")),j=this._mixin(!1,{},a),j.messages=a.messages.slice(k,l),j.onSuccess=a.onSuccess,j.onFailure=a.onFailure;for(var m=1;m0&&(f=this.setTimeout(function(){f=null,e._debug("Transport",e.getType(),"timed out while connecting to URL",a,":",h,"ms");var b={code:1e3,reason:"Connect Timeout"};e.webSocketClose(c,b.code,b.reason),e.onClose(c,b)},h));var j=function(){e._debug("WebSocket opened",c),l=!1,f&&(e.clearTimeout(f),f=null),m?(d._warn("Closing Extra WebSocket Connections",c,m),e.webSocketClose(c,1e3,"Extra Connection")):e.onOpen(c)},k=function(a){a=a||{code:1e3},e._debug("WebSocket closing",c,a),l=!1,f&&(e.clearTimeout(f),f=null),null!==m&&c!==m?e._debug("Closed Extra WebSocket Connection",c):e.onClose(c,a)},n=function(a){e._debug("WebSocket message",a,c),c!==m&&d._warn("Extra WebSocket Connections",c,m),e.onMessage(c,a)};c.onopen=j,c.onclose=k,c.onerror=function(){k({code:1002,reason:"Error"})},c.onmessage=n,this._debug("Transport",this.getType(),"configured callbacks on",c)}}function b(a,b,c){var d=org.cometd.JSON.toJSON(b.messages);a.send(d),this._debug("Transport",this.getType(),"sent",b,"metaConnect =",c);var e=this.getConfiguration().maxNetworkDelay,f=e;c&&(f+=this.getAdvice().timeout,n=!0);for(var g=this,h=[],i=0;i=0){i=!0,q.splice(r,1);var s=j[p][0],t=j[p][1];delete j[p],q.length>0&&(j[q.join(",")]=[s,t]);break}}}i&&this._debug("Transport",this.getType(),"removed envelope, envelopes",j),o.call(this,d),c&&this.webSocketClose(a,1e3,"Disconnect")},f.onClose=function(a,b){this._debug("Transport",this.getType(),"closed",a,b),g=i&&h;for(var c in k)this.clearTimeout(k[c]);k={};for(var d in j){var e=j[d][0],f=j[d][1];f&&(n=!1),e.onFailure(a,e.messages,{websocketCode:b.code,reason:b.reason})}j={},m=null},f.registered=function(a,b){e.registered(a,b),d=b},f.accept=function(a,b,c){return g&&!!org.cometd.WebSocket&&d.websocketEnabled!==!1},f.send=function(a,b){this._debug("Transport",this.getType(),"sending",a,"metaConnect =",b);for(var d=[],e=0;e0)fa._info("Appending message type to URI "+f+g+" is not supported, disabling 'appendMessageTypeToURL' configuration"),xa.appendMessageTypeToURL=!1;else{var h=f.split("/"),i=h.length-1;f.match(/\/$/)&&(i-=1),h[i].indexOf(".")>=0&&(fa._info("Appending message type to URI "+f+" is not supported, disabling 'appendMessageTypeToURL' configuration"),xa.appendMessageTypeToURL=!1)}}function g(a){if(a){var b=pa[a.channel];b&&b[a.id]&&(delete b[a.id],fa._debug("Removed",a.listener?"listener":"subscription",a))}}function h(a){a&&!a.listener&&g(a)}function i(){for(var a in pa){var b=pa[a];if(b)for(var c=0;c",a),ja=a)}function k(){return"disconnecting"===ja||"disconnected"===ja}function l(){return++ka}function m(a,b,c,e,f){try{return b.call(a,e)}catch(a){fa._debug("Exception during execution of extension",c,a);var g=fa.onExtensionException;if(d(g)){fa._debug("Invoking extension exception callback",c,a);try{g.call(fa,a,c,f,e)}catch(a){fa._info("Exception during execution of exception callback in extension",c,a)}}return e}}function n(a){for(var b=0;b0)for(var e=0;e0;--e){var f=c.slice(0,e).join("/")+"/*";e===d&&p(f,b),f+="*",p(f,b)}}function r(){null!==ra&&org.cometd.Utils.clearTimeout(ra),ra=null}function s(a){r();var b=ta.interval+qa;fa._debug("Function scheduled in",b,"ms, interval =",ta.interval,"backoff =",qa,a),ra=org.cometd.Utils.setTimeout(fa,a,b)}function t(a,b,c,e){for(var f=0;f0||oa===!0?na.push(a):t(!1,[a],!1)}function v(){qa=0}function w(){qa0&&t(!1,a,!1)}function z(){if(--ma,ma<0)throw"Calls to startBatch() and endBatch() are not paired";0!==ma||k()||oa||y()}function A(){if(!k()){var a={channel:"/meta/connect",connectionType:ca.getType()};wa||(a.advice={timeout:0}),j("connecting"),fa._debug("Connect sent",a),t(!1,[a],!0,"connect"),j("connected")}}function B(){j("connecting"),s(function(){A()})}function C(a){a&&(ta=fa._mixin(!1,{},xa.advice,a),fa._debug("New advice",ta))}function D(a){if(r(),a&&ca&&ca.abort(),la=null,j("disconnected"),ma=0,v(),ca=null,na.length>0){var b=na;na=[],za.call(fa,void 0,b,{reason:"Disconnected"})}}function E(a,b,c){var e=fa.onTransportFailure;if(d(e)){fa._debug("Invoking transport failure callback",a,b,c);try{e.call(fa,a,b,c)}catch(a){fa._info("Exception during execution of transport failure callback",a)}}}function F(a,b){d(a)&&(b=a,a=void 0),la=null,i(),k()?(ia.reset(),C(xa.advice)):C(fa._mixin(!1,ta,{reconnect:"retry"})),ma=0,oa=!0,da=a,ea=b;var c="1.0",e=fa.getURL(),f=ia.findTransportTypes(c,ha,e),g={version:c,minimumVersion:c,channel:"/meta/handshake",supportedConnectionTypes:f,_callback:b,advice:{timeout:ta.timeout,interval:ta.interval}},h=fa._mixin(!1,{},da,g);if(!ca&&(ca=ia.negotiateTransport(f,c,ha,e),!ca)){var l="Could not find initial transport among: "+ia.getTransportTypes();throw fa._warn(l),l}fa._debug("Initial transport is",ca.getType()),j("handshaking"),fa._debug("Handshake sent",h),t(!1,[h],!1,"handshake")}function G(){j("handshaking"),oa=!0,s(function(){F(da,ea)})}function H(a){var b=ua[a.id];d(b)&&(delete ua[a.id],b.call(fa,a))}function I(a){H(a),q("/meta/handshake",a),q("/meta/unsuccessful",a);var b=!k()&&"none"!==ta.reconnect;b?(w(),G()):D(!1)}function J(a){if(a.successful){la=a.clientId;var b=fa.getURL(),c=ia.negotiateTransport(a.supportedConnectionTypes,a.version,ha,b);if(null===c){var d="Could not negotiate transport with server; client=["+ia.findTransportTypes(a.version,ha,b)+"], server=["+a.supportedConnectionTypes+"]",e=fa.getTransport();return E(e.getType(),null,{reason:d,connectionType:e.getType(),transport:e}),fa._warn(d),void D(!0)}ca!==c&&(fa._debug("Transport",ca.getType(),"->",c.getType()),ca=c),oa=!1,y(),a.reestablish=va,va=!0,H(a),q("/meta/handshake",a);var f=k()?"none":ta.reconnect;switch(f){case"retry":v(),B();break;case"none":D(!1);break;default:throw"Unrecognized advice action "+f}}else I(a)}function K(a){var b="1.0",c=fa.getURL(),d=fa.getTransport(),e=ia.findTransportTypes(b,ha,c),f=ia.negotiateTransport(e,b,ha,c);f?(fa._debug("Transport",d.getType(),"->",f.getType()),E(d.getType(),f.getType(),a.failure),I(a),ca=f):(E(d.getType(),null,a.failure),fa._warn("Could not negotiate transport; client=["+e+"]"),D(!0),I(a))}function L(a){q("/meta/connect",a),q("/meta/unsuccessful",a);var b=k()?"none":ta.reconnect;switch(b){case"retry":B(),w();break;case"handshake":ia.reset(),v(),G();break;case"none":D(!1);break;default:throw"Unrecognized advice action"+b}}function M(a){if(wa=a.successful){q("/meta/connect",a);var b=k()?"none":ta.reconnect;switch(b){case"retry":v(),B();break;case"none":D(!1);break;default:throw"Unrecognized advice action "+b}}else L(a)}function N(a){wa=!1,L(a)}function O(a){D(!0),H(a),q("/meta/disconnect",a),q("/meta/unsuccessful",a)}function P(a){a.successful?(D(!1),H(a),q("/meta/disconnect",a)):O(a)}function Q(a){O(a)}function R(a){var b=pa[a.subscription];if(b)for(var c=b.length-1;c>=0;--c){var d=b[c];if(d&&!d.listener){delete b[c],fa._debug("Removed failed subscription",d);break}}H(a),q("/meta/subscribe",a),q("/meta/unsuccessful",a)}function S(a){a.successful?(H(a),q("/meta/subscribe",a)):R(a)}function T(a){R(a)}function U(a){H(a),q("/meta/unsubscribe",a),q("/meta/unsuccessful",a)}function V(a){a.successful?(H(a),q("/meta/unsubscribe",a)):U(a)}function W(a){U(a)}function X(a){H(a),q("/meta/publish",a),q("/meta/unsuccessful",a)}function Y(a){void 0===a.successful?void 0!==a.data?q(a.channel,a):fa._warn("Unknown Bayeux Message",a):a.successful?(H(a),q("/meta/publish",a)):X(a)}function Z(a){X(a)}function $(a){if(a=n(a),void 0!==a&&null!==a){C(a.advice);var b=a.channel;switch(b){case"/meta/handshake":J(a);break;case"/meta/connect":M(a);break;case"/meta/disconnect":P(a);break;case"/meta/subscribe":S(a);break;case"/meta/unsubscribe":V(a);break;default:Y(a)}}}function _(a){var b=pa[a];if(b)for(var c=0;cLogged in as: ' + credentials.userName + ''); _this.assetAdapter = new F.service.Asset(server); @@ -66,7 +66,7 @@ $(function() { var imageExtensions = ['png', 'jpg', 'jpeg', 'gif', 'svg', 'bmp']; $('.images').empty(); - this.assetAdapter.list({ scope: scope }).done(function (response) { + this.assetAdapter.list({ scope: scope }).then(function (response) { $.each(response, function () { var url = this; var filename = url.substring(url.lastIndexOf('/') + 1); @@ -103,7 +103,7 @@ $(function() { try { // filename will be ignored if it's a multipart/form-data request - this.assetAdapter.create(filename, data, { scope: scope }).done(function () { + this.assetAdapter.create(filename, data, { scope: scope }).then(function () { $('.status').html('
File uploaded! Reloading assets...
'); $('.list [name=scope]').val(scope); _this.listAssets(); @@ -130,7 +130,7 @@ $(function() { // var aa = new F.service.Asset($.extend(server, extra)); // aa.create('test.txt', { encoding: 'BASE_64', data: 'VGhpcyBpcyBhIHRlc3QgZmlsZS4=' }, { scope: 'user' }); //aa.delete('test1.txt', { scope: 'user' }); - // aa.list({ scope: 'user', fullUrl: true }).done(function (response) { + // aa.list({ scope: 'user', fullUrl: true }).then(function (response) { // console.log(response); // }); }); diff --git a/tests/integration/multiplayer/test-script.js b/tests/integration/multiplayer/test-script.js index da394356..a3962e2e 100644 --- a/tests/integration/multiplayer/test-script.js +++ b/tests/integration/multiplayer/test-script.js @@ -24,7 +24,7 @@ $(function () { password: $('#txtPassword').val(), account: $('#txtAccount').val(), project: $('#txtProject').val() - }).done(function () { + }).then(function () { window.alert('login successful'); }); }); diff --git a/tests/spec/test-multiplayer-strategy.js b/tests/spec/test-multiplayer-strategy.js index e148fb28..80da769f 100644 --- a/tests/spec/test-multiplayer-strategy.js +++ b/tests/spec/test-multiplayer-strategy.js @@ -51,24 +51,20 @@ server = sinon.fakeServer.create(); setupResponse('GET', queryMatchers.worldEndpoint, 200, worlds || []); - setupResponse('POST', /multiplayer\/world\/worldid1\/run/, 201, 'run1'); setupResponse('POST', /multiplayer\/world\/worldid2\/run/, 201, 'run2'); - server.autorespond = true; + server.respondImmediately = true; }; var teardownServer = function () { server.restore(); }; - - - describe('Multiplayer strategy', function () { + describe.only('Multiplayer strategy', function () { beforeEach(_.partial(setupServer, worldSet)); afterEach(teardownServer); - function createRunManager(options) { var rm = new F.manager.RunManager(_.extend({ strategy: 'multiplayer', @@ -82,42 +78,34 @@ // this is briddle, it knows too much about the internals of the run manager // but replace the cookie store with a stub rm.strategy._auth = fakeAuth; - return rm; } describe('with world/users setup correctly', function () { it('should get the list of worlds for the current user first', function () { - createRunManager().getRun(); - - var req = server.requests.pop(); - req.method.toUpperCase().should.equal('GET'); - - req.url.should.match(queryMatchers.getWorlds); + return createRunManager().getRun().then(function () { + var req = server.requests.pop(); + req.method.toUpperCase().should.equal('GET'); + req.url.should.match(queryMatchers.getWorlds); + }); }); it('should post to the run endpoint after getting the world', function () { - createRunManager().getRun(); - - server.respond(); - var req = server.requests.pop(); - - req.method.toUpperCase().should.equal('POST'); - req.url.should.match(/multiplayer\/world\/worldid2\/run/); - + return createRunManager().getRun().then(function () { + var req = server.requests.pop(); + req.method.toUpperCase().should.equal('POST'); + req.url.should.match(/multiplayer\/world\/worldid2\/run/); + }); }); }); describe('with two worlds for the user', function () { it('should use the latest world to retore the run', function () { - createRunManager().getRun(); - - server.respond(); - var req = server.requests.pop(); - - req.method.toUpperCase().should.equal('POST'); - - req.url.should.match(/multiplayer\/world\/worldid2/); + return createRunManager().getRun().then(function () { + var req = server.requests.pop(); + req.method.toUpperCase().should.equal('POST'); + req.url.should.match(/multiplayer\/world\/worldid2/); + }); }); }); @@ -125,14 +113,12 @@ beforeEach(_.partial(setupServer, [])); it('should fail the getRun request with proper error', function () { var callback = sinon.spy(); - createRunManager().getRun() - .fail(callback); - - server.respond(); - - callback.called.should.be.true; + return createRunManager().getRun() + .then(null, callback) + .then(function () { + callback.called.should.be.true; + }); }); }); - }); })(); diff --git a/tests/spec/test-run-api-service.js b/tests/spec/test-run-api-service.js index efad2073..d5684896 100644 --- a/tests/spec/test-run-api-service.js +++ b/tests/spec/test-run-api-service.js @@ -18,7 +18,7 @@ return include; }; - describe.only('Run API Service', function () { + describe('Run API Service', function () { var server; before(function () { server = sinon.fakeServer.create(); @@ -588,7 +588,7 @@ ret.should.throw(Error); }); - it.only('should send multiple operations calls one by one', function () { + it('should send multiple operations calls one by one', function () { var rs = new RunService({ account: account, project: 'js-libs', filter: { saved: true } }); return rs.serial([{ first: [1, 2] }, { second: [2, 3] }]).then(function () { server.requests.length.should.equal(2); From 26e83b09dc9e7e65286648608dc36f9fcce36984 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 14:28:00 -0700 Subject: [PATCH 10/16] fix context --- src/managers/run-strategies/multiplayer-strategy.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/managers/run-strategies/multiplayer-strategy.js b/src/managers/run-strategies/multiplayer-strategy.js index 40c42420..9bd4afb1 100644 --- a/src/managers/run-strategies/multiplayer-strategy.js +++ b/src/managers/run-strategies/multiplayer-strategy.js @@ -49,7 +49,7 @@ var Strategy = classFrom(IdentityStrategy, { var loadRunFromWorld = function (world) { if (!world) { - return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: this.options, session: session }); + return dtd.reject({ statusCode: 404, error: 'The user is not in any world.' }, { options: _this.options, session: session }); } return worldApi.getCurrentRunId({ model: model, filter: world.id }) From e99c509b75cf2acea8130d006d90b4d896c51d96 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 16:24:52 -0700 Subject: [PATCH 11/16] Then really only takes 1 parameters, so we should not be resolving with multiples --- dist/components/assignment/assignment.js | 4 ++-- dist/components/login/login.js | 2 -- src/components/assignment/js/users-collection.js | 2 +- src/managers/world-manager.js | 2 +- src/service/asset-api-adapter.js | 2 +- src/service/world-api-adapter.js | 4 ++-- tests/spec/test-conditional-creation-strategy.js | 2 -- tests/spec/test-world-manager.js | 5 +---- 8 files changed, 8 insertions(+), 15 deletions(-) diff --git a/dist/components/assignment/assignment.js b/dist/components/assignment/assignment.js index 369e2736..3ca45965 100644 --- a/dist/components/assignment/assignment.js +++ b/dist/components/assignment/assignment.js @@ -850,7 +850,7 @@ module.exports = classFrom(Base, { .then(function (users) { users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); }); _this.set(users); - dtd.resolve(users, _this); + dtd.resolve(users); }); return dtd.promise(); @@ -1231,4 +1231,4 @@ module.exports = function (base, props, staticProps) { }; },{}]},{},[6]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/components/assignment/js/assignment-row.js","src/components/assignment/js/assignment.js","src/components/assignment/js/base-collection.js","src/components/assignment/js/base-model.js","src/components/assignment/js/defaults.js","src/components/assignment/js/index.js","src/components/assignment/js/project-model.js","src/components/assignment/js/service-locator.js","src/components/assignment/js/templates.js","src/components/assignment/js/user-model.js","src/components/assignment/js/users-collection.js","src/components/assignment/js/world-model.js","src/components/assignment/js/worlds-collection.js","src/util/ajax-queue.js","src/util/inherit.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nvar templates = require('./templates');\n\nvar AssignmentRow = function (options) {\n    this.$el = $('<tr>');\n    this.el = this.$el[0];\n    this.$ = _.partialRight($, this.$el);\n\n    this.model = options.model;\n    this.options = options;\n    this.worlds = options.worlds;\n    this.project = options.project;\n\n    _.bindAll(this, ['setEditMode', 'removeEditMode', 'saveEdit', 'cancelEdit', 'updateData']);\n\n    this.bindEvents();\n\n};\n\n_.extend(AssignmentRow.prototype, {\n\n    template: templates['user-row'],\n\n    editTemplate: templates['edit-user-row'],\n\n    bindEvents: function () {\n        this.$el.on('click', 'button.edit', this.setEditMode);\n        this.$el.on('click', 'button.save', this.saveEdit);\n        this.$el.on('click', 'button.cancel', this.cancelEdit);\n    },\n\n    remove: function () {\n        this.$el.off('click', null, null);\n        // this only gives a delay to remove the tr\n        // animation of height of the tr does not work\n        this.$(':checkbox').attr('checked', false);\n        this.$el\n            .css({ opacity: 0.3 })\n            .animate({ height: 0 }, {\n                duration: 300,\n                complete: function () {\n                    this.remove();\n                }\n            });\n    },\n\n    makeInactive: function () {\n        return this.model.makeInactive();\n    },\n\n    setEditMode: function () {\n        this.model.set('edit-mode', true);\n        this.render();\n    },\n\n    removeEditMode: function () {\n        this.model.set('edit-mode', false);\n        this.render();\n    },\n\n    saveEdit: function () {\n        var _this = this;\n        this.updateData();\n        this.worlds\n            .updateUser(this.model)\n            .then(function () {\n                _this.removeEditMode();\n                _this.$el.trigger('update', _this);\n            });\n    },\n\n    cancelEdit: function () {\n        this.removeEditMode();\n    },\n\n    render: function () {\n        var templ = this.model.get('edit-mode') ? this.editTemplate : this.template;\n        var vm = _.extend({\n            roles: this.project.get('roles'),\n            optionalRoles: this.project.get('optionalRoles'),\n            worlds: this.worlds.getWorldNames(),\n            newWorld: this.worlds.getNextWorldName()\n        }, this.model.toJSON());\n\n        this.$el.html(templ(vm));\n\n        return this;\n    },\n\n    updateData: function () {\n        var _this = this;\n        this.$('[data-field]').each(function () {\n            var el = $(this);\n            var field = el.data('field');\n            var val = el.val();\n\n            _this.model.set(field, val);\n        });\n    }\n});\n\n\nmodule.exports = AssignmentRow;","'use strict';\n\nvar UsersCollection = require('./users-collection');\nvar WorldsCollection = require('./worlds-collection');\nvar ProjectModel = require('./project-model');\nvar AssignemntRow = require('./assignment-row');\nvar env = require('./defaults');\nvar AjaxQueue = require('../../../util/ajax-queue');\n\nfunction setEnvironment(options) {\n    env.set(_.omit(options, 'el'));\n}\n\nvar Assignment = function (options) {\n    setEnvironment(options);\n    this.initialize(options);\n};\n\nAssignment.prototype = {\n\n    initialize: function (options) {\n        this.el = typeof options.el === 'string' ? $(options.el)[0] : options.el;\n        this.$el = $(this.el);\n        this.$ = _.partialRight($, this.el);\n\n        this.users = new UsersCollection();\n        this.worlds = new WorldsCollection();\n        this.project = new ProjectModel();\n\n        _.bindAll(this, ['render', 'renderTable', 'toggleControlls', 'saveEdit', 'selectAll', 'usassignSelected', '_showUpdating', '_hideUpdating', 'autoAssignAll', 'makeUserInactive']);\n\n        this.bindEvents();\n    },\n\n    bindEvents: function () {\n        this.$el.on('update', 'tr', this.saveEdit);\n        this.$el.on('click', 'input:checkbox:not(#select-all)', this.toggleControlls);\n        this.$el.on('click', '#select-all', this.selectAll);\n        this.$el.on('click', '.unassign-user', this.usassignSelected);\n        this.$el.on('click', '.auto-assign-all', this.autoAssignAll);\n        this.$el.on('click', '.make-user-inactive', this.makeUserInactive);\n    },\n\n    load: function () {\n\n        var join = function () {\n            this.worlds.setUsersCollection(this.users);\n            this.worlds.joinUsers();\n            this.render();\n        }.bind(this);\n\n        return $.when(\n            this.worlds.fetch(),\n            this.users.fetch(),\n            this.project.fetch()\n        ).then(join);\n\n    },\n\n    saveEdit: function () {\n        this.worlds.fetch()\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n                this.updateControls();\n            }.bind(this));\n    },\n\n    autoAssignAll: function () {\n        this._showUpdating();\n        var maxUsers = +this.$('#max-users').val();\n        return this.worlds.autoAssignAll({ maxUsers: maxUsers })\n            .then(this._hideUpdating)\n            .fail(this._hideUpdating)\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n            }.bind(this));\n    },\n\n    getSelectedIds: function () {\n        return this.$('tbody :checkbox:checked').map(function () {\n            return $(this).data('id');\n        });\n    },\n\n    findRowViews: function (ids) {\n        return _.map(ids, function (id) {\n            return this.rowViews[id];\n        }, this);\n    },\n\n    unassignUsers: function (ids) {\n        var dtd = $.Deferred();\n        var done = function () {\n            dtd.resolve();\n        };\n\n        // for now we need to sequence the calls to unassign users from worlds\n        var queue = new AjaxQueue();\n\n        _.each(ids, function (userId) {\n            var user = this.users.getById(userId);\n            user.set('world', '');\n            user.set('role', '');\n            queue.add(_.partial(_.bind(this.worlds.updateUser, this.worlds), user));\n        }, this);\n\n        queue.execute(this).then(done);\n\n        return dtd.promise();\n    },\n\n    usassignSelected: function (e) {\n        e.preventDefault();\n\n        var ids = this.getSelectedIds();\n\n        var done = function () {\n            this.worlds.fetch().then(function () {\n                this.worlds.joinUsers();\n                this._hideUpdating();\n                this.render();\n\n            }.bind(this));\n        }.bind(this);\n\n        this._showUpdating();\n\n        return this.unassignUsers(ids).then(done);\n    },\n\n    makeUserInactive: function (e) {\n        e.preventDefault();\n        var ids = this.getSelectedIds();\n        var done = function () {\n            this.toggleControlls();\n        }.bind(this);\n\n        var makeUsersInactive = function () {\n            var rows = this.findRowViews(ids);\n            // for now we need to sequence the calls to patch the users\n            // since the API can only operate on one call per group at a time\n            var queue = new AjaxQueue();\n            _.each(rows, function (view) {\n                var user = view.model;\n                queue.add(function () {\n                    return view.makeInactive()\n                        .then(function () {\n                            user.remove();\n                            view.remove();\n                        });\n                    });\n\n            }, this);\n\n            queue.execute(this).then(done);\n        }.bind(this);\n\n        return this.unassignUsers(ids)\n            .then(makeUsersInactive);\n\n\n    },\n\n    render: function () {\n        this.$('table tbody').empty();\n        this.renderTable();\n        this.toggleControlls();\n    },\n\n    renderTable: function () {\n        this.rowViews = {};\n        var rows = [];\n        this.users.each(function (u) {\n            var view = new AssignemntRow({ model: u, worlds: this.worlds, project: this.project });\n            this.rowViews[u.get('id')] = view;\n            rows.push(view.render().el);\n        }, this);\n\n        this.$('table tbody').append(rows);\n    },\n\n\n    updateControls: function () {\n        this.updateControlsForSelection();\n        this.updateAutoAssignButton();\n        this.updateStatus();\n    },\n\n    updateStatus: function () {\n        var incolpleteWorlds = this.worlds.getIncompleteWorldsCount();\n        var unassignedUsers = this.users.getUnassignedUsersCount();\n        var totalWorlds = this.worlds.size();\n\n        var usersText = unassignedUsers ? unassignedUsers === 1 ? '1 user needs assignment.' : unassignedUsers + ' users need assignment.' : 'All users have been assigned.';\n        var worldsText = !totalWorlds ? 'No worlds have been created.' : !incolpleteWorlds ? 'All worlds are complete.' : incolpleteWorlds === 1 ? '1 incomplete world needs attention.' : incolpleteWorlds + ' incomplete worlds need attention.';\n\n        this.$('#users-status .text').text(usersText);\n        this.$('#worlds-status .text').text(worldsText);\n\n        if (unassignedUsers) {\n            this.$('#users-status').addClass('incomplete');\n        } else {\n            this.$('#users-status').removeClass('incomplete');\n        }\n\n        if (incolpleteWorlds || !totalWorlds) {\n            this.$('#worlds-status').addClass('incomplete');\n        } else {\n            this.$('#worlds-status').removeClass('incomplete');\n        }\n\n        this.$('.status-widget').addClass('visible');\n    },\n\n    updateControlsForSelection: function () {\n        var numSelected = this.$('tbody :checkbox:checked').length;\n        this.$('.component.controls')[numSelected ? 'addClass' : 'removeClass']('visible');\n    },\n\n    updateAutoAssignButton: function () {\n\n        if (this.project.isDynamicAssignment()) {\n            var hasRoles = this.project.hasRoles();\n            this.$('.table-controls .single').hide();\n            this.$('.table-controls .dynamic').show();\n            this.$('.table-controls .dynamic-no-roles-text')[hasRoles ? 'hide' : 'show']();\n            this.$('.table-controls .no-roles')[hasRoles ? 'hide' : 'show']();\n        } else {\n            this.$('.table-controls .dynamic').hide();\n            this.$('.table-controls .dynamic-no-roles-text').hide();\n            this.$('.table-controls .single').show();\n            this.$('.table-controls .no-roles').show();\n\n        }\n\n        if (this.users.allUsersAssigned()) {\n            this.$('.table-controls').removeClass('visible');\n        } else {\n            this.$('.table-controls').addClass('visible');\n        }\n    },\n\n    selectAll: function (e) {\n        this.$('tbody :checkbox').prop('checked', e.target.checked);\n        this.updateControls();\n    },\n\n    toggleControlls: function (e) {\n        var total = this.$('tbody :checkbox');\n        var checked = this.$('tbody :checkbox:checked');\n\n        if (total.length === checked.length) {\n            this.$('#select-all').attr('checked', 'checked');\n        } else {\n            this.$('#select-all').removeAttr('checked');\n        }\n\n        this.updateControls();\n    },\n\n    _showUpdating: function () {\n        this.$el.css({ opacity: 0.4 });\n    },\n\n    _hideUpdating: function () {\n        this.$el.css({ opacity: 1 });\n    }\n\n};\n\nmodule.exports = Assignment;","'use strict';\n\nvar BaseCollection = function (models, options) {\n    this._models = [];\n    this.options = options;\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseCollection.prototype, {\n    idAttribute: 'id',\n\n    initialize: function (models, options) {\n    },\n\n    create: function (attr, options) {\n        var m = new this.model(attr, options);\n        this.set(m);\n        return m;\n    },\n\n    reset: function (models, options) {\n        this._models.length = 0;\n        this.set(models);\n    },\n\n    remove: function (model) {\n        _.remove(this._models, function (m) {\n            return m === model;\n        });\n\n        delete model.collection;\n\n        return model;\n    },\n\n    set: function (models) {\n        if (!models) {\n            return;\n        }\n\n        models = [].concat(models);\n\n        if (!models.length) {\n            return;\n        }\n\n        _.each(models, function (m) {\n            if (!(m instanceof this.model)) {\n                m = new this.model(m);\n            }\n\n            m.collection = this;\n\n            this._models.push(m);\n        }, this);\n\n        this.sort.call(this);\n\n        return models;\n    },\n\n    sortFn: function (a, b) {\n        return b._data[this.idAttribute] - a._data[this.idAttribute];\n    },\n\n    sort: function () {\n        this._models = this._models.sort(this.sortFn.bind(this));\n\n        return this._models;\n    },\n\n    getById: function (id) {\n        return _.find(this._models, function (m) {\n            return m.get(this.idAttribute) === id;\n        }, this);\n    },\n\n    each: function (cb, ctx) {\n        return _.each(this._models, cb, ctx || this);\n    },\n\n    all: function (cb, ctx) {\n        return _.all(this._models, cb, ctx || this);\n    },\n\n    toJSON: function () {\n        return _.invoke(this._models, 'toJSON');\n    },\n\n    find: function (fn) {\n        return _.find(this._models, fn);\n    },\n\n    filter: function (fn) {\n        return _.filter(this._models, fn);\n    },\n\n    size: function () {\n        return this._models.length;\n    },\n\n    map: function (fn, ctx) {\n        return _.map(this._models, function (model) {\n            return fn.call(ctx, model.toJSON());\n        });\n    },\n\n    pluck: function (field) {\n        return this.map(function (m) {\n            return m[field];\n        });\n    }\n\n});\n\nmodule.exports = BaseCollection;","'use strict';\n\n\nvar BaseModel = function (attr, options) {\n    attr = _.defaults({}, attr, _.result(this, 'defaults'));\n    this._data = {};\n    this.set(attr, options);\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseModel.prototype, {\n    initialize: function (attr, options) {\n\n    },\n\n    set: function (key, val, options) {\n\n        if (key == null) {\n            return this;\n        }\n\n        var attrs;\n        if (typeof key === 'object') {\n            attrs = key;\n            options = val;\n        } else {\n            (attrs = {})[key] = val;\n        }\n\n        options = options || {};\n\n        _.extend(this._data, attrs);\n\n        return this;\n    },\n\n    get: function (key, options) {\n        return this._data[key];\n    },\n\n    remove: function () {\n        if (this.collection) {\n            this.collection.remove(this);\n        }\n\n        return this;\n    },\n\n    toJSON: function () {\n        return this._data;\n    },\n\n    pick: function (keys) {\n        return _.pick(this._data, keys);\n    }\n\n});\n\nmodule.exports = BaseModel;","'use strict';\n\nvar env = {\n    account: '',\n    project: '',\n    group: '',\n    groupId: '',\n    token: '',\n    server: {\n        host: 'api.forio.com',\n        protocol: 'https'\n    }\n};\n\nmodule.exports = {\n    set: function (options) {\n        env = _.merge(env, options);\n    },\n\n    get: function () {\n        return env;\n    }\n};","(function () {\n    'use strict';\n    var App = require('./assignment.js');\n\n    window.forio = window.forio || {};\n    window.forio.MultiplayerAssignmentComponent = App;\n})();\n","'use strict';\n\nvar serviceLocator = require('./service-locator');\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\n// var __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    isDynamicAssignment: function () {\n        return this.get('worlds') === 'dynamic';\n    },\n\n    hasRoles: function () {\n        var roles = this.get('roles');\n        return roles && !!roles.length;\n    },\n\n    fetch: function () {\n        var api = serviceLocator.worldApi();\n\n        return api.getProjectSettings().then(function (settings) {\n            this.set(settings);\n        }.bind(this));\n    }\n});","'use strict';\n\nvar env = require('./defaults.js');\n\nvar cache = {};\n\nmodule.exports = {\n    worldApi: function () {\n        if (!cache.worldApi) {\n            cache.worldApi = new F.service.World(env.get());\n        }\n\n        return cache.worldApi;\n    },\n\n    memberApi: function () {\n        if (!cache.memberApi) {\n            cache.memberApi = new F.service.Member(_.pick(env.get(), ['groupId', 'server']));\n        }\n\n        return cache.memberApi;\n    },\n\n    userApi: function () {\n        if (!cache.userApi) {\n            cache.userApi = new F.service.User(_.pick(env.get(), ['account', 'server']));\n        }\n\n        return cache.userApi;\n    }\n};","exports[\"edit-user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape, __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id )) == null ? '' : __t) +\n'\"</td>\\n<td></td>\\n<td>\\n    <select name=\"worlds\" class=\"form-control\" data-field=\"world\">\\n\\n    ';\n _.each(worlds, function (w) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( w )) == null ? '' : __t) +\n'\" ' +\n((__t = ( w === world ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( w )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n        <option value=\"' +\n((__t = ( newWorld )) == null ? '' : __t) +\n'\" class=\"new-world-text\"><i>' +\n((__t = ( newWorld )) == null ? '' : __t) +\n' - New -</i></option>\\n    </select>\\n</td>\\n<td>\\n    <select name=\"roles\" class=\"form-control\" data-field=\"role\">\\n    ';\n _.each(roles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n\\n    ';\n _.each(optionalRoles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n' <i>(Optional)</i></option>\\n    ';\n }); ;\n__p += '\\n    </select>\\n</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\">\\n    <button class=\"btn btn-primary btn-tools btn-save save\">Save</button>\\n    <button class=\"btn btn-tools btn-cancel cancel\">Cancel</button>\\n</td>';\n\n}\nreturn __p\n};\nexports[\"user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape;\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id)) == null ? '' : __t) +\n'\"</td>\\n<td>' +\n((__t = ( !isWorldComplete ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( world )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( role )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\"><button class=\"btn edit btn-edit btn-tools auto-hide\">Edit</button></td>';\n\n}\nreturn __p\n};","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar serviceLocator = require('./service-locator');\n\n\nmodule.exports = classFrom(Base, {\n    defaults: {\n        world: '',\n        role: '',\n        active: true,\n        isWorldComplete: true,\n        firstName: '',\n        lastName: ''\n    },\n\n    makeActive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', true);\n\n        return memberApi.makeUserActive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    },\n\n    makeInactive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', false);\n\n        return memberApi.makeUserInactive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    }\n\n});\n","'use strict';\n\nvar classFrom = require('../../../util/inherit');\n\nvar Model = require('./user-model');\nvar Base = require('./base-collection');\nvar env = require('./defaults');\nvar serviceLocator = require('./service-locator');\n\n\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    sortFn: function (a, b) {\n        var aw = a.get('world').toLowerCase();\n        var bw = b.get('world').toLowerCase();\n        if (aw !== bw) {\n            return aw < bw ? -1 : 1;\n        }\n\n        return b.get('userName') > a.get('userName') ? -1 : 1;\n    },\n\n    initialize: function () {\n        $.ajaxSetup({\n            headers: {\n                Authorization: 'Bearer ' + env.get().token\n            }\n        });\n    },\n\n    allUsersAssigned: function () {\n        return this.all(function (u) {\n            return !!u.get('world');\n        });\n    },\n\n    getUnassignedUsersCount: function () {\n        return this.filter(function (u) {\n            return !u.get('world');\n        }).length;\n    },\n\n    fetch: function () {\n        var dtd = $.Deferred();\n        var _this = this;\n        var groupId = env.get().groupId;\n\n        var getGroupUsers = function () {\n            var memberApi = serviceLocator.memberApi();\n            var userApi = serviceLocator.userApi();\n\n            var loadGroupMembers = function () {\n                return memberApi.getGroupDetails();\n            };\n\n            var loadUsersInfo = function (group) {\n                var nonFacAndActive = function (u) { return u.active && u.role !== 'facilitator'; };\n                var users = _.pluck(_.filter(group.members, nonFacAndActive), 'userId');\n                return userApi.get({ id: users });\n            };\n\n            return loadGroupMembers()\n                .then(loadUsersInfo)\n                .fail(dtd.reject);\n        };\n\n        getGroupUsers()\n            .then(function (users) {\n                users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); });\n                _this.set(users);\n                dtd.resolve(users, _this);\n            });\n\n        return dtd.promise();\n    }\n\n});\n","'use strict';\nvar serviceLocator = require('./service-locator');\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    defaults: {\n        users: null,\n        model: 'model.eqn'\n    },\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n\n        this._data.users = this._data.users || [];\n\n        this._worldApi = serviceLocator.worldApi();\n\n        var id = this.get('id');\n        if (id) {\n            this._worldApi.updateConfig({ filter: id });\n        }\n    },\n\n    addUser: function (user) {\n        var users = this.get('users');\n        users.push(user);\n\n        return this.save();\n    },\n\n    removeUser: function (user) {\n        var id = this.get('id');\n        var checkWorld = function () {\n            if (!this.get('users').length) {\n                this.remove();\n                return this._worldApi.updateConfig({ filter: id }).delete();\n            }\n        }.bind(this);\n\n        _.remove(this.get('users'), function (u) {\n            return u.get('id') === user.get('id');\n        });\n\n        return this._worldApi\n            .updateConfig({ filter: id })\n            .removeUser({ userId: user.get('id') })\n            .then(checkWorld);\n    },\n\n    save: function () {\n        var _this = this;\n        var mapUsers = function () {\n            return _.map(this.get('users'), function (u) {\n                var res = { userId: u.get('id') };\n                var role = u.get('role');\n\n                if (role) {\n                    res.role = role;\n                }\n\n                return res;\n            });\n        }.bind(this);\n\n        var createWorld = _.partial(this._worldApi.create, this.pick(['model', 'name', 'minUsers']));\n        var addUsers = _.partial(_this._worldApi.addUsers, mapUsers(), { filter: _this.get('id') });\n        var savedUsers = this.get('users');\n        if (this.isNew()) {\n            // we need to create the world in the API and then add the users\n            return createWorld()\n                .then(function (world) {\n                    _this.set(world);\n                    _this._worldApi.updateConfig({ filter: world.id });\n                })\n                .then(addUsers)\n                .then(function (users) {\n                    // since we re-set the world, re-set the users\n                    _this.set('users', savedUsers);\n                });\n        } else {\n            // the world is already created just add the users\n            return addUsers();\n        }\n    },\n\n    isNew: function () {\n        return !this.get('lastModified');\n    }\n\n});","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Model = require('./world-model');\nvar UserModel = require('./user-model');\nvar serviceLocator = require('./service-locator');\n\nvar Base = require('./base-collection');\nvar __super = Base.prototype;\n\nvar doneFn = function (dtd, after) {\n    return _.after(after, dtd.resolve);\n};\n\nvar worldApi;\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n        worldApi = serviceLocator.worldApi();\n    },\n\n    autoAssignAll: function (options) {\n        return worldApi.autoAssign(options)\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    getIncompleteWorldsCount: function () {\n        return this.filter(function (w) {\n            return !w.get('complete');\n        }).length;\n    },\n\n    updateUser: function (user) {\n        var worldName = user.get('world');\n        var dtd = $.Deferred();\n        var prevWorld = this.getWorldByUser(user);\n        var curWorld = this.getOrCreateWorld(worldName);\n        var done = doneFn(dtd, 1);\n\n        // check if there's anything to do\n        if (!prevWorld && !curWorld) {\n            return dtd.resolve().promise();\n        }\n\n        if (prevWorld) {\n            prevWorld.removeUser(user)\n                .then(function () {\n                    if (curWorld) {\n                        return curWorld.addUser(user);\n                    }\n                })\n                .then(done);\n        } else if (curWorld) {\n            curWorld.addUser(user)\n                .then(done);\n        }\n\n        return dtd.promise();\n    },\n\n    getOrCreateWorld: function (worldName) {\n        if (!worldName) {\n            return;\n        }\n\n        var world = this.getWordByName(worldName);\n\n        if (!world) {\n            world = this.create({ name: worldName });\n        }\n\n        return world;\n    },\n\n    getWordByName: function (worldName) {\n        return this.find(function (world) {\n            return world.get('name') === worldName;\n        });\n    },\n\n    getWorldByUser: function (user) {\n        if (!user.get) {\n            throw new Error('getWorldByUser expectes a model (' + user + ')');\n        }\n\n        var id = user.get('id');\n        return this.getWorldByUserId(id);\n    },\n\n    getWorldByUserId: function (userId) {\n        return this.find(function (world) {\n            return _.find(world.get('users'), function (u) {\n                return u.get('id') === userId;\n            });\n        });\n    },\n\n    getWorldNames: function () {\n        return this.pluck('name');\n    },\n\n    getNextWorldName: function () {\n        var pad = function (num, places) {\n            var zeros = '000000000000000000';\n            var digits = num.toString().length;\n            var needed = places - digits;\n            return zeros.substr(0, needed) + num;\n        };\n\n        var worlds = this.getWorldNames();\n\n        if (!worlds.length) {\n            return 'World001';\n        }\n\n        var properNames = _.filter(worlds, function (w) { return /World\\d\\d\\d/.test(w); }).sort();\n        var lastWorld = properNames[properNames.length - 1];\n        var numWorld = +lastWorld.match(/World(\\d\\d\\d)/)[1];\n        return 'World' + pad(numWorld + 1, 3);\n    },\n\n    setUsersCollection: function (usersCollection) {\n        this.usersCollection = usersCollection;\n    },\n\n    joinUsers: function () {\n        var usersHash = {};\n        var usersCollection = this.usersCollection;\n        usersCollection.each(function (u) {\n            u.set({ isWorldComplete: true });\n            return (usersHash[u.get('id')] = u);\n        });\n\n        this.each(function (w, i) {\n            var name = w.get('name');\n            var isComplete = w.get('complete');\n            w.set({ index: i, name: name || (i + 1) + '' });\n            _.each(w.get('users'), function (u) {\n                if (usersHash[u.get('userId')]) {\n                    usersHash[u.get('userId')].set({ world: name, role: u.get('role'), isWorldComplete: isComplete });\n                }\n            });\n        }, this);\n\n        usersCollection.sort();\n    },\n\n    fetch: function () {\n        return worldApi.list()\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    parse: function (worlds) {\n        if (worlds.length) {\n            worlds = _.map(worlds, function (w) {\n                var users = _.map(w.users, function (u) {\n                    // in the world api users Ids comes as userId\n                    // make sure we add it as id so we can use the\n                    // same code to access models that come from the\n                    // member/local api as with the world api\n                    u.id = u.userId;\n                    return new UserModel(u);\n                });\n\n                w.users = users;\n\n                return w;\n            });\n        }\n\n        return worlds;\n    }\n});\n","'use strict';\n\n/**\n* Utility class to make ajax calls sequencial\n*/\nfunction AjaxQueue () {\n    this.queue = [];\n}\n\n$.extend(AjaxQueue.prototype, {\n    add: function (fn) {\n        return this.queue.push(fn);\n    },\n\n    execute: function (context) {\n        var dtd = $.Deferred();\n        var _this = this;\n        context = context || this;\n\n        function next() {\n            if (_this.queue.length) {\n                var fn = _this.queue.shift();\n\n                fn.call(context)\n                    .then(next)\n                    .fail(dtd.reject);\n            } else {\n                dtd.resolve();\n            }\n        }\n\n        next();\n\n        return dtd.promise();\n    }\n});\n\n\nmodule.exports = AjaxQueue;","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j<obj.length; j++) {\n        if (!(current = obj[j])) {\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        /*jshint -W089 */\n        for (var key in current) {\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n"]} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/components/assignment/js/assignment-row.js","src/components/assignment/js/assignment.js","src/components/assignment/js/base-collection.js","src/components/assignment/js/base-model.js","src/components/assignment/js/defaults.js","src/components/assignment/js/index.js","src/components/assignment/js/project-model.js","src/components/assignment/js/service-locator.js","src/components/assignment/js/templates.js","src/components/assignment/js/user-model.js","src/components/assignment/js/users-collection.js","src/components/assignment/js/world-model.js","src/components/assignment/js/worlds-collection.js","src/util/ajax-queue.js","src/util/inherit.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nvar templates = require('./templates');\n\nvar AssignmentRow = function (options) {\n    this.$el = $('<tr>');\n    this.el = this.$el[0];\n    this.$ = _.partialRight($, this.$el);\n\n    this.model = options.model;\n    this.options = options;\n    this.worlds = options.worlds;\n    this.project = options.project;\n\n    _.bindAll(this, ['setEditMode', 'removeEditMode', 'saveEdit', 'cancelEdit', 'updateData']);\n\n    this.bindEvents();\n\n};\n\n_.extend(AssignmentRow.prototype, {\n\n    template: templates['user-row'],\n\n    editTemplate: templates['edit-user-row'],\n\n    bindEvents: function () {\n        this.$el.on('click', 'button.edit', this.setEditMode);\n        this.$el.on('click', 'button.save', this.saveEdit);\n        this.$el.on('click', 'button.cancel', this.cancelEdit);\n    },\n\n    remove: function () {\n        this.$el.off('click', null, null);\n        // this only gives a delay to remove the tr\n        // animation of height of the tr does not work\n        this.$(':checkbox').attr('checked', false);\n        this.$el\n            .css({ opacity: 0.3 })\n            .animate({ height: 0 }, {\n                duration: 300,\n                complete: function () {\n                    this.remove();\n                }\n            });\n    },\n\n    makeInactive: function () {\n        return this.model.makeInactive();\n    },\n\n    setEditMode: function () {\n        this.model.set('edit-mode', true);\n        this.render();\n    },\n\n    removeEditMode: function () {\n        this.model.set('edit-mode', false);\n        this.render();\n    },\n\n    saveEdit: function () {\n        var _this = this;\n        this.updateData();\n        this.worlds\n            .updateUser(this.model)\n            .then(function () {\n                _this.removeEditMode();\n                _this.$el.trigger('update', _this);\n            });\n    },\n\n    cancelEdit: function () {\n        this.removeEditMode();\n    },\n\n    render: function () {\n        var templ = this.model.get('edit-mode') ? this.editTemplate : this.template;\n        var vm = _.extend({\n            roles: this.project.get('roles'),\n            optionalRoles: this.project.get('optionalRoles'),\n            worlds: this.worlds.getWorldNames(),\n            newWorld: this.worlds.getNextWorldName()\n        }, this.model.toJSON());\n\n        this.$el.html(templ(vm));\n\n        return this;\n    },\n\n    updateData: function () {\n        var _this = this;\n        this.$('[data-field]').each(function () {\n            var el = $(this);\n            var field = el.data('field');\n            var val = el.val();\n\n            _this.model.set(field, val);\n        });\n    }\n});\n\n\nmodule.exports = AssignmentRow;","'use strict';\n\nvar UsersCollection = require('./users-collection');\nvar WorldsCollection = require('./worlds-collection');\nvar ProjectModel = require('./project-model');\nvar AssignemntRow = require('./assignment-row');\nvar env = require('./defaults');\nvar AjaxQueue = require('../../../util/ajax-queue');\n\nfunction setEnvironment(options) {\n    env.set(_.omit(options, 'el'));\n}\n\nvar Assignment = function (options) {\n    setEnvironment(options);\n    this.initialize(options);\n};\n\nAssignment.prototype = {\n\n    initialize: function (options) {\n        this.el = typeof options.el === 'string' ? $(options.el)[0] : options.el;\n        this.$el = $(this.el);\n        this.$ = _.partialRight($, this.el);\n\n        this.users = new UsersCollection();\n        this.worlds = new WorldsCollection();\n        this.project = new ProjectModel();\n\n        _.bindAll(this, ['render', 'renderTable', 'toggleControlls', 'saveEdit', 'selectAll', 'usassignSelected', '_showUpdating', '_hideUpdating', 'autoAssignAll', 'makeUserInactive']);\n\n        this.bindEvents();\n    },\n\n    bindEvents: function () {\n        this.$el.on('update', 'tr', this.saveEdit);\n        this.$el.on('click', 'input:checkbox:not(#select-all)', this.toggleControlls);\n        this.$el.on('click', '#select-all', this.selectAll);\n        this.$el.on('click', '.unassign-user', this.usassignSelected);\n        this.$el.on('click', '.auto-assign-all', this.autoAssignAll);\n        this.$el.on('click', '.make-user-inactive', this.makeUserInactive);\n    },\n\n    load: function () {\n\n        var join = function () {\n            this.worlds.setUsersCollection(this.users);\n            this.worlds.joinUsers();\n            this.render();\n        }.bind(this);\n\n        return $.when(\n            this.worlds.fetch(),\n            this.users.fetch(),\n            this.project.fetch()\n        ).then(join);\n\n    },\n\n    saveEdit: function () {\n        this.worlds.fetch()\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n                this.updateControls();\n            }.bind(this));\n    },\n\n    autoAssignAll: function () {\n        this._showUpdating();\n        var maxUsers = +this.$('#max-users').val();\n        return this.worlds.autoAssignAll({ maxUsers: maxUsers })\n            .then(this._hideUpdating)\n            .fail(this._hideUpdating)\n            .then(function () {\n                this.worlds.joinUsers();\n                this.render();\n            }.bind(this));\n    },\n\n    getSelectedIds: function () {\n        return this.$('tbody :checkbox:checked').map(function () {\n            return $(this).data('id');\n        });\n    },\n\n    findRowViews: function (ids) {\n        return _.map(ids, function (id) {\n            return this.rowViews[id];\n        }, this);\n    },\n\n    unassignUsers: function (ids) {\n        var dtd = $.Deferred();\n        var done = function () {\n            dtd.resolve();\n        };\n\n        // for now we need to sequence the calls to unassign users from worlds\n        var queue = new AjaxQueue();\n\n        _.each(ids, function (userId) {\n            var user = this.users.getById(userId);\n            user.set('world', '');\n            user.set('role', '');\n            queue.add(_.partial(_.bind(this.worlds.updateUser, this.worlds), user));\n        }, this);\n\n        queue.execute(this).then(done);\n\n        return dtd.promise();\n    },\n\n    usassignSelected: function (e) {\n        e.preventDefault();\n\n        var ids = this.getSelectedIds();\n\n        var done = function () {\n            this.worlds.fetch().then(function () {\n                this.worlds.joinUsers();\n                this._hideUpdating();\n                this.render();\n\n            }.bind(this));\n        }.bind(this);\n\n        this._showUpdating();\n\n        return this.unassignUsers(ids).then(done);\n    },\n\n    makeUserInactive: function (e) {\n        e.preventDefault();\n        var ids = this.getSelectedIds();\n        var done = function () {\n            this.toggleControlls();\n        }.bind(this);\n\n        var makeUsersInactive = function () {\n            var rows = this.findRowViews(ids);\n            // for now we need to sequence the calls to patch the users\n            // since the API can only operate on one call per group at a time\n            var queue = new AjaxQueue();\n            _.each(rows, function (view) {\n                var user = view.model;\n                queue.add(function () {\n                    return view.makeInactive()\n                        .then(function () {\n                            user.remove();\n                            view.remove();\n                        });\n                    });\n\n            }, this);\n\n            queue.execute(this).then(done);\n        }.bind(this);\n\n        return this.unassignUsers(ids)\n            .then(makeUsersInactive);\n\n\n    },\n\n    render: function () {\n        this.$('table tbody').empty();\n        this.renderTable();\n        this.toggleControlls();\n    },\n\n    renderTable: function () {\n        this.rowViews = {};\n        var rows = [];\n        this.users.each(function (u) {\n            var view = new AssignemntRow({ model: u, worlds: this.worlds, project: this.project });\n            this.rowViews[u.get('id')] = view;\n            rows.push(view.render().el);\n        }, this);\n\n        this.$('table tbody').append(rows);\n    },\n\n\n    updateControls: function () {\n        this.updateControlsForSelection();\n        this.updateAutoAssignButton();\n        this.updateStatus();\n    },\n\n    updateStatus: function () {\n        var incolpleteWorlds = this.worlds.getIncompleteWorldsCount();\n        var unassignedUsers = this.users.getUnassignedUsersCount();\n        var totalWorlds = this.worlds.size();\n\n        var usersText = unassignedUsers ? unassignedUsers === 1 ? '1 user needs assignment.' : unassignedUsers + ' users need assignment.' : 'All users have been assigned.';\n        var worldsText = !totalWorlds ? 'No worlds have been created.' : !incolpleteWorlds ? 'All worlds are complete.' : incolpleteWorlds === 1 ? '1 incomplete world needs attention.' : incolpleteWorlds + ' incomplete worlds need attention.';\n\n        this.$('#users-status .text').text(usersText);\n        this.$('#worlds-status .text').text(worldsText);\n\n        if (unassignedUsers) {\n            this.$('#users-status').addClass('incomplete');\n        } else {\n            this.$('#users-status').removeClass('incomplete');\n        }\n\n        if (incolpleteWorlds || !totalWorlds) {\n            this.$('#worlds-status').addClass('incomplete');\n        } else {\n            this.$('#worlds-status').removeClass('incomplete');\n        }\n\n        this.$('.status-widget').addClass('visible');\n    },\n\n    updateControlsForSelection: function () {\n        var numSelected = this.$('tbody :checkbox:checked').length;\n        this.$('.component.controls')[numSelected ? 'addClass' : 'removeClass']('visible');\n    },\n\n    updateAutoAssignButton: function () {\n\n        if (this.project.isDynamicAssignment()) {\n            var hasRoles = this.project.hasRoles();\n            this.$('.table-controls .single').hide();\n            this.$('.table-controls .dynamic').show();\n            this.$('.table-controls .dynamic-no-roles-text')[hasRoles ? 'hide' : 'show']();\n            this.$('.table-controls .no-roles')[hasRoles ? 'hide' : 'show']();\n        } else {\n            this.$('.table-controls .dynamic').hide();\n            this.$('.table-controls .dynamic-no-roles-text').hide();\n            this.$('.table-controls .single').show();\n            this.$('.table-controls .no-roles').show();\n\n        }\n\n        if (this.users.allUsersAssigned()) {\n            this.$('.table-controls').removeClass('visible');\n        } else {\n            this.$('.table-controls').addClass('visible');\n        }\n    },\n\n    selectAll: function (e) {\n        this.$('tbody :checkbox').prop('checked', e.target.checked);\n        this.updateControls();\n    },\n\n    toggleControlls: function (e) {\n        var total = this.$('tbody :checkbox');\n        var checked = this.$('tbody :checkbox:checked');\n\n        if (total.length === checked.length) {\n            this.$('#select-all').attr('checked', 'checked');\n        } else {\n            this.$('#select-all').removeAttr('checked');\n        }\n\n        this.updateControls();\n    },\n\n    _showUpdating: function () {\n        this.$el.css({ opacity: 0.4 });\n    },\n\n    _hideUpdating: function () {\n        this.$el.css({ opacity: 1 });\n    }\n\n};\n\nmodule.exports = Assignment;","'use strict';\n\nvar BaseCollection = function (models, options) {\n    this._models = [];\n    this.options = options;\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseCollection.prototype, {\n    idAttribute: 'id',\n\n    initialize: function (models, options) {\n    },\n\n    create: function (attr, options) {\n        var m = new this.model(attr, options);\n        this.set(m);\n        return m;\n    },\n\n    reset: function (models, options) {\n        this._models.length = 0;\n        this.set(models);\n    },\n\n    remove: function (model) {\n        _.remove(this._models, function (m) {\n            return m === model;\n        });\n\n        delete model.collection;\n\n        return model;\n    },\n\n    set: function (models) {\n        if (!models) {\n            return;\n        }\n\n        models = [].concat(models);\n\n        if (!models.length) {\n            return;\n        }\n\n        _.each(models, function (m) {\n            if (!(m instanceof this.model)) {\n                m = new this.model(m);\n            }\n\n            m.collection = this;\n\n            this._models.push(m);\n        }, this);\n\n        this.sort.call(this);\n\n        return models;\n    },\n\n    sortFn: function (a, b) {\n        return b._data[this.idAttribute] - a._data[this.idAttribute];\n    },\n\n    sort: function () {\n        this._models = this._models.sort(this.sortFn.bind(this));\n\n        return this._models;\n    },\n\n    getById: function (id) {\n        return _.find(this._models, function (m) {\n            return m.get(this.idAttribute) === id;\n        }, this);\n    },\n\n    each: function (cb, ctx) {\n        return _.each(this._models, cb, ctx || this);\n    },\n\n    all: function (cb, ctx) {\n        return _.all(this._models, cb, ctx || this);\n    },\n\n    toJSON: function () {\n        return _.invoke(this._models, 'toJSON');\n    },\n\n    find: function (fn) {\n        return _.find(this._models, fn);\n    },\n\n    filter: function (fn) {\n        return _.filter(this._models, fn);\n    },\n\n    size: function () {\n        return this._models.length;\n    },\n\n    map: function (fn, ctx) {\n        return _.map(this._models, function (model) {\n            return fn.call(ctx, model.toJSON());\n        });\n    },\n\n    pluck: function (field) {\n        return this.map(function (m) {\n            return m[field];\n        });\n    }\n\n});\n\nmodule.exports = BaseCollection;","'use strict';\n\n\nvar BaseModel = function (attr, options) {\n    attr = _.defaults({}, attr, _.result(this, 'defaults'));\n    this._data = {};\n    this.set(attr, options);\n    this.initialize.apply(this, arguments);\n};\n\n_.extend(BaseModel.prototype, {\n    initialize: function (attr, options) {\n\n    },\n\n    set: function (key, val, options) {\n\n        if (key == null) {\n            return this;\n        }\n\n        var attrs;\n        if (typeof key === 'object') {\n            attrs = key;\n            options = val;\n        } else {\n            (attrs = {})[key] = val;\n        }\n\n        options = options || {};\n\n        _.extend(this._data, attrs);\n\n        return this;\n    },\n\n    get: function (key, options) {\n        return this._data[key];\n    },\n\n    remove: function () {\n        if (this.collection) {\n            this.collection.remove(this);\n        }\n\n        return this;\n    },\n\n    toJSON: function () {\n        return this._data;\n    },\n\n    pick: function (keys) {\n        return _.pick(this._data, keys);\n    }\n\n});\n\nmodule.exports = BaseModel;","'use strict';\n\nvar env = {\n    account: '',\n    project: '',\n    group: '',\n    groupId: '',\n    token: '',\n    server: {\n        host: 'api.forio.com',\n        protocol: 'https'\n    }\n};\n\nmodule.exports = {\n    set: function (options) {\n        env = _.merge(env, options);\n    },\n\n    get: function () {\n        return env;\n    }\n};","(function () {\n    'use strict';\n    var App = require('./assignment.js');\n\n    window.forio = window.forio || {};\n    window.forio.MultiplayerAssignmentComponent = App;\n})();\n","'use strict';\n\nvar serviceLocator = require('./service-locator');\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\n// var __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    isDynamicAssignment: function () {\n        return this.get('worlds') === 'dynamic';\n    },\n\n    hasRoles: function () {\n        var roles = this.get('roles');\n        return roles && !!roles.length;\n    },\n\n    fetch: function () {\n        var api = serviceLocator.worldApi();\n\n        return api.getProjectSettings().then(function (settings) {\n            this.set(settings);\n        }.bind(this));\n    }\n});","'use strict';\n\nvar env = require('./defaults.js');\n\nvar cache = {};\n\nmodule.exports = {\n    worldApi: function () {\n        if (!cache.worldApi) {\n            cache.worldApi = new F.service.World(env.get());\n        }\n\n        return cache.worldApi;\n    },\n\n    memberApi: function () {\n        if (!cache.memberApi) {\n            cache.memberApi = new F.service.Member(_.pick(env.get(), ['groupId', 'server']));\n        }\n\n        return cache.memberApi;\n    },\n\n    userApi: function () {\n        if (!cache.userApi) {\n            cache.userApi = new F.service.User(_.pick(env.get(), ['account', 'server']));\n        }\n\n        return cache.userApi;\n    }\n};","exports[\"edit-user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape, __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id )) == null ? '' : __t) +\n'\"</td>\\n<td></td>\\n<td>\\n    <select name=\"worlds\" class=\"form-control\" data-field=\"world\">\\n\\n    ';\n _.each(worlds, function (w) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( w )) == null ? '' : __t) +\n'\" ' +\n((__t = ( w === world ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( w )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n        <option value=\"' +\n((__t = ( newWorld )) == null ? '' : __t) +\n'\" class=\"new-world-text\"><i>' +\n((__t = ( newWorld )) == null ? '' : __t) +\n' - New -</i></option>\\n    </select>\\n</td>\\n<td>\\n    <select name=\"roles\" class=\"form-control\" data-field=\"role\">\\n    ';\n _.each(roles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n'</option>\\n    ';\n }); ;\n__p += '\\n\\n    ';\n _.each(optionalRoles, function (r) { ;\n__p += '\\n        <option value=\"' +\n((__t = ( r )) == null ? '' : __t) +\n'\" ' +\n((__t = ( r === role ? 'selected' : '' )) == null ? '' : __t) +\n'>' +\n((__t = ( r )) == null ? '' : __t) +\n' <i>(Optional)</i></option>\\n    ';\n }); ;\n__p += '\\n    </select>\\n</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\">\\n    <button class=\"btn btn-primary btn-tools btn-save save\">Save</button>\\n    <button class=\"btn btn-tools btn-cancel cancel\">Cancel</button>\\n</td>';\n\n}\nreturn __p\n};\nexports[\"user-row\"] = function(obj) {\nobj || (obj = {});\nvar __t, __p = '', __e = _.escape;\nwith (obj) {\n__p += '<td><input type=\"checkbox\" class=\"select\" data-id=\"' +\n((__t = ( id)) == null ? '' : __t) +\n'\"</td>\\n<td>' +\n((__t = ( !isWorldComplete ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( world )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( role )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( lastName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( userName )) == null ? '' : __t) +\n'</td>\\n<td>' +\n((__t = ( !world ? '<em class=\"f-icon f-warning\"></em>' : '' )) == null ? '' : __t) +\n'</td>\\n<td class=\"actions\"><button class=\"btn edit btn-edit btn-tools auto-hide\">Edit</button></td>';\n\n}\nreturn __p\n};","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar serviceLocator = require('./service-locator');\n\n\nmodule.exports = classFrom(Base, {\n    defaults: {\n        world: '',\n        role: '',\n        active: true,\n        isWorldComplete: true,\n        firstName: '',\n        lastName: ''\n    },\n\n    makeActive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', true);\n\n        return memberApi.makeUserActive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    },\n\n    makeInactive: function () {\n        var memberApi = serviceLocator.memberApi();\n        var params = {\n            userId: this.get('id'),\n            groupId: this.get('groupId')\n        };\n\n        var original = this.get('active');\n        this.set('active', false);\n\n        return memberApi.makeUserInactive(params)\n            .fail(function () {\n                // revert the change\n                this.set('active', original);\n            }.bind(this));\n    }\n\n});\n","'use strict';\n\nvar classFrom = require('../../../util/inherit');\n\nvar Model = require('./user-model');\nvar Base = require('./base-collection');\nvar env = require('./defaults');\nvar serviceLocator = require('./service-locator');\n\n\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    sortFn: function (a, b) {\n        var aw = a.get('world').toLowerCase();\n        var bw = b.get('world').toLowerCase();\n        if (aw !== bw) {\n            return aw < bw ? -1 : 1;\n        }\n\n        return b.get('userName') > a.get('userName') ? -1 : 1;\n    },\n\n    initialize: function () {\n        $.ajaxSetup({\n            headers: {\n                Authorization: 'Bearer ' + env.get().token\n            }\n        });\n    },\n\n    allUsersAssigned: function () {\n        return this.all(function (u) {\n            return !!u.get('world');\n        });\n    },\n\n    getUnassignedUsersCount: function () {\n        return this.filter(function (u) {\n            return !u.get('world');\n        }).length;\n    },\n\n    fetch: function () {\n        var dtd = $.Deferred();\n        var _this = this;\n        var groupId = env.get().groupId;\n\n        var getGroupUsers = function () {\n            var memberApi = serviceLocator.memberApi();\n            var userApi = serviceLocator.userApi();\n\n            var loadGroupMembers = function () {\n                return memberApi.getGroupDetails();\n            };\n\n            var loadUsersInfo = function (group) {\n                var nonFacAndActive = function (u) { return u.active && u.role !== 'facilitator'; };\n                var users = _.pluck(_.filter(group.members, nonFacAndActive), 'userId');\n                return userApi.get({ id: users });\n            };\n\n            return loadGroupMembers()\n                .then(loadUsersInfo)\n                .fail(dtd.reject);\n        };\n\n        getGroupUsers()\n            .then(function (users) {\n                users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); });\n                _this.set(users);\n                dtd.resolve(users);\n            });\n\n        return dtd.promise();\n    }\n\n});\n","'use strict';\nvar serviceLocator = require('./service-locator');\nvar classFrom = require('../../../util/inherit');\nvar Base = require('./base-model');\nvar __super = Base.prototype;\n\nmodule.exports = classFrom(Base, {\n\n    defaults: {\n        users: null,\n        model: 'model.eqn'\n    },\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n\n        this._data.users = this._data.users || [];\n\n        this._worldApi = serviceLocator.worldApi();\n\n        var id = this.get('id');\n        if (id) {\n            this._worldApi.updateConfig({ filter: id });\n        }\n    },\n\n    addUser: function (user) {\n        var users = this.get('users');\n        users.push(user);\n\n        return this.save();\n    },\n\n    removeUser: function (user) {\n        var id = this.get('id');\n        var checkWorld = function () {\n            if (!this.get('users').length) {\n                this.remove();\n                return this._worldApi.updateConfig({ filter: id }).delete();\n            }\n        }.bind(this);\n\n        _.remove(this.get('users'), function (u) {\n            return u.get('id') === user.get('id');\n        });\n\n        return this._worldApi\n            .updateConfig({ filter: id })\n            .removeUser({ userId: user.get('id') })\n            .then(checkWorld);\n    },\n\n    save: function () {\n        var _this = this;\n        var mapUsers = function () {\n            return _.map(this.get('users'), function (u) {\n                var res = { userId: u.get('id') };\n                var role = u.get('role');\n\n                if (role) {\n                    res.role = role;\n                }\n\n                return res;\n            });\n        }.bind(this);\n\n        var createWorld = _.partial(this._worldApi.create, this.pick(['model', 'name', 'minUsers']));\n        var addUsers = _.partial(_this._worldApi.addUsers, mapUsers(), { filter: _this.get('id') });\n        var savedUsers = this.get('users');\n        if (this.isNew()) {\n            // we need to create the world in the API and then add the users\n            return createWorld()\n                .then(function (world) {\n                    _this.set(world);\n                    _this._worldApi.updateConfig({ filter: world.id });\n                })\n                .then(addUsers)\n                .then(function (users) {\n                    // since we re-set the world, re-set the users\n                    _this.set('users', savedUsers);\n                });\n        } else {\n            // the world is already created just add the users\n            return addUsers();\n        }\n    },\n\n    isNew: function () {\n        return !this.get('lastModified');\n    }\n\n});","'use strict';\n\nvar classFrom = require('../../../util/inherit');\nvar Model = require('./world-model');\nvar UserModel = require('./user-model');\nvar serviceLocator = require('./service-locator');\n\nvar Base = require('./base-collection');\nvar __super = Base.prototype;\n\nvar doneFn = function (dtd, after) {\n    return _.after(after, dtd.resolve);\n};\n\nvar worldApi;\n\nmodule.exports = classFrom(Base, {\n    model: Model,\n\n    initialize: function () {\n        __super.initialize.apply(this, arguments);\n        worldApi = serviceLocator.worldApi();\n    },\n\n    autoAssignAll: function (options) {\n        return worldApi.autoAssign(options)\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    getIncompleteWorldsCount: function () {\n        return this.filter(function (w) {\n            return !w.get('complete');\n        }).length;\n    },\n\n    updateUser: function (user) {\n        var worldName = user.get('world');\n        var dtd = $.Deferred();\n        var prevWorld = this.getWorldByUser(user);\n        var curWorld = this.getOrCreateWorld(worldName);\n        var done = doneFn(dtd, 1);\n\n        // check if there's anything to do\n        if (!prevWorld && !curWorld) {\n            return dtd.resolve().promise();\n        }\n\n        if (prevWorld) {\n            prevWorld.removeUser(user)\n                .then(function () {\n                    if (curWorld) {\n                        return curWorld.addUser(user);\n                    }\n                })\n                .then(done);\n        } else if (curWorld) {\n            curWorld.addUser(user)\n                .then(done);\n        }\n\n        return dtd.promise();\n    },\n\n    getOrCreateWorld: function (worldName) {\n        if (!worldName) {\n            return;\n        }\n\n        var world = this.getWordByName(worldName);\n\n        if (!world) {\n            world = this.create({ name: worldName });\n        }\n\n        return world;\n    },\n\n    getWordByName: function (worldName) {\n        return this.find(function (world) {\n            return world.get('name') === worldName;\n        });\n    },\n\n    getWorldByUser: function (user) {\n        if (!user.get) {\n            throw new Error('getWorldByUser expectes a model (' + user + ')');\n        }\n\n        var id = user.get('id');\n        return this.getWorldByUserId(id);\n    },\n\n    getWorldByUserId: function (userId) {\n        return this.find(function (world) {\n            return _.find(world.get('users'), function (u) {\n                return u.get('id') === userId;\n            });\n        });\n    },\n\n    getWorldNames: function () {\n        return this.pluck('name');\n    },\n\n    getNextWorldName: function () {\n        var pad = function (num, places) {\n            var zeros = '000000000000000000';\n            var digits = num.toString().length;\n            var needed = places - digits;\n            return zeros.substr(0, needed) + num;\n        };\n\n        var worlds = this.getWorldNames();\n\n        if (!worlds.length) {\n            return 'World001';\n        }\n\n        var properNames = _.filter(worlds, function (w) { return /World\\d\\d\\d/.test(w); }).sort();\n        var lastWorld = properNames[properNames.length - 1];\n        var numWorld = +lastWorld.match(/World(\\d\\d\\d)/)[1];\n        return 'World' + pad(numWorld + 1, 3);\n    },\n\n    setUsersCollection: function (usersCollection) {\n        this.usersCollection = usersCollection;\n    },\n\n    joinUsers: function () {\n        var usersHash = {};\n        var usersCollection = this.usersCollection;\n        usersCollection.each(function (u) {\n            u.set({ isWorldComplete: true });\n            return (usersHash[u.get('id')] = u);\n        });\n\n        this.each(function (w, i) {\n            var name = w.get('name');\n            var isComplete = w.get('complete');\n            w.set({ index: i, name: name || (i + 1) + '' });\n            _.each(w.get('users'), function (u) {\n                if (usersHash[u.get('userId')]) {\n                    usersHash[u.get('userId')].set({ world: name, role: u.get('role'), isWorldComplete: isComplete });\n                }\n            });\n        }, this);\n\n        usersCollection.sort();\n    },\n\n    fetch: function () {\n        return worldApi.list()\n            .then(function (worlds) {\n                this.reset(this.parse(worlds));\n            }.bind(this));\n    },\n\n    parse: function (worlds) {\n        if (worlds.length) {\n            worlds = _.map(worlds, function (w) {\n                var users = _.map(w.users, function (u) {\n                    // in the world api users Ids comes as userId\n                    // make sure we add it as id so we can use the\n                    // same code to access models that come from the\n                    // member/local api as with the world api\n                    u.id = u.userId;\n                    return new UserModel(u);\n                });\n\n                w.users = users;\n\n                return w;\n            });\n        }\n\n        return worlds;\n    }\n});\n","'use strict';\n\n/**\n* Utility class to make ajax calls sequencial\n*/\nfunction AjaxQueue () {\n    this.queue = [];\n}\n\n$.extend(AjaxQueue.prototype, {\n    add: function (fn) {\n        return this.queue.push(fn);\n    },\n\n    execute: function (context) {\n        var dtd = $.Deferred();\n        var _this = this;\n        context = context || this;\n\n        function next() {\n            if (_this.queue.length) {\n                var fn = _this.queue.shift();\n\n                fn.call(context)\n                    .then(next)\n                    .fail(dtd.reject);\n            } else {\n                dtd.resolve();\n            }\n        }\n\n        next();\n\n        return dtd.promise();\n    }\n});\n\n\nmodule.exports = AjaxQueue;","/**\n/* Inherit from a class (using prototype borrowing)\n*/\n'use strict';\n\nfunction inherit(C, P) {\n    var F = function () {};\n    F.prototype = P.prototype;\n    C.prototype = new F();\n    C.__super = P.prototype;\n    C.prototype.constructor = C;\n}\n\n/**\n* Shallow copy of an object\n*/\nvar extend = function (dest /*, var_args*/) {\n    var obj = Array.prototype.slice.call(arguments, 1);\n    var current;\n    for (var j = 0; j<obj.length; j++) {\n        if (!(current = obj[j])) {\n            continue;\n        }\n\n        // do not wrap inner in dest.hasOwnProperty or bad things will happen\n        /*jshint -W089 */\n        for (var key in current) {\n            dest[key] = current[key];\n        }\n    }\n\n    return dest;\n};\n\nmodule.exports = function (base, props, staticProps) {\n    var parent = base;\n    var child;\n\n    child = props && props.hasOwnProperty('constructor') ? props.constructor : function () { return parent.apply(this, arguments); };\n\n    // add static properties to the child constructor function\n    extend(child, parent, staticProps);\n\n    // associate prototype chain\n    inherit(child, parent);\n\n    // add instance properties\n    if (props) {\n        extend(child.prototype, props);\n    }\n\n    // done\n    return child;\n};\n"]} diff --git a/dist/components/login/login.js b/dist/components/login/login.js index 743bbf26..98cd82b0 100644 --- a/dist/components/login/login.js +++ b/dist/components/login/login.js @@ -92,8 +92,6 @@ $(function () { }) .then(function () { window.location = action; - }) - .done(function () { $('.group-selection-dialog').hide(); }); }); diff --git a/src/components/assignment/js/users-collection.js b/src/components/assignment/js/users-collection.js index 67c2e7d6..078fe3a6 100644 --- a/src/components/assignment/js/users-collection.js +++ b/src/components/assignment/js/users-collection.js @@ -70,7 +70,7 @@ module.exports = classFrom(Base, { .then(function (users) { users = _.map(users, function (u) { return _.extend(u, { groupId: groupId }); }); _this.set(users); - dtd.resolve(users, _this); + dtd.resolve(users); }); return dtd.promise(); diff --git a/src/managers/world-manager.js b/src/managers/world-manager.js index 4ef9e7a3..7601cd02 100644 --- a/src/managers/world-manager.js +++ b/src/managers/world-manager.js @@ -70,7 +70,7 @@ function buildStrategy(worldId, dtd) { return _this.runService.load(runId); }) .then(function (run) { - dtd.resolve.call(this, run, _this.runService); + dtd.resolveWith(_this, [run]); }) .fail(dtd.reject); } diff --git a/src/service/asset-api-adapter.js b/src/service/asset-api-adapter.js index 4d552453..bf18d560 100644 --- a/src/service/asset-api-adapter.js +++ b/src/service/asset-api-adapter.js @@ -302,7 +302,7 @@ module.exports = function (config) { var fullPathFiles = $.map(files, function (file) { return buildUrl(file, urlOptions); }); - dtd.resolve(fullPathFiles, me); + dtd.resolveWith(me, [fullPathFiles]); }) .fail(dtd.reject); diff --git a/src/service/world-api-adapter.js b/src/service/world-api-adapter.js index af1b07eb..89bdee36 100644 --- a/src/service/world-api-adapter.js +++ b/src/service/world-api-adapter.js @@ -585,10 +585,10 @@ module.exports = function (config) { var currentWorld = worlds[0]; if (currentWorld) { - serviceOptions.filter = currentWorld.id; + serviceOptions.filter = currentWorld.id; } - dtd.resolve(currentWorld, me); + dtd.resolveWith(me, [currentWorld]); }) .fail(dtd.reject); diff --git a/tests/spec/test-conditional-creation-strategy.js b/tests/spec/test-conditional-creation-strategy.js index f4862be8..f94b9916 100644 --- a/tests/spec/test-conditional-creation-strategy.js +++ b/tests/spec/test-conditional-creation-strategy.js @@ -33,9 +33,7 @@ var setupServer = function () { server = sinon.fakeServer.create(); - setupResponse('GET', /run\/forio-dev\/js-libs/, 200, runs || []); - server.autorespond = true; }; diff --git a/tests/spec/test-world-manager.js b/tests/spec/test-world-manager.js index 0793148f..0ea575b6 100644 --- a/tests/spec/test-world-manager.js +++ b/tests/spec/test-world-manager.js @@ -140,12 +140,9 @@ describe('getCurrentRun', function (done) { it('should return the current run object and runService of the run of the current world', function (done) { createWorldManager().getCurrentRun('model.py') - .then(function (run, runService) { + .then(function (run) { run.should.not.be.null; run.id.should.be.equal('run2'); - - runService.should.not.be.null; - done(); }) .fail(function () { From 1c94786beef5e0789c2cef143eaf74480ba7b2af Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 16:25:05 -0700 Subject: [PATCH 12/16] passing tests --- tests/spec/test-multiplayer-strategy.js | 19 +++++++++++++------ 1 file changed, 13 insertions(+), 6 deletions(-) diff --git a/tests/spec/test-multiplayer-strategy.js b/tests/spec/test-multiplayer-strategy.js index 80da769f..8ec51cc2 100644 --- a/tests/spec/test-multiplayer-strategy.js +++ b/tests/spec/test-multiplayer-strategy.js @@ -25,7 +25,8 @@ var setupResponse = function (verb, endpoint, statusCode, resp, respHeaders) { server.respondWith(verb, endpoint, function (xhr, id) { - var headers = _.extend({}, { 'Content-Type': 'application/json' }, respHeaders); + var ct = typeof resp === 'string' ? {} : { 'Content-Type': 'application/json' }; + var headers = _.extend({}, ct, respHeaders); var body = typeof resp === 'object' ? JSON.stringify(resp) : resp; xhr.respond(statusCode, headers, body); }); @@ -49,7 +50,10 @@ var setupServer = function (worlds) { server = sinon.fakeServer.create(); - + server.respondWith('GET', /(.*)\/run\/(.*)\/(.*)/, function (xhr, id) { + xhr.respond(200, { 'Content-Type': 'application/json' }, JSON.stringify({ url: xhr.url })); + return true; + }); setupResponse('GET', queryMatchers.worldEndpoint, 200, worlds || []); setupResponse('POST', /multiplayer\/world\/worldid1\/run/, 201, 'run1'); setupResponse('POST', /multiplayer\/world\/worldid2\/run/, 201, 'run2'); @@ -61,7 +65,7 @@ server.restore(); }; - describe.only('Multiplayer strategy', function () { + describe('Multiplayer strategy', function () { beforeEach(_.partial(setupServer, worldSet)); afterEach(teardownServer); @@ -84,15 +88,18 @@ describe('with world/users setup correctly', function () { it('should get the list of worlds for the current user first', function () { return createRunManager().getRun().then(function () { - var req = server.requests.pop(); + var req = server.requests[0]; req.method.toUpperCase().should.equal('GET'); req.url.should.match(queryMatchers.getWorlds); + }).fail(function () { + var s = server; + console.log(arguments); }); }); it('should post to the run endpoint after getting the world', function () { return createRunManager().getRun().then(function () { - var req = server.requests.pop(); + var req = server.requests[1]; req.method.toUpperCase().should.equal('POST'); req.url.should.match(/multiplayer\/world\/worldid2\/run/); }); @@ -102,7 +109,7 @@ describe('with two worlds for the user', function () { it('should use the latest world to retore the run', function () { return createRunManager().getRun().then(function () { - var req = server.requests.pop(); + var req = server.requests[1]; req.method.toUpperCase().should.equal('POST'); req.url.should.match(/multiplayer\/world\/worldid2/); }); From 012c0c3f15ad75c9ac0f795995767a97f0762f27 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 16:26:48 -0700 Subject: [PATCH 13/16] passing tests --- tests/spec/test-env-load.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/tests/spec/test-env-load.js b/tests/spec/test-env-load.js index dc060f0f..137f06db 100644 --- a/tests/spec/test-env-load.js +++ b/tests/spec/test-env-load.js @@ -10,7 +10,7 @@ var server; beforeEach(function () { server = sinon.fakeServer.create(); - server.autoRespond = true; + server.respondImmediately = true; }); afterEach(function () { @@ -39,7 +39,6 @@ delete F.service.URL.host; //done(); }); - //server.respond(); }); it('it should set protocol and host to api.forio.com when the config request fails', function () { @@ -54,8 +53,9 @@ F.service.URL.host.should.equal('api.forio.com'); delete F.service.URL.protocol; delete F.service.URL.host; - }).fail(callback); - callback.should.have.been.called; + }).then(null, callback).then(function () { + callback.should.have.been.called; + }); }); }); }); From e284bb730c4156c9c12248caaaebfcb120202125 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 16:32:31 -0700 Subject: [PATCH 14/16] autorespond => respondImmediately --- tests/spec/test-ajax-http-transport.js | 4 +- tests/spec/test-auth-api-service.js | 3 +- tests/spec/test-auth-manager.js | 39 ++++++++++--------- .../test-conditional-creation-strategy.js | 11 +----- tests/spec/test-group-api-service.js | 2 +- tests/spec/test-member-api-service.js | 2 +- tests/spec/test-user-api-adapter.js | 2 +- tests/spec/test-world-api-service.js | 2 +- tests/spec/test-world-manager.js | 5 +-- 9 files changed, 29 insertions(+), 41 deletions(-) diff --git a/tests/spec/test-ajax-http-transport.js b/tests/spec/test-ajax-http-transport.js index ba636417..f08041cd 100644 --- a/tests/spec/test-ajax-http-transport.js +++ b/tests/spec/test-ajax-http-transport.js @@ -13,7 +13,7 @@ server.respondWith('GET', /api\.fail/, function (xhr, id) { xhr.respond(500, { 'Content-Type': 'application/json' }, JSON.stringify({ url: xhr.url })); }); - //server.autoRespond = true; + server.respondImmediately = true; }); @@ -96,7 +96,6 @@ ajax.get({ a:2,b:3 }, { success: callback }); - server.respond(); callback.called.should.equal(true); }); it('should call fail callback on success', function () { @@ -106,7 +105,6 @@ ajax.get({ a:2,b:3 }, { error: callback }); - server.respond(); callback.called.should.equal(true); }); diff --git a/tests/spec/test-auth-api-service.js b/tests/spec/test-auth-api-service.js index aff4d0b6..c92803cf 100644 --- a/tests/spec/test-auth-api-service.js +++ b/tests/spec/test-auth-api-service.js @@ -14,7 +14,7 @@ { 'refresh_token':'snip-refresh','access_token': token,'expires':43199 } )); }); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { @@ -27,7 +27,6 @@ var as = new AuthService({ transport: { beforeSend: callback } }); as.login({ userName: 'john', password: 'y' }); - server.respond(); server.requests.pop(); callback.should.have.been.called; }); diff --git a/tests/spec/test-auth-manager.js b/tests/spec/test-auth-manager.js index f34d0122..4588eeca 100644 --- a/tests/spec/test-auth-manager.js +++ b/tests/spec/test-auth-manager.js @@ -86,26 +86,29 @@ var response = multipleGroupsResponse ? multipleGroups : singleGroup; xhr.respond(200, { 'Content-Type': 'application/json' }, JSON.stringify(response)); }); - server.autoRespond = true; + server.respondImmediately = true; }); + afterEach(function () { + server.requests = []; + }); after(function () { server.restore(); }); describe('Login', function () { - it ('It should construct the right authenticaton request', function () { + it('It should construct the right authenticaton request', function () { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', }); am.login({ userName: 'test', password: 'test' }); - var req = server.requests.pop(); + var req = server.requests[0]; req.method.toUpperCase().should.equal('POST'); req.url.should.match(/https:\/\/api\.forio\.com\/authentication\/?/); }); - it ('It should call members API on sucessful login', function (done) { + it('It should call members API on sucessful login', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -121,7 +124,7 @@ }); }); - it ('it should set the session', function (done) { + it('it should set the session', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -141,7 +144,7 @@ }); }); - it ('it should fail when the user has multiple groups', function (done) { + it('it should fail when the user has multiple groups', function (done) { multipleGroupsResponse = true; var am = new F.manager.AuthManager({ account: 'accountName', @@ -157,7 +160,7 @@ }); }); - it ('it should work when a group is specified', function (done) { + it('it should work when a group is specified', function (done) { multipleGroupsResponse = true; var am = new F.manager.AuthManager({ account: 'accountName', @@ -175,7 +178,7 @@ }); }); - it ('it should not work when a wrong group is used', function (done) { + it('it should not work when a wrong group is used', function (done) { multipleGroupsResponse = true; var am = new F.manager.AuthManager({ account: 'accountName', @@ -191,7 +194,7 @@ }); }); - it ('should log a team member and get all the groups in the project', function (done) { + it('should log a team member and get all the groups in the project', function (done) { teamMemberResponse = true; var am = new F.manager.AuthManager({ account: 'accountName', @@ -212,7 +215,7 @@ }); - it ('it should fail with the list of groups on a team member login with no group', function (done) { + it('it should fail with the list of groups on a team member login with no group', function (done) { multipleGroupsResponse = true; teamMemberResponse = true; var am = new F.manager.AuthManager({ @@ -237,7 +240,7 @@ }); describe('Logout', function () { - it ('It should remove the epicenter cookie', function (done) { + it('It should remove the epicenter cookie', function (done) { sinon.spy(cookie, 'set'); var am = new F.manager.AuthManager({ account: 'accountName', @@ -261,7 +264,7 @@ }); describe('#setting cookies', function () { - it ('creates cookie with the correct path name when passing in account info in consructor', function () { + it('creates cookie with the correct path name when passing in account info in consructor', function () { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -271,7 +274,7 @@ var store = am.sessionManager.getStore(); store.serviceOptions.root.should.equal('/app/accountName/projectName'); }); - it ('creates cookie with the root path in local mode', function () { + it('creates cookie with the root path in local mode', function () { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -280,7 +283,7 @@ var store = am.sessionManager.getStore(); store.serviceOptions.root.should.equal('/'); }); - it ('creates cookie with the correct path name when passing in account info in login', function (done) { + it('creates cookie with the correct path name when passing in account info in login', function (done) { var am = new F.manager.AuthManager({ isLocal: false, store: { @@ -305,7 +308,7 @@ }); describe('#addGroups', function () { - it ('it should have one group on login', function (done) { + it('it should have one group on login', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -320,7 +323,7 @@ }); }); - it ('it should accept an object', function (done) { + it('it should accept an object', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -343,7 +346,7 @@ }); }); - it ('it should accept an array', function (done) { + it('it should accept an array', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', @@ -371,7 +374,7 @@ }); }); - it ('it should override a project\'s group', function (done) { + it('it should override a project\'s group', function (done) { var am = new F.manager.AuthManager({ account: 'accountName', project: 'projectName', diff --git a/tests/spec/test-conditional-creation-strategy.js b/tests/spec/test-conditional-creation-strategy.js index f94b9916..a42826f5 100644 --- a/tests/spec/test-conditional-creation-strategy.js +++ b/tests/spec/test-conditional-creation-strategy.js @@ -34,7 +34,7 @@ var setupServer = function () { server = sinon.fakeServer.create(); setupResponse('GET', /run\/forio-dev\/js-libs/, 200, runs || []); - server.autorespond = true; + server.respondImmediately = true; }; var teardownServer = function () { @@ -42,15 +42,6 @@ }; describe('Conditional Creation Strategy', function () { - - before(function () { - - }); - - after(function () { - - }); - beforeEach(function () { setupServer(); }); diff --git a/tests/spec/test-group-api-service.js b/tests/spec/test-group-api-service.js index c692005b..e2783ac5 100644 --- a/tests/spec/test-group-api-service.js +++ b/tests/spec/test-group-api-service.js @@ -5,7 +5,7 @@ describe('Group API Service', function () { before(function () { server = sinon.fakeServer.create(); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { diff --git a/tests/spec/test-member-api-service.js b/tests/spec/test-member-api-service.js index 366977c1..9bb46595 100644 --- a/tests/spec/test-member-api-service.js +++ b/tests/spec/test-member-api-service.js @@ -15,7 +15,7 @@ xhr.respond(201, { 'Content-Type': 'application/json' }, JSON.stringify({ })); }); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { diff --git a/tests/spec/test-user-api-adapter.js b/tests/spec/test-user-api-adapter.js index b1098bec..1dbfe5f7 100644 --- a/tests/spec/test-user-api-adapter.js +++ b/tests/spec/test-user-api-adapter.js @@ -12,7 +12,7 @@ describe('User API Service', function () { before(function () { server = sinon.fakeServer.create(); - server.autoRespond = true; + server.respondImmediately = true; }); after(function () { diff --git a/tests/spec/test-world-api-service.js b/tests/spec/test-world-api-service.js index fc61a6ed..abdd9e1c 100644 --- a/tests/spec/test-world-api-service.js +++ b/tests/spec/test-world-api-service.js @@ -13,7 +13,7 @@ xhr.respond(204, { 'Content-Type': 'application/json' }, null); }); - server.autoRespond = true; + server.respondImmediately = true; }); afterEach(function () { diff --git a/tests/spec/test-world-manager.js b/tests/spec/test-world-manager.js index 0ea575b6..290dddc0 100644 --- a/tests/spec/test-world-manager.js +++ b/tests/spec/test-world-manager.js @@ -62,10 +62,7 @@ lastModified: '123' })); }); - - - - server.autoRespond = true; + server.respondImmediately = true; }); afterEach(function () { From 16357baf1f586fa01fa569400c48978faf4d1545 Mon Sep 17 00:00:00 2001 From: Naren Ranjit Date: Thu, 15 Sep 2016 16:34:05 -0700 Subject: [PATCH 15/16] linting errors --- tests/spec/test-multiplayer-strategy.js | 3 --- tests/spec/test-run-api-service-callbacks.js | 4 ++-- 2 files changed, 2 insertions(+), 5 deletions(-) diff --git a/tests/spec/test-multiplayer-strategy.js b/tests/spec/test-multiplayer-strategy.js index 8ec51cc2..cdbd467c 100644 --- a/tests/spec/test-multiplayer-strategy.js +++ b/tests/spec/test-multiplayer-strategy.js @@ -91,9 +91,6 @@ var req = server.requests[0]; req.method.toUpperCase().should.equal('GET'); req.url.should.match(queryMatchers.getWorlds); - }).fail(function () { - var s = server; - console.log(arguments); }); }); diff --git a/tests/spec/test-run-api-service-callbacks.js b/tests/spec/test-run-api-service-callbacks.js index 5a40684b..8a5b7432 100644 --- a/tests/spec/test-run-api-service-callbacks.js +++ b/tests/spec/test-run-api-service-callbacks.js @@ -43,7 +43,7 @@ var cb2 = sinon.spy(); var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); - return rs.do('add', [1,2], { success: cb1 }).then(cb2).then( function () { + return rs.do('add', [1,2], { success: cb1 }).then(cb2).then(function () { cb1.should.have.been.called; cb2.should.have.been.called; }); @@ -219,7 +219,7 @@ var rs = new RunService({ account: 'forio', project: 'js-libs', filter: { saved: true } }); return rs.save({ completed: true }, { success: cb1 }).then(cb2).then(function () { cb1.should.have.been.called; - cb2.should.have.been.called; + cb2.should.have.been.called; }); }); it('passes error callbacks', function () { From 5f7d4b3871c9ae9c3bf7d8068b1189c6441e2443 Mon Sep 17 00:00:00 2001 From: Molly Jones Date: Tue, 20 Sep 2016 11:55:58 -0700 Subject: [PATCH 16/16] epicenter-2129: updating jquery version in bundled components too --- src/components/assignment/index.html | 2 +- src/components/login/index.html | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/assignment/index.html b/src/components/assignment/index.html index 1ac536db..2d3ae940 100644 --- a/src/components/assignment/index.html +++ b/src/components/assignment/index.html @@ -8,7 +8,7 @@ - + diff --git a/src/components/login/index.html b/src/components/login/index.html index 431e42e6..e80f30db 100644 --- a/src/components/login/index.html +++ b/src/components/login/index.html @@ -14,7 +14,7 @@ - +