修改create-react-app支持多入口

使用Facebook官方脚手架create-react-app建立React应用,默认只能生成一个SPA,入口是index.html。虽然,SPA的页面切换可使用前台路由框架方便(好比React-Router)实现,这也是SPA的推荐作法;但某些状况下,仍要将页面切分为多个页面,或者在同一个工程开发多个SPA,好比一个是面向客户的SPA,一个是后台管理的SPA。javascript

官方给出的回答是:css

Sorry, but Create React App doesn't support this use case.
You can eject and then configure Webpack to have multiple entry points.html

查阅create-react-app和webpack官方文档,测试以下方式可行,步骤以下:(假设在默认的index.html外增长一个admin.html)java

1. Eject

要实现自定义配置,就须要先Eject出配置(此步骤对create-react-app工程不可逆):npm run ejectreact

运行后,package.js会被更新,工程下会多出config目录,其中有webpack有两个配置文件,分别对应开发和生产环境(/config/webpack.config.dev.js和/config/webpack.config.prod.js)。这两个配置文件都要修改,但略有不一样,下面以dev为例说明:webpack

2. 修改webpack配置支持多入口

因为是使用webpack打包,先要让webpack配置出多入口。/config/webpack.config.dev.js默认配置的入口以下:git

 entry: [
    require.resolve('react-dev-utils/webpackHotDevClient'),
    require.resolve('./polyfills'),
    require.resolve('react-error-overlay'),
    paths.appIndexJs,
  ],
  output: {
    path: paths.appBuild,
    pathinfo: true,
    filename: 'static/js/bundle.js',
    chunkFilename: 'static/js/[name].chunk.js',
    publicPath: publicPath,
    devtoolModuleFilenameTemplate: info =>
      path.resolve(info.absoluteResourcePath),
  },

根据webpack官方文档MULTIPLE ENTRY POINTS,可作以下修改:github

要点是:web

  1. entry从原来的数组扩展为对象,每一个key表明一个入口。
  2. output中的filename要区分输出名,可增长[name]变量,这样会根据entry分别编译出每一个entry的js文件。
  3. 因为path里面没有定义新的entry的路径,图方即可以直接写死为paths.appSrc + "/admin.js"
 entry: {
    index: [
      require.resolve('react-dev-utils/webpackHotDevClient'),
      require.resolve('./polyfills'),
      require.resolve('react-error-overlay'),
      paths.appIndexJs,
    ],
    admin:[
      require.resolve('react-dev-utils/webpackHotDevClient'),
      require.resolve('./polyfills'),
      require.resolve('react-error-overlay'),
      paths.appSrc + "/admin.js",
      ]
  },
  output: {
    path: paths.appBuild,
    pathinfo: true,
    filename: 'static/js/[name].bundle.js',
    chunkFilename: 'static/js/[name].chunk.js',
    publicPath: publicPath,
    devtoolModuleFilenameTemplate: info =>
      path.resolve(info.absoluteResourcePath),
  },

这样在src文件夹下,就能够再增长一个admin.js的入口,单独写新的SPA。npm

普通的webpack打包工程,到此便可实现多入口,但create-react-app流程更复杂,需继续修改。

3. 修改HtmlWebpackPlugin生成多个HTML

Webpack配置多入口后,只是编译出多个入口的JS,同时入口的HTML文件由HtmlWebpackPlugin生成,也需作配置。

原配置以下:

new HtmlWebpackPlugin({
      inject: true,
      template: paths.appHtml,
    }),

修改成:

    new HtmlWebpackPlugin({
      inject: true,
      chunks: ["index"],
      template: paths.appHtml,
    }),
    new HtmlWebpackPlugin({
      inject: true,
      chunks: ["admin"],
      template: paths.appHtml,
      filename: 'admin.html',
    }),

每调一次HtmlWebpackPlugin生成一次HTML页面,故为admin.html多增长一个节点。其余要点以下:

  • chunks,指明哪些webpack入口的JS会被注入到这个HTML页面。若是不配置,则将全部entry的JS文件都注入HTML。
  • filename,指明生成的HTML路径,若是不配置就是build/index.html,admin配置了新的filename,避免与第一个入口的index.html相互覆盖。

另外,template属性也能够修改成不一样的HTML模板,这里的例子,我么就和index入口共用HTML模板了。

 

4. 修改webpack Dev Server配置

上述配置作完后,理论就能够打包出多入口的版本;但使用npm start启动后,发现不管输入/index.html仍是/admin.html,好像都是和原来/index.html显示同样的内容。甚至输入显然不存在的/xxxx.html,也显示为/index.html的内容。

这种现象,初步判断是HTTP服务器把全部请求重定向到了/index.html。对于单页应用,这种作法是没有问题的(原本就一个页面);但咱们新增的/admin.html就能够访问了。

参考官方文档The historyApiFallback option,发现是webpack dev server的问题,还要额外作一些配置,需修改/config/webpackDevServer.config.js。

原配置以下:

 historyApiFallback: {
      disableDotRule: true,
    },

修改成:

   historyApiFallback: {
      disableDotRule: true,
      // 指明哪些路径映射到哪一个html
      rewrites: [
        { from: /^\/admin.html/, to: '/build/admin.html' },
      ]
    }

增长的rewrites节点,特别对/admin.html这个URL重定向为/build/admin.html页面(也就是HtmlWebpackPlugin输出的HTML文件路径),这样/admin.html就能够正常访问了。
至此,dev环境的多入口问题就解决了。

 

prod环境

prod环境,比dev环境更简单。因为不存在webpack Dev Server,直接在config/webpack.config.prod.js同理作2和3步骤便可。

使用npm run build构建prod版本,观察build目录,入口/admin.html的html、js和css文件俱在:

build
├── admin.html
├── asset-manifest.json
├── favicon.ico
├── index.html
├── manifest.json
├── service-worker.js
└── static
    ├── css
    │   ├── admin.d41d8cd9.css
    │   ├── admin.d41d8cd9.css.map
    │   ├── index.9a0fe4f1.css
    │   └── index.9a0fe4f1.css.map
    ├── js
    │   ├── admin.f3dca2cd.js
    │   ├── admin.f3dca2cd.js.map
    │   ├── index.4b87195c.js
    │   └── index.4b87195c.js.map
    └── media
        └── logo.5d5d9eef.svg
相关文章
相关标签/搜索