基于@vue-cli3的多页面应用改造及nignx配置

  

20200318
基于@vue-cli3的多页面应用改造及nignx配置
参考资料:https://www.jianshu.com/p/a7a742724e14

需求

一个项目型的工程。由于初期欠考虑的设计。做成了单页应用。现在项目越做越大.项目中多了许多原设计无法很好兼容的内容。这个工程原来的定位是预约审批系统的PC端,但是现在,这个工程包含着
1、预约审批功能PC端
2、审批功能钉钉H5页面
3、移动端问卷调查
4、访客账号的子账号管理功能

所以,打算将工程改造为多页应用。

修改步骤

修改目录结构

我们首先把多个模块独立出来。

原单页应用目录

实际上用@vue/cli3生成的目录都长得一个样。

项目
|
|__public
|
|__src
|   |
|   |__api
|   |
|   |__assets
|	|
|	|__components
|	|
|	|__dictionary
|	|
|	|__router
|	|
|	|__store
|	|
|	|__style
|	|
|	|__utils
|	|
|	|__views
|	|
|	|__App.vue
|	|
|	|__config.js
|
|__vue.config.js

多页应用目录

项目
|
|__public
|
|__src
|	|
|	|__appointModule // 原来的预约功能模块全部放到了appointModule目录内
|	|	|
|	|	|
|	|   |
|	|   |__api
|	|   |
|	|   |__assets
|	|	|
|	|	|__components
|	|	|
|	|	|__dictionary
|	|	|
|	|	|__router
|	|	|
|	|	|__store
|	|	|
|	|	|__style
|	|	|
|	|	|__utils
|	|	|
|	|	|__views
|	|	|
|	|	|__App.vue
|	|	|
|	|	|__config.js
|	|	|
|	|	|__main.js // 预约审批模块入口文件
|	|
|	|__systemModule // 新增的子账号管理功能当成系统模块,全部放到systemModule内
|	|	|
|	|	|__api
|	|	|
|	|	|__compoinents
|	|	|
|	|	|__dictionary
|	|	|
|	|	|__router
|	|	|
|	|	|__store
|	|	|
|	|	|__views
|	|	|
|	|	|__App.vue
|	|	|
|	|	|__config.js
|	|	|
|	|	|__systemEntry.js // 系统模块的入口文件
|
|__vue.config.js

我将原先的预约相关功能,放入了appointModule目录下,并新建了systemModule目录,用来放系统管理相关的功能。
systemModule中的入口文件,路由,组件等都是独立的。方便将来系统模块扩展。

修改组件引用路径

因为项目结构变了,所以引用组件的路径也要跟着修改。这个没啥好说的,不然组件引用不到肯定是运行不起来的。

修改路由

原单页应用

// router.js
import Vue from 'vue';
import VueRouter from 'vue-router';

Vue.use(VueRouter);

// 静态路由
const routes = [
    {
        name: '',
        path:'',
        components: ''
    },
    ....
];

// 动态路由
const asyncRoutes = [
    {
        name: '',
        path: '',
        compoents: ''
    },
    ...
]

const router = new VueRouter({
    mode: 'history',
    routes
});


// 路由守卫
router.beforeEach((from, to, next) => {
    ...
    next();
});

router.onError((error) => {
    ...
});

export router

多页应用路由

// 预约模块 appointModule/router.js

// 静态路由
...
// 动态路由
...
// 路由守卫
...


const router = new VueRouter({
    mode: 'history',
    routers,
    base: '/appointModule/' // 注意这一行
});

export router;
// 系统模块 systemModule/router.js

// 静态路由
...
// 动态路由
...
// 路由守卫
...

const router = new VueRouter({
    mode: 'history',
    routers,
    base: '/systemModule/' // 注意这一行
});

给路由添加 base 基础路径.用于区分两个模块的两个路由

路由修改后,我们访问项目所要输入的url也将发生变化。访问方式从原来的 contextPath/路由path改为了contextPath/module/路由path

所以,项目中原来路由跳转的写法也要跟着改变。

// 原写法:
this.$router.push('/login');
// 改造后写法
this.$router.push({name: 'Login'});
// 或者改成这种也行
this.$router.push('/appointModule/login');

改造路由跳转的时候,我发现,
this.$router.push({name: 'xxx'}));

这种用name的写法比直接用’/appointModule/'这种用路径的写法好用,至少我以后修改路由的base路径时候,name的写法不用我再做一次修改了。

进入预约审批模块的页面时,我们需要输入http://localhost:8080/appointModule/路由path
进入系统模块的页面时,我们需要输入http://localhost:8080/systemModule/路由path

修改vue.config.js

原单页应用


