赞
踩
移动互联网的初期,囿于设备硬件性能限制,流量以原生App为主,iOS、Android是当时两大平台。
随着硬件及OS的更新换代,H5可承载的体验逐步完善,为提高开发效率、节约资源(复用代码)以及热更新等目的,Hybrid模式成为主流;以及轻应用、服务号等平台的助推,H5网页流量暴涨,成为第三大平台。
2017年1月9日,微信发布小程序,历经3年发展,在今年主题为”未完成 Always Beta“的微信公开课 PRO上,微信团队披露,2019年小程序日活跃用户超过 3 亿,全年累计成交额达8000亿,同比增长超160%。
看到小程序如此惊人的增长力,我们有理由相信,有中国特色的小程序互联网时代已经到来,微信小程序也已成为继iOS、Android、H5之后的第四大流量平台。
平台分裂,为不同平台编写相同的业务代码,是件无趣的事情。
有追求的程序员,一直在探索代码复用的方案,Hybrid App即是代表。
而在如今的小程序时代,对于同样基于WEB技术的H5和小程序,如何实现代码复用,是很多前端工程师探索的方向。业内也已有不少成熟方案,从场景上来说,大致分为三类:
uni-app
、京东凹凸实验室的taro
最近微信官方推出的kbone,也是为了解决“把 Web 端的代码挪到小程序环境内执行”;不过,kbone 相比 mpvue 前进了一步(当然也有了新的性能缺陷),因为:
kbone实现了一个适配器,在适配层里模拟出了浏览器环境,让 Web 端的代码可以不做什么改动便可运行在小程序里。
- 复用小程序代码,转换小程序代码在web环境中运行;适用于有小程序代码沉淀,未开发H5或H5平台完善度较低的开发者;这个方向业内成熟的方案还比较少。
uni-app
近期支持了小程序自定义组件运行到H5平台,是对如上第三种场景的一种探索。
鉴于小程序的低成本获客特征,很多厂商选择先开发小程序,验证业务模式后,再扩展至H5、App等其它平台。
开发者虽可借助转换器将小程序代码转换为uni-app
项目(或其它跨端框架项目),快速实现多平台发行;但不少开发者是不敢轻易决策将跨端版本替换之前线上的小程序版本的,毕竟线上版本已稳定运行了一段时间。
常选的方案是:让原生小程序版本和uni-app
跨端版本并行一段时间,微信平台继续使用原生版本,其它平台使用uni-app
跨端版本;经过一段时间验证uni-app
版本稳定后,再使用uni-app
版替换掉原生小程序版本。
在这段并行的时间内,开发者需要同时维护微信原生、uni-app
两个版本,新增业务需编写两份逻辑相同的代码,重复劳动,成本叠加,如何改善?
借助uni-app
支持将微信小程序组件运行到H5平台的特性,我们给出一种思路:
wxml/wxss/js/json
文件到uni-app
项目下,通过uni-app
的编译器及运行时,保证小程序自定义组件在H5平台的正确运行。这个方案的好处是:
优先小程序开发,毕竟小程序早已上线,有存量用户
复用小程序组件,新增业务仅需开发一套代码即可,降低开发成本
不止自己开发的小程序组件,业内开源的三方小程序组件,均可复制到uni-app
项目项目中,运行到H5平台。
另外,部分公司的产品经理,会要求不同平台有不同的交互,但核心业务逻辑是相同的,开发者常会通过维护不同项目的方式来满足产品经理需求。此时,采取如上方案,同样可满足多个项目复用相同业务逻辑的诉求。
实际上,uni-app
之前已支持将小程序自定义组件运行到App平台,对于有小程序组件沉淀或优先小程序的开发者来说,这是个好消息,一套业务组件,快速运行到iOS、Android、H5、微信小程序这四大流量平台(实际上也可运行到QQ小程序平台)。
uni-app
项目中使用自定义组件的方法很简单,分为三步:
1、拷贝小程序自定义组件到uni-app
项目根目录下的wxcomponents
文件夹下
2、在 pages.json
对应页面的 style -> usingComponents
引入组件,如:
复制代码- {
- "pages": [
- {
- "path": "index/index",
- "style": {
- "usingComponents": {
- "custom": "/wxcomponents/custom/index"
- }
- }
- }
- ]
- }
3、在页面中使用自定义组件,如:
复制代码- <!-- 页面模板 (index.vue) -->
- <view>
- <!-- 在页面中对自定义组件进行引用 -->
- <custom name="uni-app"></custom>
- </view>
简单介绍下uni-app
的多端发行原理。
uni-app
基于Vue.js
runtime,页面文件遵循Vue
.js 单文件组件 (SFC) 规范,天然对H5的支持比较好,发行到H5平台时,先通过vue-loader
解析.vue
文件,导出Vue.js
组件选项对象,然后在运行时补充规范实现:
uni-app发行到小程序平台时,逻辑又有不同,主要工作有2块:
.vue
文件拆分成wxml/wxss/js/json
4个原生页面文件Vue.js
和小程序都是逻辑视图层框架,都有数据绑定功能;运行时会实现Vue.js
到小程序的数据同步,及小程序到Vue.js
的事件代理小程序自定义组件类似小程序原生的页面开发,一个自定义组件同样由wxml/wxss/js/json
4个文件组成,另有单独的组件规范(如Component
构造器、Behaviors
特性等)。
所以,小程序自定义组件运行到H5平台,可借助uni-app
已有平台功能快速实现:
wxml/wxss/js/json
4个文件合并为.vue
文件(类似 uni-app
发行到小程序的逆过程),然后调用uni-app
发行H5平台的编译过程,通过vue-loader
解析.vue
文件,导出 Vue.js
组件选项对象Component
构造器、Behaviors
特性,模拟自定义组件特有的生命周期小程序自定义组件发行到H5平台,在编译环节主要有2项工作:
将自定义组件的wxml/wxss/js/json
4个文件组成,编译转换成.vue
文件,即小程序转vue,可简写为mp2vue
通过vue-loader
解析.vue
文件,导出 Vue.js
组件选项对象
其中,步骤2是Vue.js
项目的标准编译过程,略过不提;我们重点阐述步骤1。
mp2vue
将4个独立wxml/wxss/js/json
的文件合并成一个.vue
文件,并组装成template
、script
、style
这种三段式的结构,流程包括:
wxml
文件生成template
节点,同时完成指令、事件等模板语法转换js/json
文件生成script
节点,同时完成组件注册过wxss
文件生成style
节点,自动转换部分css兼容语法.vue
文件具体实现上,uni-app
编译前先扫描wxcomponents
目录,若存在则认为有小程序自定义组件,启动文件转换工作(uni-migration
插件来完成):
复制代码- //加载转换器
- const migrate = require('@dcloudio/uni-migration')
- //扫描wxcomponents目录
- const wxcomponents = path.resolve(process.env.UNI_INPUT_DIR, 'wxcomponents')
- if (fs.existsSync(wxcomponents)) {
- migrate(wxcomponents, false, {
- silent: true
- }) // 转换 mp-weixin 小程序组件
- }
接着开始对wxml/wxss/js/json
文件逐个解析,并合并为一个.vue
文件:
复制代码- module.exports = function transformFile(input, options) {
- //首先转换json文件,判断是否为组件
- const [jsCode, isComponent] = transformJsonFile(filepath + '.json', deps)
- options.isComponent = isComponent
- //转换 wxml 文件
- const [templateCode, wxsCode = '', wxsFiles = []] = transformTemplateFile(filepath + templateExtname, options)
- //转换wxss文件
- const styleCode = transformStyleFile(filepath + styleExtname, options, deps) || ''
- //转换js文件
- const scriptCode = transformScriptFile(filepath + '.js', jsCode, options, deps)
- // 生成合并后的.vue文件源码
- return [
- `${commentsCode}<template>
- ${templateCode}
- </template>
- ${wxsCode}
- <script>
- ${scriptCode}
- </script>
- <style platform="mp-weixin">
- ${styleCode}
- </style>`,
- deps,
- wxsFiles
- ]
- }

进一步细节说明,wxml文件转为template节点时,需完成各项指令、事件等模板语法的转换,例如:
小程序自定义组件 | Vue组件 | 描述 |
---|---|---|
wx:if | v-if | 条件渲染 |
wx:for | v-for | 列表渲染 |
bindtap | @click | 元素点击事件 |
将一个最简自定义组件,按照如上流程转换,结果示意如下:
uni-app
的编译器并不转换小程序组件的 JS 代码,依然保留Component
构造器的写法,甚至其中的API依然是wx.
开头的方式,这些都依赖uni-app
在H5平台的运行时来解决,主要有如下几部分内容:
Component
构造器:解析小程序组件的各种选项配置,转换为Vue
组件定义,包括变通实现其中的差异部分,如小程序组件特有的”组件所在页面的生命周期“Behaviors
特性:转换为Vue的混入(mixin)setData
接口及this.data.xx = yy
的数据通讯机制wx.xx
替换为uni.xx
,这个比较简单,不详述uni-app
在H5平台定义了一个Component
函数,执行到小程序的Component
构造器函数后,开始循环解析其属性,并转换成Vue组件属性,流程示意代码如下:
复制代码- export function Component (options) {
- const componentOptions = parseComponent(options)
- componentOptions.mixins.unshift(polyfill)
- componentOptions.mpOptions.path = global['__wxRoute']
- initRelationsHandler(componentOptions)
- global['__wxComponents'][global['__wxRoute']] = componentOptions
- }
-
- export function parseComponent (mpComponentOptions) {
- const {
- data,
- options,
- methods,
- behaviors,
- lifetimes,
- observers,
- relations,
- properties,
- pageLifetimes,
- externalClasses
- } = mpComponentOptions
-
- const vueComponentOptions = {
- mixins: [],
- props: {},
- watch: {},
- mpOptions: {
- mpObservers: []
- }
- }
-
- // 开始逐个解析所有属性
- parseData(data, vueComponentOptions)
- parseOptions(options, vueComponentOptions)
- parseMethods(methods, vueComponentOptions)
- parseBehaviors(behaviors, vueComponentOptions)
- parseLifetimes(lifetimes, vueComponentOptions)
- parseObservers(observers, vueComponentOptions)
- parseRelations(relations, vueComponentOptions)
- parseProperties(properties, vueComponentOptions)
- parsePageLifetimes(pageLifetimes, vueComponentOptions)
- parseExternalClasses(externalClasses, vueComponentOptions)
- parseLifecycle(mpComponentOptions, vueComponentOptions)
- parseDefinitionFilter(mpComponentOptions, vueComponentOptions)
- // 返回 Vue 组件
- return vueComponentOptions
- }

