Compare commits
	
		
			4 commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ffa302298f | |||
| 408b1bb1d4 | |||
| f158d7a7f3 | |||
| 825b5b0d89 | 
					 103 changed files with 635 additions and 23629 deletions
				
			
		
							
								
								
									
										10
									
								
								.travis.yml
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								.travis.yml
									
										
									
									
									
								
							|  | @ -3,17 +3,11 @@ branches: | |||
|   only: | ||||
|     - master | ||||
| node_js: | ||||
|   - "9" | ||||
| sudo: required | ||||
| dist: trusty | ||||
| addons: | ||||
|   chrome: stable | ||||
|   - '6' | ||||
| sudo: false | ||||
| cache: | ||||
|   directories: | ||||
|   - "$HOME/.npm" | ||||
| env: | ||||
|   global: | ||||
|     - JOBS=1 | ||||
| before_install: | ||||
|   - npm config set spin false | ||||
|   - 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/) | ||||
| * [Node.js](https://nodejs.org/) (with NPM) | ||||
| * [Ember CLI](https://ember-cli.com/) | ||||
| * [Google Chrome](https://google.com/chrome/) | ||||
| * [PhantomJS](http://phantomjs.org/) | ||||
| 
 | ||||
| ## Installation | ||||
| 
 | ||||
|  |  | |||
|  | @ -9,11 +9,4 @@ 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); | ||||
|   } | ||||
| }); | ||||
|  |  | |||
|  | @ -1,19 +0,0 @@ | |||
| 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; | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,9 +1,11 @@ | |||
| import Application from '@ember/application'; | ||||
| import Ember from 'ember'; | ||||
| import Resolver from './resolver'; | ||||
| import loadInitializers from 'ember-load-initializers'; | ||||
| import config from './config/environment'; | ||||
| 
 | ||||
| const App = Application.extend({ | ||||
| let App; | ||||
| 
 | ||||
| App = Ember.Application.extend({ | ||||
|   modulePrefix: config.modulePrefix, | ||||
|   podModulePrefix: config.podModulePrefix, | ||||
|   Resolver | ||||
|  |  | |||
|  | @ -1,11 +1,9 @@ | |||
| import { Promise } from 'rsvp'; | ||||
| import $ from 'jquery'; | ||||
| import { get } from '@ember/object'; | ||||
| import { isEmpty } from '@ember/utils'; | ||||
| import { run } from '@ember/runloop'; | ||||
| import Ember from 'ember'; | ||||
| 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', | ||||
|  |  | |||
|  | @ -1,7 +1,8 @@ | |||
| import { isEmpty } from '@ember/utils'; | ||||
| import { get } from '@ember/object'; | ||||
| import Ember from 'ember'; | ||||
| 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'); | ||||
|  |  | |||
|  | @ -1,40 +0,0 @@ | |||
| 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')(); | ||||
|   } | ||||
| }); | ||||
|  | @ -1,3 +1,5 @@ | |||
| import Component from '@ember/component'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Component } = Ember; | ||||
| 
 | ||||
| export default Component.extend({}); | ||||
|  |  | |||
|  | @ -1,3 +0,0 @@ | |||
| import Component from '@ember/component'; | ||||
| 
 | ||||
| export default Component.extend({ }); | ||||
|  | @ -1,41 +1,5 @@ | |||
| import Component from '@ember/component'; | ||||
| import { alias } from '@ember/object/computed'; | ||||
| import { computed } from '@ember/object'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| 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'); }), | ||||
| export default Ember.Component.extend({ | ||||
|   classNames: ['row'], | ||||
| }); | ||||
|  |  | |||
|  | @ -1,13 +1,16 @@ | |||
| import Component from '@ember/component'; | ||||
| import Ember from 'ember'; | ||||
| import Table from 'ember-light-table'; | ||||
| 
 | ||||
| const { Component } = Ember; | ||||
| 
 | ||||
| export default Component.extend({ | ||||
|   // ARGS
 | ||||
|   model: null, | ||||
|   columns: null, | ||||
| 
 | ||||
|   table: null, | ||||
|   didReceiveAttrs() { | ||||
| 
 | ||||
|   classNames: ['row'], | ||||
| 
 | ||||
|   init() { | ||||
|     this._super(...arguments); | ||||
|     const table = new Table(this.get('columns'), this.get('model')); | ||||
|     this.set('table', table); | ||||
|  |  | |||
|  | @ -1,93 +0,0 @@ | |||
| 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); | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,34 +0,0 @@ | |||
| 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', }, | ||||
|   ], | ||||
| }); | ||||
|  | @ -1,20 +0,0 @@ | |||
| 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', }, | ||||
|   ], | ||||
| }); | ||||
							
								
								
									
										15
									
								
								app/components/collections-container.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								app/components/collections-container.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | |||
