这篇是我参加创作者训练营活动的最后一篇文章了。
mpVue
简介
- 美团工程师推出的基于 Vue.js 封装的用于开发小程序的框架
- 融合了原生小程序和 Vue.js 的特点
- 可完全组件化开发
特点
- 组件化开发
- 完成的 Vue.js 开发体验(前提是熟悉 Vue)
- 可使用 Vuex 管理状态
- Webpack 构建项目
- 最终 H5 转换工具将项目编译成小程序识别的文件
初始化项目
- npm install vue-cli -g 下载 vue 脚手架
- vue init mpvue/mpvue-quickstart my-project 初始化项目
- cd my-project 进入项目根目录
- npm install 根据 package.json 安装项目依赖包
- npm start || npm run dev 启动初始化项目
开启项目
注册小程序
- src/app.json 全局配置文件
- src/App.vue 等同于原生小程序中的 app.js, 可再次写小程序应用实例的声明周期函数 || 全局样式(style 中编写)
- main.js 应用入口文件, 声明组件类型,挂载组件
入口文件介绍
代码语言:javascript复制import Vue from 'vue'
import App from './App.vue'
Vue.config.productionTip = false
// 默认为 false ,用于启动项目的时候提示信息,设置为false关闭提示
Vue.config.productionTip = true
//设置 mpType值
App.mpType = 'app'
// 生成 Vue
实例
const app = new Vue(App)
// 挂载组件
编写页面 pages/index
页面需要文件介绍
- index.vue 等同于原生中的 wxml wxss js
- main.js 当前组件页面的入口文件,用于生成当前组件实例,并挂载组件
- main.json 当前页面的局部配置文件 (注意:index.vue 组件最终会被转化为main.wxml 以及 main.wxss 文件, 所以当前的 json 文件需命名 main)
- src 源文件
5.自动打包后的 dist 文件
Index.vue 文件
代码语言:javascript复制<template>
<div class="indexContainer">
<img v-show="isShow" :src="userInfo.avatarUrl" alt="">
<button open-type="getUserInfo" @getuserinfo="handleGetUserInfo" class="btn"
v-show="!isShow">获取用户信息</button>
<p>hello {{userInfo.nickName}}</p>
<button @tap="toDetail">开启小程序之旅</button>
</div>
</template>
<script>
export default {
data(){
return {
isShow: false,
userInfo: {}
}
},
beforeMount(){
console.log('组件挂载之前')
// this.getUserInfo()
this.getUserInfo();
},
methods: {
handleGetUserInfo(msg){
console.log(this);
console.log(msg);
if(msg.mp.detail.rawData){
this.getUserInfo();
}
},
getUserInfo(){
// 获取用户信息
wx.getUserInfo({
success: (data) => {
this.isShow = true;
this.userInfo = data.userInfo
},
fail: () => {
}
})
},
toDetail(){
console.log('xxx');
wx.switchTab({
url: '/pages/list/main'
})
}
}
}
</script>
<style>
page {
background: gray;
height:100%;
}
.indexContainer {
display: flex;
flex-direction: column;
align-items: center;
}
.indexContainer img {
width:200rpx;
height:200rpx;
border-radius: 100rpx;
margin:100rpx 0;
}
.btn {
width:300rpx;
height:300rpx;
border-radius: 150rpx;
margin:100rpx 0;
line-height: 300rpx;
}
.indexContainer p {
font-size: 40rpx;
font-weight: bold;
margin:100rpx 0;
}
.indexContainer button {
background: #009ee0;
font-size: 32rpx;
}
</style>
index/main.js
代码语言:javascript复制import Vue from 'vue'
import Index from './index.vue'
const index = new Vue(Index)
index.$mount()
index/main.json
代码语言:javascript复制{
"navigationBarTitleText": "主页",
"navigationBarBackgroundColor": "#8ed145"
}
注意事项
- 在每个组件中都需要使用: 组件实例.$mount() 去挂载当前组件,否则对应的页面不能 生效
- npm run dev 每次会重新打包 dist 文件,测试只能在小程序工具上
- mpvue 中绑定小程序原生事件不能使用 bind 事件名,需要使用@事件名 且要定义在 methods 中否则不生效
- 新创建的页面需要重新执行: npm run dev 才能将新的页面打包到 dist 文件中
vue 实例声明周期 && 小程序声明周期
vue 实例声明周期
- beforeCreate 实例初始化之后,数据观测(data observer) 和 event/watcher 事件配置之前被调用。
- created 实例已经创建完成之后被调用。在这一步,实例已完成以下的配置:数据观测(data observer),属性和方法的运算, watch/event 事件回调。然而,挂载阶段还 没开始,$el 属性目前不可见。
- beforeMount 在挂载开始之前被调用:相关的 render 函数首次被调用。
- mounted el 被新创建的 vm.$el 替换,并挂载到实例上去之后调用该钩子。
- beforeUpdate 数据更新时调用,发生在虚拟 DOM 重新渲染和打补丁之前。 你可以在这个钩子中进一步地更改状态,这不会触发附加的重渲染过程。
- updated 由于数据更改导致的虚拟 DOM 重新渲染和打补丁,在这之后会调用该钩子。
- beforeDestroy 实例销毁之前调用。在这一步,实例仍然完全可用。 8.destroyed Vue 实例销毁后调用。调用后,Vue 实例指示的所有东西都会解绑定,所有的事件监听器会被移除,所有的子实例也会被销毁。 该钩子在服务器端渲染期间不被调用。
小程序应用 App 实例声明周期
- onLaunch: 小程序应用初始化
- onShow: 小程序启动获取后台进入前台
- onHide: 小程序应用从前台进入后台
小程序页面 Page 实例生命周期
- onLoad 监听页面加载
- onShow: 页面显示
- onReady: 监听页面初始化渲染完成
- onHide: 监听页面隐藏,注意当前页面实例依然存活
- onUnload: 监听页面卸载
- onPullDownRefresh: 监听用户下载动作
- onReachBottom: 监听用户上拉触底操作
- onShareAppMessage: 用户点击右上角分享功能
- onPageScroll: 页面滚动
- onTabItemTap: 当前是 tab 页时,点击 tab 时触发
注意事项
.4 注意事项 除了 Vue 本身的生命周期外,mpvue 还兼容了小程序生命周期,这部分生命周期钩子的来源于 微信小程序的 Page, 除特殊情况外,不建议使用小程序的生命周期钩子。
mpvue 中使用 vue-router && axios
vue-router
- 在 mpvue 中对 vue-router 的支持不好,问题较多
- 进行页面跳转的是可使用小程序提供的 API (1) wx.navigateTo() 保留当前页面,可回退 (2) wx.redirectTo() 不保留,不能回退 (3) wx.switchTab() 使用于 tabBar 页面
axios
- 小程序中不支持使用 axios,会报错: XMLHttpRequest is not a constructor
- 原因: 小程序的环境和浏览器的环境不一样
- 解决方法: 使用其他库: flyio
mpvue VS 小程序 状态管理
原生小程序
- 在 data 中初始化状态数据
- 修改状态: this.setData({key: value})
- 页面公共状态: a. App 程序实例的 data 中定义 b. 获取状态数据: let datas = getApp() c. 修改状态数据: datas.data.xxx = value
- 或者利用 storage 本地存储
Mpvue
- 在组件中通过 getApp 无法拿到对应的数据
- mpvue 中支持 vuex,所以可以使用 vuex 集中管理状态
- vuex 几个重要的概念: a. store 对象 b. dispatch() 分发状态 c. actions 携带参与修改状态的数据,并触发 mutations d. mutations 用于修改状态,并将状态交给 store 对象 e. getter 用于动态计算状态
原生小程序 VS mpvue 对比总结
- 原生小程序运行更稳定些, 兼容性好,mpvue 可能在某些方面存在兼容性问题(vue- router)
- mpvue 支持 vue 组件化开发. 效率更高,功能更强大(双向数据绑定, vuex)
- mpvue 可基于 webpack 组件化, 工程化开发
- 原生不支持 npm 安装包,不支持 css 预处理
- 支持 computed 计算属性和 watcher 监听器;模板语法中只支持简单的 js 表达式。可以直接写 div 、span 等标签computed 的写法
- 之前会 vue 的工程师上手 mpvue 框架的成本较低