在这个过程中,需处理小程序自定义组件和 Vue组件的属性对应关系及细节差异,如小程序组件的lifetimes
:
小程序自定义组件 | Vue/uni-app | 描述 |
---|---|---|
created | onServiceCreated | 小程序的created 触发时,可以访问子组件信息,而Vue 的created 访问不到,故需uni-app 框架映射到其它时机(onServiceCreated)执行 |
attached | onServiceAttached | 同上 |
ready | mounted | Vue 生命周期 |
moved | - | Vue中不存在该钩子,暂不支持转换 |
detached | destroyed | Vue 生命周期 |
小程序的pageLifetimes (组件所在页面的生命周期)在Vue中是没有的,需要映射为uni-app 封装的页面生命周期: | 小程序自定义组件 | uni-app | 描述 |
---|---|---|---|
ready | onPageShow | 页面被展示时执行 | |
hide | onPageHide | 页面被隐藏时执行 | |
resize | onPageResize | 页面尺寸变化时执行 |
Behaviors
特性的实现过程,类似Component
构造器,不再赘述。
Vue
和小程序都有一套数据绑定系统,但机制不同,比如在Vue体系下,数据赋值是这样的:
复制代码this.a = 1
但在小程序中,数据赋值方式则是这样的:
复制代码- this.setData({
- a:1
- }) //响应式
- this.data.a = 2 //非响应式
另外,小程序和Vue
在数据的properties
、observer
等方面都存在不少差异,经过我们评估,若将小程序的数据响应用法直接映射到Vue
体系下,复杂度较高且有性能压力,故uni-app
在H5平台按照微信的语法规范,单独实现了一套数据响应系统。
复制代码- // 小程序的setData在H5平台的实现
- function setData (data, callback) {
- if (!isPlainObject(data)) {
- return
- }
- Object.keys(data).forEach(key => {
- if (setDataByExprPath(key, data[key], this.data)) {
- !hasOwn(this, key) && proxy(this, SOURCE_KEY, key);
- }
- });
- this.$forceUpdate();//数据变化,强制视图更新(响应式)
- isFn(callback) && this.$nextTick(callback);
- }
将setData
挂载到 vm 对象上,可通过this.setData
这种小程序的方式调用;同时将数据绑定到data属性上,支持this.data.xx
的访问方式。
复制代码- export function initState (vm) {
- const instanceData = JSON.parse(JSON.stringify(vm.$options.mpOptions.data || {}))
- vm[SOURCE_KEY] = instanceData
-
- //vm对象上挂载 setData 方法,实现小程序方法
- vm.setData = setData
-
- const propertyDefinition = {
- get () {
- return vm[SOURCE_KEY]
- },
- set (value) {
- vm[SOURCE_KEY] = value
- }
- }
- //小程序用法,可通过this.data.xx访问
- Object.defineProperties(vm, {
- data: propertyDefinition,
- properties: propertyDefinition
- })
-
- Object.keys(instanceData).forEach(key => {
- proxy(vm, SOURCE_KEY, key)
- })
- }

