forked from shakacode/shakapacker
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Update the HMR docs for React (#2438)
- Loading branch information
Showing
3 changed files
with
186 additions
and
42 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,183 @@ | ||
# React | ||
|
||
## Props Hydration and Server-Side Rendering (SSR) | ||
You only _need_ props hydration if you need SSR. However, there's no good reason to | ||
have your app do a second round trip to the Rails server to get props. | ||
|
||
Server-Side Rendering (SSR) results in Rails rendering HTML for your React components. | ||
The main reasons to use SSR are better SEO and pages display more quickly. | ||
|
||
### Rails and React Integration Gems | ||
If you desire more advanced React-integration, like server-side rendering, SSR with react-router, SSR with code splitting, then you should consider these gems: | ||
|
||
| Gem | Props Hydration | Server-Side-Rendering (SSR) | SSR with HMR | SSR with React-Router | SSR with Code Splitting | Node SSR | | ||
| --- | --------------- | --- | --------------------- | ----------------------| ------------------------|----| | ||
| [shakacode/react_on_rails](https://github.com/shakacode/react_on_rails) | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ | | ||
| [react-rails](https://github.com/reactjs/react-rails) | ✅ | ✅ | | | | | | | ||
| [webpacker-react](https://github.com/renchap/webpacker-react) | ✅ | | | | | | | | ||
|
||
Note, Node SSR for React on Rails requires [React on Rails Pro](https://www.shakacode.com/react-on-rails-pro). | ||
|
||
### Hydration of Props the Manual Way | ||
|
||
If you're not concerned with view helpers to pass props or server-side rendering, you can do it like this: | ||
|
||
```erb | ||
<%# views/layouts/application.html.erb %> | ||
<%= content_tag :div, | ||
id: "hello-react", | ||
data: { | ||
message: 'Hello!', | ||
name: 'David' | ||
}.to_json do %> | ||
<% end %> | ||
``` | ||
|
||
```js | ||
// app/javascript/packs/hello_react.js | ||
|
||
const Hello = props => ( | ||
<div className='react-app-wrapper'> | ||
<img src={clockIcon} alt="clock" /> | ||
<h5 className='hello-react'> | ||
{props.message} {props.name}! | ||
</h5> | ||
</div> | ||
) | ||
|
||
// Render component with data | ||
document.addEventListener('DOMContentLoaded', () => { | ||
const node = document.getElementById('hello-react') | ||
const data = JSON.parse(node.getAttribute('data')) | ||
|
||
ReactDOM.render(<Hello {...data} />, node) | ||
}) | ||
``` | ||
|
||
---- | ||
|
||
## HMR and React Hot Reloading | ||
|
||
Before turning HMR on, consider upgrading to latest stable gems and packages: | ||
https://github.com/rails/webpacker#upgrading | ||
|
||
Configure `config/webpacker.yml` file: | ||
|
||
```yaml | ||
development: | ||
extract_css: false | ||
dev_server: | ||
hmr: true | ||
inline: true | ||
``` | ||
The basic setup will have HMR working with the default webpacker setup. However, the basic will cause a full page refresh each time you save a file. | ||
Webpack's HMR allows replacement of modules for React in-place without reloading the browser. To do this, you have two options: | ||
1. Steps below for the [github.com/pmmmwh/react-refresh-webpack-plugin](https://github.com/pmmmwh/react-refresh-webpack-plugin). | ||
1. Deprecated steps below for using the [github.com/gaearon/react-hot-loader](https://github.com/gaearon/react-hot-loader). | ||
### React Refresh Webpack Plugin | ||
[github.com/pmmmwh/react-refresh-webpack-plugin](https://github.com/pmmmwh/react-refresh-webpack-plugin) | ||
You can see an example commit of adding this [here](https://github.com/shakacode/react_on_rails_tutorial_with_ssr_and_hmr_fast_refresh/commit/7e53803fce7034f5ecff335db1f400a5743a87e7). | ||
1. Add react refresh packages: | ||
`yarn add @pmmmwh/react-refresh-webpack-plugin react-refresh -D` | ||
2. Update `babel.config.js` adding | ||
```js | ||
plugins: [ | ||
process.env.WEBPACK_DEV_SERVER && 'react-refresh/babel', | ||
// other plugins | ||
``` | ||
3. Update `config/webpack/development.js`, only including the plugin if running the WEBPACK_DEV_SERVER | ||
```js | ||
const ReactRefreshWebpackPlugin = require('@pmmmwh/react-refresh-webpack-plugin'); | ||
const environment = require('./environment') | ||
const isWebpackDevServer = process.env.WEBPACK_DEV_SERVER; | ||
//plugins | ||
if (isWebpackDevServer) { | ||
environment.plugins.append( | ||
'ReactRefreshWebpackPlugin', | ||
new ReactRefreshWebpackPlugin({ | ||
overlay: { | ||
sockPort: 3035 | ||
} | ||
}) | ||
); | ||
} | ||
``` | ||
|
||
### React Hot Loader (Deprecated) | ||
|
||
1. Add the `react-hot-loader` and ` @hot-loader/react-dom` npm packages. | ||
```sh | ||
yarn add --dev react-hot-loader @hot-loader/react-dom | ||
``` | ||
|
||
2. Update your babel config, `babel.config.js`. Add the plugin `react-hot-loader/babel` | ||
with option `"safetyNet": false`: | ||
|
||
``` | ||
{ | ||
"plugins": [ | ||
[ | ||
"react-hot-loader/babel", | ||
{ | ||
"safetyNet": false | ||
} | ||
] | ||
] | ||
} | ||
``` | ||
|
||
3. Add changes like this to your entry points: | ||
|
||
```diff | ||
// app/javascript/app.jsx | ||
import React from 'react'; | ||
+ import { hot } from 'react-hot-loader/root'; | ||
const App = () => <SomeComponent(s) /> | ||
- export default App; | ||
+ export default hot(App); | ||
``` | ||
|
||
4. Adjust your webpack configuration for development so that `sourceMapContents` option for the sass | ||
loader is `false`: | ||
|
||
```diff | ||
// config/webpack/development.js | ||
process.env.NODE_ENV = process.env.NODE_ENV || 'development' | ||
const environment = require('./environment') | ||
// allows for editing sass/scss files directly in browser | ||
+ if (!module.hot) { | ||
+ environment.loaders.get('sass').use.find(item => item.loader === 'sass-loader').options.sourceMapContents = false | ||
+ } | ||
+ | ||
module.exports = environment.toWebpackConfig() | ||
``` | ||
|
||
5. Adjust your `config/webpack/environment.js` for a | ||
|
||
```diff | ||
// config/webpack/environment.js | ||
// ... | ||
// Fixes: React-Hot-Loader: react-🔥-dom patch is not detected. React 16.6+ features may not work. | ||
// https://github.com/gaearon/react-hot-loader/issues/1227#issuecomment-482139583 | ||
+ environment.config.merge({ resolve: { alias: { 'react-dom': '@hot-loader/react-dom' } } }); | ||
module.exports = environment; | ||
``` | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters