Compare commits
22 commits
Author | SHA1 | Date | |
---|---|---|---|
7548aafdb7 | |||
0cd3cd96f1 | |||
22be697ef6 | |||
bf6023a7b4 | |||
6f6148df9d | |||
437be4fb13 | |||
9d9e67868f | |||
86ab62cb0f | |||
234dc3a34c | |||
d2c485af88 | |||
ab30692021 | |||
2ead72c552 | |||
467a8d8b64 | |||
a32d147c1c | |||
c87bd953d9 | |||
6f01fbf00f | |||
eb4537afb1 | |||
39f4789a61 | |||
56f8796eaf | |||
dcbad4d54b | |||
a4264ac16c | |||
4dbfcfa98b |
60 changed files with 22387 additions and 339 deletions
10
.travis.yml
10
.travis.yml
|
@ -3,11 +3,17 @@ branches:
|
||||||
only:
|
only:
|
||||||
- master
|
- master
|
||||||
node_js:
|
node_js:
|
||||||
- '6'
|
- "9"
|
||||||
sudo: false
|
sudo: required
|
||||||
|
dist: trusty
|
||||||
|
addons:
|
||||||
|
chrome: stable
|
||||||
cache:
|
cache:
|
||||||
directories:
|
directories:
|
||||||
- "$HOME/.npm"
|
- "$HOME/.npm"
|
||||||
|
env:
|
||||||
|
global:
|
||||||
|
- JOBS=1
|
||||||
before_install:
|
before_install:
|
||||||
- npm config set spin false
|
- npm config set spin false
|
||||||
- npm install -g phantomjs-prebuilt
|
- npm install -g phantomjs-prebuilt
|
||||||
|
|
|
@ -10,7 +10,7 @@ You will need the following things properly installed on your computer.
|
||||||
* [Git](https://git-scm.com/)
|
* [Git](https://git-scm.com/)
|
||||||
* [Node.js](https://nodejs.org/) (with NPM)
|
* [Node.js](https://nodejs.org/) (with NPM)
|
||||||
* [Ember CLI](https://ember-cli.com/)
|
* [Ember CLI](https://ember-cli.com/)
|
||||||
* [PhantomJS](http://phantomjs.org/)
|
* [Google Chrome](https://google.com/chrome/)
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,9 @@
|
||||||
import Ember from 'ember';
|
import Application from '@ember/application';
|
||||||
import Resolver from './resolver';
|
import Resolver from './resolver';
|
||||||
import loadInitializers from 'ember-load-initializers';
|
import loadInitializers from 'ember-load-initializers';
|
||||||
import config from './config/environment';
|
import config from './config/environment';
|
||||||
|
|
||||||
let App;
|
const App = Application.extend({
|
||||||
|
|
||||||
App = Ember.Application.extend({
|
|
||||||
modulePrefix: config.modulePrefix,
|
modulePrefix: config.modulePrefix,
|
||||||
podModulePrefix: config.podModulePrefix,
|
podModulePrefix: config.podModulePrefix,
|
||||||
Resolver
|
Resolver
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
import Ember from 'ember';
|
import { Promise } from 'rsvp';
|
||||||
|
import $ from 'jquery';
|
||||||
|
import { get } from '@ember/object';
|
||||||
|
import { isEmpty } from '@ember/utils';
|
||||||
|
import { run } from '@ember/runloop';
|
||||||
import BaseAuthenticator from 'ember-simple-auth/authenticators/base';
|
import BaseAuthenticator from 'ember-simple-auth/authenticators/base';
|
||||||
import config from '../config/environment';
|
import config from '../config/environment';
|
||||||
|
|
||||||
const { RSVP: { Promise }, $, get, isEmpty, run } = Ember;
|
|
||||||
|
|
||||||
export default BaseAuthenticator.extend({
|
export default BaseAuthenticator.extend({
|
||||||
serverTokenEndpoint: `${config.APP.API_HOST}/api/auth/login/`,
|
serverTokenEndpoint: `${config.APP.API_HOST}/api/auth/login/`,
|
||||||
tokenAttributeName: 'data.attributes.auth-token',
|
tokenAttributeName: 'data.attributes.auth-token',
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
import Ember from 'ember';
|
import { isEmpty } from '@ember/utils';
|
||||||
|
import { get } from '@ember/object';
|
||||||
import BaseAuthorizer from 'ember-simple-auth/authorizers/base';
|
import BaseAuthorizer from 'ember-simple-auth/authorizers/base';
|
||||||
|
|
||||||
const { isEmpty, get } = Ember;
|
|
||||||
|
|
||||||
export default BaseAuthorizer.extend({
|
export default BaseAuthorizer.extend({
|
||||||
authorize(data, block) {
|
authorize(data, block) {
|
||||||
const accessToken = get(data, 'data.attributes.auth-token');
|
const accessToken = get(data, 'data.attributes.auth-token');
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
tagName: 'a',
|
tagName: 'a',
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({});
|
export default Component.extend({});
|
||||||
|
|
|
@ -1,5 +1,3 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({ });
|
export default Component.extend({ });
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
import { alias } from '@ember/object/computed';
|
||||||
const { Component, computed: { alias }, computed } = Ember;
|
import { computed } from '@ember/object';
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
|
@ -8,7 +8,7 @@ export default Component.extend({
|
||||||
|
|
||||||
// COMPUTED
|
// COMPUTED
|
||||||
meta: alias('model.meta'),
|
meta: alias('model.meta'),
|
||||||
links: alias('model.links'),
|
links: alias('meta.links'),
|
||||||
|
|
||||||
currentPage: alias('meta.pagination.page'),
|
currentPage: alias('meta.pagination.page'),
|
||||||
totalRecords: alias('meta.pagination.count'),
|
totalRecords: alias('meta.pagination.count'),
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
import Table from 'ember-light-table';
|
import Table from 'ember-light-table';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
model: null,
|
model: null,
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
import Ember from 'ember';
|
import { getProperties, set } from '@ember/object';
|
||||||
|
import Component from '@ember/component';
|
||||||
|
import { inject as service } from '@ember/service';
|
||||||
|
import { debounce } from '@ember/runloop';
|
||||||
|
import RSVP from 'rsvp';
|
||||||
import Changeset from 'ember-changeset';
|
import Changeset from 'ember-changeset';
|
||||||
import lookupValidator from 'ember-changeset-validations';
|
import lookupValidator from 'ember-changeset-validations';
|
||||||
|
import config from 'ccdb-web/config/environment';
|
||||||
const { Component, inject: { service } } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
store: service(),
|
store: service(),
|
||||||
|
@ -11,81 +14,80 @@ export default Component.extend({
|
||||||
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');
|
||||||
|
const hasMany = this.get('hasMany');
|
||||||
|
|
||||||
let changesets = {};
|
let changesets = {};
|
||||||
changesets['new'] = [];
|
changesets['new'] = [];
|
||||||
changesets['delete'] = [];
|
changesets['delete'] = [];
|
||||||
changesets['hasMany'] = [];
|
changesets['hasMany'] = {};
|
||||||
changesets['model'] = new Changeset(model,
|
changesets['model'] = new Changeset(model,
|
||||||
lookupValidator(validations['collection']),
|
lookupValidator(validations['collection']),
|
||||||
validations['collection']);
|
validations['collection']);
|
||||||
|
|
||||||
let collectionSpeciesChangesets = [];
|
hasMany.forEach((hasMany) => {
|
||||||
const collectionSpecies = model.get('collectionSpecies');
|
let relatedChangesets = [];
|
||||||
collectionSpecies.forEach((cs) => {
|
let validation = validations[hasMany];
|
||||||
const changeset = new Changeset(cs,
|
const related = model.get(hasMany);
|
||||||
lookupValidator(validations['collectionSpecies']),
|
related.forEach((r) => {
|
||||||
validations['collectionSpecies']);
|
const changeset = new Changeset(r, lookupValidator(validation),
|
||||||
collectionSpeciesChangesets.push({ model: cs, changeset: changeset });
|
validation);
|
||||||
|
relatedChangesets.push({ model: r, changeset: changeset });
|
||||||
|
});
|
||||||
|
changesets['hasMany'][hasMany] = relatedChangesets;
|
||||||
});
|
});
|
||||||
changesets['hasMany']['collectionSpecies'] = collectionSpeciesChangesets;
|
|
||||||
|
|
||||||
let datasheetsChangesets = [];
|
|
||||||
const datasheets = model.get('datasheets');
|
|
||||||
datasheets.forEach((d) => {
|
|
||||||
const changeset = new Changeset(d,
|
|
||||||
lookupValidator(validations['datasheet']),
|
|
||||||
validations['datasheet']);
|
|
||||||
datasheetsChangesets.push({ model: d, changeset: changeset });
|
|
||||||
});
|
|
||||||
changesets['hasMany']['datasheets'] = datasheetsChangesets;
|
|
||||||
|
|
||||||
this.set('changesets', changesets);
|
this.set('changesets', changesets);
|
||||||
|
this.set('newStudyLocationAdmin', `${config.APP.API_HOST}/admin/locations/studylocation/add/`);
|
||||||
},
|
},
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
addCollectionSpecies() {
|
addHasMany(modelName, relatedName) {
|
||||||
const store = this.get('store');
|
const store = this.get('store');
|
||||||
let changesets = this.get('changesets');
|
let changesets = this.get('changesets');
|
||||||
const validations = this.get('validations');
|
const validations = this.get('validations');
|
||||||
const collection = this.get('model');
|
const validation = validations[relatedName];
|
||||||
const cs = store.createRecord('collection-species', { collection: collection });
|
const model = this.get('model');
|
||||||
collection.get('collectionSpecies').pushObject(cs);
|
const related = store.createRecord(modelName, { collection: model });
|
||||||
changesets['new'].pushObject(cs);
|
model.get(relatedName).pushObject(related);
|
||||||
const changeset = new Changeset(cs,
|
changesets['new'].pushObject(related);
|
||||||
lookupValidator(validations['collectionSpecies']),
|
const changeset = new Changeset(related, lookupValidator(validation), validation);
|
||||||
validations['collectionSpecies']);
|
changesets['hasMany'][relatedName].pushObject({ model: related, changeset: changeset });
|
||||||
changesets['hasMany']['collectionSpecies'].pushObject({ model: cs, changeset: changeset });
|
|
||||||
},
|
},
|
||||||
|
|
||||||
deleteCollectionSpecies(changesetRecord) {
|
deleteHasMany(changesetRecord, relatedName) {
|
||||||
let changesets = this.get('changesets');
|
let changesets = this.get('changesets');
|
||||||
changesets['delete'].pushObject(changesetRecord.model);
|
changesets['delete'].pushObject(changesetRecord.model);
|
||||||
changesets['hasMany']['collectionSpecies'].removeObject(changesetRecord);
|
changesets['hasMany'][relatedName].removeObject(changesetRecord);
|
||||||
|
},
|
||||||
|
|
||||||
|
// Gross, this side-effects by saving immediately. Someday I should clean
|
||||||
|
// this up, but for now, you have been warned.
|
||||||
|
addOption(relatedModelName, optionName, collectionAttrName, relatedAttrName, term) {
|
||||||
|
const props = getProperties(this, 'store', 'options', 'changesets');
|
||||||
|
const { store, options, changesets: { model } } = props;
|
||||||
|
let payload = {};
|
||||||
|
payload[relatedAttrName] = term;
|
||||||
|
const record = store.createRecord(relatedModelName, payload)
|
||||||
|
record.save().then((record) => {
|
||||||
|
set(options, optionName, store.peekAll(relatedModelName));
|
||||||
|
set(model, collectionAttrName, record);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
updateDatasheet(changeset, event) {
|
updateDatasheet(changeset, event) {
|
||||||
changeset.set('datasheet', event.target.files[0]);
|
changeset.set('datasheet', event.target.files[0]);
|
||||||
},
|
},
|
||||||
|
|
||||||
addDatasheet() {
|
searchStudyLocation(term) {
|
||||||
const store = this.get('store');
|
return new RSVP.Promise((resolve, reject) => {
|
||||||
let changesets = this.get('changesets');
|
debounce(this, this._performSearch, 'study-location', { page_size: 500, code: term }, resolve, reject, 400);
|
||||||
const validations = this.get('validations');
|
});
|
||||||
const collection = this.get('model');
|
|
||||||
const d = store.createRecord('datasheet-attachment', { collection: collection });
|
|
||||||
collection.get('datasheets').pushObject(d);
|
|
||||||
changesets['new'].pushObject(d);
|
|
||||||
const changeset = new Changeset(d,
|
|
||||||
lookupValidator(validations['datasheets']),
|
|
||||||
validations['datasheets']);
|
|
||||||
changesets['hasMany']['datasheets'].pushObject({ model: d, changeset: changeset });
|
|
||||||
},
|
|
||||||
|
|
||||||
deleteDatasheet(changesetRecord) {
|
|
||||||
let changesets = this.get('changesets');
|
|
||||||
changesets['delete'].pushObject(changesetRecord.model);
|
|
||||||
changesets['hasMany']['datasheets'].removeObject(changesetRecord);
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_performSearch(model, payload, resolve, reject) {
|
||||||
|
this.get('store').query(model, payload).then((results) => {
|
||||||
|
resolve(results);
|
||||||
|
}, reject);
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
|
@ -24,6 +22,13 @@ export default Component.extend({
|
||||||
{ label: 'Species', valuePath: 'species.commonName' },
|
{ label: 'Species', valuePath: 'species.commonName' },
|
||||||
{ label: 'Count', valuePath: 'count' },
|
{ label: 'Count', valuePath: 'count' },
|
||||||
{ label: 'Count Estimated?', valuePath: 'countEstimated' },
|
{ label: 'Count Estimated?', valuePath: 'countEstimated' },
|
||||||
{ label: 'Sex', valuePath: 'sex' },
|
{ label: 'Sex', valuePath: 'sex.name' },
|
||||||
|
],
|
||||||
|
|
||||||
|
envMeasColumns: [
|
||||||
|
{ label: 'Date Measured', valuePath: 'dateMeasured', },
|
||||||
|
{ label: 'Time Measured', valuePath: 'timeMeasured', },
|
||||||
|
{ label: 'Water Temp (deg C)', valuePath: 'waterTempC', },
|
||||||
|
{ label: 'Air Temp (deg C)', valuePath: 'airTempC', },
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
|
|
23
app/components/confirm-button.js
Normal file
23
app/components/confirm-button.js
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
import Component from '@ember/component';
|
||||||
|
|
||||||
|
export default Component.extend({
|
||||||
|
tagName: 'span',
|
||||||
|
showConfirm: false,
|
||||||
|
initialLabel: 'LABEL',
|
||||||
|
confirmLabel: 'CONFIRM LABEL',
|
||||||
|
cancelLabel: 'Cancel',
|
||||||
|
|
||||||
|
actions: {
|
||||||
|
initial() {
|
||||||
|
this.set('showConfirm', true);
|
||||||
|
},
|
||||||
|
|
||||||
|
cancel() {
|
||||||
|
this.set('showConfirm', false);
|
||||||
|
},
|
||||||
|
|
||||||
|
confirm() {
|
||||||
|
this.get('onClick')();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
// ARGS
|
// ARGS
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
tagName: 'form',
|
tagName: 'form',
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
|
||||||
const { Component } = Ember;
|
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
classNames: ['spinner'],
|
classNames: ['spinner'],
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import Ember from 'ember';
|
import Component from '@ember/component';
|
||||||
|
import { get, computed } from '@ember/object';
|
||||||
const { Component, computed, get, isEmpty } = Ember;
|
import { isEmpty } from '@ember/utils';
|
||||||
|
|
||||||
export default Component.extend({
|
export default Component.extend({
|
||||||
classNames: ['form-group'],
|
classNames: ['form-group'],
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
import { inject as service } from '@ember/service';
|
||||||
const { Controller, inject: { service } } = Ember;
|
|
||||||
|
|
||||||
export default Controller.extend({
|
export default Controller.extend({
|
||||||
session: service('session'),
|
session: service('session'),
|
||||||
|
|
|
@ -1,19 +1,23 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
import { computed } from '@ember/object';
|
||||||
import CollectionValidations from 'ccdb-web/validations/collection';
|
import CollectionValidations from 'ccdb-web/validations/collection';
|
||||||
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
|
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
|
||||||
|
import CollectionMeasurementValidations from 'ccdb-web/validations/collection-measurement';
|
||||||
import DatasheetValidations from 'ccdb-web/validations/datasheet';
|
import DatasheetValidations from 'ccdb-web/validations/datasheet';
|
||||||
import ValidationMixin from 'ccdb-web/mixins/validation';
|
import ValidationMixin from 'ccdb-web/mixins/validation';
|
||||||
|
|
||||||
const { Controller, computed } = Ember;
|
|
||||||
|
|
||||||
export default Controller.extend(ValidationMixin, {
|
export default Controller.extend(ValidationMixin, {
|
||||||
CollectionValidations,
|
CollectionValidations,
|
||||||
CollectionSpeciesValidations,
|
CollectionSpeciesValidations,
|
||||||
DatasheetValidations,
|
DatasheetValidations,
|
||||||
|
CollectionMeasurementValidations,
|
||||||
|
|
||||||
|
hasMany: ['collectionSpecies', 'datasheets', 'envMeasurements'],
|
||||||
|
|
||||||
options: computed('projectOptions', 'studyLocationOptions',
|
options: computed('projectOptions', 'studyLocationOptions',
|
||||||
'collectionTypeOptions', 'collectionMethodOptions',
|
'collectionTypeOptions', 'collectionMethodOptions',
|
||||||
'speciesOptions', 'adfgPermitOptions', function() {
|
'speciesOptions', 'adfgPermitOptions', 'sexOptions',
|
||||||
|
function() {
|
||||||
return {
|
return {
|
||||||
projects: this.get('projectOptions'),
|
projects: this.get('projectOptions'),
|
||||||
studyLocations: this.get('studyLocationOptions'),
|
studyLocations: this.get('studyLocationOptions'),
|
||||||
|
@ -21,17 +25,22 @@ export default Controller.extend(ValidationMixin, {
|
||||||
collectionMethods: this.get('collectionMethodOptions'),
|
collectionMethods: this.get('collectionMethodOptions'),
|
||||||
species: this.get('speciesOptions'),
|
species: this.get('speciesOptions'),
|
||||||
adfgPermits: this.get('adfgPermitOptions'),
|
adfgPermits: this.get('adfgPermitOptions'),
|
||||||
|
sexes: this.get('sexOptions'),
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
onSave(changeset) {
|
onSave(changeset) {
|
||||||
const postSave = () => { this.transitionToRoute('collections.index'); };
|
const postSave = () => { this.transitionToRoute('collections.index'); };
|
||||||
return this.validationSave(changeset, postSave);
|
return this.transitionToRoute('loading').then(() => {
|
||||||
|
return this.validationSave(changeset, postSave);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
onCancel(changeset) {
|
onCancel(changeset) {
|
||||||
const postCancel = () => { this.transitionToRoute('collections.index'); };
|
const postCancel = () => { this.transitionToRoute('collections.index'); };
|
||||||
return this.validationCancel(changeset, postCancel);
|
return this.transitionToRoute('loading').then(() => {
|
||||||
|
return this.validationCancel(changeset, postCancel);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,19 +1,23 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
import { computed } from '@ember/object';
|
||||||
import CollectionValidations from 'ccdb-web/validations/collection';
|
import CollectionValidations from 'ccdb-web/validations/collection';
|
||||||
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
|
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
|
||||||
|
import CollectionMeasurementValidations from 'ccdb-web/validations/collection-measurement';
|
||||||
import DatasheetValidations from 'ccdb-web/validations/datasheet';
|
import DatasheetValidations from 'ccdb-web/validations/datasheet';
|
||||||
import ValidationMixin from 'ccdb-web/mixins/validation';
|
import ValidationMixin from 'ccdb-web/mixins/validation';
|
||||||
|
|
||||||
const { Controller, computed } = Ember;
|
|
||||||
|
|
||||||
export default Controller.extend(ValidationMixin, {
|
export default Controller.extend(ValidationMixin, {
|
||||||
CollectionValidations,
|
CollectionValidations,
|
||||||
CollectionSpeciesValidations,
|
CollectionSpeciesValidations,
|
||||||
DatasheetValidations,
|
DatasheetValidations,
|
||||||
|
CollectionMeasurementValidations,
|
||||||
|
|
||||||
|
hasMany: ['collectionSpecies', 'datasheets', 'envMeasurements'],
|
||||||
|
|
||||||
options: computed('projectOptions', 'studyLocationOptions',
|
options: computed('projectOptions', 'studyLocationOptions',
|
||||||
'collectionTypeOptions', 'collectionMethodOptions',
|
'collectionTypeOptions', 'collectionMethodOptions',
|
||||||
'speciesOptions', 'adfgPermitOptions', function() {
|
'speciesOptions', 'adfgPermitOptions', 'sexOptions',
|
||||||
|
function() {
|
||||||
return {
|
return {
|
||||||
projects: this.get('projectOptions'),
|
projects: this.get('projectOptions'),
|
||||||
studyLocations: this.get('studyLocationOptions'),
|
studyLocations: this.get('studyLocationOptions'),
|
||||||
|
@ -21,6 +25,7 @@ export default Controller.extend(ValidationMixin, {
|
||||||
collectionMethods: this.get('collectionMethodOptions'),
|
collectionMethods: this.get('collectionMethodOptions'),
|
||||||
species: this.get('speciesOptions'),
|
species: this.get('speciesOptions'),
|
||||||
adfgPermits: this.get('adfgPermitOptions'),
|
adfgPermits: this.get('adfgPermitOptions'),
|
||||||
|
sexes: this.get('sexOptions'),
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
@ -30,14 +35,18 @@ export default Controller.extend(ValidationMixin, {
|
||||||
// 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(changesets, postSave);
|
return this.transitionToRoute('loading').then(() => {
|
||||||
|
return this.validationSave(changesets, postSave);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
onCancel(changesets) {
|
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(changesets, postCancel);
|
return this.transitionToRoute('loading').then(() => {
|
||||||
|
return this.validationCancel(changesets, postCancel);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,11 +1,14 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
|
||||||
const { Controller } = Ember;
|
|
||||||
|
|
||||||
export default Controller.extend({
|
export default Controller.extend({
|
||||||
actions: {
|
actions: {
|
||||||
editCollection() {
|
editCollection() {
|
||||||
this.transitionToRoute('collections.detail.edit', this.get('model'));
|
this.transitionToRoute('collections.detail.edit', this.get('model'));
|
||||||
},
|
},
|
||||||
|
deleteCollection() {
|
||||||
|
this.get('model')[0].destroyRecord().then(() => {
|
||||||
|
this.transitionToRoute('collections');
|
||||||
|
});
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
import { set, get, computed } from '@ember/object';
|
||||||
const { Controller, computed, get, set } = Ember;
|
|
||||||
|
|
||||||
|
|
||||||
export default Controller.extend({
|
export default Controller.extend({
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Controller from '@ember/controller';
|
||||||
|
import { inject as service } from '@ember/service';
|
||||||
const { Controller, inject: { service } } = Ember;
|
|
||||||
|
|
||||||
export default Controller.extend({
|
export default Controller.extend({
|
||||||
session: service(),
|
session: service(),
|
||||||
|
|
|
@ -17,8 +17,8 @@
|
||||||
<body>
|
<body>
|
||||||
{{content-for "body"}}
|
{{content-for "body"}}
|
||||||
|
|
||||||
<script src="{{rootURL}}assets/vendor.js"></script>
|
<script integrity="" src="{{rootURL}}assets/vendor.js"></script>
|
||||||
<script src="{{rootURL}}assets/ccdb-web.js"></script>
|
<script integrity="" src="{{rootURL}}assets/ccdb-web.js"></script>
|
||||||
|
|
||||||
{{content-for "body-footer"}}
|
{{content-for "body-footer"}}
|
||||||
</body>
|
</body>
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Mixin from '@ember/object/mixin';
|
||||||
|
import { isArray } from '@ember/array';
|
||||||
const { Mixin, isArray } = Ember;
|
|
||||||
const { keys } = Object;
|
const { keys } = Object;
|
||||||
|
|
||||||
// Portions borrowed from https://github.com/funtusov/ember-cli-form-data
|
// Portions borrowed from https://github.com/funtusov/ember-cli-form-data
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
import Ember from 'ember';
|
import Mixin from '@ember/object/mixin';
|
||||||
|
import { get } from '@ember/object';
|
||||||
const { Mixin, get, RSVP } = Ember;
|
import RSVP from 'rsvp';
|
||||||
const { keys } = Object;
|
const { keys } = Object;
|
||||||
const { isArray } = Array;
|
|
||||||
|
|
||||||
export default Mixin.create({
|
export default Mixin.create({
|
||||||
validationSave(changesets, postSave) {
|
validationSave(changesets, postSave) {
|
||||||
|
@ -55,9 +54,13 @@ export default Mixin.create({
|
||||||
for (const model of changesets[key]) {
|
for (const model of changesets[key]) {
|
||||||
model.destroyRecord();
|
model.destroyRecord();
|
||||||
}
|
}
|
||||||
} else if (isArray(changesets[key])) { // hasMany
|
} else if (key === 'hasMany') {
|
||||||
for (const { changeset } of changesets[key]) {
|
const hasMany = changesets[key];
|
||||||
changeset.rollback();
|
for (const hasManyKey of keys(changesets[key])) {
|
||||||
|
const hasManyChangesets = hasMany[hasManyKey];
|
||||||
|
for (const changeset of hasManyChangesets) {
|
||||||
|
changeset.rollback();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else { // single
|
} else { // single
|
||||||
const changeset = changesets[key];
|
const changeset = changesets[key];
|
||||||
|
|
12
app/models/collection-measurement.js
Normal file
12
app/models/collection-measurement.js
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
import DS from 'ember-data';
|
||||||
|
|
||||||
|
const { Model, attr, belongsTo } = DS;
|
||||||
|
|
||||||
|
export default Model.extend({
|
||||||
|
dateMeasured: attr('ccdb-date'),
|
||||||
|
timeMeasured: attr('string'),
|
||||||
|
waterTempC: attr('number'),
|
||||||
|
airTempC: attr('number'),
|
||||||
|
|
||||||
|
collection: belongsTo('collection'),
|
||||||
|
});
|
|
@ -3,7 +3,7 @@ import DS from 'ember-data';
|
||||||
const { Model, attr, belongsTo } = DS;
|
const { Model, attr, belongsTo } = DS;
|
||||||
|
|
||||||
export default Model.extend({
|
export default Model.extend({
|
||||||
sex: attr('string'),
|
sex: belongsTo('sex'),
|
||||||
count: attr('number'),
|
count: attr('number'),
|
||||||
countEstimated: attr('boolean', { defaultValue: false }),
|
countEstimated: attr('boolean', { defaultValue: false }),
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,17 @@
|
||||||
import Ember from 'ember';
|
import { mapBy } from '@ember/object/computed';
|
||||||
|
import { computed } from '@ember/object';
|
||||||
import DS from 'ember-data';
|
import DS from 'ember-data';
|
||||||
|
|
||||||
const { computed } = Ember;
|
|
||||||
const { Model, attr, belongsTo, hasMany } = DS;
|
const { Model, attr, belongsTo, hasMany } = DS;
|
||||||
|
|
||||||
export default Model.extend({
|
export default Model.extend({
|
||||||
displayName: attr('string'),
|
displayName: attr('string'),
|
||||||
numberOfTraps: attr('number'),
|
numberOfTraps: attr('number'),
|
||||||
collectionStartDate: attr('string-null-to-empty'),
|
collectionStartDate: attr('ccdb-date'),
|
||||||
collectionStartTime: attr('string-null-to-empty'),
|
collectionStartTime: attr('string-null-to-empty'),
|
||||||
collectionEndDate: attr('string-null-to-empty'),
|
collectionEndDate: attr('ccdb-date'),
|
||||||
collectionEndTime: attr('string-null-to-empty'),
|
collectionEndTime: attr('string-null-to-empty'),
|
||||||
|
notes: attr('string', { defaultValue: '' }),
|
||||||
|
|
||||||
project: belongsTo('project'),
|
project: belongsTo('project'),
|
||||||
studyLocation: belongsTo('study-location'),
|
studyLocation: belongsTo('study-location'),
|
||||||
|
@ -20,13 +21,14 @@ export default Model.extend({
|
||||||
|
|
||||||
collectionSpecies: hasMany('collection-species'),
|
collectionSpecies: hasMany('collection-species'),
|
||||||
datasheets: hasMany('datasheet-attachment'),
|
datasheets: hasMany('datasheet-attachment'),
|
||||||
|
envMeasurements: hasMany('collection-measurement'),
|
||||||
|
|
||||||
// computed
|
// computed
|
||||||
species: computed.mapBy('collectionSpecies', 'species'),
|
species: mapBy('collectionSpecies', 'species'),
|
||||||
|
|
||||||
speciesNames: computed.mapBy('species', 'commonName'),
|
speciesNames: mapBy('species', 'commonName'),
|
||||||
|
|
||||||
counts: computed.mapBy('collectionSpecies', 'count'),
|
counts: mapBy('collectionSpecies', 'count'),
|
||||||
|
|
||||||
speciesAndCounts: computed('speciesNames', 'counts', function() {
|
speciesAndCounts: computed('speciesNames', 'counts', function() {
|
||||||
const speciesNames = this.get('speciesNames');
|
const speciesNames = this.get('speciesNames');
|
||||||
|
|
8
app/models/sex.js
Normal file
8
app/models/sex.js
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
import DS from 'ember-data';
|
||||||
|
|
||||||
|
const { Model, attr } = DS;
|
||||||
|
|
||||||
|
export default Model.extend({
|
||||||
|
name: attr('string'),
|
||||||
|
sortOrder: attr('number'),
|
||||||
|
});
|
|
@ -1,7 +1,7 @@
|
||||||
import Ember from 'ember';
|
import EmberRouter from '@ember/routing/router';
|
||||||
import config from './config/environment';
|
import config from './config/environment';
|
||||||
|
|
||||||
const Router = Ember.Router.extend({
|
const Router = EmberRouter.extend({
|
||||||
location: config.locationType,
|
location: config.locationType,
|
||||||
rootURL: config.rootURL
|
rootURL: config.rootURL
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin';
|
import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin';
|
||||||
|
|
||||||
const { Route } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend(ApplicationRouteMixin, {});
|
export default Route.extend(ApplicationRouteMixin, {});
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
|
import RSVP from 'rsvp';
|
||||||
const { Route, RSVP } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend({
|
export default Route.extend({
|
||||||
model() {
|
model() {
|
||||||
|
@ -8,11 +7,12 @@ export default Route.extend({
|
||||||
return RSVP.hash({
|
return RSVP.hash({
|
||||||
model: store.createRecord('collection'),
|
model: store.createRecord('collection'),
|
||||||
projectOptions: store.findAll('project'),
|
projectOptions: store.findAll('project'),
|
||||||
studyLocationOptions: store.findAll('study-location'),
|
studyLocationOptions: store.query('study-location', { page_size: 500 }),
|
||||||
collectionTypeOptions: store.findAll('collection-type'),
|
collectionTypeOptions: store.findAll('collection-type'),
|
||||||
collectionMethodOptions: store.findAll('collection-method'),
|
collectionMethodOptions: store.findAll('collection-method'),
|
||||||
speciesOptions: store.findAll('species'),
|
speciesOptions: store.query('species', { page_size: 500 }),
|
||||||
adfgPermitOptions: store.findAll('adfg-permit'),
|
adfgPermitOptions: store.findAll('adfg-permit'),
|
||||||
|
sexOptions: store.findAll('sex'),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -1,12 +1,11 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
|
import RSVP from 'rsvp';
|
||||||
const { Route, RSVP } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend({
|
export default Route.extend({
|
||||||
model(params) {
|
model(params) {
|
||||||
return RSVP.all([
|
return RSVP.all([
|
||||||
this.get('store').findRecord('collection', params.collection_id, {
|
this.get('store').findRecord('collection', params.collection_id, {
|
||||||
include: 'collection-species,datasheets',
|
include: 'collection-species,datasheets,env-measurements',
|
||||||
})
|
})
|
||||||
]);
|
]);
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
|
import RSVP from 'rsvp';
|
||||||
const { Route, RSVP } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend({
|
export default Route.extend({
|
||||||
model() {
|
model() {
|
||||||
|
@ -9,11 +8,12 @@ export default Route.extend({
|
||||||
return RSVP.hash({
|
return RSVP.hash({
|
||||||
model: model,
|
model: model,
|
||||||
projectOptions: store.findAll('project'),
|
projectOptions: store.findAll('project'),
|
||||||
studyLocationOptions: store.findAll('study-location'),
|
studyLocationOptions: store.query('study-location', { page_size: 500 }),
|
||||||
collectionTypeOptions: store.findAll('collection-type'),
|
collectionTypeOptions: store.findAll('collection-type'),
|
||||||
collectionMethodOptions: store.findAll('collection-method'),
|
collectionMethodOptions: store.findAll('collection-method'),
|
||||||
speciesOptions: store.findAll('species'),
|
speciesOptions: store.query('species', { page_size: 500 }),
|
||||||
adfgPermitOptions: store.findAll('adfg-permit'),
|
adfgPermitOptions: store.findAll('adfg-permit'),
|
||||||
|
sexOptions: store.findAll('sex'),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
|
import RSVP from 'rsvp';
|
||||||
const { Route, RSVP } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend({
|
export default Route.extend({
|
||||||
queryParams: {
|
queryParams: {
|
||||||
|
@ -30,10 +29,10 @@ export default Route.extend({
|
||||||
projectOptions: store.findAll('project'),
|
projectOptions: store.findAll('project'),
|
||||||
regionOptions: store.findAll('region'),
|
regionOptions: store.findAll('region'),
|
||||||
siteOptions: store.findAll('site'),
|
siteOptions: store.findAll('site'),
|
||||||
studyLocationOptions: store.findAll('study-location'),
|
studyLocationOptions: store.query('study-location', { page_size: 500 }),
|
||||||
collectionMethodOptions: store.findAll('collection-method'),
|
collectionMethodOptions: store.findAll('collection-method'),
|
||||||
adfgPermitOptions: store.findAll('adfg-permit'),
|
adfgPermitOptions: store.findAll('adfg-permit'),
|
||||||
speciesOptions: store.findAll('species'),
|
speciesOptions: store.query('species', { page_size: 500 }),
|
||||||
model: store.query('collection', Object.assign(params, opts)),
|
model: store.query('collection', Object.assign(params, opts)),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -44,27 +43,38 @@ export default Route.extend({
|
||||||
|
|
||||||
const store = this.get('store');
|
const store = this.get('store');
|
||||||
|
|
||||||
|
/* eslint-disable no-console */
|
||||||
|
|
||||||
let project = controller.get('project');
|
let project = controller.get('project');
|
||||||
|
console.log('project', project);
|
||||||
project = project.map(id => store.peekRecord('project', id));
|
project = project.map(id => store.peekRecord('project', id));
|
||||||
|
|
||||||
let region = controller.get('region');
|
let region = controller.get('region');
|
||||||
|
console.log('region', region);
|
||||||
region = region.map(id => store.peekRecord('region', id));
|
region = region.map(id => store.peekRecord('region', id));
|
||||||
|
|
||||||
let site = controller.get('site');
|
let site = controller.get('site');
|
||||||
|
console.log('site', site);
|
||||||
site = site.map(id => store.peekRecord('site', id));
|
site = site.map(id => store.peekRecord('site', id));
|
||||||
|
|
||||||
let studyLocation = controller.get('study_location');
|
let studyLocation = controller.get('study_location');
|
||||||
|
console.log('studyLocation', studyLocation);
|
||||||
studyLocation = studyLocation.map(id => store.peekRecord('study-location', id));
|
studyLocation = studyLocation.map(id => store.peekRecord('study-location', id));
|
||||||
|
|
||||||
let collectionMethod = controller.get('collection_method');
|
let collectionMethod = controller.get('collection_method');
|
||||||
|
console.log('collectionMethod', collectionMethod);
|
||||||
collectionMethod = collectionMethod.map(id => store.peekRecord('collection-method', id));
|
collectionMethod = collectionMethod.map(id => store.peekRecord('collection-method', id));
|
||||||
|
|
||||||
let adfgPermit = controller.get('adfg_permit');
|
let adfgPermit = controller.get('adfg_permit');
|
||||||
|
console.log('adfgPermit', adfgPermit);
|
||||||
adfgPermit = adfgPermit.map(id => store.peekRecord('adfg-permit', id));
|
adfgPermit = adfgPermit.map(id => store.peekRecord('adfg-permit', id));
|
||||||
|
|
||||||
let species = controller.get('species');
|
let species = controller.get('species');
|
||||||
|
console.log('species', species);
|
||||||
species = species.map(id => store.peekRecord('species', id));
|
species = species.map(id => store.peekRecord('species', id));
|
||||||
|
|
||||||
|
/* eslint-enable no-console */
|
||||||
|
|
||||||
const numberOfTraps = controller.get('number_of_traps');
|
const numberOfTraps = controller.get('number_of_traps');
|
||||||
const collectionStartDate = controller.get('collection_start_date');
|
const collectionStartDate = controller.get('collection_start_date');
|
||||||
const collectionEndDate = controller.get('collection_end_date');
|
const collectionEndDate = controller.get('collection_end_date');
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin';
|
import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin';
|
||||||
|
|
||||||
const { Route } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend(AuthenticatedRouteMixin, {
|
export default Route.extend(AuthenticatedRouteMixin, {
|
||||||
afterModel() {
|
afterModel() {
|
||||||
this.transitionTo('collections');
|
this.transitionTo('collections');
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
import UnauthenticatedRouteMixin from 'ember-simple-auth/mixins/unauthenticated-route-mixin';
|
import UnauthenticatedRouteMixin from 'ember-simple-auth/mixins/unauthenticated-route-mixin';
|
||||||
|
|
||||||
const { Route } = Ember;
|
|
||||||
|
|
||||||
export default Route.extend(UnauthenticatedRouteMixin, {});
|
export default Route.extend(UnauthenticatedRouteMixin, {});
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
import Ember from 'ember';
|
import Route from '@ember/routing/route';
|
||||||
|
import { inject as service } from '@ember/service';
|
||||||
const { Route, inject: { service }} = Ember;
|
|
||||||
|
|
||||||
export default Route.extend({
|
export default Route.extend({
|
||||||
session: service('session'),
|
session: service('session'),
|
||||||
|
|
|
@ -1,10 +1,16 @@
|
||||||
import Ember from 'ember';
|
import { capitalize } from '@ember/string';
|
||||||
import DS from 'ember-data';
|
import DS from 'ember-data';
|
||||||
|
|
||||||
const { JSONAPISerializer } = DS;
|
const { JSONAPISerializer } = DS;
|
||||||
|
|
||||||
export default JSONAPISerializer.extend({
|
export default JSONAPISerializer.extend({
|
||||||
payloadTypeFromModelName(modelName) {
|
payloadTypeFromModelName(modelName) {
|
||||||
return modelName.split('-').map(key => Ember.String.capitalize(key)).join('');
|
return modelName.split('-').map(key => capitalize(key)).join('');
|
||||||
}
|
},
|
||||||
|
|
||||||
|
normalizeArrayResponse(store, primaryModelClass, payload, id, requestType) {
|
||||||
|
let normalizedDocument = this._super(store, primaryModelClass, payload, id, requestType);
|
||||||
|
normalizedDocument.meta.links = normalizedDocument.links;
|
||||||
|
return normalizedDocument;
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -4,8 +4,10 @@
|
||||||
validations=(hash
|
validations=(hash
|
||||||
collection=CollectionValidations
|
collection=CollectionValidations
|
||||||
collectionSpecies=CollectionSpeciesValidations
|
collectionSpecies=CollectionSpeciesValidations
|
||||||
|
envMeasurements=CollectionMeasurementValidations
|
||||||
datasheet=DatasheetValidations)
|
datasheet=DatasheetValidations)
|
||||||
options=options
|
options=options
|
||||||
|
hasMany=hasMany
|
||||||
onSave=(action 'onSave')
|
onSave=(action 'onSave')
|
||||||
onCancel=(action 'onCancel')
|
onCancel=(action 'onCancel')
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -4,8 +4,10 @@
|
||||||
validations=(hash
|
validations=(hash
|
||||||
collection=CollectionValidations
|
collection=CollectionValidations
|
||||||
collectionSpecies=CollectionSpeciesValidations
|
collectionSpecies=CollectionSpeciesValidations
|
||||||
datasheet=DatasheetValidations)
|
datasheet=DatasheetValidations
|
||||||
|
envMeasurements=CollectionMeasurementValidations)
|
||||||
options=options
|
options=options
|
||||||
|
hasMany=hasMany
|
||||||
onSave=(action 'onSave')
|
onSave=(action 'onSave')
|
||||||
onCancel=(action 'onCancel')
|
onCancel=(action 'onCancel')
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -2,4 +2,5 @@
|
||||||
collection/detail-container
|
collection/detail-container
|
||||||
model=model
|
model=model
|
||||||
editCollection=(action 'editCollection')
|
editCollection=(action 'editCollection')
|
||||||
|
deleteCollection=(action 'deleteCollection')
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -21,27 +21,33 @@
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='adfgPermit' label='ADFG Permit' changeset=changeset}}
|
{{#validated-field property='adfgPermit' label='ADFG Permit' changeset=changeset}}
|
||||||
{{#power-select
|
{{#power-select-with-create
|
||||||
options=options.adfgPermits
|
options=options.adfgPermits
|
||||||
selected=changeset.adfgPermit
|
selected=changeset.adfgPermit
|
||||||
onchange=(action (mut changeset.adfgPermit))
|
onchange=(action (mut changeset.adfgPermit))
|
||||||
|
oncreate=(action 'addOption' 'adfg-permit' 'adfgPermits' 'adfgPermit' 'name')
|
||||||
searchField='name'
|
searchField='name'
|
||||||
as |adfgPermit|
|
as |adfgPermit term|
|
||||||
}}
|
}}
|
||||||
{{adfgPermit.name}}
|
{{adfgPermit.name}}
|
||||||
{{/power-select}}
|
{{/power-select-with-create}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='studyLocation' label='Study location' changeset=changeset}}
|
{{#validated-field property='studyLocation' changeset=changeset}}
|
||||||
{{#power-select
|
<label class="control-label">
|
||||||
options=options.studyLocations
|
Study location
|
||||||
selected=changeset.studyLocation
|
<a href="{{newStudyLocationAdmin}}" target="_blank">+</a>
|
||||||
onchange=(action (mut changeset.studyLocation))
|
</label>
|
||||||
searchField='name'
|
{{#power-select
|
||||||
as |studyLocation|
|
search=(action 'searchStudyLocation')
|
||||||
}}
|
options=options.studyLocations
|
||||||
{{studyLocation.name}}
|
selected=changeset.studyLocation
|
||||||
{{/power-select}}
|
onchange=(action (mut changeset.studyLocation))
|
||||||
|
searchField='code'
|
||||||
|
as |studyLocation|
|
||||||
|
}}
|
||||||
|
{{studyLocation.code}}
|
||||||
|
{{/power-select}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
|
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
|
||||||
|
@ -73,114 +79,204 @@
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
|
{{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
|
||||||
{{input value=changeset.collectionStartDate type='date' class='form-control'}}
|
{{
|
||||||
|
pikaday-input
|
||||||
|
onSelection=(action (mut changeset.collectionStartDate))
|
||||||
|
value=changeset.collectionStartDate
|
||||||
|
useUTC=true
|
||||||
|
placeholder='MM/DD/YYYY'
|
||||||
|
format='MM/DD/YYYY'
|
||||||
|
class='form-control'
|
||||||
|
}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionStartTime' label='Collection start time' changeset=changeset}}
|
{{#validated-field property='collectionStartTime' label='Collection start time' changeset=changeset}}
|
||||||
{{input value=changeset.collectionStartTime type='time' class='form-control'}}
|
{{input value=changeset.collectionStartTime type='time' class='form-control' placeholder='HH:MM:SS (24 hour)'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
|
{{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
|
||||||
{{input value=changeset.collectionEndDate type='date' class='form-control'}}
|
{{
|
||||||
|
pikaday-input
|
||||||
|
onSelection=(action (mut changeset.collectionEndDate))
|
||||||
|
value=changeset.collectionEndDate
|
||||||
|
useUTC=true
|
||||||
|
placeholder='MM/DD/YYYY'
|
||||||
|
format='MM/DD/YYYY'
|
||||||
|
class='form-control'
|
||||||
|
}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
|
|
||||||
{{#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' placeholder='HH:MM:SS (24 hour)'}}
|
||||||
{{/validated-field}}
|
{{/validated-field}}
|
||||||
{{/with}}
|
{{/with}}
|
||||||
{{/f.content}}
|
{{/f.content}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-8">
|
<div class="col-md-8">
|
||||||
<table class="table">
|
<div class="row">
|
||||||
<caption>
|
<div class="col-md-12">
|
||||||
Species / Count Info
|
{{#validated-field property='notes' label='Notes' changeset=changesets.model}}
|
||||||
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addCollectionSpecies')}}
|
{{textarea value=changesets.model.notes class='form-control'}}
|
||||||
</caption>
|
{{/validated-field}}
|
||||||
<thead>
|
</div>
|
||||||
<tr>
|
</div>
|
||||||
<th class="col-md-3">Species</th>
|
<div class="row">
|
||||||
<th class="col-md-3">Count</th>
|
<div class="col-md-12">
|
||||||
<th class="col-md-3">Count Estimated</th>
|
<table class="table">
|
||||||
<th class="col-md-3">Sex</th>
|
<caption>
|
||||||
<th class="col-md-1">Delete</th>
|
Species / Count Info
|
||||||
</tr>
|
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addHasMany' 'collection-species' 'collectionSpecies')}}
|
||||||
</thead>
|
</caption>
|
||||||
<tbody>
|
<thead>
|
||||||
{{#each changesets.hasMany.collectionSpecies as |cs|}}
|
<tr>
|
||||||
<tr class="form">
|
<th class="col-md-3">Species</th>
|
||||||
<td class="col-md-3">
|
<th class="col-md-3">Count</th>
|
||||||
{{#validated-field property='species' changeset=cs.changeset}}
|
<th class="col-md-3">Count Estimated</th>
|
||||||
{{#power-select
|
<th class="col-md-3">Sex</th>
|
||||||
options=options.species
|
<th class="col-md-1">Delete</th>
|
||||||
selected=cs.changeset.species
|
</tr>
|
||||||
onchange=(action (mut cs.changeset.species))
|
</thead>
|
||||||
searchField='commonName'
|
<tbody>
|
||||||
as |species|
|
{{#each changesets.hasMany.collectionSpecies as |cs|}}
|
||||||
}}
|
<tr class="form">
|
||||||
{{species.commonName}}
|
<td class="col-md-3">
|
||||||
{{/power-select}}
|
{{#validated-field property='species' changeset=cs.changeset}}
|
||||||
{{/validated-field}}
|
{{#power-select
|
||||||
</td>
|
options=options.species
|
||||||
<td class="col-md-3">
|
selected=cs.changeset.species
|
||||||
{{#validated-field property='count' changeset=cs.changeset}}
|
onchange=(action (mut cs.changeset.species))
|
||||||
{{input value=cs.changeset.count}}
|
searchField='commonName'
|
||||||
{{/validated-field}}
|
as |species|
|
||||||
</td>
|
}}
|
||||||
<td class="col-md-3">
|
{{species.commonName}}
|
||||||
{{#validated-field property='countEstimated' changeset=cs.changeset}}
|
{{/power-select}}
|
||||||
{{input checked=cs.changeset.countEstimated type='checkbox'}}
|
{{/validated-field}}
|
||||||
{{/validated-field}}
|
</td>
|
||||||
</td>
|
<td class="col-md-3">
|
||||||
<td class="col-md-3">
|
{{#validated-field property='count' changeset=cs.changeset}}
|
||||||
{{#validated-field property='sex' changeset=cs.changeset}}
|
{{input value=cs.changeset.count}}
|
||||||
{{input value=cs.changeset.sex}}
|
{{/validated-field}}
|
||||||
{{/validated-field}}
|
</td>
|
||||||
</td>
|
<td class="col-md-3">
|
||||||
<td class="col-md-2">
|
{{#validated-field property='countEstimated' changeset=cs.changeset}}
|
||||||
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteCollectionSpecies' cs)}}
|
{{input checked=cs.changeset.countEstimated type='checkbox'}}
|
||||||
</td>
|
{{/validated-field}}
|
||||||
</tr>
|
</td>
|
||||||
{{/each}}
|
<td class="col-md-3">
|
||||||
</tbody>
|
{{#validated-field property='sex' changeset=cs.changeset}}
|
||||||
</table>
|
{{#power-select
|
||||||
</div>
|
options=options.sexes
|
||||||
</div>
|
selected=cs.changeset.sex
|
||||||
<div class="row">
|
onchange=(action (mut cs.changeset.sex))
|
||||||
<div class="col-md-12">
|
searchField='name'
|
||||||
<form enctype="multipart/form-data">
|
as |sex|
|
||||||
<table class="table">
|
}}
|
||||||
<caption>
|
{{sex.name}}
|
||||||
Attachments
|
{{/power-select}}
|
||||||
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addDatasheet')}}
|
{{/validated-field}}
|
||||||
</caption>
|
</td>
|
||||||
<thead>
|
<td class="col-md-2">
|
||||||
<tr>
|
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteHasMany' cs 'collectionSpecies')}}
|
||||||
<th>File</th>
|
</td>
|
||||||
<th>Delete</th>
|
</tr>
|
||||||
</tr>
|
{{/each}}
|
||||||
</thead>
|
</tbody>
|
||||||
<tbody>
|
</table>
|
||||||
{{#each changesets.hasMany.datasheets as |d|}}
|
</div>
|
||||||
<tr class="form">
|
</div>
|
||||||
<td>
|
<div class="row">
|
||||||
{{#if d.model.isNew}}
|
<div class="col-md-12">
|
||||||
{{#validated-field property='datasheet' changeset=d.changeset}}
|
<table class="table">
|
||||||
<input type="file" onchange={{action 'updateDatasheet' d.changeset}} accept="image/png,image/jpeg,application/pdf">
|
<caption>
|
||||||
{{/validated-field}}
|
Environmental Measurements
|
||||||
{{else}}
|
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addHasMany' 'collection-measurement' 'envMeasurements')}}
|
||||||
<a href="{{ d.model.datasheet }}">{{ d.model.datasheet }}</a>
|
</caption>
|
||||||
{{/if}}
|
<thead>
|
||||||
</td>
|
<tr>
|
||||||
<td>
|
<th class="col-md-3">Date Measured</th>
|
||||||
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteDatasheet' d)}}
|
<th class="col-md-3">Time Measured</th>
|
||||||
</td>
|
<th class="col-md-3">Water Temperature (deg C)</th>
|
||||||
</tr>
|
<th class="col-md-3">Air Temperature (deg C)</th>
|
||||||
{{/each}}
|
<th class="col-md-1">Delete</th>
|
||||||
</tbody>
|
</tr>
|
||||||
</table>
|
</thead>
|
||||||
</form>
|
<tbody>
|
||||||
|
{{#each changesets.hasMany.envMeasurements as |cm|}}
|
||||||
|
<tr class="form">
|
||||||
|
<td class="col-md-3">
|
||||||
|
{{#validated-field property='dateMeasured' changeset=cm.changeset}}
|
||||||
|
{{
|
||||||
|
pikaday-input
|
||||||
|
onSelection=(action (mut cm.changeset.dateMeasured))
|
||||||
|
value=cm.changeset.dateMeasured
|
||||||
|
useUTC=true
|
||||||
|
placeholder='MM/DD/YYYY'
|
||||||
|
format='MM/DD/YYYY'
|
||||||
|
class='form-control'
|
||||||
|
}}
|
||||||
|
{{/validated-field}}
|
||||||
|
</td>
|
||||||
|
<td class="col-md-3">
|
||||||
|
{{#validated-field property='timeMeasured' changeset=cm.changeset}}
|
||||||
|
{{input value=cm.changeset.timeMeasured type='time' class='form-control'}}
|
||||||
|
{{/validated-field}}
|
||||||
|
</td>
|
||||||
|
<td class="col-md-3">
|
||||||
|
{{#validated-field property='waterTempC' changeset=cm.changeset}}
|
||||||
|
{{input value=cm.changeset.waterTempC class='form-control'}}
|
||||||
|
{{/validated-field}}
|
||||||
|
</td>
|
||||||
|
<td class="col-md-3">
|
||||||
|
{{#validated-field property='airTempC' changeset=cm.changeset}}
|
||||||
|
{{input value=cm.changeset.airTempC class='form-control'}}
|
||||||
|
{{/validated-field}}
|
||||||
|
</td>
|
||||||
|
<td class="col-md-2">
|
||||||
|
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteHasMany' cm 'envMeasurements')}}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{/each}}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-md-12">
|
||||||
|
<form enctype="multipart/form-data">
|
||||||
|
<table class="table">
|
||||||
|
<caption>
|
||||||
|
Attachments
|
||||||
|
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addHasMany' 'datasheet-attachment' 'datasheets')}}
|
||||||
|
</caption>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>File</th>
|
||||||
|
<th>Delete</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{#each changesets.hasMany.datasheets as |d|}}
|
||||||
|
<tr class="form">
|
||||||
|
<td>
|
||||||
|
{{#if d.model.isNew}}
|
||||||
|
{{#validated-field property='datasheet' changeset=d.changeset}}
|
||||||
|
<input type="file" onchange={{action 'updateDatasheet' d.changeset}} accept="image/png,image/jpeg,application/pdf">
|
||||||
|
{{/validated-field}}
|
||||||
|
{{else}}
|
||||||
|
<a href="{{ d.model.datasheet }}">{{ d.model.datasheet }}</a>
|
||||||
|
{{/if}}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteHasMany' d 'datasheets')}}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{/each}}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{f.save}} {{f.cancel}}
|
{{f.save}} {{f.cancel}}
|
||||||
|
|
|
@ -5,6 +5,13 @@
|
||||||
onClick=(action editCollection)
|
onClick=(action editCollection)
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
{{
|
||||||
|
confirm-button
|
||||||
|
initialLabel='Delete Collection'
|
||||||
|
confirmLabel='Yes, Delete Collection'
|
||||||
|
onClick=(action deleteCollection)
|
||||||
|
}}
|
||||||
|
|
||||||
<h3>Main Detail</h3>
|
<h3>Main Detail</h3>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-12">
|
<div class="col-md-12">
|
||||||
|
@ -33,14 +40,24 @@
|
||||||
|
|
||||||
<hr>
|
<hr>
|
||||||
|
|
||||||
|
<h3>Environmental Measurements</h3>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-md-12">
|
||||||
|
{{#ccdb-table model=model.[0].envMeasurements columns=envMeasColumns as |c|}}
|
||||||
|
{{#c.grid as |g|}}
|
||||||
|
{{g.head}}
|
||||||
|
{{g.body}}
|
||||||
|
{{/c.grid}}
|
||||||
|
{{/ccdb-table}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr>
|
||||||
|
|
||||||
<h3>Notes</h3>
|
<h3>Notes</h3>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-12">
|
<div class="col-md-12">
|
||||||
<ul>
|
{{model.[0].notes}}
|
||||||
<li>PLACEHOLDER</li>
|
|
||||||
<li>PLACEHOLDER</li>
|
|
||||||
<li>PLACEHOLDER</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -102,7 +102,8 @@
|
||||||
onSelection=(action (mut filters.collection_start_date))
|
onSelection=(action (mut filters.collection_start_date))
|
||||||
value=filters.collection_start_date
|
value=filters.collection_start_date
|
||||||
useUTC=true
|
useUTC=true
|
||||||
format='YYYY-MM-DD'
|
placeholder='MM/DD/YYYY'
|
||||||
|
format='MM/DD/YYYY'
|
||||||
class='form-control'
|
class='form-control'
|
||||||
}}
|
}}
|
||||||
</div>
|
</div>
|
||||||
|
@ -113,7 +114,8 @@
|
||||||
onSelection=(action (mut filters.collection_end_date))
|
onSelection=(action (mut filters.collection_end_date))
|
||||||
value=filters.collection_end_date
|
value=filters.collection_end_date
|
||||||
useUTC=true
|
useUTC=true
|
||||||
format='YYYY-MM-DD'
|
placeholder='MM/DD/YYYY'
|
||||||
|
format='MM/DD/YYYY'
|
||||||
class='form-control'
|
class='form-control'
|
||||||
}}
|
}}
|
||||||
</div>
|
</div>
|
||||||
|
|
6
app/templates/components/confirm-button.hbs
Normal file
6
app/templates/components/confirm-button.hbs
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
{{#if showConfirm}}
|
||||||
|
{{action-button isDanger=true label=cancelLabel onClick=(action 'cancel')}}
|
||||||
|
{{action-button isSuccess=true label=confirmLabel onClick=(action 'confirm')}}
|
||||||
|
{{else}}
|
||||||
|
{{action-button isDanger=true label=initialLabel onClick=(action 'initial')}}
|
||||||
|
{{/if}}
|
19
app/transforms/ccdb-date.js
Normal file
19
app/transforms/ccdb-date.js
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
import DS from 'ember-data';
|
||||||
|
|
||||||
|
export default DS.Transform.extend({
|
||||||
|
deserialize(serialized) {
|
||||||
|
return serialized || '';
|
||||||
|
},
|
||||||
|
|
||||||
|
serialize(date) {
|
||||||
|
if (date !== '') {
|
||||||
|
date = new Date(date);
|
||||||
|
const day = date.getUTCDate();
|
||||||
|
const month = date.getUTCMonth() + 1;
|
||||||
|
const year = date.getUTCFullYear();
|
||||||
|
return `${year}-${month}-${day}`;
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
12
app/validations/collection-measurement.js
Normal file
12
app/validations/collection-measurement.js
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
import {
|
||||||
|
validatePresence,
|
||||||
|
validateNumber,
|
||||||
|
} from 'ember-changeset-validations/validators';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
dateMeasured: validatePresence(true),
|
||||||
|
timeMeasured: validatePresence(true),
|
||||||
|
waterTempC: validateNumber({ allowBlank: true, integer: false, positive: false }),
|
||||||
|
airTempC: validateNumber({ allowBlank: true, integer: false, positive: false }),
|
||||||
|
collection: validatePresence(true),
|
||||||
|
}
|
|
@ -20,12 +20,23 @@ module.exports = function(environment) {
|
||||||
APP: {
|
APP: {
|
||||||
// Here you can pass flags/options to your application instance
|
// Here you can pass flags/options to your application instance
|
||||||
// when it is created
|
// when it is created
|
||||||
|
},
|
||||||
|
|
||||||
|
sentry: {
|
||||||
|
dsn: 'https://fd3c695fa9394de48a7c69b7a322960b@sentry.io/1186914',
|
||||||
|
globalErrorCatching: false,
|
||||||
|
},
|
||||||
|
|
||||||
|
contentSecurityPolicy: {
|
||||||
|
'script-src': "'self' 'unsafe-inline' 'unsafe-eval'",
|
||||||
|
'img-src': 'data: app.getsentry.com',
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (environment === 'development') {
|
if (environment === 'development') {
|
||||||
ENV.APP.API_HOST = 'http://localhost:8000';
|
ENV.APP.API_HOST = 'http://localhost:8000';
|
||||||
ENV.APP.API_NAMESPACE = 'api/v1';
|
ENV.APP.API_NAMESPACE = 'api/v1';
|
||||||
|
ENV.sentry.development = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (environment === 'test') {
|
if (environment === 'test') {
|
||||||
|
@ -40,7 +51,7 @@ module.exports = function(environment) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (environment === 'production') {
|
if (environment === 'production') {
|
||||||
ENV.APP.API_HOST = 'https://obscure-caverns-99102.herokuapp.com';
|
ENV.APP.API_HOST = 'https://ccdb-api.thermokar.st';
|
||||||
ENV.APP.API_NAMESPACE = 'api/v1';
|
ENV.APP.API_NAMESPACE = 'api/v1';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
21802
package-lock.json
generated
Normal file
21802
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
30
package.json
30
package.json
|
@ -17,25 +17,26 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"broccoli-asset-rev": "^2.4.5",
|
"broccoli-asset-rev": "^2.4.5",
|
||||||
"ember-ajax": "^3.0.0",
|
"ember-ajax": "^3.0.0",
|
||||||
"ember-changeset": "1.3.0",
|
"ember-changeset": "^1.3.0",
|
||||||
"ember-changeset-validations": "1.2.8",
|
"ember-changeset-validations": "^1.2.8",
|
||||||
"ember-cli": "~2.14.0",
|
"ember-cli": "^3.24.0",
|
||||||
"ember-cli-app-version": "^3.0.0",
|
"ember-cli-app-version": "^3.0.0",
|
||||||
"ember-cli-babel": "^6.3.0",
|
"ember-cli-babel": "^6.6.0",
|
||||||
"ember-cli-code-coverage": "^0.4.1",
|
"ember-cli-code-coverage": "^0.4.1",
|
||||||
"ember-cli-dependency-checker": "^1.3.0",
|
"ember-cli-dependency-checker": "^2.0.0",
|
||||||
"ember-cli-eslint": "^3.0.0",
|
"ember-cli-eslint": "^4.0.0",
|
||||||
"ember-cli-flash": "1.4.3",
|
"ember-cli-flash": "^1.4.3",
|
||||||
"ember-cli-htmlbars": "^2.0.1",
|
"ember-cli-htmlbars": "^2.0.1",
|
||||||
"ember-cli-htmlbars-inline-precompile": "^0.4.3",
|
"ember-cli-htmlbars-inline-precompile": "^1.0.0",
|
||||||
"ember-cli-inject-live-reload": "^1.4.1",
|
"ember-cli-inject-live-reload": "^1.4.1",
|
||||||
"ember-cli-moment-shim": "^3.5.0",
|
"ember-cli-moment-shim": "^3.5.0",
|
||||||
"ember-cli-qunit": "^4.0.0",
|
"ember-cli-qunit": "^4.0.0",
|
||||||
|
"ember-cli-sentry": "^3.0.0",
|
||||||
"ember-cli-shims": "^1.1.0",
|
"ember-cli-shims": "^1.1.0",
|
||||||
"ember-cli-sri": "^2.1.0",
|
"ember-cli-sri": "^2.1.0",
|
||||||
"ember-cli-uglify": "^1.2.0",
|
"ember-cli-uglify": "^2.0.0",
|
||||||
"ember-composable-helpers": "^2.0.3",
|
"ember-composable-helpers": "^2.0.3",
|
||||||
"ember-data": "~2.14.3",
|
"ember-data": "~2.16.2",
|
||||||
"ember-export-application-global": "^2.0.0",
|
"ember-export-application-global": "^2.0.0",
|
||||||
"ember-inflector": "^2.0.1",
|
"ember-inflector": "^2.0.1",
|
||||||
"ember-light-table": "^1.10.0",
|
"ember-light-table": "^1.10.0",
|
||||||
|
@ -43,16 +44,19 @@
|
||||||
"ember-moment": "7.3.1",
|
"ember-moment": "7.3.1",
|
||||||
"ember-pikaday": "^2.2.3",
|
"ember-pikaday": "^2.2.3",
|
||||||
"ember-power-select": "^1.8.5",
|
"ember-power-select": "^1.8.5",
|
||||||
"ember-power-select-with-create": "0.4.3",
|
"ember-power-select-with-create": "^0.4.3",
|
||||||
"ember-resolver": "^4.0.0",
|
"ember-resolver": "^4.0.0",
|
||||||
"ember-responsive": "^2.0.4",
|
"ember-responsive": "^2.0.4",
|
||||||
"ember-simple-auth": "1.4.0",
|
"ember-simple-auth": "1.4.0",
|
||||||
"ember-sinon": "^1.0.0",
|
"ember-sinon": "^1.0.0",
|
||||||
"ember-source": "~2.14.0",
|
"ember-source": "~2.16.0",
|
||||||
"loader.js": "^4.2.3"
|
"loader.js": "^4.2.3"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^4.5 || 6.* || >= 7.*"
|
"node": "^4.5 || 6.* || >= 7.*"
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true,
|
||||||
|
"dependencies": {
|
||||||
|
"npm": "^5.7.1"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
16
testem.js
16
testem.js
|
@ -3,10 +3,20 @@ module.exports = {
|
||||||
test_page: 'tests/index.html?hidepassed',
|
test_page: 'tests/index.html?hidepassed',
|
||||||
disable_watching: true,
|
disable_watching: true,
|
||||||
launch_in_ci: [
|
launch_in_ci: [
|
||||||
'PhantomJS'
|
'Chrome'
|
||||||
],
|
],
|
||||||
launch_in_dev: [
|
launch_in_dev: [
|
||||||
'PhantomJS',
|
|
||||||
'Chrome'
|
'Chrome'
|
||||||
]
|
],
|
||||||
|
browser_args: {
|
||||||
|
Chrome: {
|
||||||
|
mode: 'ci',
|
||||||
|
args: [
|
||||||
|
'--disable-gpu',
|
||||||
|
'--headless',
|
||||||
|
'--remote-debugging-port=9222',
|
||||||
|
'--window-size=1440,900'
|
||||||
|
]
|
||||||
|
},
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import Ember from 'ember';
|
import { run } from '@ember/runloop';
|
||||||
|
|
||||||
export default function destroyApp(application) {
|
export default function destroyApp(application) {
|
||||||
Ember.run(application, 'destroy');
|
run(application, 'destroy');
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,3 @@
|
||||||
import Ember from 'ember';
|
|
||||||
import FlashObject from 'ember-cli-flash/flash/object';
|
import FlashObject from 'ember-cli-flash/flash/object';
|
||||||
|
|
||||||
const { K } = Ember;
|
FlashObject.reopen({ init() {} });
|
||||||
|
|
||||||
FlashObject.reopen({ init: K });
|
|
||||||
|
|
|
@ -1,10 +1,8 @@
|
||||||
import { module } from 'qunit';
|
import { module } from 'qunit';
|
||||||
import Ember from 'ember';
|
import { resolve } from 'rsvp';
|
||||||
import startApp from '../helpers/start-app';
|
import startApp from '../helpers/start-app';
|
||||||
import destroyApp from '../helpers/destroy-app';
|
import destroyApp from '../helpers/destroy-app';
|
||||||
|
|
||||||
const { RSVP: { resolve } } = Ember;
|
|
||||||
|
|
||||||
export default function(name, options = {}) {
|
export default function(name, options = {}) {
|
||||||
module(name, {
|
module(name, {
|
||||||
beforeEach() {
|
beforeEach() {
|
||||||
|
|
|
@ -1,17 +1,16 @@
|
||||||
import Ember from 'ember';
|
import { registerAsyncHelper } from '@ember/test';
|
||||||
|
import { A } from '@ember/array';
|
||||||
|
import { computed } from '@ember/object';
|
||||||
|
import { classify } from '@ember/string';
|
||||||
|
import { getOwner } from '@ember/application';
|
||||||
import MediaService from 'ember-responsive/media';
|
import MediaService from 'ember-responsive/media';
|
||||||
|
|
||||||
const {
|
|
||||||
getOwner
|
|
||||||
} = Ember;
|
|
||||||
const { classify } = Ember.String;
|
|
||||||
|
|
||||||
MediaService.reopen({
|
MediaService.reopen({
|
||||||
// Change this if you want a different default breakpoint in tests.
|
// Change this if you want a different default breakpoint in tests.
|
||||||
_defaultBreakpoint: 'desktop',
|
_defaultBreakpoint: 'desktop',
|
||||||
|
|
||||||
_breakpointArr: Ember.computed('breakpoints', function() {
|
_breakpointArr: computed('breakpoints', function() {
|
||||||
return Object.keys(this.get('breakpoints')) || Ember.A([]);
|
return Object.keys(this.get('breakpoints')) || A([]);
|
||||||
}),
|
}),
|
||||||
|
|
||||||
_forceSetBreakpoint(breakpoint) {
|
_forceSetBreakpoint(breakpoint) {
|
||||||
|
@ -45,7 +44,7 @@ MediaService.reopen({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
export default Ember.Test.registerAsyncHelper('setBreakpoint', function(app, breakpoint) {
|
export default registerAsyncHelper('setBreakpoint', function(app, breakpoint) {
|
||||||
// this should use getOwner once that's supported
|
// this should use getOwner once that's supported
|
||||||
const mediaService = app.__deprecatedInstance__.lookup('service:media');
|
const mediaService = app.__deprecatedInstance__.lookup('service:media');
|
||||||
mediaService._forceSetBreakpoint(breakpoint);
|
mediaService._forceSetBreakpoint(breakpoint);
|
||||||
|
|
|
@ -1,12 +1,13 @@
|
||||||
import Ember from 'ember';
|
|
||||||
import Application from '../../app';
|
import Application from '../../app';
|
||||||
import config from '../../config/environment';
|
import config from '../../config/environment';
|
||||||
|
import { merge } from '@ember/polyfills';
|
||||||
|
import { run } from '@ember/runloop';
|
||||||
|
|
||||||
export default function startApp(attrs) {
|
export default function startApp(attrs) {
|
||||||
let attributes = Ember.merge({}, config.APP);
|
let attributes = merge({}, config.APP);
|
||||||
attributes = Ember.merge(attributes, attrs); // use defaults, but you can override;
|
attributes = merge(attributes, attrs); // use defaults, but you can override;
|
||||||
|
|
||||||
return Ember.run(() => {
|
return run(() => {
|
||||||
let application = Application.create(attributes);
|
let application = Application.create(attributes);
|
||||||
application.setupForTesting();
|
application.setupForTesting();
|
||||||
application.injectTestHelpers();
|
application.injectTestHelpers();
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
|
import $ from 'jquery';
|
||||||
import { moduleFor, test } from 'ember-qunit';
|
import { moduleFor, test } from 'ember-qunit';
|
||||||
import sinon from 'sinon';
|
import sinon from 'sinon';
|
||||||
import Ember from 'ember';
|
|
||||||
|
|
||||||
moduleFor('authenticator:application', 'Unit | application', {
|
moduleFor('authenticator:application', 'Unit | application', {
|
||||||
unit: true,
|
unit: true,
|
||||||
|
@ -35,28 +35,28 @@ test('`invalidate` should invalidate the session', function(assert) {
|
||||||
|
|
||||||
test('`makeRequest` should make a request', function(assert) {
|
test('`makeRequest` should make a request', function(assert) {
|
||||||
assert.expect(2);
|
assert.expect(2);
|
||||||
const stub = sinon.stub(Ember.$, 'ajax');
|
const stub = sinon.stub($, 'ajax');
|
||||||
stub.resolves(42);
|
stub.resolves(42);
|
||||||
const authenticator = this.subject();
|
const authenticator = this.subject();
|
||||||
authenticator.set('serverTokenEndpoint', 'foo')
|
authenticator.set('serverTokenEndpoint', 'foo')
|
||||||
const promise = authenticator.makeRequest({bar: 'baz'}).then((d) => {
|
const promise = authenticator.makeRequest({bar: 'baz'}).then((d) => {
|
||||||
assert.equal(d, 42);
|
assert.equal(d, 42);
|
||||||
});
|
});
|
||||||
assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}}));
|
assert.ok($.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}}));
|
||||||
Ember.$.ajax.restore();
|
$.ajax.restore();
|
||||||
return promise;
|
return promise;
|
||||||
});
|
});
|
||||||
|
|
||||||
test('authenticate should craft a nice payload', function(assert) {
|
test('authenticate should craft a nice payload', function(assert) {
|
||||||
assert.expect(2);
|
assert.expect(2);
|
||||||
const stub = sinon.stub(Ember.$, 'ajax');
|
const stub = sinon.stub($, 'ajax');
|
||||||
stub.resolves(42);
|
stub.resolves(42);
|
||||||
const authenticator = this.subject();
|
const authenticator = this.subject();
|
||||||
authenticator.set('serverTokenEndpoint', 'foo')
|
authenticator.set('serverTokenEndpoint', 'foo')
|
||||||
const promise = authenticator.authenticate('myusername', 'mypassword').then((d) => {
|
const promise = authenticator.authenticate('myusername', 'mypassword').then((d) => {
|
||||||
assert.equal(d, 42);
|
assert.equal(d, 42);
|
||||||
});
|
});
|
||||||
assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}}));
|
assert.ok($.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}}));
|
||||||
Ember.$.ajax.restore();
|
$.ajax.restore();
|
||||||
return promise;
|
return promise;
|
||||||
});
|
});
|
||||||
|
|
Loading…
Add table
Reference in a new issue