├── .babelrc ├── .eslintrc ├── .gitignore ├── .travis.yml ├── LICENSE ├── README.md ├── actions ├── __test__ │ ├── addVaccination.js │ ├── addVaccinationFailure.js │ ├── addVaccinationSuccess.js │ ├── fetchVaccinations.js │ ├── fetchVaccinationsFailure.js │ ├── fetchVaccinationsSuccess.js │ ├── pickVaccine.js │ ├── switchToChooseVaccineRoute.js │ ├── switchToDetailRoute.js │ └── switchToListRoute.js ├── addVaccination.js ├── addVaccinationFailure.js ├── addVaccinationSuccess.js ├── fetchVaccinations.js ├── fetchVaccinationsFailure.js ├── fetchVaccinationsSuccess.js ├── pickVaccine.js ├── switchToChooseVaccineRoute.js ├── switchToDetailRoute.js └── switchToListRoute.js ├── components ├── Button │ ├── index.js │ └── styles.js ├── Card │ ├── index.js │ └── styles.js ├── ChooseDate │ ├── DateInput │ │ ├── index.android.js │ │ ├── index.ios.js │ │ └── styles.js │ ├── index.js │ └── styles.js ├── ChooseVaccine │ ├── index.js │ └── styles.js ├── Detail │ ├── index.js │ └── styles.js ├── List │ ├── index.js │ └── styles.js └── Logo │ ├── icon.js │ └── index.js ├── constants ├── actions.js ├── storage.js └── vaccines.js ├── containers ├── ChooseDate.js ├── ChooseVaccine.js ├── Detail.js └── List.js ├── exp.json ├── main.js ├── package.json ├── reducers ├── __test__ │ ├── addForm.js │ ├── currentVaccination.js │ └── vaccinations.js ├── activeRoute.js ├── addForm.js ├── currentVaccination.js ├── index.js └── vaccinations.js ├── sagas ├── __test__ │ ├── fetchVaccinations.js │ ├── index.js │ ├── saveVaccinations.js │ └── startup.js ├── fetchVaccinations.js ├── index.js ├── saveVaccinations.js └── startup.js ├── selectors ├── __test__ │ ├── addForm.js │ ├── currentVaccination.js │ ├── vaccinations.js │ └── vaccines.js ├── addForm.js ├── currentVaccination.js ├── vaccinations.js └── vaccines.js ├── store └── configureStore.js └── testHelper.js /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | "react-native-stage-0/decorator-support" 4 | ], 5 | "env": { 6 | "development": { 7 | "plugins": [ 8 | "transform-react-jsx-source" 9 | ] 10 | } 11 | } 12 | } 13 | -------------------------------------------------------------------------------- /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "parser": "babel-eslint", 3 | "extends": "airbnb", 4 | "env": { 5 | "browser": true, 6 | "mocha": true, 7 | "node": true 8 | }, 9 | "rules": { 10 | "comma-dangle": [2, "always-multiline"], 11 | "react/prop-types": 0, 12 | "react/jsx-no-bind": 0, 13 | "react/jsx-filename-extension": 0, 14 | "new-cap": 0, 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # OSX 2 | # 3 | .DS_Store 4 | 5 | # node.js 6 | # 7 | node_modules/ 8 | npm-debug.log 9 | .exponent/* 10 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: node_js 2 | node_js: 3 | - "5" 4 | sudo: false 5 | script: 6 | - npm run lint 7 | - npm test 8 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 Nik Graf 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Carte Jaune 2 | 3 | A Redux/ExponentJS (React Native) app to keep track of your vaccinations. 4 | 5 | [![Build Status](https://travis-ci.org/nikgraf/CarteJaune.svg?branch=master)](https://travis-ci.org/nikgraf/CarteJaune) 6 | 7 | ## Install 8 | 9 | `npm Install` 10 | 11 | ## Run 12 | 13 | see ExponentJS tutorial on https://exponentjs.com/ 14 | 15 | screen shot 2016-01-26 at 08 35 38 16 | 17 | ## Video Demo 18 | 19 | [![Carte Jaune](http://img.youtube.com/vi/XaiVZ4RkZ6M/maxresdefault.jpg)](http://www.youtube.com/watch?v=XaiVZ4RkZ6M) 20 | 21 | ## Tests 22 | 23 | All of the business logics is well tested: 24 | - [Action creator tests](https://github.com/nikgraf/CarteJaune/tree/master/actions/__test__) 25 | - [Reducer tests](https://github.com/nikgraf/CarteJaune/tree/master/reducers/__test__) 26 | - [Selector tests](https://github.com/nikgraf/CarteJaune/tree/master/selectors/__test__) 27 | - [Saga tests](https://github.com/nikgraf/CarteJaune/tree/master/sagas/__test__) 28 | 29 | ## Support 30 | 31 | I built this application for the ExponentJS contest to win a React Conference Ticket. If you want to support me please download the ExponentJs Application and vote in the contest for my application. 32 | 33 | - https://itunes.com/apps/exponent 34 | - https://play.google.com/store/apps/details?id=host.exp.exponent 35 | -------------------------------------------------------------------------------- /actions/__test__/addVaccination.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import addVaccination from '../addVaccination'; 3 | import { ADD_VACCINATION } from '../../constants/actions'; 4 | 5 | describe('addVaccination', () => { 6 | it('returns ADD_VACCINATION action', () => { 7 | const expected = { 8 | type: ADD_VACCINATION, 9 | vaccineId: 2, 10 | vaccinationDate: 3, 11 | }; 12 | 13 | expect(addVaccination(2, 3)).to.deep.equal(expected); 14 | }); 15 | }); 16 | -------------------------------------------------------------------------------- /actions/__test__/addVaccinationFailure.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import addVaccinationFailure from '../addVaccinationFailure'; 3 | import { ADD_VACCINATION_FAILURE } from '../../constants/actions'; 4 | 5 | describe('addVaccinationFailure', () => { 6 | it('returns ADD_VACCINATION_FAILURE action', () => { 7 | const expected = { 8 | type: ADD_VACCINATION_FAILURE, 9 | }; 10 | 11 | expect(addVaccinationFailure()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/__test__/addVaccinationSuccess.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import addVaccinationSuccess from '../addVaccinationSuccess'; 3 | import { ADD_VACCINATION_SUCCESS } from '../../constants/actions'; 4 | 5 | describe('addVaccinationSuccess', () => { 6 | it('returns ADD_VACCINATION_SUCCESS action', () => { 7 | const expected = { 8 | type: ADD_VACCINATION_SUCCESS, 9 | }; 10 | 11 | expect(addVaccinationSuccess()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/__test__/fetchVaccinations.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import fetchVaccinations from '../fetchVaccinations'; 3 | import { FETCH_VACCINATIONS } from '../../constants/actions'; 4 | 5 | describe('fetchVaccinations', () => { 6 | it('returns FETCH_VACCINATIONS action', () => { 7 | const expected = { 8 | type: FETCH_VACCINATIONS, 9 | }; 10 | 11 | expect(fetchVaccinations()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/__test__/fetchVaccinationsFailure.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import fetchVaccinationsFailure from '../fetchVaccinationsFailure'; 3 | import { FETCH_VACCINATIONS_FAILURE } from '../../constants/actions'; 4 | 5 | describe('fetchVaccinationsFailure', () => { 6 | it('returns FETCH_VACCINATIONS_FAILURE action', () => { 7 | const expected = { 8 | type: FETCH_VACCINATIONS_FAILURE, 9 | }; 10 | 11 | expect(fetchVaccinationsFailure()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/__test__/fetchVaccinationsSuccess.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import fetchVaccinationsSuccess from '../fetchVaccinationsSuccess'; 3 | import { FETCH_VACCINATIONS_SUCCESS } from '../../constants/actions'; 4 | 5 | describe('addVaccination', () => { 6 | it('returns FETCH_VACCINATIONS_SUCCESS action', () => { 7 | const expected = { 8 | type: FETCH_VACCINATIONS_SUCCESS, 9 | vaccinations: [1, 2], 10 | }; 11 | 12 | expect(fetchVaccinationsSuccess([1, 2])).to.deep.equal(expected); 13 | }); 14 | }); 15 | -------------------------------------------------------------------------------- /actions/__test__/pickVaccine.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import pickVaccine from '../pickVaccine'; 3 | import { PICK_VACCINE } from '../../constants/actions'; 4 | 5 | describe('pickVaccine', () => { 6 | it('returns PICK_VACCINE action', () => { 7 | const expected = { 8 | type: PICK_VACCINE, 9 | vaccineId: 2, 10 | }; 11 | 12 | expect(pickVaccine(2)).to.deep.equal(expected); 13 | }); 14 | }); 15 | -------------------------------------------------------------------------------- /actions/__test__/switchToChooseVaccineRoute.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import switchToChooseVaccineRoute from '../switchToChooseVaccineRoute'; 3 | import { SWITCH_TO_CHOOSE_VACCINE_ROUTE } from '../../constants/actions'; 4 | 5 | describe('switchToChooseVaccineRoute', () => { 6 | it('returns SWITCH_TO_CHOOSE_VACCINE_ROUTE action', () => { 7 | const expected = { 8 | type: SWITCH_TO_CHOOSE_VACCINE_ROUTE, 9 | }; 10 | 11 | expect(switchToChooseVaccineRoute()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/__test__/switchToDetailRoute.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import switchToDetailRoute from '../switchToDetailRoute'; 3 | import { SWITCH_TO_DETAIL_ROUTE } from '../../constants/actions'; 4 | 5 | describe('switchToDetailRoute', () => { 6 | it('returns SWITCH_TO_DETAIL_ROUTE action', () => { 7 | const expected = { 8 | type: SWITCH_TO_DETAIL_ROUTE, 9 | id: 'abc', 10 | }; 11 | 12 | expect(switchToDetailRoute('abc')).to.deep.equal(expected); 13 | }); 14 | }); 15 | -------------------------------------------------------------------------------- /actions/__test__/switchToListRoute.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import switchToListRoute from '../switchToListRoute'; 3 | import { SWITCH_TO_LIST_ROUTE } from '../../constants/actions'; 4 | 5 | describe('switchToListRoute', () => { 6 | it('returns SWITCH_TO_LIST_ROUTE action', () => { 7 | const expected = { 8 | type: SWITCH_TO_LIST_ROUTE, 9 | }; 10 | 11 | expect(switchToListRoute()).to.deep.equal(expected); 12 | }); 13 | }); 14 | -------------------------------------------------------------------------------- /actions/addVaccination.js: -------------------------------------------------------------------------------- 1 | import { ADD_VACCINATION } from '../constants/actions'; 2 | 3 | export default (vaccineId, vaccinationDate) => ({ 4 | type: ADD_VACCINATION, 5 | vaccineId, 6 | vaccinationDate, 7 | }); 8 | -------------------------------------------------------------------------------- /actions/addVaccinationFailure.js: -------------------------------------------------------------------------------- 1 | import { ADD_VACCINATION_FAILURE } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: ADD_VACCINATION_FAILURE, 5 | }); 6 | -------------------------------------------------------------------------------- /actions/addVaccinationSuccess.js: -------------------------------------------------------------------------------- 1 | import { ADD_VACCINATION_SUCCESS } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: ADD_VACCINATION_SUCCESS, 5 | }); 6 | -------------------------------------------------------------------------------- /actions/fetchVaccinations.js: -------------------------------------------------------------------------------- 1 | import { FETCH_VACCINATIONS } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: FETCH_VACCINATIONS, 5 | }); 6 | -------------------------------------------------------------------------------- /actions/fetchVaccinationsFailure.js: -------------------------------------------------------------------------------- 1 | import { FETCH_VACCINATIONS_FAILURE } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: FETCH_VACCINATIONS_FAILURE, 5 | }); 6 | -------------------------------------------------------------------------------- /actions/fetchVaccinationsSuccess.js: -------------------------------------------------------------------------------- 1 | import { FETCH_VACCINATIONS_SUCCESS } from '../constants/actions'; 2 | 3 | export default (vaccinations) => ({ 4 | type: FETCH_VACCINATIONS_SUCCESS, 5 | vaccinations, 6 | }); 7 | -------------------------------------------------------------------------------- /actions/pickVaccine.js: -------------------------------------------------------------------------------- 1 | import { PICK_VACCINE } from '../constants/actions'; 2 | 3 | export default (vaccineId) => ({ 4 | type: PICK_VACCINE, 5 | vaccineId, 6 | }); 7 | -------------------------------------------------------------------------------- /actions/switchToChooseVaccineRoute.js: -------------------------------------------------------------------------------- 1 | import { SWITCH_TO_CHOOSE_VACCINE_ROUTE } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: SWITCH_TO_CHOOSE_VACCINE_ROUTE, 5 | }); 6 | -------------------------------------------------------------------------------- /actions/switchToDetailRoute.js: -------------------------------------------------------------------------------- 1 | import { SWITCH_TO_DETAIL_ROUTE } from '../constants/actions'; 2 | 3 | export default (id) => ({ 4 | type: SWITCH_TO_DETAIL_ROUTE, 5 | id, 6 | }); 7 | -------------------------------------------------------------------------------- /actions/switchToListRoute.js: -------------------------------------------------------------------------------- 1 | import { SWITCH_TO_LIST_ROUTE } from '../constants/actions'; 2 | 3 | export default () => ({ 4 | type: SWITCH_TO_LIST_ROUTE, 5 | }); 6 | -------------------------------------------------------------------------------- /components/Button/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | Text, 4 | TouchableHighlight, 5 | } from 'react-native'; 6 | import styles from './styles'; 7 | 8 | export default class Button extends Component { 9 | state = { 10 | active: false, 11 | }; 12 | 13 | onHighlight() { 14 | this.setState({ active: true }); 15 | } 16 | 17 | onUnhighlight() { 18 | this.setState({ active: false }); 19 | } 20 | 21 | onPress() { 22 | if (!this.props.disabled) { 23 | this.props.onPress(); 24 | } 25 | } 26 | 27 | render() { 28 | const colorStyle = { 29 | color: this.props.disabled ? '#ccc' : '#000', 30 | }; 31 | const underlayColor = this.props.disabled ? '#E0F4FF' : '#B8CCD8'; 32 | return ( 33 | 40 | {this.props.children} 41 | 42 | ); 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /components/Button/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | button: { 5 | borderRadius: 5, 6 | alignSelf: 'stretch', 7 | justifyContent: 'center', 8 | alignItems: 'center', 9 | overflow: 'hidden', 10 | backgroundColor: '#E0F4FF', 11 | borderBottomWidth: 1, 12 | borderBottomColor: '#B8CCD8', 13 | }, 14 | buttonText: { 15 | fontSize: 18, 16 | margin: 5, 17 | textAlign: 'center', 18 | }, 19 | }); 20 | -------------------------------------------------------------------------------- /components/Card/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | View, 4 | } from 'react-native'; 5 | import styles from './styles'; 6 | 7 | // eslint-disable-next-line react/prefer-stateless-function 8 | export default class Card extends Component { 9 | render() { 10 | return ( 11 | 14 | {this.props.children} 15 | 16 | ); 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /components/Card/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | container: { 5 | backgroundColor: '#FFF', 6 | marginLeft: 10, 7 | marginRight: 10, 8 | paddingLeft: 10, 9 | paddingRight: 10, 10 | borderBottomColor: '#f3f3f3', 11 | borderBottomWidth: 1, 12 | marginBottom: 10, 13 | borderRadius: 3, 14 | }, 15 | }); 16 | -------------------------------------------------------------------------------- /components/ChooseDate/DateInput/index.android.js: -------------------------------------------------------------------------------- 1 | import React, { 2 | Component, 3 | } from 'react'; 4 | import { 5 | Text, 6 | TextInput, 7 | View, 8 | } from 'react-native'; 9 | import moment from 'moment'; 10 | import dateformat from 'dateformat'; 11 | import styles from './styles'; 12 | 13 | export default class DatePickerAndroid extends Component { 14 | 15 | constructor(props) { 16 | super(); 17 | this.state = { 18 | isValid: true, 19 | date: dateformat(props.date, 'yyyy/mm/dd'), 20 | }; 21 | } 22 | 23 | updateDate(text) { 24 | const isValid = moment(text, 'YYYY/MM/DD', true).isValid(); 25 | if (!isValid) { 26 | this.setState({ 27 | isValid: false, 28 | date: text, 29 | }); 30 | this.props.onInvalidDateChange(); 31 | return; 32 | } 33 | 34 | this.setState({ 35 | date: text, 36 | isValid: true, 37 | }); 38 | this.props.onDateChange(new Date(text)); 39 | } 40 | 41 | render() { 42 | return ( 43 | 44 | 50 | 51 | {!this.state.isValid && 'Please enter a valid Date: YYYY/MM/DD'} 52 | 53 | 54 | ); 55 | } 56 | } 57 | -------------------------------------------------------------------------------- /components/ChooseDate/DateInput/index.ios.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | DatePickerIOS, 4 | } from 'react-native'; 5 | 6 | // eslint-disable-next-line react/prefer-stateless-function 7 | export default class CustomDatePickerIOS extends Component { 8 | render() { 9 | return ( 10 | 15 | ); 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /components/ChooseDate/DateInput/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | textInput: { 5 | height: 50, 6 | borderColor: '#ddd', 7 | borderWidth: 1, 8 | marginLeft: 10, 9 | marginRight: 10, 10 | marginTop: 40, 11 | backgroundColor: '#fff', 12 | borderRadius: 4, 13 | fontSize: 20, 14 | textAlign: 'center', 15 | }, 16 | errorMessage: { 17 | color: '#DD3333', 18 | marginTop: 10, 19 | height: 20, 20 | textAlign: 'center', 21 | }, 22 | }); 23 | -------------------------------------------------------------------------------- /components/ChooseDate/index.js: -------------------------------------------------------------------------------- 1 | import React, { 2 | Component, 3 | } from 'react'; 4 | import { 5 | Text, 6 | View, 7 | } from 'react-native'; 8 | import styles from './styles'; 9 | import Button from '../Button'; 10 | // eslint-disable-next-line import/no-unresolved 11 | import DateInput from './DateInput'; 12 | 13 | export default class ChooseDate extends Component { 14 | state = { 15 | vaccinationDate: new Date(), 16 | isValidDate: true, 17 | }; 18 | 19 | onUpdateDate(date) { 20 | this.setState({ 21 | vaccinationDate: date, 22 | isValidDate: true, 23 | }); 24 | } 25 | 26 | onInvalidDateChange() { 27 | this.setState({ 28 | isValidDate: false, 29 | }); 30 | } 31 | 32 | onPress() { 33 | this.props.addVaccination(this.props.addForm.get('vaccineId'), this.state.vaccinationDate); 34 | } 35 | 36 | render() { 37 | return ( 38 | 39 | 40 | Pick the Vaccination Date 41 | 42 | 43 | 48 | 55 | 56 | ); 57 | } 58 | } 59 | -------------------------------------------------------------------------------- /components/ChooseDate/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | container: { 5 | backgroundColor: '#FFFBE6', 6 | position: 'absolute', 7 | top: 0, 8 | bottom: 0, 9 | left: 0, 10 | right: 0, 11 | }, 12 | header: { 13 | fontSize: 24, 14 | paddingBottom: 20, 15 | paddingTop: 40, 16 | textAlign: 'center', 17 | }, 18 | addButton: { 19 | marginLeft: 10, 20 | marginRight: 10, 21 | marginTop: 30, 22 | paddingTop: 30, 23 | paddingBottom: 30, 24 | }, 25 | }); 26 | -------------------------------------------------------------------------------- /components/ChooseVaccine/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | ListView, 4 | Text, 5 | TouchableOpacity, 6 | View, 7 | } from 'react-native'; 8 | import styles from './styles'; 9 | 10 | export default class ChooseVaccine extends Component { 11 | 12 | state = { 13 | dataSource: new ListView.DataSource({ 14 | rowHasChanged: (row1, row2) => row1 !== row2, 15 | }), 16 | }; 17 | 18 | componentWillMount() { 19 | this.setState({ 20 | dataSource: this.state.dataSource.cloneWithRows(this.props.vaccines.toJS()), 21 | }); 22 | } 23 | 24 | renderItem(vaccine) { 25 | return ( 26 | this.props.pickVaccine(vaccine.id)}> 27 | 28 | { vaccine.name } 29 | { vaccine.diseases.join(', ')} 30 | 31 | 32 | ); 33 | } 34 | 35 | render() { 36 | return ( 37 | 38 | Pick a Vaccine 39 | 45 | 46 | ); 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /components/ChooseVaccine/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | container: { 5 | backgroundColor: '#FFFBE6', 6 | position: 'absolute', 7 | top: 0, 8 | bottom: 0, 9 | left: 0, 10 | right: 0, 11 | }, 12 | header: { 13 | fontSize: 24, 14 | paddingBottom: 20, 15 | paddingTop: 40, 16 | textAlign: 'center', 17 | }, 18 | row: { 19 | flex: 1, 20 | paddingTop: 20, 21 | paddingBottom: 20, 22 | paddingLeft: 20, 23 | paddingRight: 20, 24 | borderBottomWidth: 1, 25 | borderBottomColor: '#f6f6f6', 26 | }, 27 | listView: { 28 | backgroundColor: '#FFF', 29 | marginLeft: 10, 30 | marginRight: 10, 31 | marginBottom: 10, 32 | borderBottomColor: '#f3f3f3', 33 | borderBottomWidth: 1, 34 | borderRadius: 3, 35 | }, 36 | footer: { 37 | alignItems: 'center', 38 | paddingTop: 10, 39 | }, 40 | disease: { 41 | color: '#999', 42 | fontSize: 10, 43 | }, 44 | }); 45 | -------------------------------------------------------------------------------- /components/Detail/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | Text, 4 | View, 5 | } from 'react-native'; 6 | import dateformat from 'dateformat'; 7 | import Button from '../Button'; 8 | import Card from '../Card'; 9 | import styles from './styles'; 10 | 11 | export default class Detail extends Component { 12 | 13 | render() { 14 | const { currentVaccination } = this.props; 15 | if (!currentVaccination) { return null; } 16 | 17 | return ( 18 | 19 | 25 | 26 | {currentVaccination.getIn(['vaccine', 'name'])} 27 | 28 | { currentVaccination.getIn(['vaccine', 'diseases']).join(', ')} 29 | 30 | 31 | {dateformat(currentVaccination.get('date'), 'ddd, dS mmmm, yyyy')} 32 | 33 | 34 | 35 | ); 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /components/Detail/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | container: { 5 | backgroundColor: '#FFFBE6', 6 | position: 'absolute', 7 | top: 0, 8 | bottom: 0, 9 | left: 0, 10 | right: 0, 11 | paddingTop: 80, 12 | }, 13 | name: { 14 | fontSize: 20, 15 | textAlign: 'center', 16 | marginBottom: 50, 17 | marginTop: 50, 18 | }, 19 | vaccinationDate: { 20 | textAlign: 'center', 21 | marginBottom: 20, 22 | }, 23 | disease: { 24 | color: '#999', 25 | fontSize: 12, 26 | marginBottom: 30, 27 | textAlign: 'center', 28 | }, 29 | backButton: { 30 | paddingLeft: 30, 31 | paddingRight: 30, 32 | marginLeft: 10, 33 | marginTop: 20, 34 | marginBottom: 10, 35 | paddingTop: 10, 36 | paddingBottom: 10, 37 | alignSelf: 'flex-start', 38 | }, 39 | }); 40 | -------------------------------------------------------------------------------- /components/List/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | ListView, 4 | Text, 5 | TouchableOpacity, 6 | View, 7 | } from 'react-native'; 8 | import dateformat from 'dateformat'; 9 | import styles from './styles'; 10 | import Logo from '../Logo'; 11 | import Button from '../Button'; 12 | import Card from '../Card'; 13 | 14 | export default class List extends Component { 15 | 16 | state = { 17 | dataSource: new ListView.DataSource({ 18 | rowHasChanged: (row1, row2) => row1 !== row2, 19 | }), 20 | }; 21 | 22 | componentWillMount() { 23 | this.setState({ 24 | dataSource: this.state.dataSource.cloneWithRows(this.props.vaccinations.toJS()), 25 | }); 26 | } 27 | 28 | componentWillReceiveProps(nextProps) { 29 | this.setState({ 30 | dataSource: this.state.dataSource.cloneWithRows(nextProps.vaccinations.toJS()), 31 | }); 32 | } 33 | 34 | getDisease() { 35 | const diseases = this.props.vaccinations 36 | .toList() 37 | .map((vaccination) => vaccination.getIn(['vaccine', 'diseases'])) 38 | .flatten(true); 39 | return diseases 40 | .toSet() 41 | .toList() 42 | .sort() 43 | .join(', '); 44 | } 45 | 46 | // eslint-disable-next-line class-methods-use-this 47 | renderHeader() { 48 | return ( 49 | 50 | 51 | 52 | 53 | 54 | Carte Jaune 55 | 56 | 57 | Vaccines 58 | 59 | ); 60 | } 61 | 62 | renderDiseases() { 63 | return ( 64 | 65 | Covered Diseases 66 | 67 | {this.getDisease()} 68 | 69 | 70 | ); 71 | } 72 | 73 | renderFooter() { 74 | return ( 75 | 76 | {this.props.vaccinations.size ? this.renderDiseases() : null} 77 | 84 | Made with ♡ by Nik Graf 85 | 86 | Open Source Code: https://www.github.com/nikgraf/CarteJaune 87 | 88 | 89 | Logo made by Freepik is licensed under CC BY 3.0 90 | 91 | 92 | This App is designed for educational purposes only and is not intended 93 | to serve as medical advice. 94 | 95 | 96 | ); 97 | } 98 | 99 | renderItem(vaccination) { 100 | return ( 101 | this.props.switchToDetailRoute(vaccination.listId)}> 102 | 103 | {vaccination.vaccine.name} 104 | { vaccination.vaccine.diseases.join(', ')} 105 | 106 | {dateformat(vaccination.date, 'ddd, dS mmmm, yyyy')} 107 | 108 | 109 | 110 | ); 111 | } 112 | 113 | render() { 114 | return ( 115 | 123 | ); 124 | } 125 | } 126 | -------------------------------------------------------------------------------- /components/List/styles.js: -------------------------------------------------------------------------------- 1 | import { StyleSheet } from 'react-native'; 2 | 3 | export default StyleSheet.create({ 4 | add: { 5 | flex: 1, 6 | justifyContent: 'center', 7 | alignItems: 'center', 8 | paddingTop: 20, 9 | paddingBottom: 20, 10 | borderTopColor: '#ddd', 11 | borderTopWidth: 1, 12 | }, 13 | addText: { 14 | fontSize: 20, 15 | }, 16 | container: { 17 | flex: 1, 18 | justifyContent: 'center', 19 | alignItems: 'center', 20 | paddingBottom: 15, 21 | paddingTop: 15, 22 | }, 23 | header: { 24 | flex: 1, 25 | alignItems: 'center', 26 | flexDirection: 'row', 27 | justifyContent: 'center', 28 | marginTop: 20, 29 | }, 30 | headerText: { 31 | fontSize: 32, 32 | paddingTop: 20, 33 | paddingLeft: 10, 34 | }, 35 | logo: { 36 | width: 40, 37 | height: 40, 38 | }, 39 | listView: { 40 | paddingTop: 20, 41 | backgroundColor: '#FFFBE6', 42 | }, 43 | name: { 44 | fontSize: 20, 45 | textAlign: 'center', 46 | }, 47 | vaccinationDate: { 48 | textAlign: 'center', 49 | }, 50 | diseases: { 51 | paddingTop: 10, 52 | paddingBottom: 20, 53 | paddingLeft: 20, 54 | paddingRight: 20, 55 | }, 56 | diseasesText: { 57 | lineHeight: 24, 58 | }, 59 | label: { 60 | paddingTop: 20, 61 | textAlign: 'center', 62 | marginBottom: 10, 63 | color: '#421C00', 64 | }, 65 | addButton: { 66 | marginLeft: 20, 67 | marginRight: 20, 68 | marginTop: 50, 69 | paddingTop: 30, 70 | paddingBottom: 30, 71 | }, 72 | plus: { 73 | fontSize: 30, 74 | lineHeight: 30, 75 | marginRight: 10, 76 | }, 77 | disease: { 78 | color: '#999', 79 | fontSize: 10, 80 | marginBottom: 10, 81 | }, 82 | promo: { 83 | textAlign: 'center', 84 | fontSize: 13, 85 | marginTop: 70, 86 | marginBottom: 8, 87 | color: '#666', 88 | }, 89 | smallPromo: { 90 | textAlign: 'center', 91 | fontSize: 11, 92 | marginBottom: 8, 93 | color: '#666', 94 | }, 95 | disclaimer: { 96 | textAlign: 'center', 97 | fontSize: 9, 98 | marginBottom: 20, 99 | color: '#666', 100 | marginLeft: 100, 101 | marginRight: 100, 102 | }, 103 | }); 104 | -------------------------------------------------------------------------------- /components/Logo/icon.js: -------------------------------------------------------------------------------- 1 | /*eslint-disable */ 2 | 3 | const logo = ''; 4 | 5 | export default logo; 6 | -------------------------------------------------------------------------------- /components/Logo/index.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { 3 | Image, 4 | } from 'react-native'; 5 | import icon from './icon'; 6 | 7 | // eslint-disable-next-line react/prefer-stateless-function 8 | export default class Logo extends Component { 9 | render() { 10 | return ( 11 | 16 | ); 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /constants/actions.js: -------------------------------------------------------------------------------- 1 | export const ADD_VACCINATION = 'ADD_VACCINATION'; 2 | export const ADD_VACCINATION_SUCCESS = 'ADD_VACCINATION_SUCCESS'; 3 | export const ADD_VACCINATION_FAILURE = 'ADD_VACCINATION_FAILURE'; 4 | export const SWITCH_TO_CHOOSE_VACCINE_ROUTE = 'SWITCH_TO_CHOOSE_VACCINE_ROUTE'; 5 | export const SWITCH_TO_DETAIL_ROUTE = 'SWITCH_TO_DETAIL_ROUTE'; 6 | export const SWITCH_TO_LIST_ROUTE = 'SWITCH_TO_LIST_ROUTE'; 7 | export const PICK_VACCINE = 'PICK_VACCINE'; 8 | export const FETCH_VACCINATIONS = 'FETCH_VACCINATIONS'; 9 | export const FETCH_VACCINATIONS_SUCCESS = 'FETCH_VACCINATIONS_SUCCESS'; 10 | export const FETCH_VACCINATIONS_FAILURE = 'FETCH_VACCINATIONS_FAILURE'; 11 | -------------------------------------------------------------------------------- /constants/storage.js: -------------------------------------------------------------------------------- 1 | // eslint-disable-next-line import/prefer-default-export 2 | export const KEY = '@Vaccinations:me'; 3 | -------------------------------------------------------------------------------- /constants/vaccines.js: -------------------------------------------------------------------------------- 1 | import { fromJS, OrderedMap } from 'immutable'; 2 | 3 | const vaccines = fromJS([ 4 | { 5 | name: 'AVA (BioThrax)', 6 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 7 | diseases: ['Anthrax'], 8 | }, 9 | { 10 | name: 'VAR (Varivax)', 11 | id: 'f3e08a56-003c-4b46-9dea-216298401ca0', 12 | diseases: ['Varicella (Chickenpox)'], 13 | }, 14 | { 15 | name: 'MMRV (ProQuad)', 16 | id: '3373721d-3d14-490c-9fa9-69a223888322', 17 | diseases: [ 18 | 'Varicella (Chickenpox)', 19 | 'Measles', 20 | 'Mumps', 21 | 'Rubella (German Measles)', 22 | ], 23 | }, 24 | { 25 | name: 'HepA (Havrix, Vaqta)', 26 | id: 'a9144edf-13a2-4ce5-b6af-14eb38fd848c', 27 | diseases: ['Hepatitis A'], 28 | }, 29 | { 30 | name: 'HepA-HepB (Twinrix)', 31 | id: '6888fd1a-af4f-4f33-946d-40d4c473c9cc', 32 | diseases: ['Hepatitis A', 'Hepatitis B'], 33 | }, 34 | { 35 | name: 'HepB (Engerix-B, Recombivax HB)', 36 | id: 'ca079856-a561-4bc9-9bef-e62429ed3a38', 37 | diseases: ['Hepatitis B'], 38 | }, 39 | { 40 | name: 'Hib-HepB (Comvax)', 41 | id: '7305d769-0d1e-4bef-bd09-6998dc839825', 42 | diseases: ['Hepatitis B', 'Haemophilus influenzae type b (Hib)'], 43 | }, 44 | { 45 | name: 'Hib (ActHIB, PedvaxHIB, Hiberix)', 46 | id: 'd241f0c7-9920-4bc6-8f34-288a13e03f4d', 47 | diseases: ['Haemophilus influenzae type b (Hib)'], 48 | }, 49 | { 50 | name: 'HPV4 (Gardasil)', 51 | id: 'c2fef03c-db7f-483b-af70-50560712b189', 52 | diseases: ['Human Papillomavirus (HPV)'], 53 | }, 54 | { 55 | name: 'HPV2 (Cervarix)', 56 | id: '286f55e4-e727-4fc4-86b0-5a08ea712a77', 57 | diseases: ['Human Papillomavirus (HPV)'], 58 | }, 59 | { 60 | name: 'TIV (Afluria, Agriflu, FluLaval, Fluarix, Fluvirin, Fluzone, Fluzone High-Dose, Fluzone Intradermal)', // eslint-disable-line max-len 61 | id: '60e85a31-6a54-48e1-b0b7-deb28120675b', 62 | diseases: ['Seasonal Influenza (Flu)'], 63 | }, 64 | { 65 | name: 'LAIV (FluMist)', 66 | id: '9e67e321-9a7f-426f-ba9b-28885f93f9b9', 67 | diseases: ['Seasonal Influenza (Flu)'], 68 | }, 69 | { 70 | name: 'JE (Ixiaro)', 71 | id: '5ce00584-3350-442d-ac6c-7f19567eff8a', 72 | diseases: ['Japanese Encephalitis'], 73 | }, 74 | { 75 | name: 'MMR (M-M-R II)', 76 | id: 'd10b7bf0-d51e-4117-a6a4-08bdb5cb682a', 77 | diseases: ['Measles', 'Mumps', 'Rubella (German Measles)'], 78 | }, 79 | { 80 | name: 'MCV4 (Menactra)', 81 | id: '6295fe11-f0ce-4967-952c-f271416cc300', 82 | diseases: ['Meningococcal'], 83 | }, 84 | { 85 | name: 'MPSV4 (Menomune)', 86 | id: '65f6d6d0-6dd8-49c9-95da-ed9fa403ae96', 87 | diseases: ['Meningococcal'], 88 | }, 89 | { 90 | name: 'MODC (Menveo)', 91 | id: 'be10b480-7934-46be-a488-66540aac2881', 92 | diseases: ['Meningococcal'], 93 | }, 94 | { 95 | name: 'Tdap (Adacel, Boostrix)', 96 | id: '0c6c33fb-f4dc-44c6-8684-625099f6fa21', 97 | diseases: ['Pertussis (Whooping Cough)', 'Tetanus (Lockjaw)', 'Diphtheria'], 98 | }, 99 | { 100 | name: 'PCV13 (Prevnar13)', 101 | id: 'd8c5a723-21e2-49a6-a921-705da16563e1', 102 | diseases: ['Pneumococcal'], 103 | }, 104 | { 105 | name: 'PPSV23 (Pneumovax 23)', 106 | id: '4005de2f-8e6d-40ae-bb5f-068ac56885b8', 107 | diseases: ['Pneumococcal'], 108 | }, 109 | { 110 | name: 'Polio (Ipol)', 111 | id: '9c1582f2-8a7b-4bae-8ba5-656efe33fb29', 112 | diseases: ['Polio'], 113 | }, 114 | { 115 | name: 'Rabies (Imovax Rabies, RabAvert)', 116 | id: '2bfeeb1f-b7a7-4ce6-aae1-72e840a93e2e', 117 | diseases: ['Rabies'], 118 | }, 119 | { 120 | name: 'RV1 (Rotarix)', 121 | id: '8ddfa840-7558-469a-a53b-19a40d016518', 122 | diseases: ['Rotavirus'], 123 | }, 124 | { 125 | name: 'RV5 (RotaTeq)', 126 | id: '9281ddcb-5ef3-47e6-a249-6b2b8bee1e7f', 127 | diseases: ['Rotavirus'], 128 | }, 129 | { 130 | name: 'ZOS (Zostavax)', 131 | id: '2921b034-8a4c-46f5-9753-70a112dfec3f', 132 | diseases: ['Shingles (Herpes Zoster)'], 133 | }, 134 | { 135 | name: 'Vaccinia (ACAM2000)', 136 | id: 'e26378f4-5d07-4b5f-9c93-53816c0faf9f', 137 | diseases: ['Smallpox'], 138 | }, 139 | { 140 | name: 'DTaP (Daptacel, Infanrix)', 141 | id: 'b23e765e-a05b-4a24-8095-03d79e47a8aa', 142 | diseases: [ 143 | 'Tetanus (Lockjaw)', 144 | 'Pertussis (Whooping Cough)', 145 | 'Diphtheria', 146 | ], 147 | }, 148 | { 149 | name: 'Td (Decavac, generic)', 150 | id: '1af45230-cb2a-4242-81ac-2430cd64f8ce', 151 | diseases: ['Tetanus (Lockjaw)', 'Diphtheria'], 152 | }, 153 | { 154 | name: 'DT (-generic-)', 155 | id: '6eb77e28-aaa1-4e29-b124-5793a4bd6f1f', 156 | diseases: ['Tetanus (Lockjaw)', 'Diphtheria'], 157 | }, 158 | { 159 | name: 'TT (-generic-)', 160 | id: 'd6cf7277-831c-43c6-a1fa-7109d3325168', 161 | diseases: ['Tetanus (Lockjaw)'], 162 | }, 163 | { 164 | name: 'DTaP-IPV (Kinrix)', 165 | id: 'a8ecfef5-5f09-442c-84c3-4dfbcd99b3b8', 166 | diseases: [ 167 | 'Tetanus (Lockjaw)', 168 | 'Polio', 169 | 'Pertussis (Whooping Cough)', 170 | 'Diphtheria', 171 | ], 172 | }, 173 | { 174 | name: 'DTaP-HepB-IPV (Pediarix)', 175 | id: '10bc0626-7b0a-4a42-b1bf-2742f0435c37', 176 | diseases: [ 177 | 'Tetanus (Lockjaw)', 178 | 'Polio', 179 | 'Hepatitis B', 180 | 'Pertussis (Whooping Cough)', 181 | 'Diphtheria', 182 | ], 183 | }, 184 | { 185 | name: 'DTaP-IPV/Hib (Pentacel)', 186 | id: 'dcbb9691-1544-44fc-a9ca-351946010876', 187 | diseases: [ 188 | 'Tetanus (Lockjaw)', 189 | 'Polio', 190 | 'Haemophilus influenzae type b (Hib)', 191 | 'Pertussis (Whooping Cough)', 192 | 'Diphtheria', 193 | ], 194 | }, 195 | { 196 | name: 'DTaP/Hib', 197 | id: 'e817c55d-e3db-4963-9fec-04d5823f6915', 198 | diseases: [ 199 | 'Tetanus (Lockjaw)', 200 | 'Diphtheria', 201 | 'Haemophilus influenzae type b (Hib)', 202 | 'Pertussis (Whooping Cough)', 203 | ], 204 | }, 205 | { 206 | name: 'BCG (TICE BCG, Mycobax)', 207 | id: '8f2049a1-a1e3-44e1-947e-debbf3cafecc', 208 | diseases: ['Tuberculosis (TB)'], 209 | }, 210 | { 211 | name: 'Typhoid Oral (Vivotif)', 212 | id: '060f44be-e1e7-4575-ba0f-62611f03384b', 213 | diseases: ['Typhoid Fever'], 214 | }, 215 | { 216 | name: 'Typhoid Polysaccharide (Typhim Vi)', 217 | id: '87009829-1a48-4330-91e1-6bcd7ab04ee1', 218 | diseases: ['Typhoid Fever'], 219 | }, 220 | { 221 | name: 'YF (YF-Vax)', 222 | id: '24d5bfc4-d69a-4311-bb10-8980dddafa20', 223 | diseases: ['Yellow Fever'], 224 | }, 225 | ]); 226 | 227 | const keyedVaccines = vaccines.reduce((result, item) => ( 228 | result.set(item.get('id'), item) 229 | ), OrderedMap()); 230 | 231 | export default keyedVaccines.sortBy(vaccine => vaccine.get('name').toLowerCase()); 232 | -------------------------------------------------------------------------------- /containers/ChooseDate.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { connect } from 'react-redux'; 3 | import { createStructuredSelector } from 'reselect'; 4 | import ChooseDate from '../components/ChooseDate'; 5 | import addVaccination from '../actions/addVaccination'; 6 | import addForm from '../selectors/addForm'; 7 | 8 | const actions = { 9 | addVaccination, 10 | }; 11 | 12 | const selectors = { 13 | addForm, 14 | }; 15 | 16 | // eslint-disable-next-line react/prefer-stateless-function 17 | class ChooseDateContainer extends Component { 18 | render() { 19 | return ; 20 | } 21 | } 22 | 23 | export default connect(createStructuredSelector(selectors), actions)(ChooseDateContainer); 24 | -------------------------------------------------------------------------------- /containers/ChooseVaccine.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { connect } from 'react-redux'; 3 | import { createStructuredSelector } from 'reselect'; 4 | import ChooseVaccine from '../components/ChooseVaccine'; 5 | import pickVaccine from '../actions/pickVaccine'; 6 | import vaccines from '../selectors/vaccines'; 7 | 8 | const actions = { 9 | pickVaccine, 10 | }; 11 | 12 | const selectors = { 13 | vaccines, 14 | }; 15 | 16 | // eslint-disable-next-line react/prefer-stateless-function 17 | class ChooseVaccineContainer extends Component { 18 | render() { 19 | return ; 20 | } 21 | } 22 | 23 | export default connect(createStructuredSelector(selectors), actions)(ChooseVaccineContainer); 24 | -------------------------------------------------------------------------------- /containers/Detail.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { connect } from 'react-redux'; 3 | import { createStructuredSelector } from 'reselect'; 4 | import Detail from '../components/Detail'; 5 | import vaccines from '../selectors/vaccines'; 6 | import currentVaccination from '../selectors/currentVaccination'; 7 | import switchToListRoute from '../actions/switchToListRoute'; 8 | 9 | const actions = { 10 | switchToListRoute, 11 | }; 12 | 13 | const selectors = { 14 | currentVaccination, 15 | vaccines, 16 | }; 17 | 18 | // eslint-disable-next-line react/prefer-stateless-function 19 | class DetailContainer extends Component { 20 | render() { 21 | return ; 22 | } 23 | } 24 | 25 | export default connect(createStructuredSelector(selectors), actions)(DetailContainer); 26 | -------------------------------------------------------------------------------- /containers/List.js: -------------------------------------------------------------------------------- 1 | import React, { Component } from 'react'; 2 | import { connect } from 'react-redux'; 3 | import { createStructuredSelector } from 'reselect'; 4 | import List from '../components/List'; 5 | import switchToChooseVaccineRoute from '../actions/switchToChooseVaccineRoute'; 6 | import switchToDetailRoute from '../actions/switchToDetailRoute'; 7 | import vaccinations from '../selectors/vaccinations'; 8 | import vaccines from '../selectors/vaccines'; 9 | 10 | const actions = { 11 | switchToChooseVaccineRoute, 12 | switchToDetailRoute, 13 | }; 14 | 15 | const selectors = { 16 | vaccinations, 17 | vaccines, 18 | }; 19 | 20 | // eslint-disable-next-line react/prefer-stateless-function 21 | class ListContainer extends Component { 22 | render() { 23 | return ; 24 | } 25 | } 26 | 27 | export default connect(createStructuredSelector(selectors), actions)(ListContainer); 28 | -------------------------------------------------------------------------------- /exp.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Carte Jaune", 3 | "description": "A Redux/ExponentJS (React Native) app to keep track of your vaccinations.", 4 | "slug": "carte-jaune", 5 | "sdkVersion": "9.0.0", 6 | "version": "1.0.0", 7 | "orientation": "portrait", 8 | "primaryColor": "#cccccc", 9 | "iconUrl": "https://s3-us-west-2.amazonaws.com/votexp/carte-jaune-logo-big.png", 10 | "notification": { 11 | "iconUrl": "https://s3.amazonaws.com/exp-us-standard/placeholder-push-icon-blue-circle.png", 12 | "color": "#000000" 13 | }, 14 | "loading": { 15 | "iconUrl": "https://s3-us-west-2.amazonaws.com/votexp/carte-jaune-logo-big.png", 16 | "hideExponentText": false 17 | }, 18 | "packagerOpts": { 19 | "assetExts": [ 20 | "ttf" 21 | ] 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /main.js: -------------------------------------------------------------------------------- 1 | import Exponent from 'exponent'; // eslint-disable-line no-unused-vars 2 | import React, { Component } from 'react'; 3 | import { 4 | AppRegistry, 5 | Navigator, 6 | } from 'react-native'; 7 | import { Provider } from 'react-redux'; 8 | import { Scene, Router } from 'react-native-router-flux'; 9 | import configureStore from './store/configureStore'; 10 | import ChooseVaccine from './containers/ChooseVaccine'; 11 | import ChooseDate from './containers/ChooseDate'; 12 | import List from './containers/List'; 13 | import Detail from './containers/Detail'; 14 | 15 | const store = configureStore(); 16 | 17 | // eslint-disable-next-line react/prefer-stateless-function 18 | class ProviderWrapper extends Component { 19 | // eslint-disable-next-line class-methods-use-this 20 | render() { 21 | return ( 22 | 23 | 24 | 29 | 35 | 41 | 47 | 48 | 49 | ); 50 | } 51 | } 52 | 53 | AppRegistry.registerComponent('main', () => ProviderWrapper); 54 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "CarteJaune", 3 | "version": "0.0.0", 4 | "main": "main.js", 5 | "author": null, 6 | "repository": { 7 | "type": "git", 8 | "url": "https://github.com/nikgraf/CarteJaune.git" 9 | }, 10 | "dependencies": { 11 | "@exponent/vector-icons": "^1.0.1", 12 | "babel-core": "^6.5.2", 13 | "babel-eslint": "^6.1.2", 14 | "babel-plugin-transform-decorators-legacy": "^1.3.4", 15 | "babel-polyfill": "^6.13.0", 16 | "babel-preset-react-native": "^1.4.0", 17 | "babel-preset-react-native-stage-0": "^1.0.1", 18 | "chai": "^3.5.0", 19 | "chai-immutable": "^1.5.3", 20 | "dateformat": "^1.0.12", 21 | "dirty-chai": "^1.2.2", 22 | "eslint": "^3.5.0", 23 | "eslint-config-airbnb": "^11.0.0", 24 | "eslint-plugin-import": "^1.14.0", 25 | "eslint-plugin-jsx-a11y": "^2.2.1", 26 | "eslint-plugin-mocha": "^4.5.1", 27 | "eslint-plugin-react": "^6.2.0", 28 | "exponent": "^9.0.2", 29 | "immutable": "^3.8.1", 30 | "mocha": "^3.0.2", 31 | "moment": "^2.11.2", 32 | "react": "15.2.1", 33 | "react-native": "github:exponentjs/react-native#sdk-9.0.0", 34 | "react-native-mock": "^0.2.6", 35 | "react-native-router-flux": "^3.35.0", 36 | "react-redux": "^4.4.0", 37 | "redux": "^3.3.1", 38 | "redux-immutablejs": "0.0.8", 39 | "redux-logger": "^2.6.0", 40 | "redux-saga": "^0.9.1", 41 | "reselect": "^2.0.3", 42 | "sinon": "^1.17.3", 43 | "uuid": "^2.0.1" 44 | }, 45 | "scripts": { 46 | "lint": "eslint ./", 47 | "test": "node_modules/.bin/mocha --compilers js:babel-core/register --require testHelper.js **/__test__/*.js", 48 | "test:watch": "npm test -- --watch" 49 | }, 50 | "license": "MIT" 51 | } 52 | -------------------------------------------------------------------------------- /reducers/__test__/addForm.js: -------------------------------------------------------------------------------- 1 | import { Map } from 'immutable'; 2 | import { expect } from 'chai'; 3 | import reducer from '../addForm'; 4 | import { 5 | PICK_VACCINE, 6 | SWITCH_TO_CHOOSE_VACCINE_ROUTE, 7 | } from '../../constants/actions'; 8 | 9 | describe('addForm', () => { 10 | it('returns an initial state', () => { 11 | const nextState = reducer(undefined, { 12 | type: 'SOMETHING', 13 | }); 14 | expect(nextState).to.deep.equal(Map()); 15 | }); 16 | 17 | it('removes the id when switching to the form to have clean state', () => { 18 | const nextState = reducer(Map({ vaccineId: 'abc' }), { 19 | type: SWITCH_TO_CHOOSE_VACCINE_ROUTE, 20 | }); 21 | expect(nextState).to.deep.equal(Map()); 22 | }); 23 | 24 | it('add vaccine id to state in case it was picked', () => { 25 | const nextState = reducer(Map(), { 26 | type: PICK_VACCINE, 27 | vaccineId: 'cdf', 28 | }); 29 | expect(nextState.get('vaccineId')).to.equal('cdf'); 30 | }); 31 | }); 32 | -------------------------------------------------------------------------------- /reducers/__test__/currentVaccination.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import reducer from '../currentVaccination'; 3 | import { 4 | SWITCH_TO_DETAIL_ROUTE, 5 | SWITCH_TO_LIST_ROUTE, 6 | } from '../../constants/actions'; 7 | 8 | describe('currentVaccination', () => { 9 | it('returns an initial state', () => { 10 | const nextState = reducer(undefined, { 11 | type: 'SOMETHING', 12 | }); 13 | expect(nextState).to.be.null(); 14 | }); 15 | 16 | it('sets the current vaccination id on SWITCH_TO_DETAIL_ROUTE', () => { 17 | const nextState = reducer(undefined, { 18 | type: SWITCH_TO_DETAIL_ROUTE, 19 | id: 3225, 20 | }); 21 | expect(nextState).to.equal(3225); 22 | }); 23 | 24 | it('resets the current vaccination id to null on SWITCH_TO_LIST_ROUTE', () => { 25 | const nextState = reducer(4444, { 26 | type: SWITCH_TO_LIST_ROUTE, 27 | }); 28 | expect(nextState).to.be.null(); 29 | }); 30 | }); 31 | -------------------------------------------------------------------------------- /reducers/__test__/vaccinations.js: -------------------------------------------------------------------------------- 1 | import { OrderedMap } from 'immutable'; 2 | import { expect } from 'chai'; 3 | import reducer from '../vaccinations'; 4 | import { 5 | ADD_VACCINATION, 6 | FETCH_VACCINATIONS_SUCCESS, 7 | } from '../../constants/actions'; 8 | 9 | describe('vaccinations', () => { 10 | it('returns an initial state', () => { 11 | const nextState = reducer(undefined, { 12 | type: 'SOMETHING', 13 | }); 14 | expect(nextState).to.deep.equal(OrderedMap()); 15 | }); 16 | 17 | it('sets the current vaccination id on ADD_VACCINATION', () => { 18 | const nextState = reducer(undefined, { 19 | type: ADD_VACCINATION, 20 | vaccineId: 222, 21 | vaccinationDate: 444, 22 | }); 23 | expect(nextState.first().get('id')).to.equal(222); 24 | expect(nextState.first().get('date')).to.equal(444); 25 | }); 26 | 27 | it('sets the state to the vavvinations from the FETCH_VACCINATIONS_SUCCESS action', () => { 28 | const nextState = reducer(4444, { 29 | type: FETCH_VACCINATIONS_SUCCESS, 30 | vaccinations: OrderedMap({ id: 1, date: 3 }), 31 | }); 32 | const expected = OrderedMap({ id: 1, date: 3 }); 33 | expect(nextState).to.deep.equal(expected); 34 | }); 35 | }); 36 | -------------------------------------------------------------------------------- /reducers/activeRoute.js: -------------------------------------------------------------------------------- 1 | import { Actions } from 'react-native-router-flux'; 2 | import { 3 | ADD_VACCINATION_SUCCESS, 4 | PICK_VACCINE, 5 | SWITCH_TO_CHOOSE_VACCINE_ROUTE, 6 | SWITCH_TO_DETAIL_ROUTE, 7 | SWITCH_TO_LIST_ROUTE, 8 | } from '../constants/actions'; 9 | 10 | export default (state = 'list', action) => { 11 | switch (action.type) { 12 | case SWITCH_TO_CHOOSE_VACCINE_ROUTE: 13 | setTimeout(() => Actions.chooseVaccine(), 0); 14 | return 'add'; 15 | case ADD_VACCINATION_SUCCESS: 16 | setTimeout(() => Actions.list(), 0); 17 | return 'list'; 18 | case PICK_VACCINE: 19 | setTimeout(() => Actions.chooseDate(), 0); 20 | return 'chooseDate'; 21 | case SWITCH_TO_DETAIL_ROUTE: 22 | setTimeout(() => Actions.detail(), 0); 23 | return 'detail'; 24 | case SWITCH_TO_LIST_ROUTE: 25 | setTimeout(() => Actions.list(), 0); 26 | return 'list'; 27 | default: 28 | return state; 29 | } 30 | }; 31 | -------------------------------------------------------------------------------- /reducers/addForm.js: -------------------------------------------------------------------------------- 1 | import { Map } from 'immutable'; 2 | import { 3 | PICK_VACCINE, 4 | SWITCH_TO_CHOOSE_VACCINE_ROUTE, 5 | } from '../constants/actions'; 6 | 7 | export default (state = Map(), action) => { 8 | switch (action.type) { 9 | case SWITCH_TO_CHOOSE_VACCINE_ROUTE: 10 | return state.delete('vaccineId'); 11 | case PICK_VACCINE: 12 | return state.set('vaccineId', action.vaccineId); 13 | default: 14 | return state; 15 | } 16 | }; 17 | -------------------------------------------------------------------------------- /reducers/currentVaccination.js: -------------------------------------------------------------------------------- 1 | import { 2 | SWITCH_TO_DETAIL_ROUTE, 3 | SWITCH_TO_LIST_ROUTE, 4 | } from '../constants/actions'; 5 | 6 | export default (state = null, action) => { 7 | switch (action.type) { 8 | case SWITCH_TO_DETAIL_ROUTE: 9 | return action.id; 10 | case SWITCH_TO_LIST_ROUTE: 11 | return null; 12 | default: 13 | return state; 14 | } 15 | }; 16 | -------------------------------------------------------------------------------- /reducers/index.js: -------------------------------------------------------------------------------- 1 | import { combineReducers } from 'redux-immutablejs'; 2 | import activeRoute from './activeRoute'; 3 | import addForm from './addForm'; 4 | import vaccinations from './vaccinations'; 5 | import currentVaccination from './currentVaccination'; 6 | 7 | export default combineReducers({ 8 | activeRoute, 9 | addForm, 10 | currentVaccination, 11 | vaccinations, 12 | }); 13 | -------------------------------------------------------------------------------- /reducers/vaccinations.js: -------------------------------------------------------------------------------- 1 | import { OrderedMap, Map } from 'immutable'; 2 | import uuid from 'uuid'; 3 | import { 4 | ADD_VACCINATION, 5 | FETCH_VACCINATIONS_SUCCESS, 6 | } from '../constants/actions'; 7 | 8 | export default (state = OrderedMap(), action) => { 9 | switch (action.type) { 10 | case ADD_VACCINATION: 11 | return state.set(uuid(), Map({ 12 | id: action.vaccineId, 13 | date: action.vaccinationDate, 14 | })); 15 | case FETCH_VACCINATIONS_SUCCESS: 16 | return action.vaccinations; 17 | default: 18 | return state; 19 | } 20 | }; 21 | -------------------------------------------------------------------------------- /sagas/__test__/fetchVaccinations.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import { call, put, take } from 'redux-saga/effects'; 3 | import { 4 | executeFetchVaccinations, 5 | fetchVaccinations, 6 | watchFetchVaccinations, 7 | } from '../fetchVaccinations'; 8 | import { FETCH_VACCINATIONS } from '../../constants/actions'; 9 | import fetchVaccinationsSuccess from '../../actions/fetchVaccinationsSuccess'; 10 | import fetchVaccinationsFailure from '../../actions/fetchVaccinationsFailure'; 11 | 12 | describe('fetchVaccination', () => { 13 | describe('watchVaccination', () => { 14 | it('waits for the FETCH_VACCINATIONS action', () => { 15 | const generator = watchFetchVaccinations(); 16 | const expectTake = generator.next().value; 17 | expect(expectTake).to.deep.equal(take(FETCH_VACCINATIONS)); 18 | }); 19 | }); 20 | 21 | describe('fetchVaccination', () => { 22 | it('fetches the vaccinations from the store & triggers fetchVaccinationsSuccess', () => { 23 | const generator = fetchVaccinations(); 24 | const expectCall = generator.next().value; 25 | expect(expectCall).to.deep.equal(call(executeFetchVaccinations)); 26 | const expectPut = generator.next().value; 27 | expect(expectPut).to.deep.equal(put(fetchVaccinationsSuccess())); 28 | }); 29 | 30 | it('triggers fetchVaccinationsFailure in case an error occures', () => { 31 | const generator = fetchVaccinations(); 32 | generator.next(); 33 | const value = generator.throw({ error: 'retrieving failed' }).value; 34 | expect(value).to.deep.equal(put(fetchVaccinationsFailure())); 35 | }); 36 | }); 37 | }); 38 | -------------------------------------------------------------------------------- /sagas/__test__/index.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import { fork } from 'redux-saga/effects'; 3 | import root from '../index'; 4 | import { watchFetchVaccinations } from '../fetchVaccinations'; 5 | import { watchAddVaccination } from '../saveVaccinations'; 6 | import startup from '../startup'; 7 | 8 | describe('root', () => { 9 | it('should fork all the generators', () => { 10 | const generator = root(); 11 | const expected = [ 12 | fork(watchAddVaccination), 13 | fork(watchFetchVaccinations), 14 | fork(startup), 15 | ]; 16 | expect(generator.next().value).to.deep.equal(expected); 17 | }); 18 | }); 19 | -------------------------------------------------------------------------------- /sagas/__test__/saveVaccinations.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import { call, take, put, select } from 'redux-saga/effects'; 3 | import { 4 | executeSaveVaccinations, 5 | saveVaccinations, 6 | watchAddVaccination, 7 | } from '../saveVaccinations'; 8 | import { ADD_VACCINATION } from '../../constants/actions'; 9 | import addVaccinationSuccess from '../../actions/addVaccinationSuccess'; 10 | import addVaccinationFailure from '../../actions/addVaccinationFailure'; 11 | import vaccinationsSelector from '../../selectors/vaccinations'; 12 | 13 | describe('saveVaccination', () => { 14 | describe('watchAddVaccination', () => { 15 | it('waits for the ADD_VACCINATION action', () => { 16 | const generator = watchAddVaccination(); 17 | const takeValue = generator.next().value; 18 | expect(takeValue).to.deep.equal(take(ADD_VACCINATION)); 19 | }); 20 | }); 21 | 22 | describe('saveVaccination', () => { 23 | it('saves the vaccinations to the store and triggers addVaccinationSuccess', () => { 24 | const generator = saveVaccinations(); 25 | const selectValue = generator.next().value; 26 | expect(selectValue).to.deep.equal(select(vaccinationsSelector)); 27 | const callValue = generator.next({ 1: 2 }).value; 28 | expect(callValue).to.deep.equal(call(executeSaveVaccinations, { 1: 2 })); 29 | const putValue = generator.next().value; 30 | expect(putValue).to.deep.equal(put(addVaccinationSuccess())); 31 | }); 32 | 33 | it('triggers addVaccinationFailure in case an error occures', () => { 34 | const generator = saveVaccinations(); 35 | generator.next(); 36 | const value = generator.throw({ error: 'saving failed' }).value; 37 | expect(value).to.deep.equal(put(addVaccinationFailure())); 38 | }); 39 | }); 40 | }); 41 | -------------------------------------------------------------------------------- /sagas/__test__/startup.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import { put } from 'redux-saga/effects'; 3 | import fetchVaccinations from '../../actions/fetchVaccinations'; 4 | import startup from '../startup'; 5 | 6 | describe('startup', () => { 7 | it('should fetch the vaccinations', () => { 8 | const generator = startup(); 9 | const expectPut = generator.next().value; 10 | expect(expectPut).to.deep.equal(put(fetchVaccinations())); 11 | }); 12 | }); 13 | -------------------------------------------------------------------------------- /sagas/fetchVaccinations.js: -------------------------------------------------------------------------------- 1 | import { AsyncStorage } from 'react-native'; 2 | import { takeLatest } from 'redux-saga'; 3 | import { call, put } from 'redux-saga/effects'; 4 | import { fromJS, OrderedMap } from 'immutable'; 5 | import { FETCH_VACCINATIONS } from '../constants/actions'; 6 | import fetchVaccinationsSuccess from '../actions/fetchVaccinationsSuccess'; 7 | import fetchVaccinationsFailure from '../actions/fetchVaccinationsFailure'; 8 | import { KEY } from '../constants/storage'; 9 | 10 | /** 11 | * Returns a promise of the actual fetching from the persitent storage. 12 | */ 13 | export const executeFetchVaccinations = () => ( 14 | AsyncStorage.getItem(KEY) 15 | .then((result) => { 16 | if (result === null) return OrderedMap(); 17 | return fromJS(JSON.parse(result)); 18 | }) 19 | ); 20 | 21 | /** 22 | * Retrieves the vaccinations from the presistent storage. 23 | */ 24 | export function* fetchVaccinations() { 25 | try { 26 | const vaccinations = yield call(executeFetchVaccinations); 27 | yield put(fetchVaccinationsSuccess(vaccinations)); 28 | } catch (error) { 29 | yield put(fetchVaccinationsFailure()); 30 | } 31 | } 32 | 33 | /* 34 | * Waits for an FETCH_VACCINATIONS to trigger fetchVaccinations. 35 | * 36 | * Whenever fetchVaccinations is in progress it doesn't trigger a new call of it. 37 | */ 38 | export function* watchFetchVaccinations() { 39 | yield* takeLatest(FETCH_VACCINATIONS, fetchVaccinations); 40 | } 41 | -------------------------------------------------------------------------------- /sagas/index.js: -------------------------------------------------------------------------------- 1 | import { fork } from 'redux-saga/effects'; 2 | import { watchAddVaccination } from './saveVaccinations'; 3 | import { watchFetchVaccinations } from './fetchVaccinations'; 4 | import startup from './startup'; 5 | 6 | /* 7 | * The entry point for all the sagas used in this application. 8 | */ 9 | export default function* root() { 10 | yield [ 11 | fork(watchAddVaccination), 12 | fork(watchFetchVaccinations), 13 | fork(startup), 14 | ]; 15 | } 16 | -------------------------------------------------------------------------------- /sagas/saveVaccinations.js: -------------------------------------------------------------------------------- 1 | import { AsyncStorage } from 'react-native'; 2 | import { takeEvery } from 'redux-saga'; 3 | import { call, put, select } from 'redux-saga/effects'; 4 | import { ADD_VACCINATION } from '../constants/actions'; 5 | import addVaccinationSuccess from '../actions/addVaccinationSuccess'; 6 | import addVaccinationFailure from '../actions/addVaccinationFailure'; 7 | import { KEY } from '../constants/storage'; 8 | import vaccinationsSelector from '../selectors/vaccinations'; 9 | 10 | /* 11 | * Returns the Promise of the actual saving from the persitent storage. 12 | */ 13 | export const executeSaveVaccinations = (data) => ( 14 | AsyncStorage.setItem(KEY, JSON.stringify(data.toJS())) 15 | ); 16 | 17 | /* 18 | * Retrieves the vaccinations from the Redux store to save them in the presitent storage. 19 | */ 20 | export function* saveVaccinations() { 21 | try { 22 | const vaccinations = yield select(vaccinationsSelector); 23 | yield call(executeSaveVaccinations, vaccinations); 24 | yield put(addVaccinationSuccess()); 25 | } catch (error) { 26 | yield put(addVaccinationFailure()); 27 | } 28 | } 29 | 30 | /* 31 | * Waits for an ADD_VACCINATION to trigger saveVaccinations. 32 | * 33 | * Will trigger a new call of saveVaccinations on every ADD_VACCINATION. 34 | */ 35 | export function* watchAddVaccination() { 36 | yield* takeEvery(ADD_VACCINATION, saveVaccinations); 37 | } 38 | -------------------------------------------------------------------------------- /sagas/startup.js: -------------------------------------------------------------------------------- 1 | import { put } from 'redux-saga/effects'; 2 | import fetchVaccinations from '../actions/fetchVaccinations'; 3 | 4 | export default function* startup() { 5 | yield put(fetchVaccinations()); 6 | } 7 | -------------------------------------------------------------------------------- /selectors/__test__/addForm.js: -------------------------------------------------------------------------------- 1 | import { Map } from 'immutable'; 2 | import { expect } from 'chai'; 3 | import selector from '../addForm'; 4 | 5 | describe('addForm', () => { 6 | it('retrieves addForm from the state', () => { 7 | expect(selector(Map({ addForm: 'form' }))).to.equal('form'); 8 | }); 9 | }); 10 | -------------------------------------------------------------------------------- /selectors/__test__/currentVaccination.js: -------------------------------------------------------------------------------- 1 | import { fromJS } from 'immutable'; 2 | import { expect } from 'chai'; 3 | import selector from '../currentVaccination'; 4 | 5 | describe('currentVaccination', () => { 6 | it('retrieves currentVaccination from the state', () => { 7 | const state = fromJS({ 8 | vaccinations: { 9 | 1: { 10 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 11 | }, 12 | 2: { 13 | id: 'f3e08a56-003c-4b46-9dea-216298401ca0', 14 | }, 15 | }, 16 | currentVaccination: '1', 17 | }); 18 | const expected = fromJS({ 19 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 20 | vaccine: { 21 | name: 'AVA (BioThrax)', 22 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 23 | diseases: ['Anthrax'], 24 | }, 25 | listId: '1', 26 | }); 27 | expect(selector(state)).to.equal(expected); 28 | }); 29 | }); 30 | -------------------------------------------------------------------------------- /selectors/__test__/vaccinations.js: -------------------------------------------------------------------------------- 1 | import { fromJS } from 'immutable'; 2 | import { expect } from 'chai'; 3 | import selector from '../vaccinations'; 4 | 5 | describe('vaccinations', () => { 6 | it('retrieves with data populated vaccinations from the state', () => { 7 | const state = fromJS({ 8 | vaccinations: { 9 | 1: { 10 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 11 | }, 12 | }, 13 | }); 14 | const expected = fromJS({ 15 | 1: { 16 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 17 | vaccine: { 18 | name: 'AVA (BioThrax)', 19 | id: '8b013618-439e-4829-b88f-98a44b420ee8', 20 | diseases: ['Anthrax'], 21 | }, 22 | listId: '1', 23 | }, 24 | }); 25 | expect(selector(state)).to.equal(expected); 26 | }); 27 | }); 28 | -------------------------------------------------------------------------------- /selectors/__test__/vaccines.js: -------------------------------------------------------------------------------- 1 | import { expect } from 'chai'; 2 | import selector from '../vaccines'; 3 | import vaccines from '../../constants/vaccines'; 4 | 5 | describe('addForm', () => { 6 | it('retrieves the vaccines from constants', () => { 7 | expect(selector()).to.deep.equal(vaccines); 8 | }); 9 | }); 10 | -------------------------------------------------------------------------------- /selectors/addForm.js: -------------------------------------------------------------------------------- 1 | export default (state) => state.get('addForm'); 2 | -------------------------------------------------------------------------------- /selectors/currentVaccination.js: -------------------------------------------------------------------------------- 1 | import { createSelector } from 'reselect'; 2 | import vaccinations from './vaccinations'; 3 | 4 | const currentId = (state) => state.get('currentVaccination'); 5 | 6 | export default createSelector( 7 | [vaccinations, currentId], 8 | (list, id) => list.get(id) 9 | ); 10 | -------------------------------------------------------------------------------- /selectors/vaccinations.js: -------------------------------------------------------------------------------- 1 | import { createSelector } from 'reselect'; 2 | import { Map } from 'immutable'; 3 | import shallowVaccines from './vaccines'; 4 | 5 | const shallowVaccinations = (state) => state.get('vaccinations'); 6 | 7 | export default createSelector( 8 | [shallowVaccinations, shallowVaccines], 9 | (vaccinations, vaccines) => { 10 | if (!vaccinations) return Map(); 11 | return vaccinations.map((vaccination, id) => ( 12 | vaccination 13 | .set('vaccine', vaccines.get(vaccination.get('id'))) 14 | .set('listId', id) 15 | )); 16 | } 17 | ); 18 | -------------------------------------------------------------------------------- /selectors/vaccines.js: -------------------------------------------------------------------------------- 1 | import vaccines from '../constants/vaccines'; 2 | 3 | export default () => vaccines; 4 | -------------------------------------------------------------------------------- /store/configureStore.js: -------------------------------------------------------------------------------- 1 | import { createStore, applyMiddleware } from 'redux'; 2 | import sagaMiddleware from 'redux-saga'; 3 | import createLogger from 'redux-logger'; 4 | import reducer from '../reducers'; 5 | import sagas from '../sagas'; 6 | 7 | const loggerMiddleware = createLogger({ 8 | stateTransformer: state => state.toJS(), 9 | }); 10 | 11 | const createStoreWithMiddleware = applyMiddleware( 12 | sagaMiddleware(sagas), 13 | loggerMiddleware 14 | )(createStore); 15 | 16 | export default (initialState) => ( 17 | createStoreWithMiddleware(reducer, initialState) 18 | ); 19 | -------------------------------------------------------------------------------- /testHelper.js: -------------------------------------------------------------------------------- 1 | import chai from 'chai'; 2 | import dirtyChai from 'dirty-chai'; 3 | import chaiImmutable from 'chai-immutable'; 4 | 5 | require('babel-polyfill'); 6 | require('react-native-mock/mock'); 7 | 8 | chai.use(dirtyChai); 9 | chai.use(chaiImmutable); 10 | --------------------------------------------------------------------------------