Skip to content

Commit 0f90799

Browse files
committed
[added] react-hot-loader when developing docs
+ Enable webpack-hot-loader for css changes + Enable react-hot-loader for component changes with state retention. + Better fail over when nodemon process fail + If you change the webpack configuration this will restart the webpack-dev-server. + Video link to demo this feature in docs readme.
1 parent fdf0ec2 commit 0f90799

17 files changed

+140
-50
lines changed

docs/.eslintrc

+3
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,8 @@
33
"comma-spacing": 0,
44
"react/no-multi-comp": 0,
55
"react/prop-types": 0
6+
},
7+
"globals": {
8+
"CodeMirror": true
69
}
710
}

docs/README.md

+17-1
Original file line numberDiff line numberDiff line change
@@ -11,10 +11,26 @@ Pages](http://pages.github.com/).
1111
From the repository root run `npm run docs` and navigate your browser to
1212
`http://localhost:4000`. This will start an express base node server with
1313
webpack-dev middleware that will watch your file changes and recompile all the
14-
static assets needed to generate the site.
14+
static assets needed to generate the site. In the console output you'll see that
15+
we bind to two ports. The first port is the one you'll use to load the docs in
16+
your browser. The second is the webpack-dev-server we use to build the client
17+
side assets in watch mode. _Note: while the docs should start on port 4000 if
18+
that port is in use we progressively look for an available port. Observe
19+
console output for the actual port that we use._ We use the
20+
[webpack][webpack-hot] and [react][react-hot] hot loading functionality to allow
21+
your development experience to have quickest feedback loop possible.
22+
23+
For a demo of how the hot loader works checkout this video:
24+
25+
<a href="http://www.youtube.com/watch?feature=player_embedded&v=vViVUbyAWeY
26+
" target="_blank"><img src="http://img.youtube.com/vi/vViVUbyAWeY/0.jpg"
27+
alt="Demo of hot loader" width="240" height="180" border="10" /></a>
1528

1629
## Production
1730

1831
This site is statically published on github pages, to do this the static assets
1932
need to be generated. You can simulate a similar experience with `npm run
2033
docs-prod` and navigating your browser to `http://localhost:4000`
34+
35+
[webpack-hot]: http://webpack.github.io/docs/hot-module-replacement-with-webpack.html
36+
[react-hot]: http://gaearon.github.io/react-hot-loader/
File renamed without changes.

docs/client.js

+8
Original file line numberDiff line numberDiff line change
@@ -6,10 +6,18 @@ import './assets/carousel.png';
66
import './assets/logo.png';
77
import './assets/favicon.ico';
88

9+
import 'codemirror/mode/javascript/javascript';
10+
import 'codemirror/theme/solarized.css';
11+
import 'codemirror/lib/codemirror.css';
12+
import './assets/CodeMirror.css';
13+
914
import React from 'react';
15+
import CodeMirror from 'codemirror';
1016
import Router from 'react-router';
1117
import routes from './src/Routes';
1218

19+
global.CodeMirror = CodeMirror;
20+
1321
Router.run(routes, Router.RefreshLocation, Handler => {
1422
React.render(
1523
React.createElement(Handler, window.INITIAL_PROPS), document);

docs/dev-run

+33-33
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,7 @@ import { exec } from 'child-process-promise';
77
portfinder.basePort = 4000;
88

99
const SIGINT = 'SIGINT';
10-
let webpackDevServer;
11-
let docsServer;
10+
let processMap = {};
1211

1312
function output(prefix, message) {
1413
let formattedMessage = message.trim().split('\n')
@@ -23,22 +22,37 @@ function listen({stdout, stderr}, name) {
2322
}
2423

2524
function shutdown() {
26-
if (webpackDevServer) {
27-
webpackDevServer.kill(SIGINT);
28-
}
29-
if (docsServer) {
30-
docsServer.kill(SIGINT);
25+
for (let key in processMap) {
26+
processMap[key].kill(SIGINT);
3127
}
3228
}
3329

3430
function catchExec(name, err) {
3531
if (err.killed) {
3632
console.log('Shutdown: '.cyan + name.green);
37-
} else {
38-
console.log(`${name} -- Failed`.red);
39-
console.log(err.toString().red);
33+
shutdown();
34+
return false;
4035
}
41-
shutdown();
36+
37+
console.log(`${name} -- Failed`.red);
38+
console.log(err.toString().red);
39+
return true;
40+
}
41+
42+
function runCmd(name, cmd, options) {
43+
exec(cmd, options)
44+
.progress(childProcess => {
45+
listen(childProcess, name);
46+
processMap[name] = childProcess;
47+
return;
48+
})
49+
.then(() => console.log('Shutdown: '.cyan + name.green))
50+
.catch(err => {
51+
if (catchExec(name, err)) {
52+
// Restart if not explicitly shutdown
53+
runCmd(name, cmd, options);
54+
}
55+
});
4256
}
4357

4458
console.log('Starting docs in Development mode'.cyan);
@@ -51,28 +65,14 @@ portfinder.getPorts(2, {}, (portFinderErr, [docsPort, webpackPort]) => {
5165
process.exit(1);
5266
}
5367

54-
exec(`webpack-dev-server --quiet --config webpack.docs.js --color --port ${webpackPort}`)
55-
.progress(childProcess => {
56-
listen(childProcess, 'webpack-dev-server');
57-
webpackDevServer = childProcess;
58-
return;
59-
})
60-
.then(() => console.log('Shutdown: '.cyan + 'webpack-dev-server'.green))
61-
.catch(err => catchExec('webpack-dev-server', err));
68+
runCmd('webpack-dev-server', `nodemon --watch webpack --watch webpack.config.js --watch node_modules --exec webpack-dev-server -- --config webpack.docs.js --color --port ${webpackPort} --debug --hot`);
6269

63-
exec('nodemon --exec babel-node docs/server.js', {
64-
env: {
65-
PORT: docsPort,
66-
WEBPACK_DEV_PORT: webpackPort,
67-
...process.env
68-
}
69-
})
70-
.progress(childProcess => {
71-
listen(childProcess, 'docs-server');
72-
docsServer = childProcess;
73-
return;
74-
})
75-
.then(() => console.log('Shutdown: '.cyan + 'docs-server'.green))
76-
.catch(err => catchExec('docs-server', err));
70+
runCmd('docs-server', 'nodemon --watch docs --watch src --watch node_modules --exec babel-node docs/server.js', {
71+
env: {
72+
PORT: docsPort,
73+
WEBPACK_DEV_PORT: webpackPort,
74+
...process.env
75+
}
76+
});
7777
});
7878

docs/server.js

+4-1
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,11 @@ if (development) {
2121
});
2222

2323
app.use(function renderApp(req, res) {
24+
res.header('Access-Control-Allow-Origin', target);
25+
res.header('Access-Control-Allow-Headers', 'X-Requested-With');
26+
2427
Router.run(routes, req.url, Handler => {
25-
let html = React.renderToString(<Handler />);
28+
let html = React.renderToString(<Handler assetBaseUrl={target} />);
2629
res.send(html);
2730
});
2831
});

docs/src/CodeMirror.client.js docs/src/CodeMirrorWrapper.client.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import 'codemirror/mode/javascript/javascript';
33

44
import 'codemirror/theme/solarized.css';
55
import 'codemirror/lib/codemirror.css';
6-
import './CodeMirror.css';
6+
import './CodeMirrorWrapper.css';
77

88
export default {
99
IS_NODE: false,
File renamed without changes.

docs/src/HomePage.js

+3-5
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ import Grid from '../../src/Grid';
66
import Alert from '../../src/Alert';
77
import Glyphicon from '../../src/Glyphicon';
88

9-
const HomePage = React.createClass({
10-
render: function () {
9+
export default class HomePage extends React.Component{
10+
render() {
1111
return (
1212
<div>
1313
<NavMain activePage="home" />
@@ -34,6 +34,4 @@ const HomePage = React.createClass({
3434
</div>
3535
);
3636
}
37-
});
38-
39-
export default HomePage;
37+
}

docs/src/ReactPlayground.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,6 @@ import * as modTabPane from '../../src/TabPane';
4242
import * as modTooltip from '../../src/Tooltip';
4343
import * as modWell from '../../src/Well';
4444

45-
import {CodeMirror, IS_NODE} from './CodeMirror';
4645
import babel from 'babel-core/browser';
4746

4847
const classNames = modClassNames.default;
@@ -103,7 +102,7 @@ const IS_MOBILE = typeof navigator !== 'undefined' && (
103102

104103
const CodeMirrorEditor = React.createClass({
105104
componentDidMount() {
106-
if (IS_MOBILE || IS_NODE) {
105+
if (IS_MOBILE || CodeMirror === undefined) {
107106
return;
108107
}
109108

docs/src/Root.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,12 @@ const Root = React.createClass({
4343
}
4444
},
4545

46+
getDefaultProps() {
47+
return {
48+
assetBaseUrl: ''
49+
};
50+
},
51+
4652
render() {
4753
// Dump out our current props to a global object via a script tag so
4854
// when initialising the browser environment we can bootstrap from the
@@ -66,8 +72,8 @@ const Root = React.createClass({
6672
__html: `<title>React Bootstrap</title>
6773
<meta http-equiv='X-UA-Compatible' content='IE=edge' />
6874
<meta name='viewport' content='width=device-width, initial-scale=1.0' />
69-
<link href='assets/bundle.css' rel='stylesheet' />
70-
<link href='assets/favicon.ico' type='image/x-icon' rel='icon' />
75+
<link href='${this.props.assetBaseUrl}/assets/bundle.css' rel='stylesheet' />
76+
<link href='${this.props.assetBaseUrl}/assets/favicon.ico' type='image/x-icon' rel='icon' />
7177
<!--[if lt IE 9]>
7278
<script src='https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js'></script>
7379
<script src='https://oss.maxcdn.com/respond/1.4.2/respond.min.js'></script>
@@ -85,7 +91,7 @@ const Root = React.createClass({
8591
<Router.RouteHandler />
8692

8793
<script dangerouslySetInnerHTML={browserInitScriptObj} />
88-
<script src='assets/bundle.js' />
94+
<script src={`${this.props.assetBaseUrl}/assets/bundle.js`} />
8995
</body>
9096
</html>
9197
);

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -41,7 +41,6 @@
4141
"brfs": "^1.4.0",
4242
"chai": "^2.2.0",
4343
"child-process-promise": "^1.0.1",
44-
"client-loader": "0.0.1",
4544
"codemirror": "^5.0.0",
4645
"colors": "^1.0.3",
4746
"css-loader": "^0.12.0",
@@ -73,6 +72,7 @@
7372
"nodemon": "^1.3.7",
7473
"portfinder": "^0.4.0",
7574
"react": "^0.13.1",
75+
"react-hot-loader": "^1.2.7",
7676
"react-router": "^0.13.1",
7777
"rf-changelog": "^0.4.0",
7878
"rimraf": "^2.3.2",

webpack/docs.config.js

+17-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,18 @@
11
import config from './webpack.config';
2-
export default config({docs: true});
2+
import yargs from 'yargs';
3+
4+
const argv = yargs
5+
.alias('p', 'optimize-minimize')
6+
.alias('d', 'debug')
7+
.option('port', {
8+
default: '8080',
9+
type: 'string'
10+
})
11+
.argv;
12+
13+
export default config({
14+
docs: true,
15+
development: argv.debug,
16+
optimize: argv.optimizeMinimize,
17+
port: parseInt(argv.port)
18+
});

webpack/strategies/development.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import _ from 'lodash';
22

33
export default (config, options) => {
4-
if (options.development) {
4+
if (options.development && !options.docs) {
55
config = _.extend({}, config, {
66
devtool: 'sourcemap'
77
});
+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import _ from 'lodash';
2+
import webpack from 'webpack';
3+
4+
function addWebpackDevServerScripts(entries, webpackDevServerAddress) {
5+
let clientScript = `webpack-dev-server/client?${webpackDevServerAddress}`;
6+
let webpackScripts = ['webpack/hot/dev-server', clientScript];
7+
return _.mapValues(entries, entry => webpackScripts.concat(entry));
8+
}
9+
10+
export default (config, options) => {
11+
if (options.development && options.docs) {
12+
let webpackDevServerAddress = `http://localhost:${options.port}`;
13+
config = _.extend({}, config, {
14+
entry: addWebpackDevServerScripts(config.entry, webpackDevServerAddress),
15+
output: _.extend({}, config.output, {
16+
publicPath: `${webpackDevServerAddress}/assets/`
17+
}),
18+
module: _.extend({}, config.module, {
19+
loaders: config.module.loaders.map(value => {
20+
if (/js/.test(value.test.toString())) {
21+
return _.extend({}, value, {
22+
loader: 'react-hot!' + value.loader
23+
});
24+
}
25+
else {
26+
return value;
27+
}
28+
})
29+
}),
30+
// Remove extract text plugin from dev workflow so hot reload works on css.
31+
plugins: config.plugins.concat([
32+
new webpack.NoErrorsPlugin()
33+
])
34+
});
35+
36+
return config;
37+
}
38+
39+
return config;
40+
};

webpack/strategies/docs.js

-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,6 @@ export default (config, options) => {
2727
jsLoader = value.loader;
2828

2929
return _.extend({}, value, {
30-
loader: jsLoader + '!client',
3130
exclude: /node_modules|Samples\.js/
3231
});
3332
}

webpack/strategies/index.js

+2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import development from './development';
22
import docs from './docs';
3+
import docsDevelopment from './docs-development';
34
import ie8 from './ie8';
45
import optimize from './optimize';
56
import test from './test';
@@ -9,5 +10,6 @@ export default [
910
docs,
1011
ie8,
1112
development,
13+
docsDevelopment,
1214
test
1315
];

0 commit comments

Comments
 (0)