| 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', }, | ||||
|   ], | ||||
| }); | ||||
|  | @ -1,23 +0,0 @@ | |||
| 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 +0,0 @@ | |||
| import Component from '@ember/component'; | ||||
| 
 | ||||
| export default Component.extend({ | ||||
|   // ARGS
 | ||||
|   changesets: null, | ||||
| }); | ||||
							
								
								
									
										5
									
								
								app/components/filter-collections.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/components/filter-collections.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| import Ember from 'ember'; | ||||
| 
 | ||||
| export default Ember.Component.extend({ | ||||
|   tagName: 'form', | ||||
| }); | ||||
|  | @ -1,5 +0,0 @@ | |||
| import Component from '@ember/component'; | ||||
| 
 | ||||
| export default Component.extend({ | ||||
|   tagName: 'form', | ||||
| }); | ||||
|  | @ -1,4 +1,6 @@ | |||
| import Component from '@ember/component'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Component } = Ember; | ||||
| 
 | ||||
| export default Component.extend({ | ||||
|   classNames: ['spinner'], | ||||
|  |  | |||
|  | @ -1,18 +0,0 @@ | |||
| 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')); | ||||
|   }), | ||||
| }); | ||||
|  | @ -1,5 +1,6 @@ | |||
| import Controller from '@ember/controller'; | ||||
| import { inject as service } from '@ember/service'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Controller, inject: { service }} = Ember; | ||||
| 
 | ||||
| export default Controller.extend({ | ||||
|   session: service('session'), | ||||
|  |  | |||
|  | @ -1,46 +0,0 @@ | |||
| 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); | ||||
|       }); | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,52 +0,0 @@ | |||
| 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); | ||||
|       }); | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,14 +0,0 @@ | |||
| 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'); | ||||
|       }); | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,85 +0,0 @@ | |||
| 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); | ||||
|       }); | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,5 +1,6 @@ | |||
| import Controller from '@ember/controller'; | ||||
| import { inject as service } from '@ember/service'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Controller, inject: { service } } = Ember; | ||||
| 
 | ||||
| export default Controller.extend({ | ||||
|   session: service(), | ||||
|  |  | |||
|  | @ -17,8 +17,8 @@ | |||
|   <body> | ||||
|     {{content-for "body"}} | ||||
| 
 | ||||
|     <script integrity="" src="{{rootURL}}assets/vendor.js"></script> | ||||
|     <script integrity="" src="{{rootURL}}assets/ccdb-web.js"></script> | ||||
|     <script src="{{rootURL}}assets/vendor.js"></script> | ||||
|     <script src="{{rootURL}}assets/ccdb-web.js"></script> | ||||
| 
 | ||||
|     {{content-for "body-footer"}} | ||||
|   </body> | ||||
|  |  | |||
|  | @ -1,62 +0,0 @@ | |||
| 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); | ||||
|     } | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,73 +0,0 @@ | |||
| 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(); | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,10 +0,0 @@ | |||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| const { Model, attr, hasMany } = DS; | ||||
| 
 | ||||
| export default Model.extend({ | ||||
|   name:      attr('string'), | ||||
|   sortOrder: attr('number'), | ||||
| 
 | ||||
|   collection: hasMany('collection'), | ||||
| }); | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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'), | ||||
| }); | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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'), | ||||
| }); | ||||
|  | @ -1,41 +1,21 @@ | |||
| import { mapBy } from '@ember/object/computed'; | ||||
| import { computed } from '@ember/object'; | ||||
| import Ember from 'ember'; | ||||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| const { Model, attr, belongsTo, hasMany } = DS; | ||||
| const { computed } = Ember; | ||||
| const { Model, attr, belongsTo } = DS; | ||||
| 
 | ||||
