服务器端渲染不会渲染UI库。

4

我正在尝试为我的应用程序实现服务器端渲染。它以前是使用ReactDom进行客户端渲染的。在客户端渲染中,一切都运行良好,但现在当我将其变为服务器端渲染时,没有任何CSS样式表和UI库(Reactstrap + Material UI)被渲染。UI完全崩溃。

我将我的App.js(包含所有UI的实际前端页面)注入到我的html.js(一个模板)中。

我的服务器:

import express from 'express';
import React from 'react';
import { renderToString } from 'react-dom/server'
import App from '../../src/App'
import Html from '../../src/html.js'
const app = express();

// console.log that your server is up and running
app.listen(3001, () => console.log(`Listening on port 3001`));

app.get('/',(req,res) => {
  const title="2FA SDK"
  const body = renderToString(<App/>)
  res.send(
    Html({
      body,
      title,
    })
  )
})

这是我的html.js文件(只是一个简单的模板):

const Html = ({ body, styles, title }) => `
  <!DOCTYPE html>
  <html>
    <head>
      <title>${title}</title>
    </head>
    <body style="margin:0">
      <div id="app">${body}</div>
    </body>
  </html>
`;

我的app.js(使用material ui,最简版)

render(){
    return(
        <div className="container">
            <div>
                <TextField 
                    label="Application Name"
                    type="text" 
                    variant="outlined"
                    onChange={this.handleApplicationName}
                />
            </div>
            <div>
                <FormLabel>Phone</FormLabel>
                <div>
                    <OutlinedInput
                        value={this.state.textmaskArray[2]}
                        onChange={(e) => this.handlePhoneNumber(e,2)}
                        labelWidth={200}
                    />
                </div>
            </div>
            <Button variant="contained" color="primary" onClick={this.props.buttonAction}>
            </Button>
        </div>

使用reactstrap的App.js部分:

render(){
    return(
      <CardControl
        title={"Code Authention"}
        message={"For your security, we need to verify your identity by sending a code to your phone number"}
        >
        <Container>
            <Row>
                <Col>
                    <div>{this.props.phoneList}</div>
                </Col>
                <Col>
                    <div>{this.props.methodList}</div>
                </Col>
            </Row>
            <Button className="sm-2" disabled={this.props.disabled} color="info" onClick={this.props.buttonAction}>{this.props.buttonName}</Button>
        </Container>
      </CardControl>

这里是我的webpack.config.js:

const MiniCssExtractPlugin = require("mini-css-extract-plugin");
const HtmlWebPackPlugin = require("html-webpack-plugin");
const nodeExternals = require("webpack-node-externals");

module.exports = [
    {
        /*Config for backend code*/ 
        entry: './src/server/server.js',
        target: 'node',
        output: {
            filename: 'server.js'
        },
        externals: [nodeExternals()],
        module: {
            rules: [
                {
                    test: /\.(js|jsx)$/,
                    exclude: /node_modules/,
                    use: {
                        loader: "babel-loader"
                    }
                },
                {
                    test: /\.html$/,
                    use: {
                        loader: "html-loader",
                        options: { minimize: true }
                    }
                },
                {
                    test: /\.css$/,
                    use: [MiniCssExtractPlugin.loader,"css-loader"]
                }
            ]
        },
        plugins: [
            new HtmlWebPackPlugin({
                template: "./public/index.html",
                filename:"./index.html"
            }),
            new MiniCssExtractPlugin({
                filename: "[name].css",
                chunkFilename:"[id].css"
            })
        ]
    },
]

我使用的工具: Create-react-app

我尝试过的方法: 我安装了CSS加载器,但它根本没有起作用。我还查看了material-ui关于服务器端渲染的文档,感觉我只是缺少一个小部分,他们提供的指令可能有点过头了。

我需要帮助的地方: 我正在尝试弄清楚为什么来自material-ui或bootstrap的UI组件根本没有被渲染出来。我怀疑nodeExternal将它们排除在外,但我不确定。


你是否在任何地方为渲染的元素添加水合功能? - apokryfos
@apokryfos 不,让我发布我的app.js。 - ZpfSysn
1个回答

0

为了让React接管,您需要在客户端对呈现的HTML进行水合处理。

创建一个client.js文件,其中包含:

import React from 'react';
import ReactDOM from 'react-dom';
ReactDOM.hydrate(
  <App/>,
  document.querySelector('#app'),
);

请注意,`hydrate`应该与`renderToString`渲染的内容相匹配。
接下来,在您的webpack配置中添加这个额外的条目:
module.exports = [
{
        /*Config for backend code*/ 
        entry: './src/server/server.js',
        target: 'node',
        output: {
            filename: 'server.js'
        },
        externals: [nodeExternals()],
        module: {
            rules: [
                {
                    test: /\.(js|jsx)$/,
                    exclude: /node_modules/,
                    use: {
                        loader: "babel-loader"
                    }
                },
                {
                    test: /\.html$/,
                    use: {
                        loader: "html-loader",
                        options: { minimize: true }
                    }
                },
                {
                    test: /\.css$/,
                    use: [MiniCssExtractPlugin.loader,"css-loader"]
                }
            ]
        },
        plugins: [
            new HtmlWebPackPlugin({
                template: "./public/index.html",
                filename:"./index.html"
            }),


  new MiniCssExtractPlugin({
            filename: "[name].css",
            chunkFilename:"[id].css"
        })
    ]
},
{ 
   entry: './client.js',
   output: {
      filename: 'bundle.js',
   },
   module: {
      rules: [ {
          test: /\.js$/,
          exclude: /node_modules/,
          loader: 'babel-loader',
      },
   ],
 },
]

将您的HTML渲染代码更改为包括以下内容:

const Html = ({ body, styles, title }) => `
  <!DOCTYPE html>
  <html>
    <head>
      <title>${title}</title>
    </head>
    <body style="margin:0">
      <div id="app">${body}</div>
      <script async src="/bundle.js"></script>
    </body>
  </html>
`;

我不完全确定这段代码是否能够正常工作,但是这是一般的想法。


我应该运行服务器还是运行bundle.js? - ZpfSysn
Webpack应该创建两个文件。在运行服务器时,当它提供HTML时,还应该发送bundle.js。 - apokryfos
我正在运行时遇到了“document is not define”错误。我认为这是因为服务器端渲染没有文档所致。 - ZpfSysn
服务器端没有 document,但是 client.js 应该只在客户端加载。根据我的理解,<script async src="/bundle.js"></script> 是发送给客户端的标记的一部分。 - apokryfos

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接