虽然数据响应是uni-app
自己实现的,但渲染依然使用了Vue框架的render
函数,此时需小程序规范中的this.data.xx
和Vue规范中的this.xx
保持一致,通过代理的方式实现:
复制代码- // mp/polyfill/state/proxy.js
- const sharedPropertyDefinition = {
- enumerable: true,
- configurable: true
- };
-
- function proxy (target, sourceKey, key) {
- sharedPropertyDefinition.get = function proxyGetter () {
- return this[sourceKey][key]
- };
- sharedPropertyDefinition.set = function proxySetter (val) {
- this[sourceKey][key] = val;
- };
- Object.defineProperty(target, key, sharedPropertyDefinition);
- }
这里仅列出了主要的几步,中间涉及细节很多;部分无法通过Vue
扩展机制实现的功能,只好修改Vue.js
的内核源码,比如updateProperties
支持、小程序wxs
、externalClasses
等功能在H5平台的支持,都需要定制部分 Vue.js runtime 源码。
本文分享了uni-app
将微信小程序自定义组件发行到H5平台的实现思路,希望对大家有所启发。
但这种方案,归根到底是为了解决多套项目并存时的业务重复开发的问题。
如果你是从头开发,我们建议直接选择业内成熟的跨端框架,既可以保持一套代码,更省力的维护,还可以借助框架的成熟生态(如跨端UI库及插件市场),基于成熟轮子,快速完成业务的上线开发;
uni-app
框架代码,包括小程序组件发行到H5平台的代码,全部开源在github,如果大家对本文逻辑有疑问,欢迎提交issue交流。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。