| export default Model.extend({ | ||||
|   displayName:         attr('string'), | ||||
|   numberOfTraps:       attr('number'), | ||||
|   collectionStartDate: attr('ccdb-date'), | ||||
|   collectionStartDate: attr('string-null-to-empty'), | ||||
|   collectionStartTime: attr('string-null-to-empty'), | ||||
|   collectionEndDate:   attr('ccdb-date'), | ||||
|   collectionEndDate:   attr('string-null-to-empty'), | ||||
|   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(', '); | ||||
|   }), | ||||
|   project:          belongsTo('project'), | ||||
|   studyLocation:    belongsTo('study-location'), | ||||
|   collectionMethod: belongsTo('collection-method'), | ||||
|   collectionType:   belongsTo('collection-type'), | ||||
| 
 | ||||
|   startDateTime: computed('collectionStartDate', 'collectionStartTime', | ||||
|     function() { return this._mergeDateTime('Start'); }), | ||||
|  |  | |||
|  | @ -1,9 +0,0 @@ | |||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| const { Model, attr, belongsTo } = DS; | ||||
| 
 | ||||
| export default Model.extend({ | ||||
|   datasheet: attr('file'), | ||||
| 
 | ||||
|   collection: belongsTo('collection'), | ||||
| }); | ||||
|  | @ -1,11 +0,0 @@ | |||
| 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'), | ||||
| }); | ||||
|  | @ -1,8 +0,0 @@ | |||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| const { Model, attr } = DS; | ||||
| 
 | ||||
| export default Model.extend({ | ||||
|   name:      attr('string'), | ||||
|   sortOrder: attr('number'), | ||||
| }); | ||||
|  | @ -1,13 +0,0 @@ | |||
| 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'), | ||||
| }); | ||||
|  | @ -1,11 +0,0 @@ | |||
| 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'), | ||||
| }); | ||||
|  | @ -1,6 +1,6 @@ | |||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| const { Model, attr, belongsTo } = DS; | ||||
| const { Model, attr } = DS; | ||||
| 
 | ||||
| export default Model.extend({ | ||||
|   name:               attr('string'), | ||||
|  | @ -10,6 +10,4 @@ export default Model.extend({ | |||
|   collectingLocation: attr('string'), | ||||
|   description:        attr('string'), | ||||
|   sortOrder:          attr('number'), | ||||
| 
 | ||||
|   site: belongsTo('site'), | ||||
| }); | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| import EmberRouter from '@ember/routing/router'; | ||||
| import Ember from 'ember'; | ||||
| import config from './config/environment'; | ||||
| 
 | ||||
| const Router = EmberRouter.extend({ | ||||
| const Router = Ember.Router.extend({ | ||||
|   location: config.locationType, | ||||
|   rootURL: config.rootURL | ||||
| }); | ||||
|  | @ -10,10 +10,8 @@ Router.map(function() { | |||
|   this.route('login'); | ||||
|   this.route('logout'); | ||||
|   this.route('collections', function() { | ||||
|     this.route('create'); | ||||
|     this.route('detail', { path: '/:collection_id' }, function() { | ||||
|       this.route('edit'); | ||||
|     }); | ||||
|     this.route('1'); | ||||
|     this.route('new'); | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,4 +1,6 @@ | |||
| import Route from '@ember/routing/route'; | ||||
| import Ember from 'ember'; | ||||
| import ApplicationRouteMixin from 'ember-simple-auth/mixins/application-route-mixin'; | ||||
| 
 | ||||
| const { Route } = Ember; | ||||
| 
 | ||||
| export default Route.extend(ApplicationRouteMixin, {}); | ||||
|  |  | |||
							
								
								
									
										4
									
								
								app/routes/collections/1.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								app/routes/collections/1.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | |||
| import Ember from 'ember'; | ||||
| 
 | ||||
| export default Ember.Route.extend({ | ||||
| }); | ||||
|  | @ -1,23 +0,0 @@ | |||
| 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); | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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', | ||||
|       }) | ||||
|     ]); | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,26 +0,0 @@ | |||
| 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); | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,96 +1,11 @@ | |||
| import Route from '@ember/routing/route'; | ||||
| import RSVP from 'rsvp'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Route } = Ember; | ||||
| 
 | ||||
| export default Route.extend({ | ||||
|   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)), | ||||
|   model() { | ||||
|     return this.get('store').findAll('collection', { | ||||
|       include: 'project,study-location,collection-method,collection-type' | ||||
|     }); | ||||
|   }, | ||||
| 
 | ||||
|   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); | ||||
|   }, | ||||
|   } | ||||
| }); | ||||
|  |  | |||
							
								
								
									
										4
									
								
								app/routes/collections/new.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								app/routes/collections/new.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | |||
