ENH: Collections (update) (#40)

Fixes #36
This commit is contained in:
Matthew Ryan Dillon 2017-10-07 17:37:24 -07:00 committed by GitHub
parent e1abc5e4cb
commit f41f4caccd
17 changed files with 221 additions and 77 deletions

View file

@ -9,4 +9,11 @@ export default JSONAPIAdapter.extend(DataAdapterMixin, {
namespace: API_NAMESPACE, namespace: API_NAMESPACE,
host: API_HOST, host: API_HOST,
authorizer: 'authorizer:application', authorizer: 'authorizer:application',
// DRF-JSON-API returns 400 by default
handleResponse(status, headers, payload) {
if (status === 400 && payload.errors) {
return new DS.InvalidError(payload.errors);
}
return this._super(...arguments);
}
}); });

View file

@ -1,5 +1,6 @@
import Ember from 'ember'; import Ember from 'ember';
import Changeset from 'ember-changeset'; import Changeset from 'ember-changeset';
import lookupValidator from 'ember-changeset-validations';
const { Component } = Ember; const { Component } = Ember;
@ -7,6 +8,7 @@ export default Component.extend({
init() { init() {
this._super(...arguments); this._super(...arguments);
const model = this.get('model'); const model = this.get('model');
this.set('changeset', new Changeset(model)); const validations = this.get('validations');
this.set('changeset', new Changeset(model, lookupValidator(validations), validations));
}, },
}); });

View file

@ -0,0 +1,14 @@
import Ember from 'ember';
const { Component, computed, get, isEmpty } = Ember;
export default Component.extend({
classNames: ['form-group'],
classNameBindings: ['isValid::has-error'],
isValid: computed('changeset.error', 'property', function() {
const changeset = this.get('changeset');
const property = this.get('property');
return isEmpty(get(changeset, `error.${property}`));
}),
});

View file

@ -1,16 +1,21 @@
import Ember from 'ember'; import Ember from 'ember';
import CollectionValidations from '../../validations/collection';
import { schema } from '../../models/collection';
import ValidationMixin from '../../mixins/validation';
const { Controller } = Ember; const { Controller } = Ember;
export default Controller.extend({ export default Controller.extend(ValidationMixin, {
CollectionValidations,
actions: { actions: {
onSave(changeset) { onSave(changeset) {
changeset.save(); const postSave = () => { this.transitionToRoute('collections.index'); };
this.transitionToRoute('collections.index'); return this.validationSave(changeset, schema, postSave);
}, },
onCancel(changeset) { onCancel(changeset) {
changeset.rollback(); const postCancel = () => { this.transitionToRoute('collections.index'); };
this.transitionToRoute('collections.index'); return this.validationCancel(changeset, postCancel);
}, },
}, },
}); });

View file

@ -0,0 +1,27 @@
import Ember from 'ember';
import CollectionValidations from '../../../validations/collection';
import { schema } from '../../../models/collection';
import ValidationMixin from '../../../mixins/validation';
const { Controller } = Ember;
export default Controller.extend(ValidationMixin, {
CollectionValidations,
actions: {
onSave(changeset) {
const postSave = () => {
// Use the model's ID here because of the ArrayProxy in the route
this.transitionToRoute('collections.detail', this.get('model.id'));
};
return this.validationSave(changeset, schema, postSave);
},
onCancel(changeset) {
const postCancel = () => {
// Use the model's ID here because of the ArrayProxy in the route
return this.transitionToRoute('collections.detail', this.get('model.id'));
};
return this.validationCancel(changeset, postCancel);
},
},
});

View file

@ -0,0 +1,11 @@
import Ember from 'ember';
const { Controller } = Ember;
export default Controller.extend({
actions: {
editCollection() {
this.transitionToRoute('collections.detail.edit', this.get('model'));
},
},
});

30
app/mixins/validation.js Normal file
View file

@ -0,0 +1,30 @@
import Ember from 'ember';
const { Mixin, get } = Ember;
const { keys } = Object;
export default Mixin.create({
validationSave(changeset, schema, postSave) {
return changeset
.cast(keys(schema))
.validate()
.then(() => {
if (changeset.get('isValid')) {
return changeset.save().then(postSave);
}
})
.catch((error) => {
/* eslint-disable no-console */
console.log(error);
/* eslint-enable no-console */
get(this, 'model.errors').forEach(({ attribute, message }) => {
changeset.pushErrors(attribute, message);
});
});
},
validationCancel(changeset, postCancel) {
changeset.rollback();
return postCancel();
},
});

View file

@ -11,7 +11,9 @@ Router.map(function() {
this.route('logout'); this.route('logout');
this.route('collections', function() { this.route('collections', function() {
this.route('create'); this.route('create');
this.route('detail', { path: '/:collection_id' }); this.route('detail', { path: '/:collection_id' }, function() {
this.route('edit');
});
}); });
}); });

View file

@ -0,0 +1,24 @@
import Ember from 'ember';
const { Route, RSVP } = Ember;
export default Route.extend({
model() {
const store = this.get('store');
const model = this.modelFor('collections.detail');
return RSVP.hash({
model: model,
projectOptions: store.findAll('project'),
studyLocationOptions: store.findAll('study-location'),
collectionTypeOptions: store.findAll('collection-type'),
collectionMethodOptions: store.findAll('collection-method'),
});
},
setupController(controller, models) {
this._super(...arguments);
// Unwrap the parent route's listified model
models.model = models.model[0];
controller.setProperties(models);
},
});

