Skip to content

Commit 8ef3fdb

Browse files
Berkeley MartinezBerkeley Martinez
Berkeley Martinez
authored and
Berkeley Martinez
committed
Initial move to redux
1 parent 2863efe commit 8ef3fdb

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

67 files changed

+1511
-651
lines changed

.eslintrc

+1-1
Original file line numberDiff line numberDiff line change
@@ -232,7 +232,7 @@
232232
"react/jsx-uses-vars": 1,
233233
"react/no-did-mount-set-state": 2,
234234
"react/no-did-update-set-state": 2,
235-
"react/no-multi-comp": 2,
235+
"react/no-multi-comp": [2, { "ignoreStateless": true } ],
236236
"react/prop-types": 2,
237237
"react/react-in-jsx-scope": 1,
238238
"react/self-closing-comp": 1,

client/err-saga.js

-9
This file was deleted.

client/history-saga.js

-69
This file was deleted.

client/index.js

+46-74
Original file line numberDiff line numberDiff line change
@@ -1,99 +1,71 @@
1-
import unused from './es6-shims'; // eslint-disable-line
1+
import './es6-shims';
22
import Rx from 'rx';
33
import React from 'react';
4-
import Fetchr from 'fetchr';
5-
import debugFactory from 'debug';
4+
import debug from 'debug';
65
import { Router } from 'react-router';
6+
import { routeReducer as routing, syncHistory } from 'react-router-redux';
77
import { createLocation, createHistory } from 'history';
8-
import { hydrate } from 'thundercats';
9-
import { render$ } from 'thundercats-react';
108

119
import app$ from '../common/app';
12-
import historySaga from './history-saga';
13-
import errSaga from './err-saga';
10+
import provideStore from '../common/app/provide-store';
1411

15-
const debug = debugFactory('fcc:client');
12+
// client specific sagas
13+
import sagas from './sagas';
14+
15+
// render to observable
16+
import render from '../common/app/utils/render';
17+
18+
const log = debug('fcc:client');
1619
const DOMContianer = document.getElementById('fcc');
17-
const catState = window.__fcc__.data || {};
18-
const services = new Fetchr({
19-
xhrPath: '/services'
20-
});
20+
const initialState = window.__fcc__.data;
21+
22+
const serviceOptions = { xhrPath: '/services' };
2123

2224
Rx.config.longStackSupport = !!debug.enabled;
2325
const history = createHistory();
2426
const appLocation = createLocation(
2527
location.pathname + location.search
2628
);
29+
const routingMiddleware = syncHistory(history);
2730

28-
// returns an observable
29-
app$({ history, location: appLocation })
30-
.flatMap(
31-
({ AppCat }) => {
32-
// instantiate the cat with service
33-
const appCat = AppCat(null, services, history);
34-
// hydrate the stores
35-
return hydrate(appCat, catState).map(() => appCat);
36-
},
37-
// not using nextLocation at the moment but will be used for
38-
// redirects in the future
39-
({ nextLocation, props }, appCat) => ({ nextLocation, props, appCat })
40-
)
41-
.doOnNext(({ appCat }) => {
42-
const appStore$ = appCat.getStore('appStore');
43-
44-
const {
45-
toast,
46-
updateLocation,
47-
goTo,
48-
goBack
49-
} = appCat.getActions('appActions');
50-
51-
52-
const routerState$ = appStore$
53-
.map(({ location }) => location)
54-
.filter(location => !!location);
31+
const devTools = window.devToolsExtension ? window.devToolsExtension() : f => f;
32+
const shouldRouterListenForReplays = !!window.devToolsExtension;
5533

56-
// set page title
57-
appStore$
58-
.pluck('title')
59-
.distinctUntilChanged()
60-
.doOnNext(title => document.title = title)
61-
.subscribe(() => {});
34+
const clientSagaOptions = { doc: document };
6235

63-
historySaga(
64-
history,
65-
updateLocation,
66-
goTo,
67-
goBack,
68-
routerState$
69-
);
70-
71-
const err$ = appStore$
72-
.pluck('err')
73-
.filter(err => !!err)
74-
.distinctUntilChanged();
36+
// returns an observable
37+
app$({
38+
location: appLocation,
39+
history,
40+
serviceOptions,
41+
initialState,
42+
middlewares: [
43+
routingMiddleware,
44+
...sagas.map(saga => saga(clientSagaOptions))
45+
],
46+
reducers: { routing },
47+
enhancers: [ devTools ]
48+
})
49+
.flatMap(({ props, store }) => {
7550

76-
errSaga(err$, toast);
77-
})
78-
// allow store subscribe to subscribe to actions
79-
.delay(10)
80-
.flatMap(({ props, appCat }) => {
51+
// because of weirdness in react-routers match function
52+
// we replace the wrapped returned in props with the first one
53+
// we passed in. This might be fixed in react-router 2.0
8154
props.history = history;
8255

83-
return render$(
84-
appCat,
85-
React.createElement(Router, props),
56+
if (shouldRouterListenForReplays && store) {
57+
log('routing middleware listening for replays');
58+
routingMiddleware.listenForReplays(store);
59+
}
60+
61+
log('rendering');
62+
return render(
63+
provideStore(React.createElement(Router, props), store),
8664
DOMContianer
8765
);
8866
})
8967
.subscribe(
90-
() => {
91-
debug('react rendered');
92-
},
93-
err => {
94-
throw err;
95-
},
96-
() => {
97-
debug('react closed subscription');
98-
}
68+
() => debug('react rendered'),
69+
err => { throw err; },
70+
() => debug('react closed subscription')
9971
);

client/sagas/README.md

Whitespace-only changes.

client/sagas/err-saga.js

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// () =>
2+
// (store: Store) =>
3+
// (next: (action: Action) => Object) =>
4+
// errSaga(action: Action) => Object|Void
5+
export default () => ({ dispatch }) => next => {
6+
return function errorSaga(action) {
7+
if (!action.error) { return next(action); }
8+
9+
console.error(action.error);
10+
dispatch({
11+
type: 'app.makeToast',
12+
payload: {
13+
type: 'error',
14+
title: 'Oops, something went wrong',
15+
message: `Something went wrong, please try again later`
16+
}
17+
});
18+
};
19+
};

client/sagas/index.js

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
import errSaga from './err-saga';
2+
import titleSaga from './title-saga';
3+
4+
export default [errSaga, titleSaga];

client/sagas/title-saga.js

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
// (doc: Object) =>
2+
// () =>
3+
// (next: (action: Action) => Object) =>
4+
// titleSage(action: Action) => Object|Void
5+
export default (doc) => () => next => {
6+
return function titleSage(action) {
7+
// get next state
8+
const result = next(action);
9+
if (action !== 'updateTitle') {
10+
return result;
11+
}
12+
const newTitle = result.app.title;
13+
doc.title = newTitle;
14+
return result;
15+
};
16+
};

0 commit comments

Comments
 (0)