| import Ember from 'ember'; | ||||
| 
 | ||||
| export default Ember.Route.extend({ | ||||
| }); | ||||
|  | @ -1,8 +1,6 @@ | |||
| import Route from '@ember/routing/route'; | ||||
| import Ember from 'ember'; | ||||
| import AuthenticatedRouteMixin from 'ember-simple-auth/mixins/authenticated-route-mixin'; | ||||
| 
 | ||||
| export default Route.extend(AuthenticatedRouteMixin, { | ||||
|   afterModel() { | ||||
|     this.transitionTo('collections'); | ||||
|   }, | ||||
| }); | ||||
| const { Route } = Ember; | ||||
| 
 | ||||
| export default Route.extend(AuthenticatedRouteMixin, {}); | ||||
|  |  | |||
|  | @ -1,4 +1,6 @@ | |||
| import Route from '@ember/routing/route'; | ||||
| import Ember from 'ember'; | ||||
| import UnauthenticatedRouteMixin from 'ember-simple-auth/mixins/unauthenticated-route-mixin'; | ||||
| 
 | ||||
| const { Route } = Ember; | ||||
| 
 | ||||
| export default Route.extend(UnauthenticatedRouteMixin, {}); | ||||
|  |  | |||
|  | @ -1,5 +1,6 @@ | |||
| import Route from '@ember/routing/route'; | ||||
| import { inject as service } from '@ember/service'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| const { Route, inject: { service }} = Ember; | ||||
| 
 | ||||
| export default Route.extend({ | ||||
|   session: service('session'), | ||||
|  |  | |||
|  | @ -1,16 +0,0 @@ | |||
| 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; | ||||
|   }, | ||||
| }); | ||||
|  | @ -1,13 +1,12 @@ | |||
| [data-ember-action]:not(:disabled) { | ||||
|   cursor: pointer; | ||||
| .content { | ||||
|   padding-left: 40px; | ||||
|   padding-right: 40px; | ||||
|   padding-top: 20px; | ||||
|   padding-bottom: 20px; | ||||
| } | ||||
| 
 | ||||
| .table-nav .pager { | ||||
|   margin-top: 10px; | ||||
| } | ||||
| 
 | ||||
| .table-stats { | ||||
|   margin-top: 10px; | ||||
| .top-buffer { | ||||
|   padding-top: 20px; | ||||
| } | ||||
| 
 | ||||
| .form-signin { | ||||
|  | @ -50,10 +49,6 @@ | |||
|   border-top-right-radius: 0; | ||||
| } | ||||
| 
 | ||||
| .top-buffer { | ||||
|   margin-top: 20px; | ||||
| } | ||||
| 
 | ||||
| /* Sidebar */ | ||||
| .sidebar { | ||||
|   position: fixed; | ||||
|  |  | |||
|  | @ -3,6 +3,10 @@ | |||
|   <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}} | ||||
|  |  | |||
							
								
								
									
										147
									
								
								app/templates/collections/1.hbs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								app/templates/collections/1.hbs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,147 @@ | |||
| <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> | ||||
|  | @ -1,13 +0,0 @@ | |||
| {{ | ||||
|   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') | ||||
| }} | ||||
|  | @ -1,13 +0,0 @@ | |||
| {{ | ||||
|   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') | ||||
| }} | ||||
|  | @ -1,6 +0,0 @@ | |||
| {{ | ||||
|   collection/detail-container | ||||
|   model=model | ||||
|   editCollection=(action 'editCollection') | ||||
|   deleteCollection=(action 'deleteCollection') | ||||
| }} | ||||
|  | @ -1,11 +1 @@ | |||
| {{ | ||||
|   collection/list-container | ||||
|   model=model | ||||
|   filters=filters | ||||
|   options=options | ||||
|   changeFilter=(action 'changeFilter') | ||||
|   resetFilter=(action 'resetFilter') | ||||
|   changePage=(action 'changePage') | ||||
|   onRowClick=(action 'rowClick') | ||||
|   createCollection=(action 'createCollection') | ||||
| }} | ||||
| {{collections-container model=model}} | ||||
|  |  | |||
|  | @ -1 +0,0 @@ | |||
| {{loading-spinner}} | ||||
							
								
								
									
										130
									
								
								app/templates/collections/new.hbs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										130
									
								
								app/templates/collections/new.hbs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,130 @@ | |||
| <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> | ||||
|  | @ -1 +0,0 @@ | |||
| {{label}} | ||||
|  | @ -1 +0,0 @@ | |||
| {{yield}} | ||||
|  | @ -1,35 +1,36 @@ | |||
| <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> | ||||
| <div class="col-md-12"> | ||||
|   <div class="pull-left" style="padding-top: 26px;"> | ||||
|     Showing 1-100 of 102 entries | ||||
|   </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 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> | ||||
|       </ul> | ||||
|     </nav> | ||||
|   </div> | ||||
| </div> | ||||
|  |  | |||
|  | @ -1,14 +1,10 @@ | |||
| {{#if hasBlock}} | ||||
|   {{yield (hash | ||||
|     grid=(component 'light-table' table=table onRowClicked=(action (optional onRowClicked))) | ||||
|     pagination=(component 'ccdb-pagination' model=model changePage=(action (optional changePage))) | ||||
|     table=(component 'light-table' table=table) | ||||
|   )}} | ||||
| {{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 onRowClick=(action (optional onRowClick))}} | ||||
|     {{/light-table}} | ||||
|   </div> | ||||
|   {{#light-table table tableClassNames="table table-striped" as |t|}} | ||||
|     {{t.head}} | ||||
|     {{t.body}} | ||||
|   {{/light-table}} | ||||
| {{/if}} | ||||
|  |  | |||
|  | @ -1,283 +0,0 @@ | |||
| {{#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}} | ||||
|  | @ -1,75 +0,0 @@ | |||
| {{ | ||||
|   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> | ||||
|  | @ -1,163 +0,0 @@ | |||
| {{ | ||||
|   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) | ||||
| }} | ||||
							
								
								
									
										5
									
								
								app/templates/components/collections-container.hbs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/templates/components/collections-container.hbs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| {{#link-to "collections.new" class="btn btn-default"}}New Collection{{/link-to}} | ||||
| <hr> | ||||
| {{filter-collections}} | ||||
| {{ccdb-pagination}} | ||||
| {{ccdb-table model=model columns=columns}} | ||||
|  | @ -1,6 +0,0 @@ | |||
| {{#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}} | ||||
|  | @ -1,15 +0,0 @@ | |||
| {{#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}} | ||||
							
								
								
									
										73
									
								
								app/templates/components/filter-collections.hbs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								app/templates/components/filter-collections.hbs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,73 @@ | |||
| <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> | ||||
|  | @ -1 +0,0 @@ | |||
| {{yield}} | ||||
|  | @ -1,13 +0,0 @@ | |||
| {{#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}} | ||||
|  | @ -1 +1 @@ | |||
| TBA | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,19 +0,0 @@ | |||
| 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; | ||||
|     } | ||||
|   } | ||||
| }); | ||||
|  | @ -1,11 +0,0 @@ | |||
| import DS from 'ember-data'; | ||||
| 
 | ||||
| export default DS.Transform.extend({ | ||||
|   deserialize(serialized) { | ||||
|     return serialized; | ||||
|   }, | ||||
| 
 | ||||
|   serialize(deserialized) { | ||||
|     return deserialized; | ||||
|   } | ||||
| }); | ||||
|  | @ -8,6 +8,6 @@ export default Transform.extend({ | |||
|   }, | ||||
| 
 | ||||
|   serialize(deserialized) { | ||||
|     return deserialized === '' ? null : deserialized; | ||||
|     return deserialized; | ||||
|   } | ||||
| }); | ||||
|  |  | |||
|  | @ -1,12 +0,0 @@ | |||
| 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), | ||||
| } | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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), | ||||
| } | ||||
|  | @ -1,16 +0,0 @@ | |||
| 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
 | ||||
| } | ||||
|  | @ -1,8 +0,0 @@ | |||
| import { | ||||
|   validatePresence, | ||||
| } from 'ember-changeset-validations/validators'; | ||||
| 
 | ||||
| export default { | ||||
|   datasheet:  validatePresence(true), | ||||
|   collection: validatePresence(true), | ||||
| } | ||||
|  | @ -9,7 +9,8 @@ module.exports = function(environment) { | |||
|     locationType: 'auto', | ||||
|     EmberENV: { | ||||
|       FEATURES: { | ||||
|         'ds-payload-type-hooks': true, | ||||
|         // Here you can enable experimental features on an ember canary build
 | ||||
|         // e.g. 'with-controller': true
 | ||||
|       }, | ||||
|       EXTEND_PROTOTYPES: { | ||||
|         // Prevent Ember Data from overriding Date.parse.
 | ||||
|  | @ -20,23 +21,12 @@ 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') { | ||||
|  | @ -51,7 +41,7 @@ module.exports = function(environment) { | |||
|   } | ||||
| 
 | ||||
|   if (environment === 'production') { | ||||
|     ENV.APP.API_HOST = 'https://ccdb-api.thermokar.st'; | ||||
|     ENV.APP.API_HOST = 'https://obscure-caverns-99102.herokuapp.com'; | ||||
|     ENV.APP.API_NAMESPACE = 'api/v1'; | ||||
|   } | ||||
| 
 | ||||
|  |  | |||
|  | @ -11,7 +11,6 @@ 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
									
									
									
								
							
							
						
						
									
										21802
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										33
									
								
								package.json
									
										
									
									
									
								
							
							
						
						
									
										33
									
								
								package.json
									
										
									
									
									
								
							|  | @ -17,46 +17,37 @@ | |||
|   "devDependencies": { | ||||
|     "broccoli-asset-rev": "^2.4.5", | ||||
|     "ember-ajax": "^3.0.0", | ||||
|     "ember-changeset": "^1.3.0", | ||||
|     "ember-changeset-validations": "^1.2.8", | ||||
|     "ember-cli": "^3.24.0", | ||||
|     "ember-cli": "~2.14.0", | ||||
|     "ember-cli-app-version": "^3.0.0", | ||||
|     "ember-cli-babel": "^6.6.0", | ||||
|     "ember-cli-babel": "^6.3.0", | ||||
|     "ember-cli-code-coverage": "^0.4.1", | ||||
|     "ember-cli-dependency-checker": "^2.0.0", | ||||
|     "ember-cli-eslint": "^4.0.0", | ||||
|     "ember-cli-flash": "^1.4.3", | ||||
|     "ember-cli-dependency-checker": "^1.3.0", | ||||
|     "ember-cli-eslint": "^3.0.0", | ||||
|     "ember-cli-flash": "1.4.3", | ||||
|     "ember-cli-htmlbars": "^2.0.1", | ||||
|     "ember-cli-htmlbars-inline-precompile": "^1.0.0", | ||||
|     "ember-cli-htmlbars-inline-precompile": "^0.4.3", | ||||
|     "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": "^2.0.0", | ||||
|     "ember-composable-helpers": "^2.0.3", | ||||
|     "ember-data": "~2.16.2", | ||||
|     "ember-cli-uglify": "^1.2.0", | ||||
|     "ember-data": "~2.14.3", | ||||
|     "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-pikaday": "^2.2.3", | ||||
|     "ember-power-select": "^1.8.5", | ||||
|     "ember-power-select-with-create": "^0.4.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.16.0", | ||||
|     "ember-source": "~2.14.0", | ||||
|     "loader.js": "^4.2.3" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": "^4.5 || 6.* || >= 7.*" | ||||
|   }, | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "npm": "^5.7.1" | ||||
|   } | ||||
|   "private": true | ||||
| } | ||||
|  |  | |||
							
								
								
									
										16
									
								
								testem.js
									
										
									
									
									
								
							
							
						
						
									
										16
									
								
								testem.js
									
										
									
									
									
								
							|  | @ -3,20 +3,10 @@ module.exports = { | |||
|   test_page: 'tests/index.html?hidepassed', | ||||
|   disable_watching: true, | ||||
|   launch_in_ci: [ | ||||
|     'Chrome' | ||||
|     'PhantomJS' | ||||
|   ], | ||||
|   launch_in_dev: [ | ||||
|     'PhantomJS', | ||||
|     'Chrome' | ||||
|   ], | ||||
|   browser_args: { | ||||
|     Chrome: { | ||||
|       mode: 'ci', | ||||
|       args: [ | ||||
|         '--disable-gpu', | ||||
|         '--headless', | ||||
|         '--remote-debugging-port=9222', | ||||
|         '--window-size=1440,900' | ||||
|       ] | ||||
|     }, | ||||
|   } | ||||
|   ] | ||||
| }; | ||||
|  |  | |||
|  | @ -1,5 +1,5 @@ | |||
| import { run } from '@ember/runloop'; | ||||
| import Ember from 'ember'; | ||||
| 
 | ||||
| export default function destroyApp(application) { | ||||
|   run(application, 'destroy'); | ||||
|   Ember.run(application, 'destroy'); | ||||
| } | ||||
|  |  | |||
|  | @ -1,3 +1,6 @@ | |||
| import Ember from 'ember'; | ||||
| import FlashObject from 'ember-cli-flash/flash/object'; | ||||
| 
 | ||||
| FlashObject.reopen({ init() {} }); | ||||
| const { K } = Ember; | ||||
| 
 | ||||
| FlashObject.reopen({ init: K }); | ||||
|  |  | |||
|  | @ -1,8 +1,10 @@ | |||
| import { module } from 'qunit'; | ||||
| import { resolve } from 'rsvp'; | ||||
| import Ember from 'ember'; | ||||
| import startApp from '../helpers/start-app'; | ||||
| import destroyApp from '../helpers/destroy-app'; | ||||
| 
 | ||||
| const { RSVP: { resolve } } = Ember; | ||||
| 
 | ||||
| export default function(name, options = {}) { | ||||
|   module(name, { | ||||
|     beforeEach() { | ||||
|  |  | |||
|  | @ -1,16 +1,17 @@ | |||
| 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 Ember from 'ember'; | ||||
| 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: computed('breakpoints', function() { | ||||
|     return Object.keys(this.get('breakpoints')) || A([]); | ||||
|   _breakpointArr: Ember.computed('breakpoints', function() { | ||||
|     return Object.keys(this.get('breakpoints')) || Ember.A([]); | ||||
|   }), | ||||
| 
 | ||||
|   _forceSetBreakpoint(breakpoint) { | ||||
|  | @ -44,7 +45,7 @@ MediaService.reopen({ | |||
|   } | ||||
| }); | ||||
| 
 | ||||
| export default registerAsyncHelper('setBreakpoint', function(app, breakpoint) { | ||||
| export default Ember.Test.registerAsyncHelper('setBreakpoint', function(app, breakpoint) { | ||||
|   // this should use getOwner once that's supported
 | ||||
|   const mediaService = app.__deprecatedInstance__.lookup('service:media'); | ||||
|   mediaService._forceSetBreakpoint(breakpoint); | ||||
|  |  | |||
|  | @ -1,13 +1,12 @@ | |||
| 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 = merge({}, config.APP); | ||||
|   attributes = merge(attributes, attrs); // use defaults, but you can override;
 | ||||
|   let attributes = Ember.merge({}, config.APP); | ||||
|   attributes = Ember.merge(attributes, attrs); // use defaults, but you can override;
 | ||||
| 
 | ||||
|   return run(() => { | ||||
|   return Ember.run(() => { | ||||
|     let application = Application.create(attributes); | ||||
|     application.setupForTesting(); | ||||
|     application.injectTestHelpers(); | ||||
|  |  | |||
							
								
								
									
										25
									
								
								tests/integration/components/ccdb-pagination-test.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								tests/integration/components/ccdb-pagination-test.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | |||
| 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'); | ||||
| }); | ||||
							
								
								
									
										25
									
								
								tests/integration/components/filter-collections-test.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								tests/integration/components/filter-collections-test.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | |||
| 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'); | ||||
| }); | ||||
|  | @ -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($, 'ajax'); | ||||
|   const stub = sinon.stub(Ember.$, '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($.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}})); | ||||
|   $.ajax.restore(); | ||||
|   assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {bar: 'baz'}})); | ||||
|   Ember.$.ajax.restore(); | ||||
|   return promise; | ||||
| }); | ||||
| 
 | ||||
| test('authenticate should craft a nice payload', function(assert) { | ||||
|   assert.expect(2); | ||||
|   const stub = sinon.stub($, 'ajax'); | ||||
|   const stub = sinon.stub(Ember.$, '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($.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}})); | ||||
|   $.ajax.restore(); | ||||
|   assert.ok(Ember.$.ajax.calledWithMatch({url: 'foo', data: {username: 'myusername', password: 'mypassword'}})); | ||||
|   Ember.$.ajax.restore(); | ||||
|   return promise; | ||||
| }); | ||||
|  |  | |||
|  | @ -1,12 +0,0 @@ | |||
| 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); | ||||
| }); | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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); | ||||
| }); | ||||
|  | @ -1,12 +0,0 @@ | |||
| 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); | ||||
| }); | ||||
							
								
								
									
										10
									
								
								tests/unit/routes/collections-test.js
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								tests/unit/routes/collections-test.js
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | |||
| 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
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue