|
1 |
| -import unused from './es6-shims'; // eslint-disable-line |
| 1 | +import './es6-shims'; |
2 | 2 | import Rx from 'rx';
|
3 | 3 | import React from 'react';
|
4 |
| -import Fetchr from 'fetchr'; |
5 |
| -import debugFactory from 'debug'; |
| 4 | +import debug from 'debug'; |
6 | 5 | import { Router } from 'react-router';
|
| 6 | +import { routeReducer as routing, syncHistory } from 'react-router-redux'; |
7 | 7 | import { createLocation, createHistory } from 'history';
|
8 |
| -import { hydrate } from 'thundercats'; |
9 |
| -import { render$ } from 'thundercats-react'; |
10 | 8 |
|
11 | 9 | 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'; |
14 | 11 |
|
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'); |
16 | 19 | 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' }; |
21 | 23 |
|
22 | 24 | Rx.config.longStackSupport = !!debug.enabled;
|
23 | 25 | const history = createHistory();
|
24 | 26 | const appLocation = createLocation(
|
25 | 27 | location.pathname + location.search
|
26 | 28 | );
|
| 29 | +const routingMiddleware = syncHistory(history); |
27 | 30 |
|
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; |
55 | 33 |
|
56 |
| - // set page title |
57 |
| - appStore$ |
58 |
| - .pluck('title') |
59 |
| - .distinctUntilChanged() |
60 |
| - .doOnNext(title => document.title = title) |
61 |
| - .subscribe(() => {}); |
| 34 | +const clientSagaOptions = { doc: document }; |
62 | 35 |
|
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 }) => { |
75 | 50 |
|
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 |
81 | 54 | props.history = history;
|
82 | 55 |
|
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), |
86 | 64 | DOMContianer
|
87 | 65 | );
|
88 | 66 | })
|
89 | 67 | .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') |
99 | 71 | );
|
0 commit comments