-
Notifications
You must be signed in to change notification settings - Fork 212
/
Copy pathupdate.md
141 lines (120 loc) · 4.06 KB
/
update.md
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
# 从1.x版本升级
本文章将针对使用了1.x版本的用户,如何升级到最新的2.x版本,改动的地方非常的少
## 修改controller
``` js
// before
const stream = await renderToStream(ctx, 'Page', ctx.app.config)
// after
const stream = await renderToStream(ctx, ctx.app.config)
```
## 修改config.default.js
直接将新的config.default.js的内容粘贴过去即可
```js
const resolvePath = (path) => require('path').resolve(process.cwd(), path)
module.exports = {
keys: 'eggssr',
type: 'ssr', // 指定运行类型可设置为csr切换为客户端渲染
static: {
prefix: '/',
dir: resolvePath('dist')
},
routes: [
{
path: '/',
exact: true,
Component: () => (require('@/page/index').default), // 这里使用一个function包裹为了让它延迟require
controller: 'page',
handler: 'index'
},
{
path: '/news/:id',
exact: true,
Component: () => (require('@/page/news').default),
controller: 'page',
handler: 'index'
}
],
baseDir: resolvePath(''),
injectCss: [
`/static/css/Page.chunk.css`
], // 客户端需要加载的静态样式表
injectScript: [
`<script src='/static/js/runtime~Page.js'></script>`,
`<script src='/static/js/vendor.chunk.js'></script>`,
`<script src='/static/js/Page.chunk.js'></script>`
], // 客户端需要加载的静态资源文件表
serverJs: resolvePath(`dist/Page.server.js`)
}
```
## 修改layout.js
直接将新的layout组件内容粘贴过去
``` js
import React from 'react'
import '@/assets/common.less'
import './index.less'
import { Link } from 'react-router-dom'
import serialize from 'serialize-javascript'
const commonNode = props => (
// 为了同时兼容ssr/csr请保留此判断,如果你的layout没有内容请使用 props.children ? <div>{ props.children }</div> : ''
props.children ? <div className='normal'><h1 className='title'><Link to='/'>Egg + React + SSR</Link><div className='author'>by ykfe</div></h1>{props.children}</div>
: ''
)
const Layout = (props) => {
if (__isBrowser__) {
return commonNode(props)
} else {
const { serverData } = props.layoutData
const { injectCss, injectScript } = props.layoutData.app.config
return (
<html lang='en'>
<head>
<meta charSet='utf-8' />
<meta name='viewport' content='width=device-width, initial-scale=1, shrink-to-fit=no' />
<meta name='theme-color' content='#000000' />
<title>React App</title>
{
injectCss && injectCss.map(item => <link rel='stylesheet' href={item} key={item} />)
}
</head>
<body>
<div id='app'>{ commonNode(props) }</div>
{
serverData && <script dangerouslySetInnerHTML={{
__html: `window.__USE_SSR__=true; window.__INITIAL_DATA__ =${serialize(serverData)}`
}} />
}
<div dangerouslySetInnerHTML={{
__html: injectScript && injectScript.join('')
}} />
</body>
</html>
)
}
}
export default Layout
```
## 修改entry.js
在serverRender方法中新增layoutData属性即可
``` js
const serverRender = async (ctx) => {
// 服务端渲染 根据ctx.path获取请求的具体组件,调用getInitialProps并渲染
const ActiveComponent = getComponent(Routes, ctx.path)()
const Layout = ActiveComponent.Layout || defaultLayout
const serverData = ActiveComponent.getInitialProps ? await ActiveComponent.getInitialProps(ctx) : {}
ctx.serverData = serverData
return <StaticRouter location={ctx.req.url} context={serverData}>
<Layout layoutData={ctx}>
<ActiveComponent {...serverData} />
</Layout>
</StaticRouter>
}
```
## 修改package.json
只需要修改 `npm script` 启动命令,加入 `ykcli`即可
``` js
"scripts": {
"start": "rimraf dist && concurrently \"npm run ssr\" \" npm run csr \"",
"ssr": "concurrently \"egg-bin dev\" \"cross-env NODE_ENV=development webpack --watch --config ./build/webpack.config.server.js\"",
"csr": "cross-env NODE_ENV=development ykcli dev",
}
```