Rough in species

This commit is contained in:
Matthew Dillon 2015-06-02 14:27:23 -08:00
parent dfc62cd1ac
commit afcf24a8d8
22 changed files with 399 additions and 2 deletions

17
app/abilities/species.js Normal file
View file

@ -0,0 +1,17 @@
import { Ability } from 'ember-can';
export default Ability.extend({
// Only admins and writers can create a new species
canAdd: function() {
let role = this.get('session.currentUser.role');
return (role === 'W') || (role === 'A');
}.property('session.currentUser.role'),
// Only admins and the person who created can edit
canEdit: function() {
let role = this.get('session.currentUser.role');
let id = this.get('session.currentUser.id');
let author = this.get('model.createdBy');
return (role === 'W' && (+id === author)) || (role === 'A');
}.property('session.currentUser.role', 'session.currentUser.id', 'model.createdBy')
});

View file

@ -0,0 +1,20 @@
import Ember from 'ember';
export default Ember.Component.extend({
classNames: ['grid-1'],
isEditing: false,
isNew: false,
actions: {
editSpecies: function() {
this.get('species').get('errors').clear();
if (this.get('isNew')) {
this.get('species').destroyRecord().then(this.sendAction());
}
this.toggleProperty('isEditing');
this.get('species').rollback();
},
saveSpecies: function() {
this.get('species').save().then(this.toggleProperty('isEditing'));
}
}
});

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

@ -0,0 +1,16 @@
import DS from 'ember-data';
export default DS.Model.extend({
speciesName: DS.attr('string'),
typeSpecies: DS.attr('boolean'),
etymology: DS.attr('string'),
genusName: DS.attr('string'),
strains: DS.hasMany('strain'),
totalStrains: DS.attr('number'),
createdAt: DS.attr('date'),
updatedAt: DS.attr('date'),
deletedAt: DS.attr('date'),
createdBy: DS.attr('number'),
updatedBy: DS.attr('number'),
deletedBy: DS.attr('number'),
});

View file

@ -8,6 +8,10 @@ var Router = Ember.Router.extend({
Router.map(function() { Router.map(function() {
this.route('login'); this.route('login');
this.route('about'); this.route('about');
this.resource('species', function() {
this.route('show', { path: ':species_id' });
this.route('new');
});
this.resource('strains', function() { this.resource('strains', function() {
this.route('new'); this.route('new');
this.route('show', { path: ':strain_id' }, function() { this.route('show', { path: ':strain_id' }, function() {

4
app/routes/species.js Normal file
View file

@ -0,0 +1,4 @@
import Ember from 'ember';
import AuthenticatedRouteMixin from 'simple-auth/mixins/authenticated-route-mixin';
export default Ember.Route.extend(AuthenticatedRouteMixin);

View file

@ -0,0 +1,7 @@
import Ember from 'ember';
export default Ember.Route.extend({
model: function() {
return this.store.findAll('species');
}
});

12
app/routes/species/new.js Normal file
View file

@ -0,0 +1,12 @@
import Ember from 'ember';
export default Ember.Route.extend({
model: function() {
return this.store.createRecord('species');
},
actions: {
cancelSpecies: function() {
this.transitionTo('species.index');
}
}
});

View file

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

View file

@ -4,6 +4,9 @@
{{/link-to}} {{/link-to}}
{{#if session.isAuthenticated}} {{#if session.isAuthenticated}}
<ul> <ul>
{{#link-to 'species' tagName='li' href=false}}
{{#link-to 'species'}}Species{{/link-to}}
{{/link-to}}
{{#link-to 'strains' tagName='li' href=false}} {{#link-to 'strains' tagName='li' href=false}}
{{#link-to 'strains'}}Strains{{/link-to}} {{#link-to 'strains'}}Strains{{/link-to}}
{{/link-to}} {{/link-to}}

View file

@ -0,0 +1,76 @@
<div class="span-1">
<fieldset {{bind-attr class=":flakes-information-box isEditing"}}>
<legend>
Species
{{#if isEditing}}
{{input value=species.speciesName}}
{{else}}
{{species.speciesName}}
{{/if}}
{{display-errors a=species.errors.speciesName}}
</legend>
{{! ROW 1 }}
<div class="grid-4">
<dl class="span-4">
<dt>Type Species?</dt>
<dd>
{{#if isEditing}}
{{input type="checkbox" checked=species.typeSpecies}}
{{/if}}
{{if species.typeSpecies 'Yes' 'No'}}
{{display-errors a=species.errors.typeSpecies}}
</dd>
</dl>
</div>
{{! ROW 2 }}
<div class="grid-4">
<dl class="span-4">
<dt>Etymology</dt>
<dd>
{{#if isEditing}}
{{textarea value=species.etymology cols="70" rows="3"}}
{{else}}
{{species.etymology}}
{{/if}}
{{display-errors a=species.errors.etymology}}
</dd>
</dl>
</div>
{{! ROW 3 }}
<div class="grid-4">
<dl class="span-1">
<dt>Record Created</dt>
<dd>{{null-time species.createdAt 'LL'}}</dd>
</dl>
<dl class="span-1">
<dt>Record Updated</dt>
<dd>{{null-time species.updatedAt 'LL'}}</dd>
</dl>
<dl class="span-1">
<dt>Record Deleted</dt>
<dd>{{null-time species.deletedAt 'LL'}}</dd>
</dl>
<dl class="span-1"></dl>
</div>
{{! ROW 4 }}
{{#if (can "edit species" strain)}}
<div class="grid-4">
<div class="span-1">
{{! Does nothing ATM }}
<a {{bind-attr class=":smaller isEditing:button-red:button-gray"}} {{action 'editSpecies'}}>
{{#if isEditing}}Cancel{{else}}Edit{{/if}}
</a>
{{#if isEditing}}
<a class="button-green smaller" {{action 'saveSpecies'}}>
Save
</a>
{{/if}}
</div>
</div>
{{/if}}
</fieldset>
</div>

View file

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

View file

@ -0,0 +1,30 @@
<h2>{{genus-name}} Species</h2>
<h3>Total species: {{controller.length}}</h3>
{{#if (can "add species")}}
{{! Does nothing ATM }}
{{#link-to 'species.new' class="button-gray smaller"}}
Add Species
{{/link-to}}
{{/if}}
<table class="flakes-table">
<thead>
<tr>
<th {{action "setSortBy" "speciesName"}}>Name</th>
<th {{action "setSortBy" "totalStrains"}}>Strains</th>
</tr>
</thead>
<tbody>
{{#each species in controller}}
<tr>
<td>
{{#link-to 'species.show' species}}
{{species.speciesName}}
{{/link-to}}
</td>
<td>{{species.totalStrains}}</td>
</tr>
{{/each}}
</tbody>
</table>

View file

@ -0,0 +1 @@
{{species/species-details species=model isEditing=true isNew=true action="cancelSpecies"}}

View file

@ -0,0 +1 @@
{{species/species-details species=model}}

View file

@ -21,6 +21,7 @@
"devDependencies": { "devDependencies": {
"body-parser": "^1.12.2", "body-parser": "^1.12.2",
"broccoli-asset-rev": "^2.0.2", "broccoli-asset-rev": "^2.0.2",
"connect-restreamer": "^1.0.2",
"ember-can": "^0.4.0", "ember-can": "^0.4.0",
"ember-cli": "0.2.3", "ember-cli": "0.2.3",
"ember-cli-app-version": "0.3.3", "ember-cli-app-version": "0.3.3",
@ -37,9 +38,9 @@
"ember-cli-uglify": "1.0.1", "ember-cli-uglify": "1.0.1",
"ember-data": "1.0.0-beta.16.1", "ember-data": "1.0.0-beta.16.1",
"ember-export-application-global": "^1.0.2", "ember-export-application-global": "^1.0.2",
"express": "^4.12.3", "express": "^4.12.4",
"glob": "^4.5.3", "glob": "^4.5.3",
"jsonwebtoken": "^5.0.0", "jsonwebtoken": "^5.0.0",
"morgan": "^1.5.2" "morgan": "^1.5.3"
} }
} }

108
server/mocks/species.js Normal file
View file

@ -0,0 +1,108 @@
module.exports = function(app) {
var express = require('express');
var speciesRouter = express.Router();
var SPECIES = [
{
id: 1,
genusName: "Hymenobacter",
speciesName: "One",
typeSpecies: true,
etymology: "Test Etymology",
strains: [1],
totalStrains: 1,
createdAt: "0001-01-01T00:00:00Z",
updatedAt: "0001-01-01T00:00:00Z",
deletedAt: null,
createdBy: 1,
updatedBy: 1,
deletedBy: null,
},
{
id: 2,
genusName: "Hymenobacter",
speciesName: "Two",
typeSpecies: true,
etymology: "Test Etymology",
strains: [2],
totalStrains: 1,
createdAt: "0001-01-01T00:00:00Z",
updatedAt: "0001-01-01T00:00:00Z",
deletedAt: null,
createdBy: 1,
updatedBy: 1,
deletedBy: null,
},
{
id: 3,
genusName: "Hymenobacter",
speciesName: "Three",
typeSpecies: true,
etymology: "Test Etymology",
strains: [3],
totalStrains: 1,
createdAt: "0001-01-01T00:00:00Z",
updatedAt: "0001-01-01T00:00:00Z",
deletedAt: null,
createdBy: 1,
updatedBy: 1,
deletedBy: null,
},
{
id: 4,
genusName: "Hymenobacter",
speciesName: "Four",
typeSpecies: true,
etymology: "Test Etymology",
strains: [4],
totalStrains: 1,
createdAt: "0001-01-01T00:00:00Z",
updatedAt: "0001-01-01T00:00:00Z",
deletedAt: null,
createdBy: 1,
updatedBy: 1,
deletedBy: null,
}
];
speciesRouter.get('/', function(req, res) {
var species;
if (req.query.ids) {
species = SPECIES.filter(function(s) {
return req.query.ids.indexOf(s.id.toString()) > -1;
});
} else {
species = SPECIES;
}
res.send({
'species': species
});
});
speciesRouter.post('/', function(req, res) {
res.status(201).end();
});
speciesRouter.get('/:id', function(req, res) {
var species = SPECIES.filter(function(s) {
return s.id == req.params.id;
});
res.send({
'species': species[0]
});
});
speciesRouter.put('/:id', function(req, res) {
res.send({
'species': {
id: req.params.id
}
});
});
speciesRouter.delete('/:id', function(req, res) {
res.status(204).end();
});
app.use('/api/hymenobacter/species', speciesRouter);
};

View file

@ -0,0 +1,21 @@
import {
moduleForComponent,
test
} from 'ember-qunit';
moduleForComponent('species/species-details', {
// Specify the other units that are required for this test
// needs: ['component:foo', 'helper:bar']
});
test('it renders', function(assert) {
assert.expect(2);
// Creates the component instance
var component = this.subject();
assert.equal(component._state, 'preRender');
// Renders the component to the page
this.render();
assert.equal(component._state, 'inDOM');
});

View file

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

View file

@ -0,0 +1,14 @@
import {
moduleFor,
test
} from 'ember-qunit';
moduleFor('route:species', {
// Specify the other units that are required for this test.
// needs: ['controller:foo']
});
test('it exists', function(assert) {
var route = this.subject();
assert.ok(route);
});

View file

@ -0,0 +1,14 @@
import {
moduleFor,
test
} from 'ember-qunit';
moduleFor('route:species/index', {
// Specify the other units that are required for this test.
// needs: ['controller:foo']
});
test('it exists', function(assert) {
var route = this.subject();
assert.ok(route);
});

View file

@ -0,0 +1,14 @@
import {
moduleFor,
test
} from 'ember-qunit';
moduleFor('route:species/new', {
// Specify the other units that are required for this test.
// needs: ['controller:foo']
});
test('it exists', function(assert) {
var route = this.subject();
assert.ok(route);
});

View file

@ -0,0 +1,14 @@
import {
moduleFor,
test
} from 'ember-qunit';
moduleFor('route:species/show', {
// Specify the other units that are required for this test.
// needs: ['controller:foo']
});
test('it exists', function(assert) {
var route = this.subject();
assert.ok(route);
});