├── src
├── js
│ ├── components
│ │ ├── core
│ │ │ ├── text-label.js
│ │ │ ├── link.js
│ │ │ ├── text-input.js
│ │ │ └── editable-text.js
│ │ ├── way-point-details.js
│ │ ├── way-point-info.js
│ │ ├── app-bar.js
│ │ ├── route.js
│ │ ├── routes.js
│ │ ├── way-point.js
│ │ ├── way-points.js
│ │ ├── route-info.js
│ │ ├── route-visual.js
│ │ └── map.js
│ ├── services
│ │ ├── storage.js
│ │ └── api.js
│ ├── middlewares
│ │ └── state-storage.js
│ ├── main.js
│ ├── reducers
│ │ ├── routes.js
│ │ ├── way-points.js
│ │ ├── way-point.js
│ │ ├── route.js
│ │ └── app.js
│ ├── util.js
│ ├── containers
│ │ ├── routes.js
│ │ └── way-points.js
│ ├── selectors
│ │ └── app.js
│ ├── actions
│ │ ├── types.js
│ │ └── index.js
│ ├── app.js
│ └── sagas
│ │ └── index.js
├── scss
│ ├── _way-point-details.scss
│ ├── _variables.scss
│ ├── fonts
│ │ ├── icomoon.eot
│ │ ├── icomoon.ttf
│ │ ├── icomoon.woff
│ │ └── icomoon.svg
│ ├── _text-input.scss
│ ├── _route.scss
│ ├── _app.scss
│ ├── _routes.scss
│ ├── _way-point-info.scss
│ ├── _route-info.scss
│ ├── _app-bar.scss
│ ├── _reset.scss
│ ├── _waypoints.scss
│ ├── _icons.scss
│ └── base.scss
├── img
│ ├── bg1.jpg
│ ├── bg10.jpg
│ ├── bg11.jpg
│ ├── bg12.jpg
│ ├── bg2.jpg
│ ├── bg3.jpg
│ ├── bg4.jpg
│ ├── bg5.jpg
│ ├── bg6.jpg
│ ├── bg7.jpg
│ ├── bg8.jpg
│ └── bg9.jpg
└── views
│ └── index.html
├── test
├── tests.webpack.js
└── hello-spec.js
├── webpack.test.js
├── .gitignore
├── webpack.prod.js
├── karma.conf.js
├── README.md
├── gruntfile.js
├── package.json
├── webpack.config.js
└── npm-shrinkwrap.json
/src/js/components/core/text-label.js:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/scss/_way-point-details.scss:
--------------------------------------------------------------------------------
1 | .way-point-details {
2 |
3 | }
--------------------------------------------------------------------------------
/src/img/bg1.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg1.jpg
--------------------------------------------------------------------------------
/src/img/bg10.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg10.jpg
--------------------------------------------------------------------------------
/src/img/bg11.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg11.jpg
--------------------------------------------------------------------------------
/src/img/bg12.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg12.jpg
--------------------------------------------------------------------------------
/src/img/bg2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg2.jpg
--------------------------------------------------------------------------------
/src/img/bg3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg3.jpg
--------------------------------------------------------------------------------
/src/img/bg4.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg4.jpg
--------------------------------------------------------------------------------
/src/img/bg5.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg5.jpg
--------------------------------------------------------------------------------
/src/img/bg6.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg6.jpg
--------------------------------------------------------------------------------
/src/img/bg7.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg7.jpg
--------------------------------------------------------------------------------
/src/img/bg8.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg8.jpg
--------------------------------------------------------------------------------
/src/img/bg9.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/img/bg9.jpg
--------------------------------------------------------------------------------
/src/scss/_variables.scss:
--------------------------------------------------------------------------------
1 | $pad: 10px;
2 | $widgetWidth: 400px;
3 | $textColor: #333333;
--------------------------------------------------------------------------------
/src/scss/fonts/icomoon.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/scss/fonts/icomoon.eot
--------------------------------------------------------------------------------
/src/scss/fonts/icomoon.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/scss/fonts/icomoon.ttf
--------------------------------------------------------------------------------
/src/scss/fonts/icomoon.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vraa/route-planner/HEAD/src/scss/fonts/icomoon.woff
--------------------------------------------------------------------------------
/test/tests.webpack.js:
--------------------------------------------------------------------------------
1 | var context = require.context(".", true, /-spec\.js$/);
2 | context.keys().forEach(context);
--------------------------------------------------------------------------------
/src/scss/_text-input.scss:
--------------------------------------------------------------------------------
1 | .text-input {
2 |
3 | input {
4 | background-color: transparent;
5 | }
6 |
7 | }
--------------------------------------------------------------------------------
/webpack.test.js:
--------------------------------------------------------------------------------
1 | let baseConfig = require('./webpack.config');
2 |
3 | module.exports = Object.assign({}, baseConfig, {});
--------------------------------------------------------------------------------
/test/hello-spec.js:
--------------------------------------------------------------------------------
1 | describe('hello test', () => {
2 | it('should pass', () => {
3 | expect(true).toBe(true);
4 | });
5 | });
--------------------------------------------------------------------------------
/src/scss/_route.scss:
--------------------------------------------------------------------------------
1 | .route {
2 | background-color: rgba(255,255,255,.8);
3 | }
4 |
5 | .route .route-name {
6 | margin-bottom: $pad;
7 | }
--------------------------------------------------------------------------------
/src/scss/_app.scss:
--------------------------------------------------------------------------------
1 | .app {
2 | background-size: cover;
3 | }
4 |
5 | .route-planner {
6 | display: grid;
7 | grid-template-rows: 100vh;
8 | }
9 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules/**
2 | resources/*.html
3 | resources/js/*.*
4 | resources/css/*.css
5 | resources/css/*.css.map
6 | *.iml
7 | .sass-cache/**
8 | .idea
9 | bash.exe.stackdump
10 |
--------------------------------------------------------------------------------
/src/scss/_routes.scss:
--------------------------------------------------------------------------------
1 | .routes {
2 | display: grid;
3 | grid-template-columns: 1fr;
4 | grid-template-rows: 1fr 1fr;
5 | box-shadow: 0 0 5px #3d3d3d;
6 | }
7 |
8 | @media screen and (min-width: 600px) {
9 | .routes {
10 | grid-template-columns: 400px 1fr;
11 | grid-template-rows: 1fr;
12 | margin: 100px 10%;
13 | }
14 | }
--------------------------------------------------------------------------------
/src/js/services/storage.js:
--------------------------------------------------------------------------------
1 | const db = window.localStorage;
2 |
3 | let storage = {
4 |
5 | set: (key, val) => {
6 | if (key && val) {
7 | db.setItem(key, val);
8 | }
9 | },
10 |
11 | get: (key) => {
12 | let val = db.getItem(key);
13 | return val ? JSON.parse(val) : null;
14 | }
15 | };
16 |
17 | export default storage;
--------------------------------------------------------------------------------
/src/js/components/way-point-details.js:
--------------------------------------------------------------------------------
1 | import React, {Component} from 'react';
2 |
3 | class WayPointDetails extends Component {
4 | render() {
5 | return (
6 |
7 |
Way Point Details
8 |
9 |
10 | )
11 | }
12 | }
13 |
14 | export default WayPointDetails;
--------------------------------------------------------------------------------
/src/js/components/core/link.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 |
3 | class Link extends React.Component {
4 |
5 | handleClick(e) {
6 | e.preventDefault();
7 | this.props.onClick();
8 | }
9 |
10 | render() {
11 | return (
12 | {this.props.children}
13 | )
14 | }
15 | }
16 |
17 | module.exports = Link;
--------------------------------------------------------------------------------
/webpack.prod.js:
--------------------------------------------------------------------------------
1 | let baseConfig = require('./webpack.config');
2 | let webpack = require('webpack');
3 |
4 | module.exports = Object.assign({}, baseConfig, {
5 | plugins: [].concat(
6 | [
7 | new webpack.DefinePlugin({
8 | 'process.env': {
9 | 'NODE_ENV': JSON.stringify('production')
10 | }
11 | })
12 | ],
13 | baseConfig.plugins
14 | ),
15 | devtool: 'none'
16 | });
--------------------------------------------------------------------------------
/src/js/middlewares/state-storage.js:
--------------------------------------------------------------------------------
1 | import storage from "../services/storage";
2 |
3 | let DB_NAME = "appState";
4 |
5 | let stateStorage = ({getState}) => next => action => {
6 | let returnVal = next(action);
7 | if (action.saveState) {
8 | storage.set(DB_NAME, JSON.stringify(getState()));
9 | }
10 | return returnVal;
11 | };
12 |
13 | let createStorageMiddleware = (dbName) => {
14 | DB_NAME = dbName;
15 | return stateStorage;
16 | };
17 | export default createStorageMiddleware;
--------------------------------------------------------------------------------
/src/js/components/way-point-info.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 |
3 | class WayPointInfo extends React.Component {
4 |
5 | render() {
6 | return (
7 |
8 | {this.props.distance}
9 | {this.props.duration}
10 |
11 | )
12 | }
13 |
14 | }
15 |
16 | module.exports = WayPointInfo;
--------------------------------------------------------------------------------
/src/js/main.js:
--------------------------------------------------------------------------------
1 | require('../scss/base.scss');
2 |
3 | let React = require('react');
4 | let ReactDOM = require('react-dom');
5 | let App = require('./app');
6 | let domReady = require('domready');
7 | let GoogleMaps = require('google-maps');
8 |
9 | domReady(()=> {
10 | GoogleMaps.LIBRARIES = ['places'];
11 | GoogleMaps.KEY='AIzaSyAkpb6H_VzRih_zDtUnFJF9tgEG46S3hqU';
12 | GoogleMaps.load((google)=> {
13 | ReactDOM.render(,
14 | document.getElementById('app'));
15 | });
16 |
17 | });
18 |
--------------------------------------------------------------------------------
/src/js/reducers/routes.js:
--------------------------------------------------------------------------------
1 | let Immutable = require('immutable');
2 | let ActionTypes = require('../actions/types');
3 | let Actions = require('../actions');
4 | let route = require('./route');
5 |
6 | let DEFAULTS = Immutable.List.of(route(undefined, Actions.addRoute(0)));
7 |
8 | const routes = (state = DEFAULTS, action) => {
9 | switch (action.type) {
10 | case ActionTypes.ADD_ROUTE:
11 | return state.push(route(undefined, action));
12 | case ActionTypes.ADD_WAY_POINT_TO_ROUTE:
13 | return state;
14 | default:
15 | return state;
16 | }
17 |
18 | };
19 |
20 | module.exports = routes;
--------------------------------------------------------------------------------
/src/scss/_way-point-info.scss:
--------------------------------------------------------------------------------
1 | @import "variables";
2 |
3 | .way-point-info {
4 | margin-top: 2 * $pad;
5 | margin-left: 4 * $pad;
6 | font-size: .8rem;
7 | color: #6d6d6d;
8 |
9 | .distance,
10 | .duration {
11 | padding: 2px 5px;
12 | color: #fff;
13 | border:1px solid;
14 | border-radius: 3px;
15 | letter-spacing: 1px;
16 | }
17 |
18 | .distance {
19 | background-color: #F8BBD0;
20 | color: #880E4F;
21 | border-color: #F48FB1;
22 | }
23 |
24 | .duration {
25 | margin-left: 2 * $pad;
26 | background-color: #E1BEE7;
27 | color: #4A148C;
28 | border-color: #CE93D8;
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/src/js/components/app-bar.js:
--------------------------------------------------------------------------------
1 | import React, {Component} from 'react';
2 |
3 | class AppBar extends Component {
4 |
5 | render() {
6 | return (
7 |
14 | )
15 | }
16 | }
17 |
18 | export default AppBar;
--------------------------------------------------------------------------------
/src/js/reducers/way-points.js:
--------------------------------------------------------------------------------
1 | var ActionTypes = require('../actions/types');
2 | var Actions = require('../actions');
3 | var wayPoint = require('./way-point');
4 |
5 | var DEFAULTS = [];
6 |
7 | const wayPoints = (state = DEFAULTS, action) => {
8 |
9 | switch (action.type) {
10 | case ActionTypes.ADD_WAY_POINT:
11 | return [
12 | ...state,
13 | wayPoint(undefined, action)
14 | ];
15 | break;
16 | case ActionTypes.REMOVE_WAY_POINT:
17 | return state.filter((wp)=> {
18 | return wp.id !== action.id
19 | });
20 | default:
21 | return state;
22 | }
23 |
24 | };
25 |
26 | module.exports = wayPoints;
--------------------------------------------------------------------------------
/karma.conf.js:
--------------------------------------------------------------------------------
1 | module.exports = function (config) {
2 |
3 | config.set({
4 |
5 | captureTimeout: 60000,
6 | browserNoActivityTimeout: 60000,
7 | colors: true,
8 | port: 9876,
9 |
10 | files: [
11 | './test/tests.webpack.js'
12 | ],
13 |
14 | preprocessors: {
15 | './test/tests.webpack.js': ['webpack']
16 | },
17 | browsers: ['Chrome'],
18 |
19 | singleRun: true,
20 |
21 | autoWatch: false,
22 |
23 | frameworks: ['jasmine'],
24 |
25 | webpack: {},
26 |
27 | webpackMiddleware: {
28 | noInfo: true,
29 | stats: {
30 | colors: true
31 | }
32 | }
33 | });
34 | };
35 |
--------------------------------------------------------------------------------
/src/js/util.js:
--------------------------------------------------------------------------------
1 | import Immutable from 'immutable';
2 |
3 | var Util = {
4 | guid: () => {
5 | return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
6 | var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
7 | return v.toString(16);
8 | });
9 | },
10 |
11 | transformSavedState: (st) => {
12 | if (!st) {
13 | return null;
14 | }
15 | let routes = st.routes.map((r) => {
16 | return Object.assign({}, r, {
17 | wayPoints: Immutable.List(r.wayPoints)
18 | });
19 | });
20 | let transformed = Object.assign({}, st, {
21 | routes: Immutable.List(routes),
22 | wayPoints: Immutable.List(st.wayPoints)
23 | });
24 | return transformed;
25 | }
26 | };
27 |
28 | module.exports = Util;
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Route Planner using Google Maps, built with React JS, Redux and Immutable.js
2 |
3 | **[Route Planner](http://veerasundar.com/route-planner)** is a simple app that helps you to plan your travel routes.
4 |
5 | * Add upto 8 way points between start and end destinations of your route.
6 | * See distance and time details for each way points and whole route.
7 |
8 | ## How to run locally?
9 |
10 | 1. Clone and cd into the repo.
11 | 2. Run `npm install` to download dependencies.
12 | 3. Run `npm start` to start the development server.
13 | 4. Open your browser and point to http://localhost:8080/
14 |
15 | The application has a web pack dev server, so as you make your changes in your code, the changes will be immeditely reflected in your browser.
16 |
17 | ## Screenshot
18 |
19 | [](http://veerasundar.com/route-planner)
20 |
21 |
--------------------------------------------------------------------------------
/src/js/reducers/way-point.js:
--------------------------------------------------------------------------------
1 | var ActionTypes = require('../actions/types');
2 | var util = require('../util');
3 | let DEFAULT_NAME = '';
4 |
5 | const wayPoint = (state = {}, action) => {
6 | switch (action.type) {
7 | case ActionTypes.ADD_WAY_POINT:
8 | return {
9 | id: util.guid(),
10 | name: DEFAULT_NAME
11 | };
12 | case ActionTypes.CHANGE_WAY_POINT_NAME:
13 | return Object.assign({}, state, {
14 | name: (action.newName.trim() || DEFAULT_NAME)
15 | });
16 | case ActionTypes.OPEN_WAY_POINT_DETAILS:
17 | return Object.assign({}, state, {
18 | detailsOpen: true
19 | });
20 | case ActionTypes.CLOSE_WAY_POINT_DETAILS:
21 | return Object.assign({}, state, {
22 | detailsOpen: false
23 | });
24 | default:
25 | return state;
26 | }
27 | };
28 |
29 | module.exports = wayPoint;
--------------------------------------------------------------------------------
/src/views/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | Route Planner - Google Maps
9 |
10 |
11 |
12 |
13 |
14 |
24 |
25 |
--------------------------------------------------------------------------------
/src/scss/_route-info.scss:
--------------------------------------------------------------------------------
1 | @import "variables";
2 |
3 | .route-info {
4 | display: grid;
5 | grid-template-columns: 1fr;
6 | justify-content: center;
7 | height: 130px;
8 | width: 90%;
9 | margin: 0 auto;
10 | padding: 20px 0;
11 | font-weight: 300;
12 |
13 | h2 {
14 | font-size: .6rem;
15 | text-transform: uppercase;
16 | letter-spacing: 2pt;
17 | margin-bottom: 10px;
18 | }
19 |
20 | .route-visual {
21 | text-align: center;
22 | margin-top: 20px;
23 | }
24 |
25 | .textual {
26 | display: grid;
27 | grid-template-columns: 1fr 1fr;
28 | }
29 |
30 | .value {
31 | font-size: 2.2rem;
32 | font-weight: 300;
33 | }
34 |
35 | .unit {
36 | font-size: .9rem;
37 | text-transform: lowercase;
38 | letter-spacing: 1pt;
39 | }
40 |
41 | .distance {
42 | text-align: center;
43 | color: #249991;
44 | }
45 |
46 | .duration {
47 | text-align: center;
48 | color: #953163;
49 | li {
50 | display: inline-block;
51 | margin-right: 2* $pad;
52 |
53 | &:last-child {
54 | margin-right: 0;
55 | }
56 | }
57 | }
58 | }
--------------------------------------------------------------------------------
/src/js/components/route.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 | var EditableText = require('./core/editable-text');
3 | var RouteInfo = require('./route-info');
4 | var WayPoints = require('../containers/way-points');
5 | import AppBar from "./app-bar";
6 |
7 | class Route extends React.Component {
8 |
9 | constructor(props) {
10 | super(props);
11 | }
12 |
13 | handleRouteNameChange(newName) {
14 | this.props.onNameChange(this.props.route.id, newName);
15 | }
16 |
17 |
18 | renderRouteName() {
19 | var route = this.props.route;
20 | return (
21 |
22 |
25 |
26 | )
27 | }
28 |
29 | render() {
30 | var route = this.props.route;
31 | return (
32 |
37 | )
38 |
39 | }
40 | }
41 |
42 | module.exports = Route;
--------------------------------------------------------------------------------
/src/scss/_app-bar.scss:
--------------------------------------------------------------------------------
1 | @import "variables";
2 |
3 | .app-bar {
4 | display: grid;
5 | grid-template-columns: 1fr 1fr;
6 | padding: 10px;
7 | }
8 |
9 | .app-bar h1 {
10 | background-image: url('');
11 | background-position: top left;
12 | background-size: 20px;
13 | background-repeat: no-repeat;
14 | font-size: .8rem;
15 | text-transform: uppercase;
16 | color: #000;
17 | font-weight: bold;
18 | letter-spacing: 2pt;
19 | padding-left: 25px;
20 | }
21 |
22 | .app-bar h1 a {
23 | text-decoration: none;
24 | color: #000;
25 | }
26 |
27 | .app-bar .author-details {
28 | font-size: .8rem;
29 | color: #6d6d6d;
30 | text-align: right;
31 | }
32 |
33 | .app-bar a {
34 | color: #4d4d4d;
35 | }
--------------------------------------------------------------------------------
/src/js/containers/routes.js:
--------------------------------------------------------------------------------
1 | var {connect} = require('react-redux');
2 | var Actions = require('../actions');
3 | var Routes = require('../components/routes');
4 |
5 | const getActiveRoute = (state) => {
6 | if (state && state.routes) {
7 | return state.routes.find((r) => {
8 | return r.id === state.activeRouteID;
9 | });
10 | }
11 | };
12 |
13 | const mapStateToProps = (state, ownProps) => {
14 |
15 | return {
16 | routes: state.routes,
17 | activeRoute: getActiveRoute(state),
18 | mapService: ownProps.mapService,
19 | mapData: state.mapData
20 | }
21 | };
22 |
23 | const mapDispatchToProps = (dispatch) => {
24 | return {
25 | onAdd: () => {
26 | dispatch(Actions.addRouteRequested())
27 | },
28 | onRemove: (id) => {
29 | dispatch(Actions.removeRoute(id))
30 | },
31 | onChangeRoute: (id) => {
32 | dispatch(Actions.changeActiveRoute(id))
33 | },
34 | onChangeRouteName: (routeId, newName) => {
35 | dispatch(Actions.changeRouteName(routeId, newName))
36 | }
37 | }
38 | };
39 |
40 | const RoutesContainer = connect(
41 | mapStateToProps,
42 | mapDispatchToProps
43 | )(Routes);
44 |
45 | module.exports = RoutesContainer;
46 |
--------------------------------------------------------------------------------
/src/scss/_reset.scss:
--------------------------------------------------------------------------------
1 | /* http://meyerweb.com/eric/tools/css/reset/
2 | v2.0 | 20110126
3 | License: none (public domain)
4 | */
5 |
6 | html, body, div, span, applet, object, iframe,
7 | h1, h2, h3, h4, h5, h6, p, blockquote, pre,
8 | a, abbr, acronym, address, big, cite, code,
9 | del, dfn, em, img, ins, kbd, q, s, samp,
10 | small, strike, strong, sub, sup, tt, var,
11 | b, u, i, center,
12 | dl, dt, dd, ol, ul, li,
13 | fieldset, form, label, legend,
14 | table, caption, tbody, tfoot, thead, tr, th, td,
15 | article, aside, canvas, details, embed,
16 | figure, figcaption, footer, header, hgroup,
17 | menu, nav, output, ruby, section, summary,
18 | time, mark, audio, video {
19 | margin: 0;
20 | padding: 0;
21 | border: 0;
22 | font-size: 100%;
23 | font: inherit;
24 | vertical-align: baseline;
25 | }
26 | /* HTML5 display-role reset for older browsers */
27 | article, aside, details, figcaption, figure,
28 | footer, header, hgroup, menu, nav, section {
29 | display: block;
30 | }
31 | body {
32 | line-height: 1;
33 | }
34 | ol, ul {
35 | list-style: none;
36 | }
37 | blockquote, q {
38 | quotes: none;
39 | }
40 | blockquote:before, blockquote:after,
41 | q:before, q:after {
42 | content: '';
43 | content: none;
44 | }
45 | table {
46 | border-collapse: collapse;
47 | border-spacing: 0;
48 | }
--------------------------------------------------------------------------------
/src/scss/_waypoints.scss:
--------------------------------------------------------------------------------
1 | @import "variables";
2 |
3 | ul.way-points {
4 | margin: 2 * $pad;
5 | position: relative;
6 | max-height: 300px;
7 | overflow: auto;
8 | a {
9 | text-decoration: none;
10 | border-bottom: 1px solid #19B5FE;
11 | &:link, &:hover, &:visited, &:active {
12 | color: $textColor;
13 | }
14 | }
15 |
16 | &.empty a {
17 | margin-left: 15px;
18 | }
19 |
20 | li {
21 | margin-bottom: 2 * $pad;
22 | position: relative;
23 | z-index: 10;
24 |
25 | .way-point-icon {
26 | position: absolute;
27 | top: 0;
28 | left: 0;
29 | color: #19B5FE;
30 | z-index: 10;
31 | }
32 |
33 | &:first-child .way-point-icon {
34 | color: #2ECC71;
35 | }
36 |
37 | &:last-child .way-point-icon {
38 | color: #CF000F;
39 | }
40 |
41 | &.add-place {
42 | margin-top: 4 * $pad;
43 | }
44 | }
45 |
46 | &:before {
47 | content: '';
48 | position: absolute;
49 | top: 0;
50 | left: 8px;
51 | height: 100%;
52 | width: 1px;
53 | border-left: 1px dashed #ddd;
54 | z-index: 5;
55 | }
56 |
57 | &.empty:before {
58 | content: none;
59 | }
60 |
61 | &.max-way-points {
62 | .add-place,
63 | .insert-way-point {
64 | display: none;
65 | }
66 | }
67 | }
--------------------------------------------------------------------------------
/src/scss/_icons.scss:
--------------------------------------------------------------------------------
1 | @font-face {
2 | font-family: 'icomoon';
3 | src: url('./fonts/icomoon.eot?rgc6vh');
4 | src: url('./fonts/icomoon.eot?rgc6vh#iefix') format('embedded-opentype'),
5 | url('./fonts/icomoon.ttf?rgc6vh') format('truetype'),
6 | url('./fonts/icomoon.woff?rgc6vh') format('woff'),
7 | url('./fonts/icomoon.svg?rgc6vh#icomoon') format('svg');
8 | font-weight: normal;
9 | font-style: normal;
10 | }
11 |
12 | [class^="icon-"], [class*=" icon-"] {
13 | /* use !important to prevent issues with browser extensions that change fonts */
14 | font-family: 'icomoon' !important;
15 | speak: none;
16 | font-style: normal;
17 | font-weight: normal;
18 | font-variant: normal;
19 | text-transform: none;
20 | line-height: 1;
21 |
22 | /* Better Font Rendering =========== */
23 | -webkit-font-smoothing: antialiased;
24 | -moz-osx-font-smoothing: grayscale;
25 | }
26 |
27 | .icon-directions_car:before {
28 | content: "\e901";
29 | }
30 | .icon-clock:before {
31 | content: "\e014";
32 | }
33 | .icon-disc:before {
34 | content: "\e019";
35 | }
36 | .icon-map:before {
37 | content: "\e072";
38 | }
39 | .icon-flag:before {
40 | content: "\e108";
41 | }
42 | .icon-trash:before {
43 | content: "\e109";
44 | }
45 | .icon-plus:before {
46 | content: "\e114";
47 | }
48 | .icon-minus:before {
49 | content: "\e115";
50 | }
51 | .icon-cross:before {
52 | content: "\e117";
53 | }
54 |
55 |
--------------------------------------------------------------------------------
/gruntfile.js:
--------------------------------------------------------------------------------
1 | module.exports = function (grunt) {
2 |
3 | grunt.initConfig({
4 | pkg: grunt.file.readJSON('package.json'),
5 | watch: {
6 | options: {
7 | atBegin: true
8 | },
9 | sass: {
10 | files: ['src/scss/**/*.scss'],
11 | tasks: ['sass']
12 | }
13 | },
14 | browserify: {
15 | publish: {
16 | files: {
17 | 'resources/js/routeplanner.js': ['src/js/index.js']
18 | },
19 | options : {
20 | transform : ['reactify']
21 | }
22 | }
23 | },
24 | sass: {
25 | dist: {
26 | files: {
27 | 'resources/css/main.css': 'src/scss/main.scss'
28 | }
29 | }
30 | },
31 | connect: {
32 | server: {
33 | options: {
34 | port: 8888,
35 | keepalive: true
36 | }
37 | },
38 | keepalive: true
39 | }
40 | });
41 |
42 | grunt.loadNpmTasks('grunt-contrib-sass');
43 | grunt.loadNpmTasks('grunt-contrib-watch');
44 | grunt.loadNpmTasks('grunt-contrib-connect');
45 | grunt.loadNpmTasks('grunt-browserify');
46 |
47 | grunt.registerTask('default', ['sass','browserify:publish']);
48 |
49 | }
--------------------------------------------------------------------------------
/src/js/selectors/app.js:
--------------------------------------------------------------------------------
1 | var AppSelectors = {
2 |
3 | recentlyAddedRoute: (state) => {
4 | let routes = state.routes;
5 | if (routes.size > 0) {
6 | return routes.last();
7 | }
8 | },
9 |
10 | firstRoute: (state) => {
11 | let routes = state.routes;
12 | if (routes.size > 0) {
13 | return routes.first();
14 | }
15 | },
16 |
17 | recentlyAddedWayPoint: (state) => {
18 | let wp = state.wayPoints;
19 | if (wp.length > 0) {
20 | return wp[wp.length - 1];
21 | }
22 | },
23 |
24 | activeRouteID: (state) => {
25 | return state.activeRouteID;
26 | },
27 |
28 | activeRoute: (state) => {
29 | if (state.routes) {
30 | let activeRouteID = state.activeRouteID;
31 | return state.routes.find((r) => r.id === activeRouteID);
32 | }
33 | },
34 |
35 | activeWayPoints: (state) => {
36 | let wayPoints = [];
37 | let activeRouteID = state.activeRouteID;
38 | let activeRoute = state.routes.find((r) => r.id === activeRouteID);
39 |
40 | if (activeRoute) {
41 | activeRoute.wayPoints.forEach((wpID) => {
42 | let wayPoint = state.wayPoints.find((wp) => wp.id === wpID);
43 | if (wayPoint) {
44 | wayPoints.push(wayPoint);
45 | }
46 | });
47 | }
48 | return wayPoints;
49 | }
50 |
51 |
52 | };
53 |
54 | module.exports = AppSelectors;
--------------------------------------------------------------------------------
/src/js/actions/types.js:
--------------------------------------------------------------------------------
1 | var ActionTypes = {
2 |
3 | ADD_ROUTE: 'ADD_ROUTE',
4 | ADD_ROUTE_REQUESTED: 'ADD_ROUTE_REQUESTED',
5 | REMOVE_ROUTE: 'REMOVE_ROUTE',
6 | REMOVE_ROUTE_REQUESTED: 'REMOVE_ROUTE_REQUESTED',
7 | CHANGE_ROUTE_NAME: 'CHANGE_ROUTE_NAME',
8 | CHANGE_ACTIVE_ROUTE: 'CHANGE_ACTIVE_ROUTE',
9 | ADD_WAY_POINT_TO_ROUTE: 'ADD_WAY_POINT_TO_ROUTE',
10 | REMOVE_WAY_POINT_FROM_ROUTE: 'REMOVE_WAY_POINT_FROM_ROUTE',
11 |
12 | ADD_WAY_POINT: 'ADD_WAY_POINT',
13 | ADD_WAY_POINT_REQUESTED: 'ADD_WAY_POINT_REQUESTED',
14 | ADD_WAY_POINT_SUCCEEDED: 'ADD_WAY_POINT_SUCCEEDED',
15 | REMOVE_WAY_POINT: 'REMOVE_WAY_POINT',
16 | REMOVE_WAY_POINT_REQUESTED: 'REMOVE_WAY_POINT_REQUESTED',
17 | REMOVE_WAY_POINT_SUCCEEDED: 'REMOVE_WAY_POINT_SUCCEEDED',
18 | CHANGE_WAY_POINT_NAME: 'CHANGE_WAY_POINT_NAME',
19 | CHANGE_WAY_POINT_NAME_REQUESTED: 'CHANGE_WAY_POINT_NAME_REQUESTED',
20 | CHANGE_WAY_POINT_NAME_SUCCEEDED: 'CHANGE_WAY_POINT_NAME_SUCCEEDED',
21 | SET_EDITING_WAY_POINT: 'SET_EDITING_WAY_POINT',
22 | UNSET_EDITING_WAY_POINT: 'UNSET_EDITING_WAY_POINT',
23 | OPEN_WAY_POINT_DETAILS: 'OPEN_WAY_POINT_DETAILS',
24 | CLOSE_WAY_POINT_DETAILS: 'CLOSE_WAY_POINT_DETAILS',
25 |
26 | REFRESH_MAP: 'REFRESH_MAP',
27 | LOAD_MAP: 'LOAD_MAP',
28 |
29 | API_FETCH_ROUTES: 'API_FETCH_ROUTES',
30 | API_FETCH_ROUTES_REQUESTED: 'API_FETCH_ROUTES_REQUESTED',
31 | API_FETCH_ROUTES_SUCCEEDED: 'API_FETCH_ROUTES_SUCCEEDED',
32 | API_FETCH_ROUTES_FAILED: 'API_FETCH_ROUTES_FAILED'
33 | };
34 |
35 | module.exports = ActionTypes;
--------------------------------------------------------------------------------
/src/js/services/api.js:
--------------------------------------------------------------------------------
1 | let mapService;
2 | let directionsService;
3 |
4 | let API = {
5 |
6 | initMapService: (google) => {
7 | mapService = google;
8 | directionsService = new mapService.maps.DirectionsService();
9 | },
10 |
11 | fetchRoutes: (wayPoints) => {
12 | return new Promise((resolve, reject)=> {
13 | if (wayPoints.length < 2) {
14 | resolve();
15 | } else {
16 | let request = {
17 | origin: wayPoints[0].name,
18 | destination: wayPoints[wayPoints.length - 1].name,
19 | travelMode: mapService.maps.TravelMode.DRIVING
20 | };
21 | let wayPointsInBetween = [];
22 | let noOfWayPointsInBetween = wayPoints.length - 2;
23 |
24 | if (noOfWayPointsInBetween > 0) {
25 | for (let i = 1; i <= noOfWayPointsInBetween; i++) {
26 | wayPointsInBetween.push({location: wayPoints[i].name});
27 | }
28 | request.waypoints = wayPointsInBetween;
29 | }
30 |
31 | directionsService.route(request, (response, status) => {
32 | if (status === 'OK') {
33 | resolve({
34 | response,
35 | status
36 | });
37 | } else {
38 | reject(status);
39 | }
40 | });
41 | }
42 | });
43 | }
44 |
45 | };
46 |
47 | module.exports = API;
--------------------------------------------------------------------------------
/src/js/reducers/route.js:
--------------------------------------------------------------------------------
1 | var ActionTypes = require('../actions/types');
2 | var util = require('../util');
3 | var Immutable = require('immutable');
4 |
5 | const route = (state = {}, action) => {
6 |
7 | switch (action.type) {
8 | case ActionTypes.ADD_ROUTE:
9 | return {
10 | id: util.guid(),
11 | name: 'New Route',
12 | wayPoints: Immutable.List()
13 | };
14 | case ActionTypes.ADD_WAY_POINT_TO_ROUTE:
15 | let wayPoints = state.wayPoints;
16 | let pos = wayPoints.findIndex((wp)=> wp === action.tgtWayPoint);
17 | if (pos === -1) {
18 | wayPoints = wayPoints.push(action.newWayPoint);
19 | } else {
20 | wayPoints = wayPoints.insert(pos + 1, action.newWayPoint);
21 | }
22 | return Object.assign({}, state, {
23 | wayPoints: wayPoints
24 | });
25 | case ActionTypes.REMOVE_WAY_POINT_FROM_ROUTE:
26 | return Object.assign({}, state, {
27 | wayPoints: state.wayPoints.filter((wp)=> {
28 | return wp !== action.wayPointID;
29 | })
30 | });
31 | case ActionTypes.CHANGE_ROUTE_NAME:
32 | return Object.assign({}, state, {
33 | name: action.newName
34 | });
35 | case ActionTypes.API_FETCH_ROUTES_SUCCEEDED:
36 | return Object.assign({}, state, {
37 | directions: action.routes
38 | });
39 | default:
40 | return state;
41 | }
42 | };
43 |
44 | module.exports = route;
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "routeplanner",
3 | "version": "2.0.0",
4 | "description": "Route Planner using Google Maps",
5 | "main": "index.js",
6 | "author": "Veera",
7 | "scripts": {
8 | "start": "webpack-dev-server --inline --colors --content-base resources/",
9 | "dev": "npm run start",
10 | "build": "webpack --config ./webpack.config.js",
11 | "build-prod": "rm -rf ./resources && webpack --config ./webpack.prod.js --optimize-minimize ",
12 | "test": "karma start"
13 | },
14 | "devDependencies": {
15 | "babel-core": "6.23.1",
16 | "babel-loader": "6.3.2",
17 | "babel-polyfill": "6.23.0",
18 | "babel-preset-es2015": "6.22.0",
19 | "babel-preset-react": "6.23.0",
20 | "babel-preset-stage-0": "6.22.0",
21 | "css-loader": "0.26.1",
22 | "file-loader": "0.10.0",
23 | "html-loader": "^0.4.4",
24 | "html-webpack-harddisk-plugin": "^0.1.0",
25 | "html-webpack-plugin": "^2.28.0",
26 | "jasmine-core": "^2.5.2",
27 | "karma": "^1.6.0",
28 | "karma-chrome-launcher": "^2.0.0",
29 | "karma-jasmine": "^1.1.0",
30 | "karma-phantomjs-launcher": "^1.0.4",
31 | "karma-webpack": "^2.0.3",
32 | "node-sass": "4.5.0",
33 | "sass-loader": "6.0.1",
34 | "style-loader": "0.13.1",
35 | "url-loader": "0.5.7",
36 | "webpack": "2.2.1",
37 | "webpack-dev-server": "2.4.1"
38 | },
39 | "dependencies": {
40 | "classnames": "2.2.5",
41 | "domready": "1.0.8",
42 | "google-maps": "3.2.1",
43 | "immutable": "3.8.1",
44 | "lodash": "4.17.4",
45 | "react": "15.4.2",
46 | "react-dom": "15.4.2",
47 | "react-redux": "5.0.2",
48 | "redux": "3.6.0",
49 | "redux-saga": "0.14.3"
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/src/js/components/core/text-input.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 | var classNames = require('classnames');
3 |
4 | const KEYS = {
5 | TAB: 9,
6 | ENTER: 13,
7 | ESC: 27
8 | };
9 |
10 | class TextInput extends React.Component {
11 |
12 | constructor(props) {
13 | super(props);
14 | }
15 |
16 | handleKeyDown(e) {
17 | switch (e.which) {
18 | case KEYS.ENTER:
19 | case KEYS.TAB:
20 | this.props.onSave(e.target.value);
21 | break;
22 | case KEYS.ESC:
23 | this.props.onCancel(this.props.oldValue);
24 | default:
25 |
26 | }
27 | }
28 |
29 | handleBlur() {
30 | this.props.onCancelEdit();
31 | }
32 |
33 | focus(input) {
34 | if (this.props.domElm) {
35 | this.props.domElm(input);
36 | }
37 | if (input != null) {
38 | input.focus();
39 | }
40 | }
41 |
42 | render() {
43 | var cx = classNames('text-input', this.props.className);
44 | return (
45 |
46 |
53 |
54 | )
55 | }
56 | }
57 |
58 | TextInput.defaultProps = {
59 | value: '',
60 | onSave: ()=> {
61 | },
62 | onCancel: (oldValue)=> {
63 | }
64 | };
65 |
66 | module.exports = TextInput;
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
1 | let path = require('path');
2 | let HtmlWebpackPlugin = require('html-webpack-plugin');
3 | let HtmlWebpackHarddiskPlugin = require('html-webpack-harddisk-plugin');
4 |
5 | module.exports = {
6 | entry: ['babel-polyfill', './src/js/main.js'],
7 | output: {
8 | path: path.join(__dirname, 'resources', 'js'),
9 | filename: 'bundle.[hash].js',
10 | publicPath: 'resources'
11 | },
12 | module: {
13 | rules: [
14 | {
15 | test: /\.js$/,
16 | exclude: /node_modules/,
17 | use: [
18 | {
19 | loader: 'babel-loader',
20 | options: {
21 | presets: ['es2015', 'react']
22 | }
23 | }
24 | ]
25 | },
26 | {
27 | test: /\.scss$/,
28 | use: ['style-loader', 'css-loader', 'sass-loader']
29 | },
30 | {
31 | test: /\.woff(2)?(\?[a-z0-9]+)?$/,
32 | use: {
33 | loader: "url-loader?limit=10000&mimetype=application/font-woff"
34 | }
35 | }, {
36 | test: /\.(ttf|eot|svg)(\?[a-z0-9]+)?$/,
37 | use: ["file-loader"]
38 | }, {
39 | test: /\.html$/,
40 | use: {
41 | loader: 'html-loader'
42 | }
43 | }
44 | ]
45 | },
46 | plugins: [
47 | new HtmlWebpackPlugin({
48 | template: 'src/views/index.html',
49 | filename: '../index.html',
50 | alwaysWriteToDisk: true
51 | }),
52 | new HtmlWebpackHarddiskPlugin()
53 | ],
54 | devtool: 'source-map'
55 | };
--------------------------------------------------------------------------------
/src/js/components/routes.js:
--------------------------------------------------------------------------------
1 | let React = require('react');
2 | let Link = require('./core/link');
3 | let Route = require('./route');
4 | let Map = require('./map');
5 |
6 |
7 | class Routes extends React.Component {
8 |
9 | shouldComponentUpdate(nextProps) {
10 | return (this.props.routes == nextProps.routes);
11 | }
12 |
13 | handleAddRoute(e) {
14 | e.preventDefault();
15 | this.props.onAdd();
16 | }
17 |
18 | handleChangeRoute(id) {
19 | this.props.onChangeRoute(id);
20 | }
21 |
22 | handleChangeRouteName(routeId, newName) {
23 | this.props.onChangeRouteName(routeId, newName);
24 | }
25 |
26 | renderRouteTabs() {
27 | return (
28 |
29 | {
30 | this.props.routes.map((r, idx) => {
31 | return (
32 | -
33 | {idx + 1}
34 |
35 | )
36 | })
37 | }
38 | -
39 | + Add Route
40 |
41 |
42 | )
43 | }
44 |
45 | render() {
46 |
47 | return (
48 |
49 |
54 |
56 | );
57 | }
58 | }
59 | ;
60 |
61 | module.exports = Routes;
--------------------------------------------------------------------------------
/src/js/containers/way-points.js:
--------------------------------------------------------------------------------
1 | let {connect} = require('react-redux');
2 | let Actions = require('../actions');
3 | let WayPoints = require('../components/way-points');
4 | let AppSelector = require('../selectors/app');
5 |
6 | const getWayPointsToShow = (state) => {
7 | let activeRoute = AppSelector.activeRoute(state);
8 | let activeWayPointsIDs = activeRoute.wayPoints;
9 | let wayPoints = [];
10 |
11 | if (activeWayPointsIDs.size === 0) {
12 | return [];
13 | } else {
14 | activeWayPointsIDs.forEach((wpID) => {
15 | let wp = state.wayPoints.find((wp) => wp.id === wpID);
16 | if (wp) {
17 | wayPoints.push(wp);
18 | }
19 | });
20 |
21 | return wayPoints;
22 | }
23 | }
24 |
25 | const mapStateToProps = (state, ownProps) => {
26 | return {
27 | editingWayPoint: state.editingWayPoint,
28 | wayPoints: getWayPointsToShow(state),
29 | mapService: ownProps.mapService,
30 | route: ownProps.route
31 | }
32 | };
33 |
34 | const mapDispatchToProps = (dispatch, ownProps) => {
35 | return {
36 | onDetailsOpen: (id) => {
37 | dispatch(Actions.openWayPointDetails(id));
38 | },
39 | onDetailsClose: (id) => {
40 | dispatch(Actions.closeWayPointDetails(id));
41 | },
42 | onAdd: (id) => {
43 | dispatch(Actions.addWayPointRequested(id));
44 | },
45 | onRemove: (id) => {
46 | dispatch(Actions.removeWayPointRequested(id));
47 | },
48 | onChangeWayPointName: (wayPointID, newName) => {
49 | dispatch(Actions.changeWayPointNameRequested(wayPointID, newName, ownProps.mapService));
50 | }
51 | }
52 | };
53 |
54 | const WayPointsContainer = connect(
55 | mapStateToProps,
56 | mapDispatchToProps
57 | )(WayPoints);
58 |
59 | module.exports = WayPointsContainer;
--------------------------------------------------------------------------------
/src/js/components/core/editable-text.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 | var classNames = require('classnames');
3 | var TextInput = require('./text-input');
4 |
5 | class EditableText extends React.Component {
6 |
7 | constructor(props) {
8 | super(props);
9 | this.state = {
10 | editing: false
11 | }
12 | }
13 |
14 | componentDidMount() {
15 | if (this.props.edit) {
16 | this.setState({
17 | editing: true
18 | });
19 | }
20 | }
21 |
22 | edit() {
23 | this.setState({
24 | editing: true
25 | });
26 | }
27 |
28 | save(value) {
29 | if (this.props.onSave) {
30 | this.props.onSave(value);
31 | }
32 | this.setState({
33 | editing: false
34 | });
35 | }
36 |
37 | cancel() {
38 | if (this.props.onSave) {
39 | this.props.onSave(this.props.value);
40 | }
41 | this.cancelEdit();
42 | }
43 |
44 | cancelEdit() {
45 | this.setState({
46 | editing: false
47 | });
48 | }
49 |
50 | render() {
51 | var textInputElm;
52 | var cx = classNames('editable-text', this.props.className, {editing: this.state.editing});
53 | var cxLabel = classNames('text-label');
54 |
55 | if (this.state.editing) {
56 | textInputElm =
64 | } else {
65 | textInputElm = {this.props.value}
66 | }
67 |
68 |
69 | return (
70 |
71 | {textInputElm}
72 |
73 | )
74 | }
75 | }
76 |
77 | EditableText.defaultProps = {
78 | value: 'Editable text'
79 | }
80 |
81 | module.exports = EditableText;
--------------------------------------------------------------------------------
/src/js/app.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import createSagaMiddleware from "redux-saga";
3 | import createStorageMiddleware from "./middlewares/state-storage";
4 | import Actions from "./actions";
5 |
6 | let {Provider} = require('react-redux');
7 | let {createStore, applyMiddleware, compose} = require('redux');
8 | let appReducer = require('./reducers/app');
9 | let Routes = require('./containers/routes');
10 |
11 | let rootSagas = require('./sagas');
12 | let APIService = require('./services/api');
13 |
14 | const TOTAL_BACKGROUND_IMAGES = 12;
15 | const DB_NAME = 'appState';
16 |
17 | const sagaMiddleware = createSagaMiddleware();
18 | const storageMiddleware = createStorageMiddleware(DB_NAME);
19 | const composeEnhancers = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose;
20 |
21 | const store = createStore(
22 | appReducer,
23 | composeEnhancers(applyMiddleware(sagaMiddleware, storageMiddleware))
24 | );
25 | sagaMiddleware.run(rootSagas);
26 |
27 | class App extends React.Component {
28 |
29 | componentWillMount() {
30 | APIService.initMapService(this.props.mapService);
31 | this.loadBackground = this.loadBackground.bind(this);
32 | this.applyBackground = this.applyBackground.bind(this);
33 | }
34 |
35 | componentDidMount() {
36 | this.loadBackground();
37 | store.dispatch(Actions.loadMap());
38 | }
39 |
40 | loadBackground() {
41 | let img = new Image();
42 | let location = window.location;
43 | let randomBg = Math.floor(Math.random() * (TOTAL_BACKGROUND_IMAGES - 1 + 1)) + 1;
44 | img.onload = () => {
45 | this.applyBackground(img);
46 | };
47 | img.src = `${location.protocol}//${location.hostname}${location.pathname}/img/bg${randomBg}.jpg`;
48 | }
49 |
50 | applyBackground(img) {
51 | let elm = document.querySelector('.app');
52 | elm.style.backgroundImage = `url(${img.src})`;
53 | }
54 |
55 | render() {
56 | return (
57 |
58 |
59 |
60 |
61 |
62 | )
63 | }
64 | }
65 |
66 | module.exports = App;
--------------------------------------------------------------------------------
/src/js/components/way-point.js:
--------------------------------------------------------------------------------
1 | import React, {Component} from 'react';
2 | import WayPointDetails from "./way-point-details";
3 |
4 | var EditableText = require('./core/editable-text');
5 | var Link = require('./core/link');
6 |
7 | class WayPoint extends Component {
8 |
9 | componentWillUnmount() {
10 | this.autoComplete = null;
11 | }
12 |
13 | cacheWayPointDomElm(elm) {
14 | let google = this.props.mapService;
15 | if (elm) {
16 | this.autoComplete = new google.maps.places.Autocomplete(elm);
17 | this.autoComplete.addListener('place_changed', this.autoCompletePlaceChange.bind(this));
18 | }
19 | }
20 |
21 | handleWayPointNameChange(newName) {
22 | this.props.onNameChange(this.props.wayPoint.id, newName);
23 | }
24 |
25 | autoCompletePlaceChange() {
26 | this.handleWayPointNameChange(this.autoComplete.getPlace().formatted_address);
27 | }
28 |
29 | renderWayPointName() {
30 | return (
31 |
32 |
38 |
39 | )
40 | }
41 |
42 | renderWayPointDetails() {
43 | return
44 | }
45 |
46 | render() {
47 | let {wayPoint} = this.props;
48 | return (
49 |
50 | {this.renderWayPointName()}
51 |
52 | details
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 | {wayPoint.detailsOpen? this.renderWayPointDetails() : null}
61 |
62 | )
63 | }
64 | }
65 |
66 | module.exports = WayPoint;
--------------------------------------------------------------------------------
/src/js/sagas/index.js:
--------------------------------------------------------------------------------
1 | import {takeEvery} from "redux-saga";
2 | import {select, put, call} from "redux-saga/effects";
3 | let Actions = require('../actions');
4 | let ActionTypes = require('../actions/types');
5 | let AppSelectors = require('../selectors/app');
6 | let APIService = require('../services/api');
7 |
8 | function* addRoute() {
9 | yield put(Actions.addRoute());
10 | let newRoute = yield select(AppSelectors.recentlyAddedRoute);
11 | yield put(Actions.changeActiveRoute(newRoute.id));
12 | }
13 |
14 | function* addWayPoint(action) {
15 | yield put(Actions.addWayPoint(action.id));
16 | let newWayPoint = yield select(AppSelectors.recentlyAddedWayPoint);
17 | yield put(Actions.addWayPointToRoute(action.id, newWayPoint.id));
18 | yield put(Actions.setEditingWayPoint(newWayPoint.id));
19 | }
20 |
21 | function* removeWayPoint(action) {
22 | yield put(Actions.removeWayPointFromRoute(action.id));
23 | yield put(Actions.removeWayPoint(action.id));
24 | yield* fetchDirections(action);
25 | }
26 |
27 | function* fetchDirections(action) {
28 | let activeRouteID = yield select(AppSelectors.activeRouteID);
29 | let activeWayPoints = yield select(AppSelectors.activeWayPoints);
30 | let data = yield call(APIService.fetchRoutes, activeWayPoints, action.mapService);
31 | if (data && data.status === 'OK') {
32 | yield put(Actions.API.fetchRoutesSucceeded(activeRouteID, data.response.routes));
33 | yield put(Actions.refreshMap(data.response));
34 | } else {
35 | console.log('fetch direction failed', data && data.status);
36 | }
37 | }
38 |
39 | function* changeWayPointName(action) {
40 | yield put(Actions.changeWayPointName(action.wayPointID, action.newName));
41 | yield put(Actions.unsetEditingWayPoint());
42 | if (action.newName.trim() === '') {
43 | yield* removeWayPoint(Actions.removeWayPointRequested(action.wayPointID));
44 | }
45 | yield* fetchDirections(action);
46 | }
47 |
48 | function* rootSaga() {
49 | yield* [
50 | takeEvery(ActionTypes.LOAD_MAP, fetchDirections),
51 | takeEvery(ActionTypes.ADD_ROUTE_REQUESTED, addRoute),
52 | takeEvery(ActionTypes.ADD_WAY_POINT_REQUESTED, addWayPoint),
53 | takeEvery(ActionTypes.REMOVE_WAY_POINT_REQUESTED, removeWayPoint),
54 | takeEvery(ActionTypes.CHANGE_WAY_POINT_NAME_REQUESTED, changeWayPointName)
55 | ];
56 |
57 | }
58 |
59 | module.exports = rootSaga;
60 |
61 |
--------------------------------------------------------------------------------
/src/scss/base.scss:
--------------------------------------------------------------------------------
1 | @import "reset";
2 | @import "icons";
3 | @import "variables";
4 | @import "text-input";
5 | @import "app";
6 | @import "app-bar";
7 | @import "routes";
8 | @import "route";
9 | @import "route-info";
10 | @import "waypoints";
11 | @import "way-point-info";
12 | @import "way-point-details";
13 |
14 | body {
15 | background-color: #ECF0F1;
16 | font: 1em "Open Sans", Sans-Serif;
17 | color: #333333;
18 | }
19 |
20 | .app {
21 |
22 | }
23 |
24 | .app-name {
25 | font-weight: 300;
26 | font-size: 1rem;
27 | text-align: center;
28 | text-transform: uppercase;
29 | color: #999999;
30 | background-image: url('');
31 | background-repeat: no-repeat;
32 | background-position: top left;
33 | background-size: 20px;
34 | padding-left: 25px;
35 | }
36 |
37 | .author {
38 | margin: $pad 0;
39 | font-size: .8rem;
40 | color: #aaaaaa;
41 | text-align: center;
42 |
43 | a {
44 | &:link, &:hover, &:visited, &:active {
45 | color: #aaaaaa;
46 | }
47 | }
48 | }
49 |
50 | .editable-text {
51 |
52 | .text-input input {
53 | width: 100%;
54 | border: none;
55 | font: 1rem "Open Sans", Sans-Serif;
56 |
57 | &:focus {
58 | outline: none;
59 | }
60 | }
61 |
62 | .text-label {
63 | text-overflow: ellipsis;
64 | white-space: nowrap;
65 | overflow: hidden;
66 | transition: margin-left .2s ease-in-out;
67 |
68 | &:hover {
69 | margin-left: 2 * $pad;
70 | }
71 | }
72 |
73 | }
74 |
75 | .routes .tabs {
76 | margin-bottom: 20px;
77 |
78 | li {
79 | display: inline-block;
80 | margin-right: $pad;
81 | }
82 | }
83 |
84 | .way-point {
85 | position: relative;
86 | padding-left: 30px;
87 |
88 | .way-point-name {
89 | padding-right: 50px;
90 | text-overflow: ellipsis;
91 | }
92 |
93 | .cta {
94 | position: absolute;
95 | top: 50%;
96 | right: 0%;
97 | transform: translateY(-50%);
98 |
99 | a {
100 | text-align: center;
101 | border-bottom: none;
102 | margin-left: 2 * $pad;
103 | display: inline-block;
104 | }
105 | }
106 | }
107 |
108 | @media screen and (min-width: 600px) {
109 | ul.way-points {
110 | max-height: 700px;
111 | overflow: auto;
112 | }
113 | }
114 |
--------------------------------------------------------------------------------
/src/js/reducers/app.js:
--------------------------------------------------------------------------------
1 | import storageService from "../services/storage";
2 | import util from "../util";
3 |
4 | let ActionTypes = require('../actions/types');
5 | let wayPoints = require('./way-points');
6 | let wayPoint = require('./way-point');
7 | let routes = require('./routes');
8 | let route = require('./route');
9 |
10 | let savedState = util.transformSavedState(storageService.get('appState'));
11 |
12 | let defaultRoutes = routes(undefined, ActionTypes.ADD_ROUTE);
13 | let activeRouteID = defaultRoutes.first().id;
14 | let DEFAULTS = {
15 | activeRouteID: activeRouteID,
16 | routes: defaultRoutes
17 | };
18 |
19 | const app = (state = savedState || DEFAULTS, action) => {
20 |
21 | switch (action.type) {
22 | case ActionTypes.ADD_ROUTE:
23 | return Object.assign({}, state, {
24 | routes: routes(state.routes, action)
25 | });
26 | case ActionTypes.CHANGE_ACTIVE_ROUTE:
27 | return Object.assign({}, state, {
28 | activeRouteID: action.id
29 | });
30 | case ActionTypes.ADD_WAY_POINT_TO_ROUTE:
31 | case ActionTypes.REMOVE_WAY_POINT_FROM_ROUTE:
32 | return Object.assign({}, state, {
33 | routes: state.routes.map((r) => {
34 | if (r.id === state.activeRouteID) {
35 | return route(r, action);
36 | }
37 | return r;
38 | })
39 | });
40 | case ActionTypes.API_FETCH_ROUTES_SUCCEEDED:
41 | case ActionTypes.CHANGE_ROUTE_NAME:
42 | return Object.assign({}, state, {
43 | routes: state.routes.map((r) => {
44 | if (r.id === action.routeID) {
45 | return route(r, action);
46 | }
47 | return r;
48 | })
49 | });
50 |
51 | case ActionTypes.ADD_WAY_POINT:
52 | case ActionTypes.REMOVE_WAY_POINT:
53 | return Object.assign({}, state, {
54 | wayPoints: wayPoints(state.wayPoints, action)
55 | });
56 | case ActionTypes.OPEN_WAY_POINT_DETAILS:
57 | case ActionTypes.CLOSE_WAY_POINT_DETAILS:
58 | case ActionTypes.CHANGE_WAY_POINT_NAME:
59 | return Object.assign({}, state, {
60 | wayPoints: state.wayPoints.map((wp) => {
61 | if (wp.id === action.wayPointID) {
62 | return wayPoint(wp, action);
63 | }
64 | return wp;
65 | })
66 | });
67 | case ActionTypes.SET_EDITING_WAY_POINT:
68 | return Object.assign({}, state, {
69 | editingWayPoint: action.wayPointID
70 | });
71 | case ActionTypes.UNSET_EDITING_WAY_POINT:
72 | return Object.assign({}, state, {
73 | editingWayPoint: undefined
74 | });
75 | case ActionTypes.REFRESH_MAP:
76 | return Object.assign({}, state, {
77 | mapData: action.data
78 | });
79 | default:
80 | return state;
81 | }
82 | }
83 |
84 | module.exports = app;
--------------------------------------------------------------------------------
/src/js/components/way-points.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 | var WayPoint = require('./way-point');
3 | var WayPointInfo = require('./way-point-info');
4 | var Link = require('./core/link');
5 | var classnames = require('classnames');
6 | var MAXIMUM_WAY_POINTS = 10;
7 |
8 | class WayPoints extends React.Component {
9 |
10 | handleChangeWayPointName(wayPointID, newName) {
11 | this.props.onChangeWayPointName(wayPointID, newName);
12 | }
13 |
14 | renderWayPointInfo(wpIdx) {
15 | let directions = this.props.route.directions;
16 | if (directions) {
17 | let direction = directions[0];
18 | let legs = direction.legs;
19 | let leg = legs[wpIdx];
20 | if (leg) {
21 | return (
22 |
23 | );
24 | }
25 | }
26 | }
27 |
28 | renderWayPoints() {
29 | let cx = classnames({
30 | 'way-points': true,
31 | 'max-way-points': this.props.wayPoints.length >= MAXIMUM_WAY_POINTS
32 | });
33 | return (
34 |
35 | {
36 | this.props.wayPoints.map((wp, idx) => {
37 | let icon = (idx === 0 || idx === (this.props.wayPoints.length - 1)) ? 'icon-flag' : 'icon-disc';
38 | return (
39 | -
40 |
41 |
51 | {this.renderWayPointInfo(idx)}
52 |
53 | )
54 | })
55 | }
56 | - Add
57 | a place
58 |
59 | )
60 | }
61 |
62 | renderEmptyWayPoints() {
63 | return (
64 |
67 | )
68 | }
69 |
70 |
71 | render() {
72 | let elm;
73 | if (this.props.wayPoints.length > 0) {
74 | elm = this.renderWayPoints();
75 | } else {
76 | elm = this.renderEmptyWayPoints();
77 | }
78 | return (
79 |
80 | {elm}
81 |
82 | )
83 | }
84 | }
85 |
86 | module.exports = WayPoints;
--------------------------------------------------------------------------------
/src/js/actions/index.js:
--------------------------------------------------------------------------------
1 | var ActionTypes = require('./types');
2 |
3 | const saveState = true;
4 |
5 | var Actions = {
6 |
7 | addRoute: () => {
8 | return {
9 | type: ActionTypes.ADD_ROUTE,
10 | saveState
11 | }
12 | },
13 |
14 | addRouteRequested: () => {
15 | return {
16 | type: ActionTypes.ADD_ROUTE_REQUESTED,
17 | }
18 | },
19 |
20 | removeRoute: (id) => {
21 | return {
22 | type: ActionTypes.REMOVE_ROUTE,
23 | id,
24 | saveState
25 | }
26 | },
27 |
28 | removeRouteRequested: (id) => {
29 | return {
30 | type: ActionTypes.REMOVE_ROUTE_REQUESTED,
31 | id
32 | }
33 | },
34 |
35 | changeActiveRoute: (id) => {
36 | return {
37 | type: ActionTypes.CHANGE_ACTIVE_ROUTE,
38 | id
39 | }
40 | },
41 |
42 | changeRouteName: (routeID, newName) => {
43 | return {
44 | type: ActionTypes.CHANGE_ROUTE_NAME,
45 | routeID,
46 | newName
47 | }
48 | },
49 |
50 | addWayPointToRoute: (tgtWayPoint, newWayPoint) => {
51 | return {
52 | type: ActionTypes.ADD_WAY_POINT_TO_ROUTE,
53 | tgtWayPoint,
54 | newWayPoint
55 | }
56 | },
57 |
58 | removeWayPointFromRoute: (wayPointID) => {
59 | return {
60 | type: ActionTypes.REMOVE_WAY_POINT_FROM_ROUTE,
61 | wayPointID
62 | }
63 | },
64 |
65 | addWayPoint: (id) => {
66 | return {
67 | type: ActionTypes.ADD_WAY_POINT,
68 | id
69 | }
70 | },
71 |
72 | addWayPointRequested: (id) => {
73 | return {
74 | type: ActionTypes.ADD_WAY_POINT_REQUESTED,
75 | id
76 | }
77 | },
78 |
79 | removeWayPoint: (id) => {
80 | return {
81 | type: ActionTypes.REMOVE_WAY_POINT,
82 | id,
83 | saveState
84 | }
85 | },
86 |
87 | removeWayPointRequested: (id) => {
88 | return {
89 | type: ActionTypes.REMOVE_WAY_POINT_REQUESTED,
90 | id
91 | }
92 | },
93 |
94 | changeWayPointName: (wayPointID, newName) => {
95 | return {
96 | type: ActionTypes.CHANGE_WAY_POINT_NAME,
97 | wayPointID,
98 | newName,
99 | saveState
100 | }
101 | },
102 |
103 | changeWayPointNameRequested: (wayPointID, newName, mapService) => {
104 | return {
105 | type: ActionTypes.CHANGE_WAY_POINT_NAME_REQUESTED,
106 | wayPointID,
107 | newName,
108 | mapService
109 | }
110 | },
111 |
112 | setEditingWayPoint: (wayPointID) => {
113 | return {
114 | type: ActionTypes.SET_EDITING_WAY_POINT,
115 | wayPointID
116 | }
117 | },
118 |
119 | unsetEditingWayPoint: () => {
120 | return {
121 | type: ActionTypes.UNSET_EDITING_WAY_POINT
122 | }
123 | },
124 |
125 | openWayPointDetails: (wayPointID) => {
126 | return {
127 | type: ActionTypes.OPEN_WAY_POINT_DETAILS,
128 | wayPointID
129 | }
130 | },
131 |
132 | closeWayPointDetails: (wayPointID) => {
133 | return {
134 | type: ActionTypes.CLOSE_WAY_POINT_DETAILS,
135 | wayPointID
136 | }
137 | },
138 |
139 | refreshMap: (data) => {
140 | return {
141 | type: ActionTypes.REFRESH_MAP,
142 | data
143 | }
144 | },
145 |
146 | loadMap: () => {
147 | return {
148 | type: ActionTypes.LOAD_MAP
149 | }
150 | },
151 |
152 | API: {
153 | fetchRoutes: () => {
154 | return {
155 | type: ActionTypes.API_FETCH_ROUTES
156 | }
157 | },
158 |
159 | fetchRoutesSucceeded: (routeID, routes) => {
160 | return {
161 | type: ActionTypes.API_FETCH_ROUTES_SUCCEEDED,
162 | routeID,
163 | routes,
164 | saveState
165 | }
166 | }
167 |
168 | }
169 |
170 | };
171 |
172 | module.exports = Actions;
--------------------------------------------------------------------------------
/src/scss/fonts/icomoon.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
--------------------------------------------------------------------------------
/npm-shrinkwrap.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "routeplanner",
3 | "version": "2.0.0",
4 | "dependencies": {
5 | "asap": {
6 | "version": "2.0.5",
7 | "from": "asap@>=2.0.3 <2.1.0",
8 | "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.5.tgz"
9 | },
10 | "classnames": {
11 | "version": "2.2.5",
12 | "from": "classnames@latest",
13 | "resolved": "https://registry.npmjs.org/classnames/-/classnames-2.2.5.tgz"
14 | },
15 | "domready": {
16 | "version": "1.0.8",
17 | "from": "domready@latest",
18 | "resolved": "https://registry.npmjs.org/domready/-/domready-1.0.8.tgz"
19 | },
20 | "encoding": {
21 | "version": "0.1.12",
22 | "from": "encoding@>=0.1.11 <0.2.0",
23 | "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.12.tgz"
24 | },
25 | "fbjs": {
26 | "version": "0.8.5",
27 | "from": "fbjs@>=0.8.4 <0.9.0",
28 | "resolved": "https://registry.npmjs.org/fbjs/-/fbjs-0.8.5.tgz",
29 | "dependencies": {
30 | "core-js": {
31 | "version": "1.2.7",
32 | "from": "core-js@>=1.0.0 <2.0.0",
33 | "resolved": "https://registry.npmjs.org/core-js/-/core-js-1.2.7.tgz"
34 | }
35 | }
36 | },
37 | "google-maps": {
38 | "version": "3.2.1",
39 | "from": "google-maps@latest",
40 | "resolved": "https://registry.npmjs.org/google-maps/-/google-maps-3.2.1.tgz"
41 | },
42 | "hoist-non-react-statics": {
43 | "version": "1.2.0",
44 | "from": "hoist-non-react-statics@>=1.0.3 <2.0.0",
45 | "resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-1.2.0.tgz"
46 | },
47 | "iconv-lite": {
48 | "version": "0.4.13",
49 | "from": "iconv-lite@>=0.4.13 <0.5.0",
50 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.13.tgz"
51 | },
52 | "immutable": {
53 | "version": "3.8.1",
54 | "from": "https://registry.npmjs.org/immutable/-/immutable-3.8.1.tgz",
55 | "resolved": "https://registry.npmjs.org/immutable/-/immutable-3.8.1.tgz"
56 | },
57 | "invariant": {
58 | "version": "2.2.1",
59 | "from": "invariant@>=2.2.0 <3.0.0",
60 | "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.1.tgz"
61 | },
62 | "is-stream": {
63 | "version": "1.1.0",
64 | "from": "is-stream@>=1.0.1 <2.0.0",
65 | "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz"
66 | },
67 | "isomorphic-fetch": {
68 | "version": "2.2.1",
69 | "from": "isomorphic-fetch@>=2.1.1 <3.0.0",
70 | "resolved": "https://registry.npmjs.org/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz"
71 | },
72 | "lodash": {
73 | "version": "4.16.2",
74 | "from": "https://registry.npmjs.org/lodash/-/lodash-4.16.2.tgz",
75 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.16.2.tgz"
76 | },
77 | "lodash-es": {
78 | "version": "4.16.2",
79 | "from": "lodash-es@>=4.2.1 <5.0.0",
80 | "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.16.2.tgz"
81 | },
82 | "loose-envify": {
83 | "version": "1.2.0",
84 | "from": "loose-envify@>=1.0.0 <2.0.0",
85 | "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.2.0.tgz",
86 | "dependencies": {
87 | "js-tokens": {
88 | "version": "1.0.3",
89 | "from": "js-tokens@>=1.0.1 <2.0.0",
90 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-1.0.3.tgz"
91 | }
92 | }
93 | },
94 | "node-fetch": {
95 | "version": "1.6.3",
96 | "from": "node-fetch@>=1.0.1 <2.0.0",
97 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-1.6.3.tgz"
98 | },
99 | "object-assign": {
100 | "version": "4.1.0",
101 | "from": "object-assign@>=4.0.1 <5.0.0",
102 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.0.tgz"
103 | },
104 | "promise": {
105 | "version": "7.1.1",
106 | "from": "promise@>=7.1.1 <8.0.0",
107 | "resolved": "https://registry.npmjs.org/promise/-/promise-7.1.1.tgz"
108 | },
109 | "react": {
110 | "version": "15.3.2",
111 | "from": "https://registry.npmjs.org/react/-/react-15.3.2.tgz",
112 | "resolved": "https://registry.npmjs.org/react/-/react-15.3.2.tgz"
113 | },
114 | "react-dom": {
115 | "version": "15.3.2",
116 | "from": "react-dom@latest",
117 | "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-15.3.2.tgz"
118 | },
119 | "react-redux": {
120 | "version": "4.4.5",
121 | "from": "react-redux@latest",
122 | "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-4.4.5.tgz"
123 | },
124 | "redux": {
125 | "version": "3.6.0",
126 | "from": "redux@latest",
127 | "resolved": "https://registry.npmjs.org/redux/-/redux-3.6.0.tgz"
128 | },
129 | "redux-saga": {
130 | "version": "0.12.0",
131 | "from": "redux-saga@latest",
132 | "resolved": "https://registry.npmjs.org/redux-saga/-/redux-saga-0.12.0.tgz"
133 | },
134 | "symbol-observable": {
135 | "version": "1.0.2",
136 | "from": "symbol-observable@>=1.0.2 <2.0.0",
137 | "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.0.2.tgz"
138 | },
139 | "ua-parser-js": {
140 | "version": "0.7.10",
141 | "from": "ua-parser-js@>=0.7.9 <0.8.0",
142 | "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.10.tgz"
143 | },
144 | "whatwg-fetch": {
145 | "version": "1.0.0",
146 | "from": "whatwg-fetch@>=0.10.0",
147 | "resolved": "https://registry.npmjs.org/whatwg-fetch/-/whatwg-fetch-1.0.0.tgz"
148 | }
149 | }
150 | }
151 |
--------------------------------------------------------------------------------
/src/js/components/route-info.js:
--------------------------------------------------------------------------------
1 | let React = require('react');
2 | import RouteVisual from './route-visual';
3 |
4 | let MINUTE = 60;
5 | let HOUR = 3600;
6 | let DAY = 86400;
7 |
8 | let toMiles = (meters) => meters * 0.000621371;
9 | let ALT_COLORS = ['#E01931', '#FEC606'];
10 |
11 | class RouteInfo extends React.Component {
12 |
13 | constructor(props) {
14 | super(props);
15 | }
16 |
17 | getCanvas() {
18 |
19 | let canvasElm = this.canvasElm;
20 | let ctx = this.ctx;
21 |
22 | let cW = canvasElm.width;
23 | let cH = canvasElm.height;
24 | let midW = cW / 2;
25 | let midH = cH / 2;
26 | let pad = 10;
27 | let env = {
28 | cW,
29 | cH,
30 | midW,
31 | midH,
32 | pad
33 | };
34 |
35 | return {
36 | ctx,
37 | env
38 | };
39 | }
40 |
41 | drawDistanceSegments(ctx, env, totalDistance = 0, segments = []) {
42 |
43 | let startX = 0;
44 | let barH = 10;
45 | let startY = env.cH - barH;
46 | let barW = env.cW;
47 |
48 | ctx.save();
49 | if (segments.length <= 1) {
50 | ctx.fillStyle = ALT_COLORS[0];
51 | ctx.fillRect(startX, startY, barW, barH);
52 | } else {
53 | let i;
54 | ctx.clearRect(0, startY, barW, barH);
55 |
56 | for (i = 0; i < segments.length - 1; i++) {
57 | let segX = startX;
58 | let segY = startY;
59 | let segW = Math.floor((barW * segments[i]) / totalDistance);
60 | ctx.fillStyle = ALT_COLORS[i % 2 === 0 ? 0 : 1];
61 | ctx.fillRect(segX, segY, segW, barH);
62 | startX += segW;
63 | }
64 | ctx.fillStyle = ALT_COLORS[i % 2 === 0 ? 0 : 1];
65 | ctx.fillRect(startX, startY, (barW - startX), barH);
66 |
67 | }
68 | }
69 |
70 | renderDistance(distance = 0) {
71 | return (
72 |
73 |
Distance
74 |
{distance > 0 ? (toMiles(distance).toFixed(2)) : 0}
75 |
{distance <= 1 ? 'mile' : 'miles'}
76 |
77 | )
78 | }
79 |
80 | renderDuration(duration = 0) {
81 | let time = {
82 | days: 0,
83 | hours: 0,
84 | minutes: 0
85 | };
86 | let elm;
87 | let dayElm;
88 | let hourElm;
89 | let minuteElm;
90 |
91 | if (duration === 0) {
92 | elm = (
93 | {0}
94 | hour
95 | );
96 | } else {
97 | // map duration to days / hours / minutes
98 | if (duration >= DAY) {
99 | time.days = Math.floor(duration / DAY);
100 | duration = duration % DAY;
101 | dayElm = (
102 | {time.days}
103 | {time.days <= 1 ? 'day' : 'days'}
104 | );
105 | }
106 |
107 | if (duration >= HOUR) {
108 | time.hours = Math.floor(duration / HOUR);
109 | duration = duration % HOUR;
110 | hourElm = (
111 | {time.hours}
112 | {time.hours <= 1 ? 'hour' : 'hours'}
113 | );
114 | }
115 |
116 | if (duration >= MINUTE) {
117 | time.minutes = Math.floor(duration / MINUTE);
118 | duration = duration % MINUTE;
119 | minuteElm = (
120 | {time.minutes}
121 | {time.minutes <= 0 ? 'minute' : 'minutes'}
122 | );
123 | }
124 |
125 | }
126 |
127 |
128 | return (
129 |
130 |
Duration
131 |
132 | {elm}
133 | {dayElm}
134 | {hourElm}
135 | {minuteElm}
136 |
137 |
138 | )
139 | }
140 |
141 | extractDisplayData(props) {
142 | let source = props || this.props;
143 | if (source.route) {
144 | let directions = source.route.directions;
145 | let distances = [];
146 | let totalDistance = 0, totalDuration = 0, direction, legs, leg;
147 | if (directions) {
148 | direction = directions[0];
149 | legs = direction.legs;
150 | for (let l = 0; l < legs.length; l++) {
151 | let leg = legs[l];
152 | distances.push(leg.distance.value);
153 | totalDistance += leg.distance.value;
154 | totalDuration += leg.duration.value;
155 | }
156 | }
157 | return {
158 | distances,
159 | totalDistance,
160 | totalDuration
161 | };
162 | }
163 | return {};
164 | }
165 |
166 | render() {
167 | let data = this.extractDisplayData();
168 | return (
169 |
170 |
171 | {this.renderDistance(data.totalDistance)}
172 | {this.renderDuration(data.totalDuration)}
173 |
174 |
175 |
176 | )
177 | }
178 | }
179 |
180 | module.exports = RouteInfo;
--------------------------------------------------------------------------------
/src/js/components/route-visual.js:
--------------------------------------------------------------------------------
1 | import React, {Component} from 'react';
2 |
3 | const COLORS = [
4 | '#4A148C',
5 | '#F57F17',
6 | '#880E4F',
7 | '#827717',
8 | '#B71C1C',
9 | '#33691E',
10 | '#1A237E',
11 | '#E65100',
12 | '#3E2723',
13 | '#01579B',
14 | '#212121',
15 | '#006064'
16 | ];
17 |
18 | class RouteVisual extends Component {
19 |
20 | renderRoute(total, segments) {
21 | let width = 350,
22 | height = 40;
23 | let mappedSegments = segments.map((seg) => (width * seg) / total);
24 | let legX = 0;
25 | let markX = ( (width / 2) - width);
26 | return (
27 |
99 | )
100 | }
101 |
102 | render() {
103 | let {data} = this.props;
104 | let {totalDistance, distances} = data;
105 |
106 | // totalDistance = 5000;
107 | // distances = [2500, 2500];
108 |
109 | return !totalDistance ? null : (
110 |
111 | {this.renderRoute(totalDistance, distances)}
112 |
113 | )
114 | }
115 | }
116 |
117 | export default RouteVisual;
--------------------------------------------------------------------------------
/src/js/components/map.js:
--------------------------------------------------------------------------------
1 | var React = require('react');
2 | var ReactDOM = require('react-dom');
3 |
4 | class Map extends React.Component {
5 |
6 | componentDidMount() {
7 | this.initMap(ReactDOM.findDOMNode(this));
8 | }
9 |
10 | componentWillReceiveProps(nextProps) {
11 | if (this.directionsDisplay !== undefined && nextProps.mapData !== undefined) {
12 | this.directionsDisplay.setDirections(nextProps.mapData);
13 | }
14 | }
15 |
16 | initMap(elm) {
17 | if (!!elm && !this.map) {
18 | let google = this.props.mapService;
19 | this.directionsDisplay = new google.maps.DirectionsRenderer;
20 | this.map = new google.maps.Map(elm, {
21 | center: {lat: -34.397, lng: 150.644},
22 | zoom: 8,
23 | styles: [
24 | {
25 | "elementType": "geometry",
26 | "stylers": [
27 | {
28 | "color": "#ebe3cd"
29 | }
30 | ]
31 | },
32 | {
33 | "elementType": "labels.text.fill",
34 | "stylers": [
35 | {
36 | "color": "#523735"
37 | }
38 | ]
39 | },
40 | {
41 | "elementType": "labels.text.stroke",
42 | "stylers": [
43 | {
44 | "color": "#f5f1e6"
45 | }
46 | ]
47 | },
48 | {
49 | "featureType": "administrative",
50 | "elementType": "geometry.stroke",
51 | "stylers": [
52 | {
53 | "color": "#c9b2a6"
54 | }
55 | ]
56 | },
57 | {
58 | "featureType": "administrative.land_parcel",
59 | "elementType": "geometry.stroke",
60 | "stylers": [
61 | {
62 | "color": "#dcd2be"
63 | }
64 | ]
65 | },
66 | {
67 | "featureType": "administrative.land_parcel",
68 | "elementType": "labels.text.fill",
69 | "stylers": [
70 | {
71 | "color": "#ae9e90"
72 | }
73 | ]
74 | },
75 | {
76 | "featureType": "landscape.natural",
77 | "elementType": "geometry",
78 | "stylers": [
79 | {
80 | "color": "#dfd2ae"
81 | }
82 | ]
83 | },
84 | {
85 | "featureType": "poi",
86 | "elementType": "geometry",
87 | "stylers": [
88 | {
89 | "color": "#dfd2ae"
90 | }
91 | ]
92 | },
93 | {
94 | "featureType": "poi",
95 | "elementType": "labels.text.fill",
96 | "stylers": [
97 | {
98 | "color": "#93817c"
99 | }
100 | ]
101 | },
102 | {
103 | "featureType": "poi.park",
104 | "elementType": "geometry.fill",
105 | "stylers": [
106 | {
107 | "color": "#a5b076"
108 | }
109 | ]
110 | },
111 | {
112 | "featureType": "poi.park",
113 | "elementType": "labels.text.fill",
114 | "stylers": [
115 | {
116 | "color": "#447530"
117 | }
118 | ]
119 | },
120 | {
121 | "featureType": "road",
122 | "elementType": "geometry",
123 | "stylers": [
124 | {
125 | "color": "#f5f1e6"
126 | }
127 | ]
128 | },
129 | {
130 | "featureType": "road.arterial",
131 | "elementType": "geometry",
132 | "stylers": [
133 | {
134 | "color": "#fdfcf8"
135 | }
136 | ]
137 | },
138 | {
139 | "featureType": "road.highway",
140 | "elementType": "geometry",
141 | "stylers": [
142 | {
143 | "color": "#f8c967"
144 | }
145 | ]
146 | },
147 | {
148 | "featureType": "road.highway",
149 | "elementType": "geometry.stroke",
150 | "stylers": [
151 | {
152 | "color": "#e9bc62"
153 | }
154 | ]
155 | },
156 | {
157 | "featureType": "road.highway.controlled_access",
158 | "elementType": "geometry",
159 | "stylers": [
160 | {
161 | "color": "#e98d58"
162 | }
163 | ]
164 | },
165 | {
166 | "featureType": "road.highway.controlled_access",
167 | "elementType": "geometry.stroke",
168 | "stylers": [
169 | {
170 | "color": "#db8555"
171 | }
172 | ]
173 | },
174 | {
175 | "featureType": "road.local",
176 | "elementType": "labels.text.fill",
177 | "stylers": [
178 | {
179 | "color": "#806b63"
180 | }
181 | ]
182 | },
183 | {
184 | "featureType": "transit.line",
185 | "elementType": "geometry",
186 | "stylers": [
187 | {
188 | "color": "#dfd2ae"
189 | }
190 | ]
191 | },
192 | {
193 | "featureType": "transit.line",
194 | "elementType": "labels.text.fill",
195 | "stylers": [
196 | {
197 | "color": "#8f7d77"
198 | }
199 | ]
200 | },
201 | {
202 | "featureType": "transit.line",
203 | "elementType": "labels.text.stroke",
204 | "stylers": [
205 | {
206 | "color": "#ebe3cd"
207 | }
208 | ]
209 | },
210 | {
211 | "featureType": "transit.station",
212 | "elementType": "geometry",
213 | "stylers": [
214 | {
215 | "color": "#dfd2ae"
216 | }
217 | ]
218 | },
219 | {
220 | "featureType": "water",
221 | "elementType": "geometry.fill",
222 | "stylers": [
223 | {
224 | "color": "#b9d3c2"
225 | }
226 | ]
227 | },
228 | {
229 | "featureType": "water",
230 | "elementType": "labels.text.fill",
231 | "stylers": [
232 | {
233 | "color": "#92998d"
234 | }
235 | ]
236 | }
237 | ]
238 | });
239 | this.directionsDisplay.setMap(this.map);
240 | }
241 | }
242 |
243 | render() {
244 | return (
245 |
246 |
247 | )
248 | }
249 | }
250 |
251 | module.exports = Map;
--------------------------------------------------------------------------------