// vue.config.js
module.exports = {
	publicPath: '/',
	outputDir: `${outputDir}`,
	lintOnSave: false,
	produc tionSourceMap: false ,
	//下面是代理表,作用是用来,建一个虚拟api服务器用来代理本机的请求,只能用于开发模式
	//一般解决跨域请求api
	devServer: {
		port: 8090 ,
		proxy: {
			'/api': {
				
				target:	'http://ip:port',
				pathRewrite: {
					'^/api': ' /api'
				},
				//指示是否跨域
				changeorigin: true,
				//如果是https接口, 需要配置这个参数
				secure: false
			}
		}
		hot: true ,
		inline: true,
		disableHostCheck: true
	},
	configurewebpack: {
		resolve: {
			alias : {
				@': resolve('src' )
			}
		},
        output: {
            filename:'js/ [name].${stamp}.js',
            chunkFilename: 'js/ [name]${stamp}.js'
        },
        plugins: [
            // new BundleAnalyzerPlugin()
        ]
    }
};

修改为多页应用

// vue.config.js

module.exports = {
    pages: {
        //预约模块
        appointModule: {
        entry:'src/ appointModule/main.js',
        template: 'public/ index . html',
        // filename: 'appointModule.html', //为了在nginx更容易配置,采用下面这种filename配置
        // filename: appointModule/index.html'
        filename: 'index.html' // 打包后的html文件会放到 rootPath/index.html中
        	// chunks: [ ' chunk -verdors',' chunk - common', ' index']
        },
        //系统管理模块
        systemModule: {
	        entry: 'src/systemModule/ systemEntry.js',
           template: 'public/index. html',
	        // filename:systemModule.html', //为了在nginx更容易配置,采用下面这种filename配置
    	    filename: 'systemModule/index.html' // 打包后的html文件会放到 rootPath/systemModule/index.html中
        }
   }

};

其它都不变,新增pages配置,如上所示。

到现在,我们的项目已经可以通过npm run server 启用了。

进入预约审批模块的页面时,我们需要输入http://localhost:8080/appointModule/路由path
进入系统模块的页面时,我们需要输入http://localhost:8080/systemModule/路由path

我们还需要关心打包后的目录变化,因为这涉及到nginx的配置。我在配置nginx的过程中走了不少弯路。

这样打包后的工程目录就会变成这样子。

// 打包后的工程
citybrain-appoint
|
|__fonts
|
|__img
|
|__js
|	|
|	|__appointModule.hash.js
|	|
|	|__chunk-verdors.hash.js
|	|
|	|__systemModule.hash.js
|
|__systemModule	// 系统模块打包后的index.html放到这里
|	|
|	|__index.html
|
|__index.html // 预约模块打包后的index.htlm仍然放在根目录下,因为这样可以让nginx做最小的改动。

修改nginx配置

原nginx配置

单页应用

server {
    listen 8900;
    server_ name citybrain-appoint;
    location / {
        root 项目目录;
        index index . html index . htm;
        try_ files Suri $uri/ /index . html;
    }
    location /api {
        proxy_ pass http://ip:port/api/;
    }
}

修改后的nginx配置

多页应用

server {
    listen 8094;
    server_name 项目名 ;
    root 项目目录;
    location / {
        # rewrite / /appointModu1e/ permanent;
        index index. html;
        try_ files $uri $uri/ / index .html;
    }
    location /systemModule {
        index index. html;
        try_ files $uri $uri/ / systemModule/ index .html;
    }
    location /api {
        proxy_ _pass http://ip:port/api/;
    }
}

nginx我们并不需要多大的改动,原来的预约模块的nginx配置我们不需要修改。我们只需要将新增的系统模块的路径加入nginx配置中即可。

Q&A

Q1、为什么当初要把不合适的功能写到同一个项目里?

主要原因有两个:
1、新功能不够复杂
以钉钉H5页面为例,实际上这个功能只有一个页面。为了写一个页面开一个新工程,意味着为了一个页面就得新部署一个工程。
2、网络环境
子账号管理功能本应该放到管理端的那个工程更合适 ,但是管理端的工程在外网无法访问。而子账号需要让甲方爸爸访问到
3、时间来不及
时间非常紧急(9点上班,23点下班,都来不及的需求和变化)。开一个新工程,意味着,登录,权限控制等功能需要重复开发。而我们没有时间这么做,当时也想不到更好的解决方法

Q2、为什么选择改造为多页面应用?

1、难度不大

一是代码改造难度不大,二是部署时不需要重新部署新的工程

2、时间问题

这个项目从开始到现在,时间一直是大问题。再加上,之前的代码耦合严重,要分离成一个新项目也是需要时间的。分离出来后还要重新测试。时间完全不够用,

相关文章