Compare commits

..

35 commits

Author SHA1 Message Date
7548aafdb7 maint: move to pingo 2021-01-10 15:39:29 -07:00
0cd3cd96f1 dep updates 2021-01-09 19:36:43 -07:00
22be697ef6
MAINT: swap vanilla sentry for ember-cli-sentry (#92) 2018-04-29 11:58:26 -07:00
bf6023a7b4
BUG: Rollback hasMany changesets sanely (#90) 2018-04-29 07:28:08 -07:00
6f6148df9d
BUG: reverting changeset version bumps (#89)
Fixes #88
2018-04-29 06:55:45 -07:00
437be4fb13
BUG: fix up broken envMeas validations (#86) 2018-04-22 06:48:00 -07:00
9d9e67868f
DEBUG: adding sentry, still need to disable in dev (#85) 2018-04-10 20:42:13 -07:00
86ab62cb0f
DEBUG: console statements for everyone! (#84) 2018-04-10 20:14:48 -07:00
234dc3a34c
BUG: collection measurement date/time field issues (#83) 2018-04-10 06:23:25 -07:00
d2c485af88
BUG: Better date parsing and handling in transform (#82) 2018-03-24 15:59:51 +02:00
ab30692021
BUG: Use datepicker in collection form (#81)
Fixes #73
2018-03-13 07:59:49 -07:00
2ead72c552
ENH: Inline adfg permit and study loc creation (#79)
Fixes #71
2018-03-04 20:34:42 -07:00
467a8d8b64
MAINT: Grab all species records in lookups (#78) 2018-03-03 16:56:44 -07:00
a32d147c1c
BUG: package lock (#77) 2018-03-03 12:45:06 -07:00
c87bd953d9
ENH: Convert sex fields to lookup (#76) 2018-03-03 12:10:12 -07:00
6f01fbf00f
ENH: Add collection delete (#69)
Fixes #37
2018-02-08 08:08:03 -07:00
eb4537afb1
MAINT: Upgrade to ember 2.16 LTS (#66) 2018-01-25 06:51:47 -07:00
39f4789a61
ENH: Collection environment measurement (#65) 2018-01-14 18:01:12 -07:00
56f8796eaf
ENH: Clarify and unify date reprs (#64) 2017-12-12 08:10:56 -07:00
dcbad4d54b
BUG: Editing collections loading indicator (#63) 2017-12-12 07:58:28 -07:00
a4264ac16c
ENH: Add collection notes (#60)
Fixes #32
2017-12-04 21:21:19 -07:00
4dbfcfa98b
ENH: Refactor hasMany relations (#59)
Fixes #52
2017-12-04 05:32:39 -07:00
af9ac976dc
BUG: Pin ember-changeset versions (#58) 2017-12-02 10:46:24 -07:00
60eadf517f
ENH: Revised collection details (#54) 2017-12-01 11:55:35 -07:00
4903a204e4
ENH: Collection attachments (#51) 2017-12-01 10:58:17 -07:00
d014fc099c
ENH: Drop Overview and add loading indicators (#49) 2017-11-30 16:17:34 -07:00
cb3bc081a6
ENH: Collection Edit (parity with reading) (#48) 2017-11-30 15:51:16 -07:00
bfae4422f4
ENH: Species and counts (#44)
Fixes #31
2017-11-19 17:15:47 -07:00
93d70d3c95
ENH: IACUC and ADFG Permits (and search reset) (#43)
Fixes #30
2017-11-14 08:30:12 -07:00
17651e071e
ENH: Collection filterings (#42)
Fixes #21
Fixes #28
Fixes #34
2017-11-10 11:18:33 -07:00
695eb65806 MAINT: Structural cleanup (#41) 2017-10-07 22:01:15 -07:00
f41f4caccd ENH: Collections (update) (#40)
Fixes #36
2017-10-07 17:37:24 -07:00
e1abc5e4cb ENH: Create collections (#38)
Fixes #35
2017-10-01 16:49:43 -07:00
09a62cf165 ENH: Simple collection details (#27) 2017-09-30 21:20:00 -07:00
578b8daa32 ENH: Support table pagination (#24)
Fixes #22
2017-09-23 14:35:50 -07:00
103 changed files with 23629 additions and 635 deletions

View file

@ -3,11 +3,17 @@ branches:
only:
- master
node_js:
- '6'
sudo: false
- "9"
sudo: required
dist: trusty
addons:
chrome: stable
cache:
directories:
- "$HOME/.npm"
env:
global:
- JOBS=1
before_install:
- npm config set spin false
- npm install -g phantomjs-prebuilt

View file

@ -10,7 +10,7 @@ You will need the following things properly installed on your computer.
* [Git](https://git-scm.com/)
* [Node.js](https://nodejs.org/) (with NPM)
* [Ember CLI](https://ember-cli.com/)
* [PhantomJS](http://phantomjs.org/)
* [Google Chrome](https://google.com/chrome/)
## Installation

View file

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

View file

@ -0,0 +1,19 @@
import ApplicationAdapter from './application';
import FileUploadAdapter from 'ccdb-web/mixins/file-upload';
export default ApplicationAdapter.extend(FileUploadAdapter, {
getFormFields(data) {
return {
'datasheet': data['data']['attributes']['datasheet'],
'collection': JSON.stringify(data['data']['relationships']['collection']['data']),
}
},
getFormKey(key) {
return key;
},
getFormValue(key, value) {
return value;
},
});

View file

@ -1,11 +1,9 @@
import Ember from 'ember';
import Application from '@ember/application';
import Resolver from './resolver';
import loadInitializers from 'ember-load-initializers';
import config from './config/environment';
let App;
App = Ember.Application.extend({
const App = Application.extend({
modulePrefix: config.modulePrefix,
podModulePrefix: config.podModulePrefix,
Resolver

View file

@ -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 config from '../config/environment';
const { RSVP: { Promise }, $, get, isEmpty, run } = Ember;
export default BaseAuthenticator.extend({
serverTokenEndpoint: `${config.APP.API_HOST}/api/auth/login/`,
tokenAttributeName: 'data.attributes.auth-token',

View file

@ -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';
const { isEmpty, get } = Ember;
export default BaseAuthorizer.extend({
authorize(data, block) {
const accessToken = get(data, 'data.attributes.auth-token');

View file

@ -0,0 +1,40 @@
import Component from '@ember/component';
export default Component.extend({
tagName: 'a',
classNames: ['btn'],
classNameBindings: [
// Styles
'isDefault:btn-default',
'isPrimary:btn-primary',
'isSuccess:btn-success',
'isInfo:btn-info',
'isWarning:btn-warning',
'isDanger:btn-danger',
'isLink:btn-link',
// Sizes
'isLarge:btn-lg',
'isSmall:btn-sm',
'isXSmall:btn-xs',
],
// ARGS
// Styles
isDefault: false,
isPrimary: false,
isSuccess: false,
isInfo: false,
isWarning: false,
isDanger: false,
isLink: false,
// Sizes
isLarge: false,
isSmall: false,
isXSmall: false,
label: 'LABEL',
click() {
this.get('onClick')();
}
});

View file

@ -1,5 +1,3 @@
import Ember from 'ember';
const { Component } = Ember;
import Component from '@ember/component';
export default Component.extend({});

View file

@ -0,0 +1,3 @@
import Component from '@ember/component';
export default Component.extend({ });

View file

@ -1,5 +1,41 @@
import Ember from 'ember';
import Component from '@ember/component';
import { alias } from '@ember/object/computed';
import { computed } from '@ember/object';
export default Ember.Component.extend({
classNames: ['row'],
export default Component.extend({
// ARGS
model: null,
// COMPUTED
meta: alias('model.meta'),
links: alias('meta.links'),
currentPage: alias('meta.pagination.page'),
totalRecords: alias('meta.pagination.count'),
firstLink: alias('links.first'),
lastLink: alias('links.last'),
nextLink: alias('links.next'),
prevLink: alias('links.prev'),
_getPage(link) {
link = this.get(link);
if (link === null) {
return null;
}
const url = new URL(link);
return parseInt(url.searchParams.get('page'));
},
_notEqual(a, b) {
return this.get(a) !== this.get(b);
},
first: computed('firstLink', function() { return this._getPage('firstLink'); }),
last: computed('lastLink', function() { return this._getPage('lastLink'); }),
next: computed('nextLink', function() { return this._getPage('nextLink'); }),
prev: computed('prevLink', function() { return this._getPage('prevLink'); }),
notOnFirst: computed('first', 'currentPage', function() { return this._notEqual('first', 'currentPage'); }),
notOnLast: computed('last', 'currentPage', function() { return this._notEqual('last', 'currentPage'); }),
});

View file

@ -1,16 +1,13 @@
import Ember from 'ember';
import Component from '@ember/component';
import Table from 'ember-light-table';
const { Component } = Ember;
export default Component.extend({
// ARGS
model: null,
columns: null,
table: null,
classNames: ['row'],
init() {
didReceiveAttrs() {
this._super(...arguments);
const table = new Table(this.get('columns'), this.get('model'));
this.set('table', table);

View file

@ -0,0 +1,93 @@
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 lookupValidator from 'ember-changeset-validations';
import config from 'ccdb-web/config/environment';
export default Component.extend({
store: service(),
init() {
this._super(...arguments);
const model = this.get('model');
const validations = this.get('validations');
const hasMany = this.get('hasMany');
let changesets = {};
changesets['new'] = [];
changesets['delete'] = [];
changesets['hasMany'] = {};
changesets['model'] = new Changeset(model,
lookupValidator(validations['collection']),
validations['collection']);
hasMany.forEach((hasMany) => {
let relatedChangesets = [];
let validation = validations[hasMany];
const related = model.get(hasMany);
related.forEach((r) => {
const changeset = new Changeset(r, lookupValidator(validation),
validation);
relatedChangesets.push({ model: r, changeset: changeset });
});
changesets['hasMany'][hasMany] = relatedChangesets;
});
this.set('changesets', changesets);
this.set('newStudyLocationAdmin', `${config.APP.API_HOST}/admin/locations/studylocation/add/`);
},
actions: {
addHasMany(modelName, relatedName) {
const store = this.get('store');
let changesets = this.get('changesets');
const validations = this.get('validations');
const validation = validations[relatedName];
const model = this.get('model');
const related = store.createRecord(modelName, { collection: model });
model.get(relatedName).pushObject(related);
changesets['new'].pushObject(related);
const changeset = new Changeset(related, lookupValidator(validation), validation);
changesets['hasMany'][relatedName].pushObject({ model: related, changeset: changeset });
},
deleteHasMany(changesetRecord, relatedName) {
let changesets = this.get('changesets');
changesets['delete'].pushObject(changesetRecord.model);
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) {
changeset.set('datasheet', event.target.files[0]);
},
searchStudyLocation(term) {
return new RSVP.Promise((resolve, reject) => {
debounce(this, this._performSearch, 'study-location', { page_size: 500, code: term }, resolve, reject, 400);
});
},
},
_performSearch(model, payload, resolve, reject) {
this.get('store').query(model, payload).then((results) => {
resolve(results);
}, reject);
},
});

View file

@ -0,0 +1,34 @@
import Component from '@ember/component';
export default Component.extend({
// ARGS
model: null,
mainColumns: [
{ label: 'Project', valuePath: 'project.name', },
{ label: 'IACUC', valuePath: 'project.iacucNumber', },
{ label: 'Region', valuePath: 'studyLocation.site.region.name', },
{ label: 'Site', valuePath: 'studyLocation.site.name', },
{ label: 'Study Location', valuePath: 'studyLocation.code', },
{ label: 'Method', valuePath: 'collectionMethod.code', },
{ label: 'Type', valuePath: 'collectionType.name', },
{ label: '# of Traps', valuePath: 'numberOfTraps', },
{ label: 'Start', valuePath: 'startDateTime', },
{ label: 'End', valuePath: 'endDateTime', },
{ label: 'ADFG Permit', valuePath: 'adfgPermit.name', },
],
collectionSpeciesColumns: [
{ label: 'Species', valuePath: 'species.commonName' },
{ label: 'Count', valuePath: 'count' },
{ label: 'Count Estimated?', valuePath: 'countEstimated' },
{ 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', },
],
});

View file

@ -0,0 +1,20 @@
import Component from '@ember/component';
export default Component.extend({
// ARGS
model: null,
columns: [
{ label: 'Project', valuePath: 'project.name', },
{ label: 'IACUC', valuePath: 'project.iacucNumber', },
{ label: 'Species', valuePath: 'speciesAndCounts', },
{ label: 'Region', valuePath: 'studyLocation.site.region.name', },
{ label: 'Site', valuePath: 'studyLocation.site.name', },
{ label: 'Study Location', valuePath: 'studyLocation.code', },
{ label: 'Method', valuePath: 'collectionMethod.name', },
{ label: '# of Traps', valuePath: 'numberOfTraps', },
{ label: 'Start', valuePath: 'startDateTime', },
{ label: 'End', valuePath: 'endDateTime', },
{ label: 'ADFG Permit', valuePath: 'adfgPermit.name', },
],
});

View file

@ -1,15 +0,0 @@
import Ember from 'ember';
const { Component } = Ember;
export default Component.extend({
columns: [
{ label: 'Project', valuePath: 'project.name', },
{ label: 'Study Location', valuePath: 'studyLocation.code', },
{ label: 'Method', valuePath: 'collectionMethod.code', },
{ label: 'Type', valuePath: 'collectionType.name', },
{ label: '# of Traps', valuePath: 'numberOfTraps', },
{ label: 'Start', valuePath: 'startDateTime', },
{ label: 'End', valuePath: 'endDateTime', },
],
});

View 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')();
},
},
});

View file

@ -0,0 +1,6 @@
import Component from '@ember/component';
export default Component.extend({
// ARGS
changesets: null,
});

View file

@ -1,5 +0,0 @@
import Ember from 'ember';
export default Ember.Component.extend({
tagName: 'form',
});

View file

@ -0,0 +1,5 @@
import Component from '@ember/component';
export default Component.extend({
tagName: 'form',
});

View file

@ -1,6 +1,4 @@
import Ember from 'ember';
const { Component } = Ember;
import Component from '@ember/component';
export default Component.extend({
classNames: ['spinner'],

View file

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

View file

@ -1,6 +1,5 @@
import Ember from 'ember';
const { Controller, inject: { service }} = Ember;
import Controller from '@ember/controller';
import { inject as service } from '@ember/service';
export default Controller.extend({
session: service('session'),

View file

@ -0,0 +1,46 @@
import Controller from '@ember/controller';
import { computed } from '@ember/object';
import CollectionValidations from 'ccdb-web/validations/collection';
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
import CollectionMeasurementValidations from 'ccdb-web/validations/collection-measurement';
import DatasheetValidations from 'ccdb-web/validations/datasheet';
import ValidationMixin from 'ccdb-web/mixins/validation';
export default Controller.extend(ValidationMixin, {
CollectionValidations,
CollectionSpeciesValidations,
DatasheetValidations,
CollectionMeasurementValidations,
hasMany: ['collectionSpecies', 'datasheets', 'envMeasurements'],
options: computed('projectOptions', 'studyLocationOptions',
'collectionTypeOptions', 'collectionMethodOptions',
'speciesOptions', 'adfgPermitOptions', 'sexOptions',
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'),
sexes: this.get('sexOptions'),
};
}),
actions: {
onSave(changeset) {
const postSave = () => { this.transitionToRoute('collections.index'); };
return this.transitionToRoute('loading').then(() => {
return this.validationSave(changeset, postSave);
});
},
onCancel(changeset) {
const postCancel = () => { this.transitionToRoute('collections.index'); };
return this.transitionToRoute('loading').then(() => {
return this.validationCancel(changeset, postCancel);
});
},
},
});

View file

@ -0,0 +1,52 @@
import Controller from '@ember/controller';
import { computed } from '@ember/object';
import CollectionValidations from 'ccdb-web/validations/collection';
import CollectionSpeciesValidations from 'ccdb-web/validations/collection-species';
import CollectionMeasurementValidations from 'ccdb-web/validations/collection-measurement';
import DatasheetValidations from 'ccdb-web/validations/datasheet';
import ValidationMixin from 'ccdb-web/mixins/validation';
export default Controller.extend(ValidationMixin, {
CollectionValidations,
CollectionSpeciesValidations,
DatasheetValidations,
CollectionMeasurementValidations,
hasMany: ['collectionSpecies', 'datasheets', 'envMeasurements'],
options: computed('projectOptions', 'studyLocationOptions',
'collectionTypeOptions', 'collectionMethodOptions',
'speciesOptions', 'adfgPermitOptions', 'sexOptions',
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'),
sexes: this.get('sexOptions'),
};
}),
actions: {
onSave(changesets) {
const postSave = () => {
// Use the model's ID here because of the ArrayProxy in the route
this.transitionToRoute('collections.detail', this.get('model.id'));
};
return this.transitionToRoute('loading').then(() => {
return this.validationSave(changesets, postSave);
});
},
onCancel(changesets) {
const postCancel = () => {
// Use the model's ID here because of the ArrayProxy in the route
return this.transitionToRoute('collections.detail', this.get('model.id'));
};
return this.transitionToRoute('loading').then(() => {
return this.validationCancel(changesets, postCancel);
});
},
},
});

View file

@ -0,0 +1,14 @@
import Controller from '@ember/controller';
export default Controller.extend({
actions: {
editCollection() {
this.transitionToRoute('collections.detail.edit', this.get('model'));
},
deleteCollection() {
this.get('model')[0].destroyRecord().then(() => {
this.transitionToRoute('collections');
});
},
},
});

View file

@ -0,0 +1,85 @@
import Controller from '@ember/controller';
import { set, get, computed } from '@ember/object';
export default Controller.extend({
queryParams: ['page', 'project', 'region', 'site', 'study_location',
'collection_method', 'number_of_traps', 'collection_start_date',
'collection_end_date', 'adfg_permit', 'species'],
page: 1,
project: [],
region: [],
site: [],
study_location: [],
collection_method: [],
adfg_permit: [],
species: [],
number_of_traps: '',
collection_start_date: '',
collection_end_date: '',
options: computed('projectOptions', 'regionOptions', 'siteOptions',
'studyLocationOptions', 'collectionMethodOptions',
'adfgPermitOptions', 'speciesOptions', function() {
return {
projects: this.get('projectOptions'),
regions: this.get('regionOptions'),
sites: this.get('siteOptions'),
studyLocations: this.get('studyLocationOptions'),
collectionMethods: this.get('collectionMethodOptions'),
adfgPermits: this.get('adfgPermitOptions'),
species: this.get('speciesOptions'),
};
}),
_coerceId(model) {
return +get(model, 'id');
},
actions: {
changePage(page) {
this.set('page', page);
},
rowClick(row) {
this.transitionToRoute('collections.detail', row.get('id'));
},
createCollection() {
this.transitionToRoute('collections.create');
},
resetFilter() {
set(this, 'page', 1);
['project', 'region', 'site', 'study_location', 'collection_method',
'adfg_permit', 'species'].forEach((field) => {
set(this, field, []);
});
['number_of_traps', 'collection_start_date', 'collection_end_date'].forEach((field) => {
set(this, field, '');
});
},
changeFilter(filter) {
// Need to reset the page so that things don't get weird
set(this, 'page', 1);
const filterModelFields = ['project', 'region', 'site', 'study_location',
'collection_method', 'adfg_permit', 'species'];
filterModelFields.forEach((field) => {
let fields = get(filter, field);
fields = fields.map(this._coerceId);
set(this, field, fields);
});
set(this, 'number_of_traps', get(filter, 'number_of_traps'));
['collection_start_date', 'collection_end_date'].forEach((field) => {
let value = get(filter, field);
if (value) {
value = value.toJSON().split('T')[0];
} else {
value = '';
}
set(this, field, value);
});
},
},
});

View file

@ -1,6 +1,5 @@
import Ember from 'ember';
const { Controller, inject: { service } } = Ember;
import Controller from '@ember/controller';
import { inject as service } from '@ember/service';
export default Controller.extend({
session: service(),

View file

@ -17,8 +17,8 @@
<body>
{{content-for "body"}}
<script src="{{rootURL}}assets/vendor.js"></script>
<script src="{{rootURL}}assets/ccdb-web.js"></script>
<script integrity="" src="{{rootURL}}assets/vendor.js"></script>
<script integrity="" src="{{rootURL}}assets/ccdb-web.js"></script>
{{content-for "body-footer"}}
</body>

62
app/mixins/file-upload.js Normal file
View file

@ -0,0 +1,62 @@
import Mixin from '@ember/object/mixin';
import { isArray } from '@ember/array';
const { keys } = Object;
// Portions borrowed from https://github.com/funtusov/ember-cli-form-data
// (that project has an MIT license listed, but no copyright holder explicitly identified)
export default Mixin.create({
formDataTypes: ['POST', 'PUT', 'PATCH'],
ajaxOptions(url, type, options) {
let data;
if (options && 'data' in options) { data = options.data; }
let hash = this._super.apply(this, arguments);
if (typeof FormData !== 'undefined' && data && this.formDataTypes.indexOf(type) >= 0) {
hash.processData = false;
hash.contentType = false;
hash.data = this._getFormData(data);
}
return hash;
},
getFormFields(data) {
this._root = this._root || keys(data)[0];
return data[this._root];
},
getFormKey(key) {
return `${this._root}[${key}]`;
},
getFormValue(key, value) {
return value;
},
_getFormData(data) {
let formData = new FormData();
const fields = this.getFormFields(data);
keys(fields).forEach((key) => {
this._appendValue(
this.getFormValue(key, fields[key]),
this.getFormKey(key, fields[key]),
formData);
});
return formData;
},
_appendValue(value, formKey, formData) {
if (isArray(value)) {
value.forEach((item) => {
this._appendValue(item, `${formKey}[]`, formData);
});
} else if (value && value.constructor === Object) {
keys(value).forEach((key) => {
this._appendValue(value[key], `${formKey}[${key}]`, formData);
});
} else if (typeof value !== 'undefined'){
formData.append(formKey, value === null ? '' : value);
}
},
});

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

@ -0,0 +1,73 @@
import Mixin from '@ember/object/mixin';
import { get } from '@ember/object';
import RSVP from 'rsvp';
const { keys } = Object;
export default Mixin.create({
validationSave(changesets, postSave) {
let promises = [], changes = [], saves = [], isValid = true;
let modelChangeset = changesets['model'];
// first, delete anything that needs to be removed
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);
}
}
return RSVP.all(promises);
}).then(() => { // don't need the promises, just that they are done.
for (let changeset of changes) {
if (get(changeset, 'isValid')) {
let saver = changeset.save().catch((error) => {
/* eslint-disable no-console */
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(changesets, postCancel) {
delete changesets['delete'];
for (const key of keys(changesets)) {
if (key === 'new') {
for (const model of changesets[key]) {
model.destroyRecord();
}
} else if (key === 'hasMany') {
const hasMany = changesets[key];
for (const hasManyKey of keys(changesets[key])) {
const hasManyChangesets = hasMany[hasManyKey];
for (const changeset of hasManyChangesets) {
changeset.rollback();
}
}
} else { // single
const changeset = changesets[key];
changeset.rollback();
}
}
return postCancel();
},
});

10
app/models/adfg-permit.js Normal file
View file

@ -0,0 +1,10 @@
import DS from 'ember-data';
const { Model, attr, hasMany } = DS;
export default Model.extend({
name: attr('string'),
sortOrder: attr('number'),
collection: hasMany('collection'),
});

View 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'),
});

View file

@ -0,0 +1,12 @@
import DS from 'ember-data';
const { Model, attr, belongsTo } = DS;
export default Model.extend({
sex: belongsTo('sex'),
count: attr('number'),
countEstimated: attr('boolean', { defaultValue: false }),
collection: belongsTo('collection'),
species: belongsTo('species'),
});

View file

@ -1,21 +1,41 @@
import Ember from 'ember';
import { mapBy } from '@ember/object/computed';
import { computed } from '@ember/object';
import DS from 'ember-data';
const { computed } = Ember;
const { Model, attr, belongsTo } = DS;
const { Model, attr, belongsTo, hasMany } = DS;
export default Model.extend({
displayName: attr('string'),
numberOfTraps: attr('number'),
collectionStartDate: attr('string-null-to-empty'),
collectionStartDate: attr('ccdb-date'),
collectionStartTime: attr('string-null-to-empty'),
collectionEndDate: attr('string-null-to-empty'),
collectionEndDate: attr('ccdb-date'),
collectionEndTime: attr('string-null-to-empty'),
notes: attr('string', { defaultValue: '' }),
project: belongsTo('project'),
studyLocation: belongsTo('study-location'),
collectionMethod: belongsTo('collection-method'),
collectionType: belongsTo('collection-type'),
adfgPermit: belongsTo('adfg-permit'),
collectionSpecies: hasMany('collection-species'),
datasheets: hasMany('datasheet-attachment'),
envMeasurements: hasMany('collection-measurement'),
// computed
species: mapBy('collectionSpecies', 'species'),
speciesNames: mapBy('species', 'commonName'),
counts: mapBy('collectionSpecies', 'count'),
speciesAndCounts: computed('speciesNames', 'counts', function() {
const speciesNames = this.get('speciesNames');
let counts = this.get('counts');
counts = counts.map(c => c !== null ? c : 'No Count');
return speciesNames.map((n, i) => `${n} (${counts[i]})`).join(', ');
}),
startDateTime: computed('collectionStartDate', 'collectionStartTime',
function() { return this._mergeDateTime('Start'); }),

View file

@ -0,0 +1,9 @@
import DS from 'ember-data';
const { Model, attr, belongsTo } = DS;
export default Model.extend({
datasheet: attr('file'),
collection: belongsTo('collection'),
});

11
app/models/region.js Normal file
View file

@ -0,0 +1,11 @@
import DS from 'ember-data';
const { Model, attr, hasMany } = DS;
export default Model.extend({
name: attr('string'),
code: attr('string'),
sortOrder: attr('number'),
site: hasMany('site'),
});

8
app/models/sex.js Normal file
View file

@ -0,0 +1,8 @@
import DS from 'ember-data';
const { Model, attr } = DS;
export default Model.extend({
name: attr('string'),
sortOrder: attr('number'),
});

13
app/models/site.js Normal file
View file

@ -0,0 +1,13 @@
import DS from 'ember-data';
const { Model, attr, hasMany, belongsTo } = DS;
export default Model.extend({
name: attr('string'),
code: attr('string'),
description: attr('string'),
sortOrder: attr('number'),
region: belongsTo('region'),
studyLocation: hasMany('study-location'),
});

11
app/models/species.js Normal file
View file

@ -0,0 +1,11 @@
import DS from 'ember-data';
const { Model, attr } = DS;
export default Model.extend({
commonName: attr('string'),
genus: attr('string'),
species: attr('string'),
parasite: attr('boolean'),
sortOrder: attr('number'),
});

View file

@ -1,6 +1,6 @@
import DS from 'ember-data';
const { Model, attr } = DS;
const { Model, attr, belongsTo } = DS;
export default Model.extend({
name: attr('string'),
@ -10,4 +10,6 @@ export default Model.extend({
collectingLocation: attr('string'),
description: attr('string'),
sortOrder: attr('number'),
site: belongsTo('site'),
});

View file

@ -1,7 +1,7 @@
import Ember from 'ember';
import EmberRouter from '@ember/routing/router';
import config from './config/environment';
const Router = Ember.Router.extend({
const Router = EmberRouter.extend({
location: config.locationType,
rootURL: config.rootURL
});
@ -10,8 +10,10 @@ Router.map(function() {
this.route('login');
this.route('logout');
this.route('collections', function() {
this.route('1');
this.route('new');
this.route('create');
this.route('detail', { path: '/:collection_id' }, function() {
this.route('edit');
});
});
});

View file

@ -1,6 +1,4 @@
import Ember from 'ember';
import Route from '@ember/routing/route';
import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin';
const { Route } = Ember;
export default Route.extend(ApplicationRouteMixin, {});

View file

@ -1,4 +0,0 @@
import Ember from 'ember';
export default Ember.Route.extend({
});

View file

@ -0,0 +1,23 @@
import Route from '@ember/routing/route';
import RSVP from 'rsvp';
export default Route.extend({
model() {
const store = this.get('store');
return RSVP.hash({
model: store.createRecord('collection'),
projectOptions: store.findAll('project'),
studyLocationOptions: store.query('study-location', { page_size: 500 }),
collectionTypeOptions: store.findAll('collection-type'),
collectionMethodOptions: store.findAll('collection-method'),
speciesOptions: store.query('species', { page_size: 500 }),
adfgPermitOptions: store.findAll('adfg-permit'),
sexOptions: store.findAll('sex'),
});
},
setupController(controller, models) {
this._super(...arguments);
controller.setProperties(models);
},
});

View file

@ -0,0 +1,12 @@
import Route from '@ember/routing/route';
import RSVP from 'rsvp';
export default Route.extend({
model(params) {
return RSVP.all([
this.get('store').findRecord('collection', params.collection_id, {
include: 'collection-species,datasheets,env-measurements',
})
]);
},
});

View file

@ -0,0 +1,26 @@
import Route from '@ember/routing/route';
import RSVP from 'rsvp';
export default Route.extend({
model() {
const store = this.get('store');
const model = this.modelFor('collections.detail');
return RSVP.hash({
model: model,
projectOptions: store.findAll('project'),
studyLocationOptions: store.query('study-location', { page_size: 500 }),
collectionTypeOptions: store.findAll('collection-type'),
collectionMethodOptions: store.findAll('collection-method'),
speciesOptions: store.query('species', { page_size: 500 }),
adfgPermitOptions: store.findAll('adfg-permit'),
sexOptions: store.findAll('sex'),
});
},
setupController(controller, models) {
this._super(...arguments);
// Unwrap the parent route's listified model
models.model = models.model[0];
controller.setProperties(models);
},
});

View file

@ -1,11 +1,96 @@
import Ember from 'ember';
const { Route } = Ember;
import Route from '@ember/routing/route';
import RSVP from 'rsvp';
export default Route.extend({
model() {
return this.get('store').findAll('collection', {
include: 'project,study-location,collection-method,collection-type'
queryParams: {
// qps are snake_case for the django api
page: { refreshModel: true },
project: { refreshModel: true },
region: { refreshModel: true },
site: { refreshModel: true },
study_location: { refreshModel: true },
collection_method: { refreshModel: true },
number_of_traps: { refreshModel: true },
collection_start_date: { refreshModel: true },
collection_end_date: { refreshModel: true },
adfg_permit: { refreshModel: true },
species: { refreshModel: true },
},
model(params) {
const store = this.get('store');
const includes = ['project', 'study-location', 'study-location.site', 'site',
'collection-method', 'adfg-permit', 'collection-species', 'collection-species.species'];
const opts = {
include: includes.join(','),
};
return RSVP.hash({
projectOptions: store.findAll('project'),
regionOptions: store.findAll('region'),
siteOptions: store.findAll('site'),
studyLocationOptions: store.query('study-location', { page_size: 500 }),
collectionMethodOptions: store.findAll('collection-method'),
adfgPermitOptions: store.findAll('adfg-permit'),
speciesOptions: store.query('species', { page_size: 500 }),
model: store.query('collection', Object.assign(params, opts)),
});
},
setupController(controller, models) {
this._super(...arguments);
controller.setProperties(models);
const store = this.get('store');
/* eslint-disable no-console */
let project = controller.get('project');
console.log('project', project);
project = project.map(id => store.peekRecord('project', id));
let region = controller.get('region');
console.log('region', region);
region = region.map(id => store.peekRecord('region', id));
let site = controller.get('site');
console.log('site', site);
site = site.map(id => store.peekRecord('site', id));
let studyLocation = controller.get('study_location');
console.log('studyLocation', studyLocation);
studyLocation = studyLocation.map(id => store.peekRecord('study-location', id));
let collectionMethod = controller.get('collection_method');
console.log('collectionMethod', collectionMethod);
collectionMethod = collectionMethod.map(id => store.peekRecord('collection-method', id));
let adfgPermit = controller.get('adfg_permit');
console.log('adfgPermit', adfgPermit);
adfgPermit = adfgPermit.map(id => store.peekRecord('adfg-permit', id));
let species = controller.get('species');
console.log('species', species);
species = species.map(id => store.peekRecord('species', id));
/* eslint-enable no-console */
const numberOfTraps = controller.get('number_of_traps');
const collectionStartDate = controller.get('collection_start_date');
const collectionEndDate = controller.get('collection_end_date');
let filter = {
project,
region,
site,
study_location: studyLocation,
collection_method: collectionMethod,
number_of_traps: numberOfTraps,
collection_start_date: collectionStartDate,
collection_end_date: collectionEndDate,
adfg_permit: adfgPermit,
species,
}
controller.set('filters', filter);
},
});

View file

@ -1,4 +0,0 @@
import Ember from 'ember';
export default Ember.Route.extend({
});

View file

@ -1,6 +1,8 @@
import Ember from 'ember';
import Route from '@ember/routing/route';
import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin';
const { Route } = Ember;
export default Route.extend(AuthenticatedRouteMixin, {});
export default Route.extend(AuthenticatedRouteMixin, {
afterModel() {
this.transitionTo('collections');
},
});

View file

@ -1,6 +1,4 @@
import Ember from 'ember';
import Route from '@ember/routing/route';
import UnauthenticatedRouteMixin from 'ember-simple-auth/mixins/unauthenticated-route-mixin';
const { Route } = Ember;
export default Route.extend(UnauthenticatedRouteMixin, {});

View file

@ -1,6 +1,5 @@
import Ember from 'ember';
const { Route, inject: { service }} = Ember;
import Route from '@ember/routing/route';
import { inject as service } from '@ember/service';
export default Route.extend({
session: service('session'),

View file

@ -0,0 +1,16 @@
import { capitalize } from '@ember/string';
import DS from 'ember-data';
const { JSONAPISerializer } = DS;
export default JSONAPISerializer.extend({
payloadTypeFromModelName(modelName) {
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;
},
});

View file

@ -1,12 +1,13 @@
.content {
padding-left: 40px;
padding-right: 40px;
padding-top: 20px;
padding-bottom: 20px;
[data-ember-action]:not(:disabled) {
cursor: pointer;
}
.top-buffer {
padding-top: 20px;
.table-nav .pager {
margin-top: 10px;
}
.table-stats {
margin-top: 10px;
}
.form-signin {
@ -49,6 +50,10 @@
border-top-right-radius: 0;
}
.top-buffer {
margin-top: 20px;
}
/* Sidebar */
.sidebar {
position: fixed;

View file

@ -3,10 +3,6 @@
<div class="row">
<div class="col-md-2 sidebar">
<ul class="nav nav-sidebar">
{{#link-to 'index' tagName='li' href=false}}
{{link-to 'Overview' 'index'}}
{{/link-to}}
<li><a href="#">Experiments</a></li>
{{#link-to 'collections' tagName='li' href=false}}
{{link-to 'Collections' 'collections'}}
{{/link-to}}

View file

@ -1,147 +0,0 @@
<h3>Collection Detail</h3>
<div class="row">
<div class="panel panel-default col-md-12">
<div class="panel-body table-responsive">
<table class="table table-border table-hover table-condensed">
<thead>
<tr>
<th>Project</th>
<th>Study Location</th>
<th>Method</th>
<th>Type</th>
<th># of Traps</th>
<th>Start</th>
<th>End</th>
<th>Experiments</th>
</tr>
</thead>
<tbody>
<tr>
<td>Perchlorate</td>
<td>RS</td>
<td>MT</td>
<td>Embryos</td>
<td></td>
<td>2010-06-07</td>
<td>2010-06-07</td>
<td>Perchlorate diel; Perchlorate rescue</td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
<div class="row">
<div class="panel panel-default col-md-12">
<div class="panel-body">
<ul>
<li><a href="">Attachment 1: Photo</a></li>
<li><a href="">Attachment 2: Field Sheets</a></li>
<li><a href="">Attachment 3: Misc Notes</a></li>
</ul>
</div>
</div>
</div>
<div class="row">
<div class="col-md-12">
<form class="form-inline">
<div class="form-group">
<label>Experiment</label>
<select class="form-control">
<option>All</option>
<option selected=selected>Perchlorate diel</option>
<option>Perchlorate rescue</option>
</select>
</div>
</form>
</div>
</div>
<div class="row top-buffer">
<div class="col-md-12">
<!-- Nav tabs -->
<ul class="nav nav-tabs" role="tablist">
<li role="presentation" class="active"><a href="#treatments" role="tab" data-toggle="tab">Treatments</a></li>
<li role="presentation"><a href="#treatment-replicates" role="tab" data-toggle="tab">Treatment Replicates</a></li>
</ul>
<!-- Tab panes -->
<div class="tab-content">
<div role="tabpanel" class="tab-pane active" id="treatments">
<table class="table table-border table-hover table-condensed">
<thead>
<tr>
<th>Experiment</th>
<th>Type</th>
<th>Species</th>
<th>Sex</th>
<th>Container</th>
<th>Replicates</th>
<th>Flaw</th>
</tr>
</thead>
<tbody>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
<tr>
<td>Perchlorate diel</td>
<td>10 ppm perchlorate, iodide supplemented</td>
<td>Threespine stickleback</td>
<td>Both Sexes</td>
<td>Jar</td>
<td>1; 2; 3</td>
<td></td>
</tr>
</tbody>
</table>
</div>
<div role="tabpanel" class="tab-pane" id="treatment-replicates">...</div>
</div>
</div>
</div>

View file

@ -0,0 +1,13 @@
{{
collection/create-container
model=model
validations=(hash
collection=CollectionValidations
collectionSpecies=CollectionSpeciesValidations
envMeasurements=CollectionMeasurementValidations
datasheet=DatasheetValidations)
options=options
hasMany=hasMany
onSave=(action 'onSave')
onCancel=(action 'onCancel')
}}

View file

@ -0,0 +1,13 @@
{{
collection/create-container
model=model
validations=(hash
collection=CollectionValidations
collectionSpecies=CollectionSpeciesValidations
datasheet=DatasheetValidations
envMeasurements=CollectionMeasurementValidations)
options=options
hasMany=hasMany
onSave=(action 'onSave')
onCancel=(action 'onCancel')
}}

View file

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

View file

@ -1 +1,11 @@
{{collections-container model=model}}
{{
collection/list-container
model=model
filters=filters
options=options
changeFilter=(action 'changeFilter')
resetFilter=(action 'resetFilter')
changePage=(action 'changePage')
onRowClick=(action 'rowClick')
createCollection=(action 'createCollection')
}}

View file

@ -0,0 +1 @@
{{loading-spinner}}

View file

@ -1,130 +0,0 @@
<form class="form-horizontal">
<div class="form-group">
<label class="col-sm-2 control-label">
Start Date
</label>
<div class="col-sm-10">
<input type="date" class="form-control">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Start Time
</label>
<div class="col-sm-10">
<input type="date" class="form-control">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
End Date
</label>
<div class="col-sm-10">
<input type="date" class="form-control">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
End Time
</label>
<div class="col-sm-10">
<input type="date" class="form-control">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Collectors
</label>
<div class="col-sm-10">
<select multiple=true class="form-control">
<option>Frank</option>
<option>Loren</option>
<option>Danielle</option>
<option>Matt</option>
<option>Lauren</option>
<option>Amy</option>
</select>
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Site
</label>
<div class="col-sm-10">
<select class="form-control">
<option>--- Select a Site ---</option>
<option>SQ12</option>
<option>SQ1</option>
<option>SQ3</option>
</select>
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Method
</label>
<div class="col-sm-10">
<select class="form-control">
<option>--- Select a Method ---</option>
<option>MT</option>
</select>
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Type
</label>
<div class="col-sm-10">
<select class="form-control">
<option>--- Select a Type ---</option>
<option>Embryos</option>
<option>Adults</option>
</select>
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
# Of Traps
</label>
<div class="col-sm-10">
<input type="text" class="form-control">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Experiments
</label>
<div class="col-sm-10">
<select multiple=true class="form-control">
<option>Perchlorate diel</option>
<option>Perchlorate rescue</option>
<option>Ship creek monitoring</option>
<option>St Lawrence island monitoring</option>
</select>
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">
Attachments
</label>
<div class="col-sm-10">
<input type="file">
</div>
</div>
<div class="form-group">
<div class="col-sm-offset-2 col-sm-10">
<button type="submit" class="btn btn-default">Save</button>
</div>
</div>
</form>

View file

@ -0,0 +1 @@
{{label}}

View file

@ -0,0 +1 @@
{{yield}}

View file

@ -1,36 +1,35 @@
<div class="col-md-12">
<div class="pull-left" style="padding-top: 26px;">
Showing 1-100 of 102 entries
</div>
<div class="pull-right">
<nav aria-label="Page navigation">
<ul class="pagination">
<li>
<a href="#">
<span aria-hidden="true">First</span>
</a>
</li>
<li>
<a href="#">
<span aria-hidden="true">Previous</span>
</a>
</li>
<li><a href="#">1</a></li>
<li><a href="#">2</a></li>
<li><a href="#">3</a></li>
<li><a href="#">4</a></li>
<li><a href="#">5</a></li>
<li>
<a href="#">
<span aria-hidden="true">Next</span>
</a>
</li>
<li>
<a href="#">
<span aria-hidden="true">Last</span>
</a>
</li>
<div class="row">
<div class="col-md-6 table-nav">
<ul class="pager pull-left">
{{#if notOnFirst}}
<li><a {{action (action changePage first)}}>First</a></li>
{{else}}
<li class="disabled"><a>First</a></li>
{{/if}}
{{#if prev}}
<li><a {{action (action changePage prev)}}>Previous</a></li>
{{else}}
<li class="disabled"><a>Previous</a></li>
{{/if}}
{{#if next}}
<li><a {{action (action changePage next)}}>Next</a></li>
{{else}}
<li class="disabled"><a>Next</a></li>
{{/if}}
{{#if notOnLast}}
<li><a {{action (action changePage last)}}>Last</a></li>
{{else}}
<li class="disabled"><a>Last</a></li>
{{/if}}
</ul>
</nav>
</div>
<div class="col-md-6">
<div class="pull-right table-stats">
<span class="label label-default">Current Page: {{currentPage}}</span>
<span class="label label-default">Total Records: {{totalRecords}}</span>
</div>
</div>
</div>

View file

@ -1,10 +1,14 @@
{{#if hasBlock}}
{{yield (hash
table=(component 'light-table' table=table)
grid=(component 'light-table' table=table onRowClicked=(action (optional onRowClicked)))
pagination=(component 'ccdb-pagination' model=model changePage=(action (optional changePage)))
)}}
{{else}}
{{ccdb-pagination model=model changePage=(action (optional changePage))}}
<div class="table-responsive">
{{#light-table table tableClassNames="table table-striped" as |t|}}
{{t.head}}
{{t.body}}
{{t.body onRowClick=(action (optional onRowClick))}}
{{/light-table}}
</div>
{{/if}}

View file

@ -0,0 +1,283 @@
{{#crud-form
changesets=changesets
onSave=(action onSave)
onCancel=(action onCancel) as |f|
}}
<div class="row">
<div class="col-md-4">
<div class="well">
{{#f.content class='form'}}
{{#with changesets.model as |changeset|}}
{{#validated-field property='project' label='Project' changeset=changeset}}
{{#power-select
options=options.projects
selected=changeset.project
onchange=(action (mut changeset.project))
searchField='name'
as |project|
}}
{{project.name}}
{{/power-select}}
{{/validated-field}}
{{#validated-field property='adfgPermit' label='ADFG Permit' changeset=changeset}}
{{#power-select-with-create
options=options.adfgPermits
selected=changeset.adfgPermit
onchange=(action (mut changeset.adfgPermit))
oncreate=(action 'addOption' 'adfg-permit' 'adfgPermits' 'adfgPermit' 'name')
searchField='name'
as |adfgPermit term|
}}
{{adfgPermit.name}}
{{/power-select-with-create}}
{{/validated-field}}
{{#validated-field property='studyLocation' changeset=changeset}}
<label class="control-label">
Study location
<a href="{{newStudyLocationAdmin}}" target="_blank">+</a>
</label>
{{#power-select
search=(action 'searchStudyLocation')
options=options.studyLocations
selected=changeset.studyLocation
onchange=(action (mut changeset.studyLocation))
searchField='code'
as |studyLocation|
}}
{{studyLocation.code}}
{{/power-select}}
{{/validated-field}}
{{#validated-field property='collectionType' label='Collection type' changeset=changeset}}
{{#power-select
options=options.collectionTypes
selected=changeset.collectionType
onchange=(action (mut changeset.collectionType))
searchField='name'
as |collectionType|
}}
{{collectionType.name}}
{{/power-select}}
{{/validated-field}}
{{#validated-field property='collectionMethod' label='Collection method' changeset=changeset}}
{{#power-select
options=options.collectionMethods
selected=changeset.collectionMethod
onchange=(action (mut changeset.collectionMethod))
searchField='name'
as |collectionMethod|
}}
{{collectionMethod.name}}
{{/power-select}}
{{/validated-field}}
{{#validated-field property='numberOfTraps' label='Number of traps' changeset=changeset}}
{{input value=changeset.numberOfTraps type='number' class='form-control'}}
{{/validated-field}}
{{#validated-field property='collectionStartDate' label='Collection start date' changeset=changeset}}
{{
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 property='collectionStartTime' label='Collection start time' changeset=changeset}}
{{input value=changeset.collectionStartTime type='time' class='form-control' placeholder='HH:MM:SS (24 hour)'}}
{{/validated-field}}
{{#validated-field property='collectionEndDate' label='Collection end date' changeset=changeset}}
{{
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 property='collectionEndTime' label='Collection end time' changeset=changeset}}
{{input value=changeset.collectionEndTime type='time' class='form-control' placeholder='HH:MM:SS (24 hour)'}}
{{/validated-field}}
{{/with}}
{{/f.content}}
</div>
</div>
<div class="col-md-8">
<div class="row">
<div class="col-md-12">
{{#validated-field property='notes' label='Notes' changeset=changesets.model}}
{{textarea value=changesets.model.notes class='form-control'}}
{{/validated-field}}
</div>
</div>
<div class="row">
<div class="col-md-12">
<table class="table">
<caption>
Species / Count Info
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addHasMany' 'collection-species' 'collectionSpecies')}}
</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}}
{{#power-select
options=options.sexes
selected=cs.changeset.sex
onchange=(action (mut cs.changeset.sex))
searchField='name'
as |sex|
}}
{{sex.name}}
{{/power-select}}
{{/validated-field}}
</td>
<td class="col-md-2">
{{action-button isDanger=true isXSmall=true label='X' onClick=(action 'deleteHasMany' cs 'collectionSpecies')}}
</td>
</tr>
{{/each}}
</tbody>
</table>
</div>
</div>
<div class="row">
<div class="col-md-12">
<table class="table">
<caption>
Environmental Measurements
{{action-button isSuccess=true isXSmall=true label='+' onClick=(action 'addHasMany' 'collection-measurement' 'envMeasurements')}}
</caption>
<thead>
<tr>
<th class="col-md-3">Date Measured</th>
<th class="col-md-3">Time Measured</th>
<th class="col-md-3">Water Temperature (deg C)</th>
<th class="col-md-3">Air Temperature (deg C)</th>
<th class="col-md-1">Delete</th>
</tr>
</thead>
<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>
{{f.save}} {{f.cancel}}
{{/crud-form}}

View file

@ -0,0 +1,75 @@
{{
action-button
isPrimary=true
label='Edit Collection'
onClick=(action editCollection)
}}
{{
confirm-button
initialLabel='Delete Collection'
confirmLabel='Yes, Delete Collection'
onClick=(action deleteCollection)
}}
<h3>Main Detail</h3>
<div class="row">
<div class="col-md-12">
{{#ccdb-table model=model columns=mainColumns as |c|}}
{{#c.grid as |g|}}
{{g.head}}
{{g.body}}
{{/c.grid}}
{{/ccdb-table}}
</div>
</div>
<hr>
<h3>Species/Counts</h3>
<div class="row">
<div class="col-md-12">
{{#ccdb-table model=model.[0].collectionSpecies columns=collectionSpeciesColumns as |c|}}
{{#c.grid as |g|}}
{{g.head}}
{{g.body}}
{{/c.grid}}
{{/ccdb-table}}
</div>
</div>
<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>
<div class="row">
<div class="col-md-12">
{{model.[0].notes}}
</div>
</div>
<hr>
<h3>Attachments</h3>
<div class="row">
<div class="col-md-12">
<ul>
{{#each model.[0].datasheets as |d|}}
<li><a href="{{ d.datasheet }}">{{ d.datasheet }}</a></li>
{{/each}}
</ul>
</div>
</div>

View file

@ -0,0 +1,163 @@
{{
action-button
isPrimary=true
label='New Collection'
onClick=(action createCollection)
}}
<hr>
{{#ccdb-filter
filters=filters
options=options
}}
<div class="well">
<div class="row">
<div class="col-md-3">
<label>Projects</label>
{{#power-select-multiple
options=options.projects
selected=filters.project
onchange=(action (mut filters.project))
searchField='name'
as |project|
}}
{{project.name}}
{{/power-select-multiple}}
</div>
<div class="col-md-3">
<label>Regions</label>
{{#power-select-multiple
options=options.regions
selected=filters.region
onchange=(action (mut filters.region))
searchField='name'
as |region|
}}
{{region.name}}
{{/power-select-multiple}}
</div>
<div class="col-md-3">
<label>Sites</label>
{{#power-select-multiple
options=options.sites
selected=filters.site
onchange=(action (mut filters.site))
searchField='name'
as |site|
}}
{{site.name}}
{{/power-select-multiple}}
</div>
<div class="col-md-3">
<label>Study Locations</label>
{{#power-select-multiple
options=options.studyLocations
selected=filters.study_location
onchange=(action (mut filters.study_location))
searchField='code'
as |studyLocation|
}}
{{studyLocation.code}}
{{/power-select-multiple}}
</div>
</div>
<div class="row">
<div class="col-md-4">
<label>Collection Methods</label>
{{#power-select-multiple
options=options.collectionMethods
selected=filters.collection_method
onchange=(action (mut filters.collection_method))
searchField='name'
as |collectionMethod|
}}
{{collectionMethod.name}}
{{/power-select-multiple}}
</div>
<div class="col-md-4">
<label>Number of Traps</label>
{{input type="text" class="form-control" value=filters.number_of_traps}}
</div>
<div class="col-md-4">
<label>ADFG Permit</label>
{{#power-select-multiple
options=options.adfgPermits
selected=filters.adfg_permit
onchange=(action (mut filters.adfg_permit))
searchField='name'
as |adfgPermit|
}}
{{adfgPermit.name}}
{{/power-select-multiple}}
</div>
</div>
<div class="row">
<div class="col-md-4">
<label>Start Date</label>
{{
pikaday-input
onSelection=(action (mut filters.collection_start_date))
value=filters.collection_start_date
useUTC=true
placeholder='MM/DD/YYYY'
format='MM/DD/YYYY'
class='form-control'
}}
</div>
<div class="col-md-4">
<label>End Date</label>
{{
pikaday-input
onSelection=(action (mut filters.collection_end_date))
value=filters.collection_end_date
useUTC=true
placeholder='MM/DD/YYYY'
format='MM/DD/YYYY'
class='form-control'
}}
</div>
<div class="col-md-4">
<label>Species</label>
{{#power-select-multiple
options=options.species
selected=filters.species
onchange=(action (mut filters.species))
searchField='commonName'
as |species|
}}
{{species.commonName}}
{{/power-select-multiple}}
</div>
</div>
<div class="row top-buffer">
<div class="col-md-12">
<div class="btn-group">
{{
action-button
isSuccess=true
label='Search'
onClick=(action changeFilter filters)
}}
{{
action-button
isWarning=true
label='Reset'
onClick=(action resetFilter)
}}
</div>
</div>
</div>
</div>
{{/ccdb-filter}}
{{
ccdb-table
model=model
columns=columns
changePage=(action changePage)
onRowClick=(action onRowClick)
}}

View file

@ -1,5 +0,0 @@
{{#link-to "collections.new" class="btn btn-default"}}New Collection{{/link-to}}
<hr>
{{filter-collections}}
{{ccdb-pagination}}
{{ccdb-table model=model columns=columns}}

View 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}}

View file

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

View file

@ -1,73 +0,0 @@
<div class="well well-sm">
<div class="row">
<div class="col-md-3">
<label>Project</label>
<select class="form-control">
<option selected=selected>Perchlorate</option>
<option>Ship Creek</option>
<option>St. Lawrence Island</option>
</select>
</div>
<div class="col-md-3">
<label>Study Location</label>
<select class="form-control">
<option selected=selected>TL5</option>
<option>SQ12</option>
<option>CPISBV</option>
</select>
</div>
<div class="col-md-3">
<label>Method</label>
<select class="form-control">
<option selected=selected>Fyke Net</option>
<option>Seine</option>
<option>Minnow Trap</option>
</select>
</div>
<div class="col-md-3">
<label>Type</label>
<select class="form-control">
<option selected=selected>Adults</option>
<option>Embryos</option>
</select>
</div>
</div>
<div class="row top-buffer">
<div class="col-md-4">
<label># of Traps</label>
<input type="text" class="form-control" placeholder="3">
</div>
<div class="col-md-4">
<label>Start</label>
<input type="datetime" class="form-control">
</div>
<div class="col-md-4">
<label>End</label>
<input type="datetime" class="form-control">
</div>
</div>
<div class="row top-buffer">
<div class="col-md-12">
<div class="btn-group" role="group">
<button type="submit" class="btn btn-default">
Search
</button>
<button type="button" class="btn btn-default">
Reset
</button>
</div>
</div>
</div>
</div>

View file

@ -0,0 +1 @@
{{yield}}

View file

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

View file

@ -1 +1 @@
TBA

View 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;
}
}
});

11
app/transforms/file.js Normal file
View file

@ -0,0 +1,11 @@
import DS from 'ember-data';
export default DS.Transform.extend({
deserialize(serialized) {
return serialized;
},
serialize(deserialized) {
return deserialized;
}
});

View file

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

View 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),
}

View file

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

View file

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

View file

@ -0,0 +1,8 @@
import {
validatePresence,
} from 'ember-changeset-validations/validators';
export default {
datasheet: validatePresence(true),
collection: validatePresence(true),
}

View file

@ -9,8 +9,7 @@ module.exports = function(environment) {
locationType: 'auto',
EmberENV: {
FEATURES: {
// Here you can enable experimental features on an ember canary build
// e.g. 'with-controller': true
'ds-payload-type-hooks': true,
},
EXTEND_PROTOTYPES: {
// Prevent Ember Data from overriding Date.parse.
@ -21,12 +20,23 @@ module.exports = function(environment) {
APP: {
// Here you can pass flags/options to your application instance
// 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') {
ENV.APP.API_HOST = 'http://localhost:8000';
ENV.APP.API_NAMESPACE = 'api/v1';
ENV.sentry.development = true;
}
if (environment === 'test') {
@ -41,7 +51,7 @@ module.exports = function(environment) {
}
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';
}

View file

@ -11,6 +11,7 @@ module.exports = function(defaults) {
});
app.import('bower_components/bootstrap/dist/css/bootstrap.min.css');
app.import('bower_components/bootstrap/dist/css/bootstrap-theme.min.css');
return app.toTree();
};

21802
package-lock.json generated Normal file

File diff suppressed because it is too large Load diff

View file

@ -17,37 +17,46 @@
"devDependencies": {
"broccoli-asset-rev": "^2.4.5",
"ember-ajax": "^3.0.0",
"ember-cli": "~2.14.0",
"ember-changeset": "^1.3.0",
"ember-changeset-validations": "^1.2.8",
"ember-cli": "^3.24.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-dependency-checker": "^1.3.0",
"ember-cli-eslint": "^3.0.0",
"ember-cli-flash": "1.4.3",
"ember-cli-dependency-checker": "^2.0.0",
"ember-cli-eslint": "^4.0.0",
"ember-cli-flash": "^1.4.3",
"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-moment-shim": "^3.5.0",
"ember-cli-qunit": "^4.0.0",
"ember-cli-sentry": "^3.0.0",
"ember-cli-shims": "^1.1.0",
"ember-cli-sri": "^2.1.0",
"ember-cli-uglify": "^1.2.0",
"ember-data": "~2.14.3",
"ember-cli-uglify": "^2.0.0",
"ember-composable-helpers": "^2.0.3",
"ember-data": "~2.16.2",
"ember-export-application-global": "^2.0.0",
"ember-inflector": "^2.0.1",
"ember-light-table": "^1.10.0",
"ember-load-initializers": "^1.0.0",
"ember-moment": "7.3.1",
"ember-power-select": "1.8.5",
"ember-power-select-with-create": "0.4.3",
"ember-pikaday": "^2.2.3",
"ember-power-select": "^1.8.5",
"ember-power-select-with-create": "^0.4.3",
"ember-resolver": "^4.0.0",
"ember-responsive": "^2.0.4",
"ember-simple-auth": "1.4.0",
"ember-sinon": "^1.0.0",
"ember-source": "~2.14.0",
"ember-source": "~2.16.0",
"loader.js": "^4.2.3"
},
"engines": {
"node": "^4.5 || 6.* || >= 7.*"
},
"private": true
"private": true,
"dependencies": {
"npm": "^5.7.1"
}
}

View file

@ -3,10 +3,20 @@ module.exports = {
test_page: 'tests/index.html?hidepassed',
disable_watching: true,
launch_in_ci: [
'PhantomJS'
'Chrome'
],
launch_in_dev: [
'PhantomJS',
'Chrome'
],
browser_args: {
Chrome: {
mode: 'ci',
args: [
'--disable-gpu',
'--headless',
'--remote-debugging-port=9222',
'--window-size=1440,900'
]
},
}
};

View file

@ -1,5 +1,5 @@
import Ember from 'ember';
import { run } from '@ember/runloop';
export default function destroyApp(application) {
Ember.run(application, 'destroy');
run(application, 'destroy');
}

View file

@ -1,6 +1,3 @@
import Ember from 'ember';
import FlashObject from 'ember-cli-flash/flash/object';
const { K } = Ember;
FlashObject.reopen({ init: K });
FlashObject.reopen({ init() {} });

View file

@ -1,10 +1,8 @@
import { module } from 'qunit';
import Ember from 'ember';
import { resolve } from 'rsvp';
import startApp from '../helpers/start-app';
import destroyApp from '../helpers/destroy-app';
const { RSVP: { resolve } } = Ember;
export default function(name, options = {}) {
module(name, {
beforeEach() {

View file

@ -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';
const {
getOwner
} = Ember;
const { classify } = Ember.String;
MediaService.reopen({
// Change this if you want a different default breakpoint in tests.
_defaultBreakpoint: 'desktop',
_breakpointArr: Ember.computed('breakpoints', function() {
return Object.keys(this.get('breakpoints')) || Ember.A([]);
_breakpointArr: computed('breakpoints', function() {
return Object.keys(this.get('breakpoints')) || A([]);
}),
_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
const mediaService = app.__deprecatedInstance__.lookup('service:media');
mediaService._forceSetBreakpoint(breakpoint);

View file

@ -1,12 +1,13 @@
import Ember from 'ember';
import Application from '../../app';
import config from '../../config/environment';
import { merge } from '@ember/polyfills';
import { run } from '@ember/runloop';
export default function startApp(attrs) {
let attributes = Ember.merge({}, config.APP);
attributes = Ember.merge(attributes, attrs); // use defaults, but you can override;
let attributes = merge({}, config.APP);
attributes = merge(attributes, attrs); // use defaults, but you can override;
return Ember.run(() => {
return run(() => {
let application = Application.create(attributes);
application.setupForTesting();
application.injectTestHelpers();

View file

@ -1,25 +0,0 @@
import { moduleForComponent, test } from 'ember-qunit';
import hbs from 'htmlbars-inline-precompile';
moduleForComponent('ccdb-pagination', 'Integration | Component | ccdb pagination', {
integration: true
});
test('it renders', function(assert) {
// Set any properties with this.set('myProperty', 'value');
// Handle any actions with this.on('myAction', function(val) { ... });
this.render(hbs`{{ccdb-pagination}}`);
assert.equal(this.$().text().trim(), '');
// Template block usage:
this.render(hbs`
{{#ccdb-pagination}}
template block text
{{/ccdb-pagination}}
`);
assert.equal(this.$().text().trim(), 'template block text');
});

View file

@ -1,25 +0,0 @@
import { moduleForComponent, test } from 'ember-qunit';
import hbs from 'htmlbars-inline-precompile';
moduleForComponent('filter-collections', 'Integration | Component | filter collections', {
integration: true
});
test('it renders', function(assert) {
// Set any properties with this.set('myProperty', 'value');
// Handle any actions with this.on('myAction', function(val) { ... });
this.render(hbs`{{filter-collections}}`);
assert.equal(this.$().text().trim(), '');
// Template block usage:
this.render(hbs`
{{#filter-collections}}
template block text
{{/filter-collections}}
`);
assert.equal(this.$().text().trim(), 'template block text');
});

View file

@ -1,6 +1,6 @@
import $ from 'jquery';
import { moduleFor, test } from 'ember-qunit';
import sinon from 'sinon';
import Ember from 'ember';
moduleFor('authenticator:application', 'Unit | application', {
unit: true,
@ -35,28 +35,28 @@ test('`invalidate` should invalidate the session', function(assert) {
test('`makeRequest` should make a request', function(assert) {
assert.expect(2);
const stub = sinon.stub(Ember.$, 'ajax');
const stub = sinon.stub($, 'ajax');
stub.resolves(42);
const authenticator = this.subject();
authenticator.set('serverTokenEndpoint', 'foo')
const promise = authenticator.makeRequest({bar: 'baz'}).then((d) => {
assert.equal(d, 42);
});
assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}}));
Ember.$.ajax.restore();
assert.ok($.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}}));
$.ajax.restore();
return promise;
});
test('authenticate should craft a nice payload', function(assert) {
assert.expect(2);
const stub = sinon.stub(Ember.$, 'ajax');
const stub = sinon.stub($, 'ajax');
stub.resolves(42);
const authenticator = this.subject();
authenticator.set('serverTokenEndpoint', 'foo')
const promise = authenticator.authenticate('myusername', 'mypassword').then((d) => {
assert.equal(d, 42);
});
assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}}));
Ember.$.ajax.restore();
assert.ok($.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}}));
$.ajax.restore();
return promise;
});

View file

@ -0,0 +1,12 @@
import { moduleForModel, test } from 'ember-qunit';
moduleForModel('adfg-permit', 'Unit | Model | adfg permit', {
// Specify the other units that are required for this test.
needs: []
});
test('it exists', function(assert) {
let model = this.subject();
// let store = this.store();
assert.ok(!!model);
});

View file

@ -0,0 +1,12 @@
import { moduleForModel, test } from 'ember-qunit';
moduleForModel('region', 'Unit | Model | region', {
// Specify the other units that are required for this test.
needs: []
});
test('it exists', function(assert) {
let model = this.subject();
// let store = this.store();
assert.ok(!!model);
});

View file

@ -0,0 +1,12 @@
import { moduleForModel, test } from 'ember-qunit';
moduleForModel('site', 'Unit | Model | site', {
// Specify the other units that are required for this test.
needs: []
});
test('it exists', function(assert) {
let model = this.subject();
// let store = this.store();
assert.ok(!!model);
});

View file

@ -1,10 +0,0 @@
import { moduleFor, test } from 'ember-qunit';
moduleFor('route:collections', 'Unit | Route | collections', {
unit: true,
});
test('it exists', function(assert) {
const route = this.subject();
assert.ok(route);
});

Some files were not shown because too many files have changed in this diff Show more