diff --git a/app/pods/protected/strains/edit/controller.js b/app/pods/protected/strains/edit/controller.js
index 4d68665..509b1e7 100644
--- a/app/pods/protected/strains/edit/controller.js
+++ b/app/pods/protected/strains/edit/controller.js
@@ -1,36 +1,61 @@
import Ember from 'ember';
-import SaveModel from '../../../../mixins/save-model';
-import ajaxError from '../../../../utils/ajax-error';
-const { Controller } = Ember;
+const { Controller, RSVP, inject: { service } } = Ember;
+
+export default Controller.extend({
+ ajaxError: service('ajax-error'),
-export default Controller.extend(SaveModel, {
- // Required for SaveModel mixin
fallbackRouteSave: 'protected.strains.show',
fallbackRouteCancel: 'protected.strains.show',
actions: {
- addCharacteristic: function() {
- return this.store.createRecord('measurement', {
- characteristic: this.store.createRecord('characteristic', { sortOrder: -999 }),
+ save: function(properties, deleteQueue, updateQueue) {
+ let promises = [];
+ properties.measurements.forEach((measurement) => {
+ if (measurement.get('isNew')) {
+ promises.push(measurement.save());
+ }
+ });
+
+ updateQueue.forEach((measurement) => {
+ promises.push(measurement.save());
+ });
+
+ deleteQueue.forEach((measurement) => {
+ promises.push(measurement.destroyRecord());
+ });
+
+ const model = this.get('model');
+ const fallbackRoute = this.get('fallbackRouteSave');
+
+ RSVP.all(promises).then(() => {
+ // Can't call _super inside promise, have to reproduce save-model
+ // mixin here :-(
+ model.setProperties(properties);
+ model.save().then((model) => {
+ this.get('flashMessages').clearMessages();
+ this.transitionToRoute(fallbackRoute, model);
+ });
+ }, (errors) => {
+ this.get('ajaxError').alert(errors);
});
},
- saveMeasurement: function(measurement, properties) {
- measurement.setProperties(properties);
- measurement.save().then(() => {
- this.get('flashMessages').clearMessages();
- }, () => {
- ajaxError(measurement.get('errors'), this.get('flashMessages'));
- });
- },
+ cancel: function() {
+ const model = this.get('model');
- deleteMeasurement: function(measurement) {
- const characteristic = measurement.get('characteristic');
- if (characteristic.get('isNew')) {
- characteristic.destroyRecord();
+ model.get('errors').clear();
+ model.rollbackAttributes();
+
+ if (model.get('isNew')) {
+ this.transitionToRoute(this.get('fallbackRouteCancel'));
+ } else {
+ this.transitionToRoute(this.get('fallbackRouteCancel'), model);
}
- measurement.destroyRecord();
+ },
+
+ addMeasurement: function() {
+ return this.store.createRecord('measurement');
},
},
diff --git a/app/pods/protected/strains/edit/template.hbs b/app/pods/protected/strains/edit/template.hbs
index 59d1633..334ad07 100644
--- a/app/pods/protected/strains/edit/template.hbs
+++ b/app/pods/protected/strains/edit/template.hbs
@@ -2,10 +2,8 @@
protected/strains/strain-form
strain=model
speciesList=speciesList
- add-characteristic=(action "addCharacteristic")
+ add-measurement=(action "addMeasurement")
allCharacteristics=allCharacteristics
- save-measurement=(action "saveMeasurement")
- delete-measurement=(action "deleteMeasurement")
on-save=(action "save")
on-cancel=(action "cancel")
}}
diff --git a/app/pods/protected/strains/measurements-table-row/component.js b/app/pods/protected/strains/measurements-table-row/component.js
index ad63dc9..6b67968 100644
--- a/app/pods/protected/strains/measurements-table-row/component.js
+++ b/app/pods/protected/strains/measurements-table-row/component.js
@@ -10,6 +10,8 @@ export default Component.extend({
allCharacteristics: null,
measurement: null,
isDirty: null,
+ isNew: false,
+ isQueued: false,
// Actions
"save-measurement": null,
@@ -22,11 +24,23 @@ export default Component.extend({
notes: null,
resetOnInit: Ember.on('init', function() {
+ this._resetProperties();
+ }),
+
+ _resetProperties: function() {
this.get('propertiesList').forEach((field) => {
const valueInMeasurement = this.get('measurement').get(field);
this.set(field, valueInMeasurement);
});
- }),
+ // Read-only attributes
+ this.set('isNew', this.get('measurement.isNew'));
+ if (this.get('isNew') && !this.get('isQueued')) {
+ this.set('isEditing', true);
+ } else {
+ this.set('isEditing', false);
+ }
+ this.set('isDirty', false);
+ },
updateField: function(property, value) {
this.set(property, value);
@@ -40,12 +54,22 @@ export default Component.extend({
actions: {
edit: function() {
- this.toggleProperty('isEditing');
+ this.set('isEditing', true);
},
save: function() {
this.attrs['save-measurement'](this.get('measurement'), this.getProperties(this.get('propertiesList')));
- this.toggleProperty('isEditing');
+ this.set('isQueued', true);
+ this._resetProperties();
+ },
+
+ cancel: function() {
+ if (this.get('isNew')) {
+ this.attrs['delete-measurement'](this.get('measurement'));
+ } else {
+ this._resetProperties();
+ this.set('isEditing', false);
+ }
},
delete: function() {
diff --git a/app/pods/protected/strains/measurements-table-row/loading/template.hbs b/app/pods/protected/strains/measurements-table-row/loading/template.hbs
new file mode 100644
index 0000000..e5a3e05
--- /dev/null
+++ b/app/pods/protected/strains/measurements-table-row/loading/template.hbs
@@ -0,0 +1 @@
+{{loading-panel}}
diff --git a/app/pods/protected/strains/measurements-table-row/template.hbs b/app/pods/protected/strains/measurements-table-row/template.hbs
index 0dc03b7..7a51e78 100644
--- a/app/pods/protected/strains/measurements-table-row/template.hbs
+++ b/app/pods/protected/strains/measurements-table-row/template.hbs
@@ -1,5 +1,7 @@
{{#if isEditing}}
-
|
+
+ {{{characteristic.characteristicTypeName}}}
+ |
|
{{#if canEdit}}
- {{#if isDirty}}
-
- {{else}}
- |
{{/if}}
diff --git a/app/pods/protected/strains/measurements-table/component.js b/app/pods/protected/strains/measurements-table/component.js
index e4fd3a1..87bf403 100644
--- a/app/pods/protected/strains/measurements-table/component.js
+++ b/app/pods/protected/strains/measurements-table/component.js
@@ -5,13 +5,13 @@ const { sort } = computed;
export default Component.extend({
// Passed in
- strain: null,
+ measurements: null,
allCharacteristics: null,
canEdit: false,
canAdd: false,
// Actions
- "add-characteristic": null,
+ "add-measurement": null,
"save-measurement": null,
"delete-measurement": null,
@@ -19,15 +19,11 @@ export default Component.extend({
sortParams: ['characteristic.characteristicTypeName', 'characteristic.sortOrder', 'characteristic.characteristicName'],
sortAsc: true,
paramsChanged: false,
- sortedMeasurements: sort('strain.measurements', 'sortParams'),
- measurementsPresent: computed('strain.measurements', function() {
- return this.get('strain.measurements.length') > 0;
- }),
+ sortedMeasurements: sort('measurements', 'sortParams'),
actions: {
- addCharacteristic: function() {
- const newChar = this.attrs['add-characteristic']();
- this.get('strain.measurements').addObject(newChar);
+ addMeasurement: function() {
+ return this.attrs['add-measurement']();
},
changeSortParam: function(col) {
diff --git a/app/pods/protected/strains/measurements-table/template.hbs b/app/pods/protected/strains/measurements-table/template.hbs
index 9e22393..9d648b4 100644
--- a/app/pods/protected/strains/measurements-table/template.hbs
+++ b/app/pods/protected/strains/measurements-table/template.hbs
@@ -1,17 +1,16 @@
{{#if canAdd}}
-
- Add characteristic
+
+ Add measurement
{{/if}}
-{{#if measurementsPresent}}
- {{#if paramsChanged}}
-
- Reset sort
-
- {{/if}}
+{{#if paramsChanged}}
+
+ Reset sort
+
+{{/if}}
{{#if canEdit}}
@@ -48,9 +47,10 @@
allCharacteristics=allCharacteristics
canEdit=canEdit
}}
+ {{else}}
+
+ No Measurements on Record |
+
{{/each}}
-{{else}}
-No measurements on record.
-{{/if}}
diff --git a/app/pods/protected/strains/show/strain-card/template.hbs b/app/pods/protected/strains/show/strain-card/template.hbs
index 517256d..4629f23 100644
--- a/app/pods/protected/strains/show/strain-card/template.hbs
+++ b/app/pods/protected/strains/show/strain-card/template.hbs
@@ -71,11 +71,11 @@
{{! ROW 5 }}
- - Characteristics
+ - Characteristic Measurements
-
{{
protected/strains/measurements-table
- strain=strain
+ measurements=strain.measurements
canEdit=false
canAdd=false
}}
diff --git a/app/pods/protected/strains/strain-form/component.js b/app/pods/protected/strains/strain-form/component.js
index ebb157c..f0fbab7 100644
--- a/app/pods/protected/strains/strain-form/component.js
+++ b/app/pods/protected/strains/strain-form/component.js
@@ -9,22 +9,22 @@ export default Component.extend(SetupMetaData, {
isNew: null,
isDirty: false,
speciesList: null,
- allCharacteristics: null,
+ allCharacteristics: [],
+ updateQueue: [],
+ deleteQueue: [],
// Actions
"on-save": null,
"on-cancel": null,
"on-update": null,
- "add-characteristic": null,
- "save-measurement": null,
- "delete-measurement": null,
+ "add-measurements": null,
// CPs
sortParams: ['sortOrder'],
sortedSpeciesList: sort('speciesList', 'sortParams'),
// Property mapping
- propertiesList: ['strainName', 'typeStrain', 'species', 'isolatedFrom', 'accessionNumbers', 'genbank', 'wholeGenomeSequence', 'notes'],
+ propertiesList: ['strainName', 'typeStrain', 'species', 'isolatedFrom', 'accessionNumbers', 'genbank', 'wholeGenomeSequence', 'notes', 'measurements'],
strainName: null,
typeStrain: null,
species: null,
@@ -33,15 +33,55 @@ export default Component.extend(SetupMetaData, {
genbank: null,
wholeGenomeSequence: null,
notes: null,
+ measurements: [],
+
+ // Dropdown menu
+ characteristics: [],
+ charSortParams: ['characteristicTypeName', 'sortOrder', 'characteristicName'],
+ sortedCharacteristics: sort('characteristics', 'charSortParams'),
+ setupCharacteristics: Ember.on('init', function() {
+ const tempArray = this._resetArray(this.get('allCharacteristics'));
+ this.set('characteristics', tempArray);
+ }),
resetOnInit: Ember.on('init', function() {
+ this._resetProperties();
+ }),
+
+ _resetArray: function(arr) {
+ let tempArray = [];
+ arr.forEach((val) => {
+ if (!val.get('isNew')) {
+ tempArray.push(val);
+ }
+ });
+ return tempArray;
+ },
+
+ _resetProperties: function() {
+ // Still some coupling going on here because of adding strain to measurement
+ this.get('measurements').forEach((val) => {
+ if (val.get('hasDirtyAttributes')) {
+ val.rollbackAttributes();
+ }
+ if (val.get('isNew')) {
+ this.get('strain.measurements').removeObject(val);
+ }
+ });
this.get('propertiesList').forEach((field) => {
const valueInStrain = this.get('strain').get(field);
- this.set(field, valueInStrain);
+ if (field === 'measurements') {
+ const tempArray = this._resetArray(valueInStrain);
+ this.set(field, tempArray);
+ } else {
+ this.set(field, valueInStrain);
+ }
});
+ this.set('updateQueue', []);
+ this.set('deleteQueue', []);
// Read-only attributes
this.set('isNew', this.get('strain.isNew'));
- }),
+ },
updateField: function(property, value) {
this.set(property, value);
@@ -55,23 +95,32 @@ export default Component.extend(SetupMetaData, {
actions: {
save: function() {
- return this.attrs['on-save'](this.getProperties(this.get('propertiesList')));
+ return this.attrs['on-save'](this.getProperties(this.get('propertiesList')), this.get('deleteQueue'), this.get('updateQueue'));
},
cancel: function() {
+ this._resetProperties();
return this.attrs['on-cancel']();
},
- addCharacteristic: function() {
- return this.attrs['add-characteristic']();
+ addMeasurement: function() {
+ const measurement = this.attrs['add-measurement']();
+ this.get('measurements').pushObject(measurement);
},
saveMeasurement: function(measurement, properties) {
- return this.attrs['save-measurement'](measurement, properties);
+ measurement.setProperties(properties);
+ measurement.set('strain', this.get('strain'));
+ if (!measurement.get('isNew')) {
+ this.get('updateQueue').pushObject(measurement);
+ }
+ this.set('isDirty', true);
},
deleteMeasurement: function(measurement) {
- return this.attrs['delete-measurement'](measurement);
+ this.get('deleteQueue').pushObject(measurement);
+ this.get('measurements').removeObject(measurement);
+ this.set('isDirty', true);
},
strainNameDidChange: function(value) {
diff --git a/app/pods/protected/strains/strain-form/template.hbs b/app/pods/protected/strains/strain-form/template.hbs
index e44096f..99b729c 100644
--- a/app/pods/protected/strains/strain-form/template.hbs
+++ b/app/pods/protected/strains/strain-form/template.hbs
@@ -61,9 +61,9 @@
{{
protected/strains/measurements-table
- strain=strain
- add-characteristic=(action "addCharacteristic")
- allCharacteristics=allCharacteristics
+ measurements=measurements
+ add-measurement=(action "addMeasurement")
+ allCharacteristics=sortedCharacteristics
save-measurement=(action "saveMeasurement")
delete-measurement=(action "deleteMeasurement")
canEdit=strain.canEdit
diff --git a/app/pods/protected/users/changepassword/controller.js b/app/pods/protected/users/changepassword/controller.js
index 638961f..ddfd685 100644
--- a/app/pods/protected/users/changepassword/controller.js
+++ b/app/pods/protected/users/changepassword/controller.js
@@ -1,11 +1,11 @@
import Ember from 'ember';
-import ajaxErrorNew from '../../../../utils/ajax-error-new';
const { Controller, inject: { service } } = Ember;
export default Controller.extend({
session: service(),
ajax: service(),
+ ajaxError: service('ajax-error'),
currentUser: service('session-account'),
actions: {
@@ -15,12 +15,13 @@ export default Controller.extend({
this.get('ajax').post('/users/password', { data: data }).then(() => {
this.transitionToRoute('protected.users.show', id);
this.get('flashMessages').information('Your password has been changed.');
- }, (error) => {
- ajaxErrorNew(error, this.get('flashMessages'));
+ }, (errors) => {
+ this.get('ajaxError').alert(errors);
});
},
cancel: function() {
+ this.get('flashMessages').clearMessages();
this.transitionToRoute('protected.users.show', this.get('currentUser.account.id'));
},
},
diff --git a/app/serializers/application.js b/app/serializers/application.js
index d40a2bf..4a316f0 100644
--- a/app/serializers/application.js
+++ b/app/serializers/application.js
@@ -5,24 +5,41 @@ const { RESTSerializer } = DS;
const { isNone } = Ember;
export default RESTSerializer.extend({
- isNewSerializerAPI: true,
-
serializeBelongsTo: function(snapshot, json, relationship) {
- let key = relationship.key;
- const belongsTo = snapshot.belongsTo(key);
- key = this.keyForRelationship ? this.keyForRelationship(key, "belongsTo", "serialize") : key;
- json[key] = isNone(belongsTo) ? belongsTo : +belongsTo.record.id;
+ const key = relationship.key;
+ if (this._canSerialize(key)) {
+ const belongsToId = snapshot.belongsTo(key, { id: true });
+ let payloadKey = this._getMappedKey(key, snapshot.type);
+ if (payloadKey === key && this.keyForRelationship) {
+ payloadKey = this.keyForRelationship(key, "belongsTo", "serialize");
+ }
+ if (isNone(belongsToId)) {
+ json[payloadKey] = null;
+ } else {
+ json[payloadKey] = +belongsToId;
+ }
+
+ if (relationship.options.polymorphic) {
+ this.serializePolymorphicType(snapshot, json, relationship);
+ }
+ }
},
serializeHasMany: function(snapshot, json, relationship) {
- let key = relationship.key;
- const hasMany = snapshot.hasMany(key);
- key = this.keyForRelationship ? this.keyForRelationship(key, "hasMany", "serialize") : key;
-
- json[key] = [];
- hasMany.forEach((item) => {
- json[key].push(+item.id);
- });
+ const key = relationship.key;
+ if (this._shouldSerializeHasMany(snapshot, key, relationship)) {
+ const hasMany = snapshot.hasMany(key, { ids: true });
+ if (hasMany !== undefined) {
+ let payloadKey = this._getMappedKey(key, snapshot.type);
+ if (payloadKey === key && this.keyForRelationship) {
+ payloadKey = this.keyForRelationship(key, "hasMany", "serialize");
+ }
+ json[payloadKey] = [];
+ hasMany.forEach((item) => {
+ json[payloadKey].push(+item);
+ });
+ }
+ }
},
});
diff --git a/app/services/ajax-error.js b/app/services/ajax-error.js
new file mode 100644
index 0000000..6cfd29a
--- /dev/null
+++ b/app/services/ajax-error.js
@@ -0,0 +1,19 @@
+import Ember from 'ember';
+
+const { Service, inject: { service } } = Ember;
+
+export default Service.extend({
+ flashMessages: service(),
+
+ alert: function(error) {
+ const flash = this.get('flashMessages');
+
+ flash.clearMessages();
+ window.scrollTo(0,0);
+ error.errors.forEach((error) => {
+ console.error(error);
+ const source = error.source.pointer.split('/');
+ flash.error(`${source[source.length-1].replace(/([A-Z])/g, ' $1').capitalize()} - ${error.detail}`);
+ });
+ }
+});
diff --git a/app/utils/ajax-error-new.js b/app/utils/ajax-error-new.js
deleted file mode 100644
index 37031f2..0000000
--- a/app/utils/ajax-error-new.js
+++ /dev/null
@@ -1,7 +0,0 @@
-export default function ajaxErrorNew(error, flash) {
- flash.clearMessages();
- error.errors.forEach((error) => {
- const source = error.source.pointer.split('/');
- flash.error(`${source[source.length-1].replace(/([A-Z])/g, ' $1').capitalize()} - ${error.detail}`);
- });
-}