react 应用多入口配置及实践总结


Posted in Javascript onOctober 17, 2018

背景

还是之前的那个项目,做完国际化没多久,还没来得及划水, 又有新的活了 -- 移动端的兼容。 考虑到后期的复杂度, 需要做两套资源。 具体的目标是:同一个URL,PC打开就显示PC的那一套, M端打开就显示Mobile的页面。 create-react-app 脚手架本身不支持多入口, 需要改造,今天下午研究了一下,改造了一波, 基本达到了预期, 在这里简单把经验总结分享下。

先睹为快

Mobile:

react 应用多入口配置及实践总结

PC:

react 应用多入口配置及实践总结

react 应用多入口配置及实践总结

输出之后的文件, 相比之前的index.html, 多了一个额外的mobile.html.

最终的源代码目录:

react 应用多入口配置及实践总结

具体的改造步骤如下:

Steps

step1: Eject

在下之前图方便, 直接用了create-react-app, 现在需要更改配置, 需要弹出默认配置:

在终端执行:yarn eject.

step2: 修改webpack config

原本的 webpack.config.dev.js:

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),
},

需要修改为:

entry: {
  index: [
   require.resolve('./polyfills'),
   require.resolve('react-dev-utils/webpackHotDevClient'),
   paths.appIndexJs,
  ],
  mobile: [
   require.resolve('./polyfills'),
   require.resolve('react-dev-utils/webpackHotDevClient'),
   paths.appSrc + "/mobile/index.js",
  ]
 },
 output: {
  pathinfo: true,
  filename: 'static/js/[name].bundle.js',
  chunkFilename: 'static/js/[name].chunk.js',
  publicPath: publicPath,
  devtoolModuleFilenameTemplate: info =>
   path.resolve(info.absoluteResourcePath).replace(/\\/g, '/'),
 },

可能需要注意的几点:

  • entry从原来的数组扩展为对象,每个key代表一个入口。
  • output中的filename要区分输出名,可增加[name]变量,这样会根据entry分别编译出每个entry的js文件。

这样你就可以在src 目录下新起一个民目录开发新的SPA:

react 应用多入口配置及实践总结

step3: 生成多个html入口文件

Webpack配置多入口后,只是编译出多个入口的Js,入口的HTML文件也需要配置, 可以用HtmlWebpackPlugin来生成。

webpack.config.dev.js 原配置:

// Generates an `index.html` file with the <script> injected.
  new HtmlWebpackPlugin({
   inject: true,
   chunks: ["index"],
   template: paths.appHtml,
  }),

需要加多一个配置, 改成:

// Generates an `index.html` file with the <script> injected.
  new HtmlWebpackPlugin({
   inject: true,
   chunks: ["index"],
   template: paths.appHtml,
  }),
  new HtmlWebpackPlugin({
   inject: true,
   chunks: ["mobile"],
   template: paths.appHtml,
   filename: 'mobile.html',
  }),
  • 每调一次HtmlWebpackPlugin生成一次HTML页面,这里有两个,所以就会多增加一个mobile.html节点。
  • chunks: 指明哪些webpack入口的js会被注入到这个HTML页面。如果不配置,则将所有entry的JS文件都注入HTML。
  • filename: 指明生成的HTML路径,如果不配置就是dist/index.html。mobile配置了新的filename,避免与第一个入口相互覆盖。
  • template: 指定模版, 我这里因为用的这两个模版都一样, 所以就指定的同一个appHtml, 如有特殊需求, 就新建一个html 文件, 在template字段里引用即可。

Step4: 配置webpack Dev Server

上述配置做完后,理论就可以打包出多入口的版本;但使用npm start启动后,发现无论输入/index.html还是/mobile.html,好像都是和原来/index.html显示一样的内容。

甚至输入显然不存在的/xxxx.html,也显示为/index.html的内容。

这种现象,初步判断是HTTP服务器把所有请求重定向到了/index.html。

对于单页应用,这种做法是没有问题的(本来就一个页面), 但我们新增的/mobile.html就可以访问到了。

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

原配置:

historyApiFallback: {
 // Paths with dots should still use the history fallback.
 // See https://github.com/facebookincubator/create-react-app/issues/387.
 disableDotRule: true,
},

修改为:

historyApiFallback: {
 // Paths with dots should still use the history fallback.
 // See https://github.com/facebookincubator/create-react-app/issues/387.
 disableDotRule: true,
 // 指明哪些路径映射到哪个html
 rewrites: [
  { from: /^\/mobile.html/, to: '/dist/mobile.html' },
 ]
},

增加的rewrites节点,特别对/admin.html这个URL重定向为/dist/mobile.html页面(也就是HtmlWebpackPlugin输出的HTML文件路径),这样/mobile.html就可以正常访问了。

至此,dev环境的多入口问题就解决了。

step5: Prod 环境配置

prod环境,比dev环境更简单。由于不存在webpack Dev Server,直接在config/webpack.config.prod.js同理做2和3步骤即可。

这时候你就可以通过手动修改URl 来访问了:

react 应用多入口配置及实践总结

react 应用多入口配置及实践总结

