ENH: Collection Edit (parity with reading) (#48)
This commit is contained in:
parent
bfae4422f4
commit
cb3bc081a6
19 changed files with 337 additions and 121 deletions
|
@ -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',
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -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', },
|
||||||
|
|
|
@ -4,5 +4,5 @@ const { Component } = Ember;
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
changeset: null,
|
changesets: null,
|
||||||
});
|
});
|
||||||
|
|
|
@ -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'));
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|
|
@ -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'); };
|
||||||
|
|
|
@ -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);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -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());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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);
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
.catch((error) => {
|
return RSVP.all(promises);
|
||||||
/* eslint-disable no-console */
|
}).then(() => { // don't need the promises, just that they are done.
|
||||||
console.log(error);
|
for (let changeset of changes) {
|
||||||
/* eslint-enable no-console */
|
if (get(changeset, 'isValid')) {
|
||||||
get(this, 'model.errors').forEach(({ attribute, message }) => {
|
let saver = changeset.save().catch((error) => {
|
||||||
changeset.pushErrors(attribute, message);
|
/* eslint-disable no-console */
|
||||||
});
|
console.log(error);
|
||||||
});
|
/* eslint-enable no-console */
|
||||||
|
// TODO: do something with server-side non-attr errors
|
||||||
|
});
|
||||||
|
saves.push(saver);
|
||||||
|
} else {
|
||||||
|
isValid = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return RSVP.all(saves);
|
||||||
|
}).then(() => {
|
||||||
|
if (isValid) { return postSave(); }
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
validationCancel(changeset, postCancel) {
|
validationCancel(changesets, postCancel) {
|
||||||
changeset.rollback();
|
delete changesets['delete'];
|
||||||
return postCancel();
|
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();
|
||||||
|
}
|
||||||
|
} else { // single
|
||||||
|
const changeset = changesets[key];
|
||||||
|
changeset.rollback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return postCancel();
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -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'),
|
||||||
|
|
|
@ -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'),
|
||||||
|
|
||||||
|
speciesNames: computed.mapBy('species', 'commonName'),
|
||||||
|
|
||||||
|
counts: computed.mapBy('collectionSpecies', 'count'),
|
||||||
|
|
||||||
species: computed.mapBy('collectionSpecies', 'species'),
|
|
||||||
speciesNames: computed.mapBy('species', 'commonName'),
|
|
||||||
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();
|
||||||
},
|
},
|
||||||
}));
|
});
|
||||||
|
|
|
@ -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'),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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'),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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')
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -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')
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -1,80 +1,150 @@
|
||||||
{{#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="well">
|
<div class="row">
|
||||||
{{#f.content class='form'}}
|
<div class="col-md-4">
|
||||||
{{#validated-field property='project' label='Project' changeset=changeset}}
|
<div class="well">
|
||||||
{{#power-select
|
{{#f.content class='form'}}
|
||||||
options=projectOptions
|
{{#with changesets.model as |changeset|}}
|
||||||
selected=changeset.project
|
{{#validated-field property='project' label='Project' changeset=changeset}}
|
||||||
onchange=(action (mut changeset.project))
|
{{#power-select
|
||||||
searchField='name'
|
options=options.projects
|
||||||
as |project|
|
selected=changeset.project
|
||||||
}}
|
onchange=(action (mut changeset.project))
|
||||||
{{project.name}}
|
searchField='name'
|
||||||
{{/power-select}}
|
as |project|
|
||||||
{{/validated-field}}
|
}}
|
||||||
|
{{project.name}}
|
||||||
|
{{/power-select}}
|
||||||
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='studyLocation' label='Study location' changeset=changeset}}
|
{{#validated-field property='adfgPermit' label='ADFG Permit' changeset=changeset}}
|
||||||
{{#power-select
|
{{#power-select
|
||||||
options=studyLocationOptions
|
options=options.adfgPermits
|
||||||
selected=changeset.studyLocation
|
selected=changeset.adfgPermit
|
||||||
onchange=(action (mut changeset.studyLocation))
|
onchange=(action (mut changeset.adfgPermit))
|
||||||
searchField='name'
|
searchField='name'
|
||||||
as |studyLocation|
|
as |adfgPermit|
|
||||||
}}
|
}}
|
||||||
{{studyLocation.name}}
|
{{adfgPermit.name}}
|
||||||
{{/power-select}}
|
{{/power-select}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
|
{{#validated-field property='studyLocation' label='Study location' changeset=changeset}}
|
||||||
{{#power-select
|
{{#power-select
|
||||||
options=collectionTypeOptions
|
options=options.studyLocations
|
||||||
selected=changeset.collectionType
|
selected=changeset.studyLocation
|
||||||
onchange=(action (mut changeset.collectionType))
|
onchange=(action (mut changeset.studyLocation))
|
||||||
searchField='name'
|
searchField='name'
|
||||||
as |collectionType|
|
as |studyLocation|
|
||||||
}}
|
}}
|
||||||
{{collectionType.name}}
|
{{studyLocation.name}}
|
||||||
{{/power-select}}
|
{{/power-select}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}}
|
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
|
||||||
{{#power-select
|
{{#power-select
|
||||||
options=collectionMethodOptions
|
options=options.collectionTypes
|
||||||
selected=changeset.collectionMethod
|
selected=changeset.collectionType
|
||||||
onchange=(action (mut changeset.collectionMethod))
|
onchange=(action (mut changeset.collectionType))
|
||||||
searchField='name'
|
searchField='name'
|
||||||
as |collectionMethod|
|
as |collectionType|
|
||||||
}}
|
}}
|
||||||
{{collectionMethod.name}}
|
{{collectionType.name}}
|
||||||
{{/power-select}}
|
{{/power-select}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='numberOfTraps' label='Number of traps' changeset=changeset}}
|
{{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}}
|
||||||
{{input value=changeset.numberOfTraps type='number' class='form-control'}}
|
{{#power-select
|
||||||
{{/validated-field}}
|
options=options.collectionMethods
|
||||||
|
selected=changeset.collectionMethod
|
||||||
|
onchange=(action (mut changeset.collectionMethod))
|
||||||
|
searchField='name'
|
||||||
|
as |collectionMethod|
|
||||||
|
}}
|
||||||
|
{{collectionMethod.name}}
|
||||||
|
{{/power-select}}
|
||||||
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
|
{{#validated-field property='numberOfTraps' label='Number of traps' changeset=changeset}}
|
||||||
{{input value=changeset.collectionStartDate type='date' class='form-control'}}
|
{{input value=changeset.numberOfTraps type='number' class='form-control'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionStartTime' label='Collection start time' changeset=changeset}}
|
{{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
|
||||||
{{input value=changeset.collectionStartTime type='time' class='form-control'}}
|
{{input value=changeset.collectionStartDate type='date' class='form-control'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
|
{{#validated-field property='collectionStartTime' label='Collection start time' changeset=changeset}}
|
||||||
{{input value=changeset.collectionEndDate type='date' class='form-control'}}
|
{{input value=changeset.collectionStartTime type='time' class='form-control'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionEndTime' label='Collection end time' changeset=changeset}}
|
{{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
|
||||||
{{input value=changeset.collectionEndTime type='time' class='form-control'}}
|
{{input value=changeset.collectionEndDate type='date' class='form-control'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{/f.content}}
|
{{#validated-field property='collectionEndTime' label='Collection end time' changeset=changeset}}
|
||||||
|
{{input value=changeset.collectionEndTime type='time' class='form-control'}}
|
||||||
{{f.save}} {{f.cancel}}
|
{{/validated-field}}
|
||||||
|
{{/with}}
|
||||||
|
{{/f.content}}
|
||||||
|
</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>
|
</div>
|
||||||
|
{{f.save}} {{f.cancel}}
|
||||||
{{/crud-form}}
|
{{/crud-form}}
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -1,4 +1,7 @@
|
||||||
<label class="control-label">{{label}}</label>
|
{{#if hasLabel}}
|
||||||
|
<label class="control-label">{{label}}</label>
|
||||||
|
{{/if}}
|
||||||
|
|
||||||
{{yield}}
|
{{yield}}
|
||||||
|
|
||||||
{{#if (get changeset.error property)}}
|
{{#if (get changeset.error property)}}
|
||||||
|
|
|
@ -8,6 +8,6 @@ export default Transform.extend({
|
||||||
},
|
},
|
||||||
|
|
||||||
serialize(deserialized) {
|
serialize(deserialized) {
|
||||||
return deserialized;
|
return deserialized === '' ? null : deserialized;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
12
app/validations/collection-species.js
Normal file
12
app/validations/collection-species.js
Normal 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),
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue