ENH: Collection Edit (parity with reading) (#48)

This commit is contained in:
Matthew Ryan Dillon 2017-11-30 15:51:16 -07:00 committed by GitHub
parent bfae4422f4
commit cb3bc081a6
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
19 changed files with 337 additions and 121 deletions

View file

@ -6,6 +6,7 @@ export default Component.extend({
tagName: 'a', tagName: 'a',
classNames: ['btn'], classNames: ['btn'],
classNameBindings: [ classNameBindings: [
// Styles
'isDefault:btn-default', 'isDefault:btn-default',
'isPrimary:btn-primary', 'isPrimary:btn-primary',
'isSuccess:btn-success', 'isSuccess:btn-success',
@ -13,9 +14,14 @@ export default Component.extend({
'isWarning:btn-warning', 'isWarning:btn-warning',
'isDanger:btn-danger', 'isDanger:btn-danger',
'isLink:btn-link', 'isLink:btn-link',
// Sizes
'isLarge:btn-lg',
'isSmall:btn-sm',
'isXSmall:btn-xs',
], ],
// ARGS // ARGS
// Styles
isDefault: false, isDefault: false,
isPrimary: false, isPrimary: false,
isSuccess: false, isSuccess: false,
@ -23,6 +29,10 @@ export default Component.extend({
isWarning: false, isWarning: false,
isDanger: false, isDanger: false,
isLink: false, isLink: false,
// Sizes
isLarge: false,
isSmall: false,
isXSmall: false,
label: 'LABEL', label: 'LABEL',

View file

@ -2,13 +2,57 @@ import Ember from 'ember';
import Changeset from 'ember-changeset'; import Changeset from 'ember-changeset';
import lookupValidator from 'ember-changeset-validations'; import lookupValidator from 'ember-changeset-validations';
const { Component } = Ember; const { Component, inject: { service } } = Ember;
export default Component.extend({ export default Component.extend({
store: service(),
init() { init() {
this._super(...arguments); this._super(...arguments);
const model = this.get('model'); const model = this.get('model');
const validations = this.get('validations'); const validations = this.get('validations');
this.set('changeset', new Changeset(model, lookupValidator(validations), validations));
let changesets = {};
changesets['new'] = [];
changesets['delete'] = [];
changesets['hasMany'] = [];
changesets['model'] = new Changeset(model,
lookupValidator(validations['collection']),
validations['collection']);
// TODO: gross, just grab these data in the route.
model.get('collectionSpecies').then((collectionSpecies) => {
let collectionSpeciesChangesets = [];
collectionSpecies.forEach((cs) => {
const changeset = new Changeset(cs,
lookupValidator(validations['collectionSpecies']),
validations['collectionSpecies']);
collectionSpeciesChangesets.push({ model: cs, changeset: changeset });
});
changesets['hasMany']['collectionSpecies'] = collectionSpeciesChangesets;
this.set('changesets', changesets);
});
},
actions: {
addCollectionSpecies() {
const store = this.get('store');
let changesets = this.get('changesets');
const validations = this.get('validations');
const collection = this.get('model');
const cs = store.createRecord('collection-species', { collection: collection });
collection.get('collectionSpecies').pushObject(cs);
changesets['new'].pushObject(cs);
const changeset = new Changeset(cs,
lookupValidator(validations['collectionSpecies']),
validations['collectionSpecies']);
changesets['hasMany']['collectionSpecies'].pushObject({ model: cs, changeset: changeset });
},
deleteCollectionSpecies(changesetRecord) {
let changesets = this.get('changesets');
changesets['delete'].pushObject(changesetRecord.model);
changesets['hasMany']['collectionSpecies'].removeObject(changesetRecord);
},
}, },
}); });

View file

@ -9,6 +9,7 @@ export default Component.extend({
columns: [ columns: [
{ label: 'Project', valuePath: 'project.name', }, { label: 'Project', valuePath: 'project.name', },
{ label: 'IACUC', valuePath: 'project.iacucNumber', }, { label: 'IACUC', valuePath: 'project.iacucNumber', },
{ label: 'Species', valuePath: 'speciesAndCounts', },
{ label: 'Region', valuePath: 'studyLocation.site.region.name', }, { label: 'Region', valuePath: 'studyLocation.site.region.name', },
{ label: 'Site', valuePath: 'studyLocation.site.name', }, { label: 'Site', valuePath: 'studyLocation.site.name', },
{ label: 'Study Location', valuePath: 'studyLocation.code', }, { label: 'Study Location', valuePath: 'studyLocation.code', },

View file

@ -4,5 +4,5 @@ const { Component } = Ember;
export default Component.extend({ export default Component.extend({
// ARGS // ARGS
changeset: null, changesets: null,
}); });

View file

@ -11,4 +11,8 @@ export default Component.extend({
const property = this.get('property'); const property = this.get('property');
return isEmpty(get(changeset, `error.${property}`)); return isEmpty(get(changeset, `error.${property}`));
}), }),
hasLabel: computed('label', function() {
return !isEmpty(get(this, 'label'));
}),
}); });

View file

@ -1,17 +1,31 @@
import Ember from 'ember'; import Ember from 'ember';
import CollectionValidations from 'ccdb-web/validations/collection'; import CollectionValidations from 'ccdb-web/validations/collection';
import { schema } from 'ccdb-web/models/collection'; import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
import ValidationMixin from 'ccdb-web/mixins/validation'; import ValidationMixin from 'ccdb-web/mixins/validation';
const { Controller } = Ember; const { Controller, computed } = Ember;
export default Controller.extend(ValidationMixin, { export default Controller.extend(ValidationMixin, {
CollectionValidations, CollectionValidations,
CollectionSpeciesValidations,
options: computed('projectOptions', 'studyLocationOptions',
'collectionTypeOptions', 'collectionMethodOptions',
'speciesOptions', 'adfgPermitOptions', function() {
return {
projects: this.get('projectOptions'),
studyLocations: this.get('studyLocationOptions'),
collectionTypes: this.get('collectionTypeOptions'),
collectionMethods: this.get('collectionMethodOptions'),
species: this.get('speciesOptions'),
adfgPermits: this.get('adfgPermitOptions'),
};
}),
actions: { actions: {
onSave(changeset) { onSave(changeset) {
const postSave = () => { this.transitionToRoute('collections.index'); }; const postSave = () => { this.transitionToRoute('collections.index'); };
return this.validationSave(changeset, schema, postSave); return this.validationSave(changeset, postSave);
}, },
onCancel(changeset) { onCancel(changeset) {
const postCancel = () => { this.transitionToRoute('collections.index'); }; const postCancel = () => { this.transitionToRoute('collections.index'); };

View file

@ -1,27 +1,41 @@
import Ember from 'ember'; import Ember from 'ember';
import CollectionValidations from 'ccdb-web/validations/collection'; import CollectionValidations from 'ccdb-web/validations/collection';
import { schema } from 'ccdb-web/models/collection'; import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
import ValidationMixin from 'ccdb-web/mixins/validation'; import ValidationMixin from 'ccdb-web/mixins/validation';
const { Controller } = Ember; const { Controller, computed } = Ember;
export default Controller.extend(ValidationMixin, { export default Controller.extend(ValidationMixin, {
CollectionValidations, CollectionValidations,
CollectionSpeciesValidations,
options: computed('projectOptions', 'studyLocationOptions',
'collectionTypeOptions', 'collectionMethodOptions',
'speciesOptions', 'adfgPermitOptions', function() {
return {
projects: this.get('projectOptions'),
studyLocations: this.get('studyLocationOptions'),
collectionTypes: this.get('collectionTypeOptions'),
collectionMethods: this.get('collectionMethodOptions'),
species: this.get('speciesOptions'),
adfgPermits: this.get('adfgPermitOptions'),
};
}),
actions: { actions: {
onSave(changeset) { onSave(changesets) {
const postSave = () => { const postSave = () => {
// Use the model's ID here because of the ArrayProxy in the route // Use the model's ID here because of the ArrayProxy in the route
this.transitionToRoute('collections.detail', this.get('model.id')); this.transitionToRoute('collections.detail', this.get('model.id'));
}; };
return this.validationSave(changeset, schema, postSave); return this.validationSave(changesets, postSave);
}, },
onCancel(changeset) { onCancel(changesets) {
const postCancel = () => { const postCancel = () => {
// Use the model's ID here because of the ArrayProxy in the route // Use the model's ID here because of the ArrayProxy in the route
return this.transitionToRoute('collections.detail', this.get('model.id')); return this.transitionToRoute('collections.detail', this.get('model.id'));
}; };
return this.validationCancel(changeset, postCancel); return this.validationCancel(changesets, postCancel);
}, },
}, },
}); });

View file

@ -1,30 +1,70 @@
import Ember from 'ember'; import Ember from 'ember';
const { Mixin, get } = Ember; const { Mixin, get, RSVP } = Ember;
const { keys } = Object; const { keys } = Object;
const { isArray } = Array;
export default Mixin.create({ export default Mixin.create({
validationSave(changeset, schema, postSave) { validationSave(changesets, postSave) {
return changeset let promises = [], changes = [], saves = [], isValid = true;
.cast(keys(schema))
.validate() let modelChangeset = changesets['model'];
.then(() => {
if (changeset.get('isValid')) { // first, delete anything that needs to be removed
return changeset.save().then(postSave); for (const model of changesets['delete']) {
promises.push(model.destroyRecord());
} }
})
.catch((error) => { // second, handle changes on parent model (this is important if new)
modelChangeset.validate().then(() => {
if (modelChangeset.get('isValid')) {
return modelChangeset.save();
}
}).then(() => {
for (const hasMany of keys(changesets['hasMany'])) {
for (const { changeset } of changesets['hasMany'][hasMany]) {
promises.push(changeset.validate());
changes.push(changeset);
}
}
return RSVP.all(promises);
}).then(() => { // don't need the promises, just that they are done.
for (let changeset of changes) {
if (get(changeset, 'isValid')) {
let saver = changeset.save().catch((error) => {
/* eslint-disable no-console */ /* eslint-disable no-console */
console.log(error); console.log(error);
/* eslint-enable no-console */ /* eslint-enable no-console */
get(this, 'model.errors').forEach(({ attribute, message }) => { // TODO: do something with server-side non-attr errors
changeset.pushErrors(attribute, message);
}); });
saves.push(saver);
} else {
isValid = false;
}
}
return RSVP.all(saves);
}).then(() => {
if (isValid) { return postSave(); }
}); });
}, },
validationCancel(changeset, postCancel) { validationCancel(changesets, postCancel) {
delete changesets['delete'];
for (const key of keys(changesets)) {
if (key === 'new') {
for (const model of changesets[key]) {
model.destroyRecord();
}
} else if (isArray(changesets[key])) { // hasMany
for (const { changeset } of changesets[key]) {
changeset.rollback(); changeset.rollback();
}
} else { // single
const changeset = changesets[key];
changeset.rollback();
}
}
return postCancel(); return postCancel();
}, },
}); });

View file

@ -5,7 +5,7 @@ const { Model, attr, belongsTo } = DS;
export default Model.extend({ export default Model.extend({
sex: attr('string'), sex: attr('string'),
count: attr('number'), count: attr('number'),
countEstimated: attr('boolean'), countEstimated: attr('boolean', { defaultValue: false }),
collection: belongsTo('collection'), collection: belongsTo('collection'),
species: belongsTo('species'), species: belongsTo('species'),

View file

@ -4,7 +4,7 @@ import DS from 'ember-data';
const { computed } = Ember; const { computed } = Ember;
const { Model, attr, belongsTo, hasMany } = DS; const { Model, attr, belongsTo, hasMany } = DS;
export const schema = { export default Model.extend({
displayName: attr('string'), displayName: attr('string'),
numberOfTraps: attr('number'), numberOfTraps: attr('number'),
collectionStartDate: attr('string-null-to-empty'), collectionStartDate: attr('string-null-to-empty'),
@ -18,20 +18,22 @@ export const schema = {
collectionType: belongsTo('collection-type'), collectionType: belongsTo('collection-type'),
adfgPermit: belongsTo('adfg-permit'), adfgPermit: belongsTo('adfg-permit'),
collectionSpecies: hasMany('collection-species', { async: false }), collectionSpecies: hasMany('collection-species'),
// computed
species: computed.mapBy('collectionSpecies', 'species'), species: computed.mapBy('collectionSpecies', 'species'),
speciesNames: computed.mapBy('species', 'commonName'), speciesNames: computed.mapBy('species', 'commonName'),
counts: computed.mapBy('collectionSpecies', 'count'), counts: computed.mapBy('collectionSpecies', 'count'),
speciesAndCounts: computed('speciesNames', 'counts', function() { speciesAndCounts: computed('speciesNames', 'counts', function() {
const speciesNames = this.get('speciesNames'); const speciesNames = this.get('speciesNames');
let counts = this.get('counts'); let counts = this.get('counts');
counts = counts.map(c => c !== null ? c : 'No Count'); counts = counts.map(c => c !== null ? c : 'No Count');
return speciesNames.map((n, i) => `${n} (${counts[i]})`).join(', '); return speciesNames.map((n, i) => `${n} (${counts[i]})`).join(', ');
}), }),
};
export default Model.extend(Object.assign({}, schema, {
startDateTime: computed('collectionStartDate', 'collectionStartTime', startDateTime: computed('collectionStartDate', 'collectionStartTime',
function() { return this._mergeDateTime('Start'); }), function() { return this._mergeDateTime('Start'); }),
@ -43,4 +45,4 @@ export default Model.extend(Object.assign({}, schema, {
const time = this.get(`collection${timepoint}Time`); const time = this.get(`collection${timepoint}Time`);
return `${date} ${time}`.trim(); return `${date} ${time}`.trim();
}, },
})); });

View file

@ -11,6 +11,8 @@ export default Route.extend({
studyLocationOptions: store.findAll('study-location'), studyLocationOptions: store.findAll('study-location'),
collectionTypeOptions: store.findAll('collection-type'), collectionTypeOptions: store.findAll('collection-type'),
collectionMethodOptions: store.findAll('collection-method'), collectionMethodOptions: store.findAll('collection-method'),
speciesOptions: store.findAll('species'),
adfgPermitOptions: store.findAll('adfg-permit'),
}); });
}, },

View file

@ -12,6 +12,8 @@ export default Route.extend({
studyLocationOptions: store.findAll('study-location'), studyLocationOptions: store.findAll('study-location'),
collectionTypeOptions: store.findAll('collection-type'), collectionTypeOptions: store.findAll('collection-type'),
collectionMethodOptions: store.findAll('collection-method'), collectionMethodOptions: store.findAll('collection-method'),
speciesOptions: store.findAll('species'),
adfgPermitOptions: store.findAll('adfg-permit'),
}); });
}, },

View file

@ -1,11 +1,10 @@
{{ {{
collection/create-container collection/create-container
model=model model=model
validations=CollectionValidations validations=(hash
projectOptions=projectOptions collection=CollectionValidations
studyLocationOptions=studyLocationOptions collectionSpecies=CollectionSpeciesValidations)
collectionTypeOptions=collectionTypeOptions options=options
collectionMethodOptions=collectionMethodOptions
onSave=(action 'onSave') onSave=(action 'onSave')
onCancel=(action 'onCancel') onCancel=(action 'onCancel')
}} }}

View file

@ -1,11 +1,10 @@
{{ {{
collection/create-container collection/create-container
model=model model=model
validations=CollectionValidations validations=(hash
projectOptions=projectOptions collection=CollectionValidations
studyLocationOptions=studyLocationOptions collectionSpecies=CollectionSpeciesValidations)
collectionTypeOptions=collectionTypeOptions options=options
collectionMethodOptions=collectionMethodOptions
onSave=(action 'onSave') onSave=(action 'onSave')
onCancel=(action 'onCancel') onCancel=(action 'onCancel')
}} }}

View file

@ -1,13 +1,16 @@
{{#crud-form {{#crud-form
changeset=changeset changesets=changesets
onSave=(action onSave) onSave=(action onSave)
onCancel=(action onCancel) as |f| onCancel=(action onCancel) as |f|
}} }}
<div class="row">
<div class="col-md-4">
<div class="well"> <div class="well">
{{#f.content class='form'}} {{#f.content class='form'}}
{{#with changesets.model as |changeset|}}
{{#validated-field property='project' label='Project' changeset=changeset}} {{#validated-field property='project' label='Project' changeset=changeset}}
{{#power-select {{#power-select
options=projectOptions options=options.projects
selected=changeset.project selected=changeset.project
onchange=(action (mut changeset.project)) onchange=(action (mut changeset.project))
searchField='name' searchField='name'
@ -17,9 +20,21 @@
{{/power-select}} {{/power-select}}
{{/validated-field}} {{/validated-field}}
{{#validated-field property='adfgPermit' label='ADFG Permit' changeset=changeset}}
{{#power-select
options=options.adfgPermits
selected=changeset.adfgPermit
onchange=(action (mut changeset.adfgPermit))
searchField='name'
as |adfgPermit|
}}
{{adfgPermit.name}}
{{/power-select}}
{{/validated-field}}
{{#validated-field property='studyLocation' label='Study location' changeset=changeset}} {{#validated-field property='studyLocation' label='Study location' changeset=changeset}}
{{#power-select {{#power-select
options=studyLocationOptions options=options.studyLocations
selected=changeset.studyLocation selected=changeset.studyLocation
onchange=(action (mut changeset.studyLocation)) onchange=(action (mut changeset.studyLocation))
searchField='name' searchField='name'
@ -31,7 +46,7 @@
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}} {{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
{{#power-select {{#power-select
options=collectionTypeOptions options=options.collectionTypes
selected=changeset.collectionType selected=changeset.collectionType
onchange=(action (mut changeset.collectionType)) onchange=(action (mut changeset.collectionType))
searchField='name' searchField='name'
@ -43,7 +58,7 @@
{{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}} {{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}}
{{#power-select {{#power-select
options=collectionMethodOptions options=options.collectionMethods
selected=changeset.collectionMethod selected=changeset.collectionMethod
onchange=(action (mut changeset.collectionMethod)) onchange=(action (mut changeset.collectionMethod))
searchField='name' searchField='name'
@ -72,9 +87,64 @@
{{#validated-field property='collectionEndTime' label='Collection end time' changeset=changeset}} {{#validated-field property='collectionEndTime' label='Collection end time' changeset=changeset}}
{{input value=changeset.collectionEndTime type='time' class='form-control'}} {{input value=changeset.collectionEndTime type='time' class='form-control'}}
{{/validated-field}} {{/validated-field}}
{{/with}}
{{/f.content}} {{/f.content}}
{{f.save}} {{f.cancel}}
</div> </div>
</div>
<div class="col-md-8">
<table class="table">
<caption>
Species / Count Info
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addCollectionSpecies')}}
</caption>
<thead>
<tr>
<th class="col-md-3">Species</th>
<th class="col-md-3">Count</th>
<th class="col-md-3">Count Estimated</th>
<th class="col-md-3">Sex</th>
<th class="col-md-1">Delete</th>
</tr>
</thead>
<tbody>
{{#each changesets.hasMany.collectionSpecies as |cs|}}
<tr class="form">
<td class="col-md-3">
{{#validated-field property='species' changeset=cs.changeset}}
{{#power-select
options=options.species
selected=cs.changeset.species
onchange=(action (mut cs.changeset.species))
searchField='commonName'
as |species|
}}
{{species.commonName}}
{{/power-select}}
{{/validated-field}}
</td>
<td class="col-md-3">
{{#validated-field property='count' changeset=cs.changeset}}
{{input value=cs.changeset.count}}
{{/validated-field}}
</td>
<td class="col-md-3">
{{#validated-field property='countEstimated' changeset=cs.changeset}}
{{input checked=cs.changeset.countEstimated type='checkbox'}}
{{/validated-field}}
</td>
<td class="col-md-3">
{{#validated-field property='sex' changeset=cs.changeset}}
{{input value=cs.changeset.sex}}
{{/validated-field}}
</td>
<th class="col-md-2">
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteCollectionSpecies' cs)}}
</th>
</tr>
{{/each}}
</tbody>
</table>
</div>
</div>
{{f.save}} {{f.cancel}}
{{/crud-form}} {{/crud-form}}

View file

@ -1,14 +1,14 @@
{{#if hasBlock}} {{#if hasBlock}}
{{yield (hash {{yield (hash
content=(component 'form-content' changeset=changeset) content=(component 'form-content')
cancel=(component 'action-button' cancel=(component 'action-button'
label='Cancel' label='Cancel'
isDanger=true isDanger=true
onClick=(action onCancel changeset)) onClick=(action onCancel changesets))
save=(component 'action-button' save=(component 'action-button'
label='Save' label='Save'
isSuccess=true isSuccess=true
onClick=(action onSave changeset)) onClick=(action onSave changesets))
)}} )}}
{{else}} {{else}}
MISSING CONTENT BLOCK MISSING CONTENT BLOCK

View file

@ -1,4 +1,7 @@
{{#if hasLabel}}
<label class="control-label">{{label}}</label> <label class="control-label">{{label}}</label>
{{/if}}
{{yield}} {{yield}}
{{#if (get changeset.error property)}} {{#if (get changeset.error property)}}

View file

@ -8,6 +8,6 @@ export default Transform.extend({
}, },
serialize(deserialized) { serialize(deserialized) {
return deserialized; return deserialized === '' ? null : deserialized;
} }
}); });

View file

@ -0,0 +1,12 @@
import {
validatePresence,
validateNumber,
} from 'ember-changeset-validations/validators';
export default {
sex: validatePresence(true),
count: validateNumber({ allowBlank: true, integer: true, positive: true }),
countEstimated: validatePresence(true),
species: validatePresence(true),
collection: validatePresence(true),
}