路由相关

到这, 已经可以手动修改URL 来访问pc 和 mobile的页面了。 还有一个问题没有解决:

URL最后肯定是不能给你手动改来改去的, 需要根据设备的情况自己判断, 这里有两个思路:

1: 配置 Nginx 的路径的时候, 加多一个alias 的映射。
2: 前端根据UA自行配置。

为了快速出效果, 简单搞了一下, 具体代码如下:

react 应用多入口配置及实践总结

这样, 不用手动修改URL 也能根据UA自动显示不同的页面了, 具体的效果图在文章开头的先睹为快中。

其他

上面的路由就为了简单的出个效果, 比较粗暴, 仅供参考。

结语

以上就是全部的细节了, 达到了预期的效果, 但是也有很大优化空间。 等后面一波需求做完了, 再来做补充吧。希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
May 24 Javascript
THREE.JS入门教程(2)着色器-上
Jan 24 Javascript
用js的document.write输出的广告无阻塞加载的方法
Jun 05 Javascript
浅析Bootstrap表格的使用
Jun 23 Javascript
JavaScript中的对象继承关系
Aug 01 Javascript
js删除数组中的元素delete和splice的区别详解
Feb 03 Javascript
详解使用vue-admin-template的优化历程
May 20 Javascript
微信小程序蓝牙连接小票打印机实例代码详解
Jun 03 Javascript
JavaScript 俄罗斯方块游戏实现方法与代码解释
Apr 08 Javascript
Jquery ajax书写方法代码实例解析
Jun 12 jQuery
js实现3D粒子酷炫动态旋转特效
Sep 13 Javascript
vue实现树状表格效果
Dec 29 Vue.js
vue+echarts实现动态绘制图表及异步加载数据的方法
Oct 17 #Javascript
手把手教你写一个微信小程序(推荐)
Oct 17 #Javascript
微信小程序开发之tabbar图标和颜色的实现
Oct 17 #Javascript
解决百度Echarts图表坐标轴越界的方法
Oct 17 #Javascript
vue实现循环切换动画
Oct 17 #Javascript
element-ui upload组件多文件上传的示例代码
Oct 17 #Javascript
bootstrap table合并行数据并居中对齐效果
Oct 17 #Javascript
You might like
Linux下编译redis和phpredis的方法
2016/04/07 PHP
浅谈PHP中new self()和new static()的区别
2017/08/11 PHP
Laravel学习教程之View模块详解
2017/09/18 PHP
PHP实现图的邻接矩阵表示及几种简单遍历算法分析
2017/11/24 PHP
PHP实现浏览器格式化显示XML的方法示例
2019/01/22 PHP
PHP7 整型处理机制修改
2021/03/09 PHP
JQuery DataTable删除行后的页面更新利用Ajax解决
2013/05/17 Javascript
jquery序列化表单以及回调函数的使用示例
2014/07/02 Javascript
jquery中each遍历对象和数组示例
2014/08/05 Javascript
基于Angularjs实现分页功能
2016/05/30 Javascript
JS中对数组元素进行增删改移的方法总结
2016/12/15 Javascript
jqgrid实现简单的单行编辑功能
2017/09/30 Javascript
JavaScript设计模式之观察者模式(发布订阅模式)原理与实现方法示例
2018/07/27 Javascript
使用JS实现导航切换时高亮显示的示例讲解
2018/08/22 Javascript
创建echart多个联动的示例代码
2018/11/23 Javascript
BootStrap table实现表格行拖拽效果
2018/12/01 Javascript
element ui分页多选,翻页记忆的实例
2019/09/03 Javascript
vue 组件间的通信之子组件向父组件传值的方式
2020/07/29 Javascript
Vue 实现创建全局组件,并且使用Vue.use() 载入方式
2020/08/11 Javascript
[36:37]2014 DOTA2华西杯精英邀请赛5 24 VG VS iG
2014/05/25 DOTA
[01:22:10]Ti4 循环赛第二日 DK vs Empire
2014/07/11 DOTA
[14:56]教你分分钟做大人:巫医
2014/10/30 DOTA
[36:43]NB vs Optic 2018国际邀请赛小组赛BO1 B组加赛 8.19
2018/08/21 DOTA
将pandas.dataframe的数据写入到文件中的方法
2018/12/07 Python
Python+OpenCV感兴趣区域ROI提取方法
2019/01/10 Python
在python plt图表中文字大小调节的方法
2019/07/08 Python
TFRecord文件查看包含的所有Features代码
2020/02/17 Python
Python opencv相机标定实现原理及步骤详解
2020/04/09 Python
Python就将所有的英文单词首字母变成大写
2021/02/12 Python
采用冷却技术的超自然舒适度:GhostBed床垫
2018/09/18 全球购物
公司财务自我评价分享
2013/12/17 职场文书
简历自我评价怎么写呢?
2014/01/06 职场文书
钢琴师观后感
2015/06/12 职场文书
幼儿园家长心得体会
2016/01/21 职场文书
初二数学教学反思
2016/02/17 职场文书
Python还能这么玩之用Python修改了班花的开机密码
2021/06/04 Python