View file

@ -1,6 +1,7 @@
{{ {{
collection-create-container collection-create-container
model=model model=model
validations=CollectionValidations
projectOptions=projectOptions projectOptions=projectOptions
studyLocationOptions=studyLocationOptions studyLocationOptions=studyLocationOptions
collectionTypeOptions=collectionTypeOptions collectionTypeOptions=collectionTypeOptions

View file

@ -1 +0,0 @@
{{collection-detail-container model=model}}

View file

@ -0,0 +1,11 @@
{{
collection-create-container
model=model
validations=CollectionValidations
projectOptions=projectOptions
studyLocationOptions=studyLocationOptions
collectionTypeOptions=collectionTypeOptions
collectionMethodOptions=collectionMethodOptions
onSave=(action 'onSave')
onCancel=(action 'onCancel')
}}

View file

@ -0,0 +1,5 @@
{{
collection-detail-container
model=model
editCollection=(action 'editCollection')
}}

View file

@ -4,10 +4,8 @@
onCancel=(action onCancel) as |f| onCancel=(action onCancel) as |f|
}} }}
<div class="well"> <div class="well">
{{#f.content class='form-horizontal'}} {{#f.content class='form'}}
<div class="form-group"> {{#validated-field property='project' label='Project' changeset=changeset}}
<label class="col-md-2 control-label">Project</label>
<div class="col-md-10">
{{#power-select {{#power-select
options=projectOptions options=projectOptions
selected=changeset.project selected=changeset.project
@ -17,12 +15,9 @@
}} }}
{{project.name}} {{project.name}}
{{/power-select}} {{/power-select}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='studyLocation' label='Study location' changeset=changeset}}
<label class="col-md-2 control-label">Study location</label>
<div class="col-md-10">
{{#power-select {{#power-select
options=studyLocationOptions options=studyLocationOptions
selected=changeset.studyLocation selected=changeset.studyLocation
@ -32,12 +27,9 @@
}} }}
{{studyLocation.name}} {{studyLocation.name}}
{{/power-select}} {{/power-select}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
<label class="col-md-2 control-label">Collection type</label>
<div class="col-md-10">
{{#power-select {{#power-select
options=collectionTypeOptions options=collectionTypeOptions
selected=changeset.collectionType selected=changeset.collectionType
@ -47,12 +39,9 @@
}} }}
{{collectionType.name}} {{collectionType.name}}
{{/power-select}} {{/power-select}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}}
<label class="col-md-2 control-label">Collection method</label>
<div class="col-md-10">
{{#power-select {{#power-select
options=collectionMethodOptions options=collectionMethodOptions
selected=changeset.collectionMethod selected=changeset.collectionMethod
@ -62,43 +51,27 @@
}} }}
{{collectionMethod.name}} {{collectionMethod.name}}
{{/power-select}} {{/power-select}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='numberOfTraps' label='Number of traps' changeset=changeset}}
<label class="col-md-2 control-label">Number of traps</label>
<div class="col-md-10">
{{input value=changeset.numberOfTraps type='number' class='form-control'}} {{input value=changeset.numberOfTraps type='number' class='form-control'}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
<label class="col-md-2 control-label">Collection start date</label>
<div class="col-md-10">
{{input value=changeset.collectionStartDate type='date' class='form-control'}} {{input value=changeset.collectionStartDate type='date' class='form-control'}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionStartTime' label='Collection start time' changeset=changeset}}
<label class="col-md-2 control-label">Collection start time</label>
<div class="col-md-10">
{{input value=changeset.collectionStartTime type='time' class='form-control'}} {{input value=changeset.collectionStartTime type='time' class='form-control'}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
<label class="col-md-2 control-label">Collection end date</label>
<div class="col-md-10">
{{input value=changeset.collectionEndDate type='date' class='form-control'}} {{input value=changeset.collectionEndDate type='date' class='form-control'}}
</div> {{/validated-field}}
</div>
<div class="form-group"> {{#validated-field property='collectionEndTime' label='Collection end time' changeset=changeset}}
<label class="col-md-2 control-label">Collection end time</label>
<div class="col-md-10">
{{input value=changeset.collectionEndTime type='time' class='form-control'}} {{input value=changeset.collectionEndTime type='time' class='form-control'}}
</div> {{/validated-field}}
</div>
{{/f.content}} {{/f.content}}

View file

@ -1,3 +1,10 @@
{{
action-button
isPrimary=true
label='Edit Collection'
onClick=(action editCollection)
}}
{{#ccdb-table model=model columns=columns as |c|}} {{#ccdb-table model=model columns=columns as |c|}}
{{#c.grid as |g|}} {{#c.grid as |g|}}
{{g.head}} {{g.head}}

View file

@ -0,0 +1,10 @@
<label class="control-label">{{label}}</label>
{{yield}}
{{#if (get changeset.error property)}}
<ul class="help-block">
{{#each (get (get changeset.error property) "validation") as |message|}}
<li>{{message}}</li>
{{/each}}
</ul>
{{/if}}

View file

@ -0,0 +1,16 @@
import {
validatePresence,
validateNumber,
} from 'ember-changeset-validations/validators';
export default {
project: validatePresence(true),
studyLocation: validatePresence(true),
collectionMethod: validatePresence(true),
collectionType: validatePresence(true),
numberOfTraps: validateNumber({ allowBlank: true, integer: true, positive: true }),
collectionStartDate: validatePresence(true),
collectionEndDate: validatePresence(true),
// TODO: Fix time formats
}