在开发小程序的时候,咱们老是指望用以往的技术规范和语法特色来书写当前的小程序,因此才会有各色的小程序框架,例如 mpvue、taro 等这些编译型框架。固然这些框架自己对于新开发的项目是有所帮助。而对于老项目,咱们又想要利用 vue 的语法特性进行维护,又该如何呢?
在此我研究了一下youzan的 vant-weapp。而发现该项目中的组件是如此编写的。html
import { VantComponent } from '../common/component'; VantComponent({ mixins: [], props: { name: String, size: String }, // 可使用 watch 来监控 props 变化 // 其实就是把properties中的observer提取出来 watch: { name(newVal) { ... }, // 能够直接使用字符串 代替函数调用 size: 'changeSize' }, // 使用计算属性 来 获取数据,能够在 wxml直接使用 computed: { bigSize() { return this.data.size + 100 } }, data: { size: 0 }, methods: { onClick() { this.$emit('click'); }, changeSize(size) { // 使用set this.set(size) } }, // 对应小程序组件 created 周期 beforeCreate() {}, // 对应小程序组件 attached 周期 created() {}, // 对应小程序组件 ready 周期 mounted() {}, // 对应小程序组件 detached 周期 destroyed: {} });
竟然发现该组件写法总体上相似于 Vue 语法。而自己却没有任何编译。看来问题是出在了导入的 VantComponet 这个方法上。下面咱们开始详细介绍一下如何利用 VantComponet 来对老项目进行维护。vue
小程序组件写法这里就再也不介绍。这里咱们给出利用 VantComponent 写 Page 的代码风格。c++
import { VantComponent } from '../common/component'; VantComponent({ mixins: [], props: { a: String, b: Number }, // 在页面这里 watch 基本上是没有做用了,由于只作了props 变化的watch,page不会出现 props 变化 // 后面会详细说明为什么 watch: {}, // 计算属性仍旧可用 computed: { d() { return c++ } }, methods: { onLoad() {} }, created() {}, // 其余组件生命周期 })
这里你可能感到疑惑,VantComponet 不是对组件 Component 生效的吗?怎么会对页面 Page 生效呢。事实上,咱们是可使用组件来构造小程序页面的。
在官方文档中,咱们能够看到 使用 Component 构造器构造页面
事实上,小程序的页面也能够视为自定义组件。于是,页面也可使用 Component 构造器构造,拥有与普通组件同样的定义段与实例方法。代码编写以下:小程序
Component({ // 可使用组件的 behaviors 机制,虽然 React 以为 mixins 并非一个很好的方案 // 可是在某种程度该方案的确能够复用相同的逻辑代码 behaviors: [myBehavior], // 对应于page的options,与此自己是有类型的,而从options 取得数据均为 string类型 // 访问 页面 /pages/index/index?paramA=123¶mB=xyz // 若是声明有属性 paramA 或 paramB ,则它们会被赋值为 123 或 xyz,而不是 string类型 properties: { paramA: Number, paramB: String, }, methods: { // onLoad 不须要 option // 可是页面级别的生命周期却只能写道 methods中来 onLoad() { this.data.paramA // 页面参数 paramA 的值 123 this.data.paramB // 页面参数 paramB 的值 ’xyz’ } } })
那么组件的生命周期和页面的生命周期又是怎么对应的呢。通过一番测试,得出结果为: (为了简便。只会列出 重要的的生命周期)缓存
// 组件实例被建立 到 组件实例进入页面节点树 component created -> component attched -> // 页面页面加载 到 组件在视图层布局完成 page onLoad -> component ready -> // 页面卸载 到 组件实例被从页面节点树移除 page OnUnload -> component detached
固然 咱们重点不是在 onload 和 onunload 中间的状态,由于中间状态的时候,咱们能够在页面中使用页面生命周期来操做更好。
某些时候咱们的一些初始化代码不该该放在 onload 里面,咱们能够考虑放在 component create 进行操做,甚至能够利用 behaviors 来复用初始化代码。
某种方面来讲,若是不须要 Vue 风格,咱们在老项目中直接利用 Component 代替 Page 也不失为一个不错的维护方案。毕竟官方标准,不用担忧其余一系列后续问题。app
此时,咱们对 VantComponent 开始进行解析框架
// 赋值,根据 map 的 key 和 value 来进行操做 function mapKeys(source: object, target: object, map: object) { Object.keys(map).forEach(key => { if (source[key]) { // 目标对象 的 map[key] 对应 源数据对象的 key target[map[key]] = source[key]; } }); } // ts代码,也就是 泛型 function VantComponent<Data, Props, Watch, Methods, Computed>( vantOptions: VantComponentOptions< Data, Props, Watch, Methods, Computed, CombinedComponentInstance<Data, Props, Watch, Methods, Computed> > = {} ): void { const options: any = {}; // 用function 来拷贝 新的数据,也就是咱们能够用的 Vue 风格 mapKeys(vantOptions, options, { data: 'data', props: 'properties', mixins: 'behaviors', methods: 'methods', beforeCreate: 'created', created: 'attached', mounted: 'ready', relations: 'relations', destroyed: 'detached', classes: 'externalClasses' }); // 对组件间关系进行编辑,可是page不须要,能够删除 const { relation } = vantOptions; if (relation) { options.relations = Object.assign(options.relations || {}, { [`../${relation.name}/index`]: relation }); } // 对组件默认添加 externalClasses,可是page不须要,能够删除 // add default externalClasses options.externalClasses = options.externalClasses || []; options.externalClasses.push('custom-class'); // 对组件默认添加 basic,封装了 $emit 和小程序节点查询方法,能够删除 // add default behaviors options.behaviors = options.behaviors || []; options.behaviors.push(basic); // map field to form-field behavior // 默认添加 内置 behavior wx://form-field // 它使得这个自定义组件有相似于表单控件的行为。 // 能够研究下文给出的 内置behaviors if (vantOptions.field) { options.behaviors.push('wx://form-field'); } // add default options // 添加组件默认配置,多slot options.options = { multipleSlots: true,// 在组件定义时的选项中启用多slot支持 // 若是这个 Component 构造器用于构造页面 ,则默认值为 shared // 组件的apply-shared,能够研究下文给出的 组件样式隔离 addGlobalClass: true }; // 监控 vantOptions observe(vantOptions, options); // 把当前从新配置的options 放入Component Component(options); }
刚刚咱们谈到 basic behaviors,代码以下所示xss
export const basic = Behavior({ methods: { // 调用 $emit组件 其实是使用了 triggerEvent $emit() { this.triggerEvent.apply(this, arguments); }, // 封装 程序节点查询 getRect(selector: string, all: boolean) { return new Promise(resolve => { wx.createSelectorQuery() .in(this)[all ? 'selectAll' : 'select'](selector) .boundingClientRect(rect => { if (all && Array.isArray(rect) && rect.length) { resolve(rect); } if (!all && rect) { resolve(rect); } }) .exec(); }); } } });
小程序 watch 和 computed的 代码解析函数
export function observe(vantOptions, options) { // 从传入的 option中获得 watch computed const { watch, computed } = vantOptions; // 添加 behavior options.behaviors.push(behavior); /// 若是有 watch 对象 if (watch) { const props = options.properties || {}; // 例如: // props: { // a: String // }, // watch: { // a(val) { // // 每次val变化时候打印 // consol.log(val) // } } Object.keys(watch).forEach(key => { // watch只会对prop中的数据进行 监视 if (key in props) { let prop = props[key]; if (prop === null || !('type' in prop)) { prop = { type: prop }; } // prop的observer被watch赋值,也就是小程序组件自己的功能。 prop.observer = watch[key]; // 把当前的key 放入prop props[key] = prop; } }); // 通过此方法 // props: { // a: { // type: String, // observer: (val) { // console.log(val) // } // } // } options.properties = props; } // 对计算属性进行封装 if (computed) { options.methods = options.methods || {}; options.methods.$options = () => vantOptions; if (options.properties) { // 监视props,若是props发生改变,计算属性自己也要变 observeProps(options.properties); } } }
如今剩下的也就是 observeProps 以及 behavior 两个文件了,这两个都是为了计算属性而生成的,这里咱们先解释 observeProps 代码
export function observeProps(props) { if (!props) { return; } Object.keys(props).forEach(key => { let prop = props[key]; if (prop === null || !('type' in prop)) { prop = { type: prop }; } // 保存以前的 observer,也就是上一个代码生成的prop let { observer } = prop; prop.observer = function() { if (observer) { if (typeof observer === 'string') { observer = this[observer]; } // 调用以前保存的 observer observer.apply(this, arguments); } // 在发生改变的时候调用一次 set 来重置计算属性 this.set(); }; // 把修改的props 赋值回去 props[key] = prop; }); }
最终 behavior,也就算 computed 实现机制
// 异步调用 setData function setAsync(context: Weapp.Component, data: object) { return new Promise(resolve => { context.setData(data, resolve); }); }; export const behavior = Behavior({ created() { if (!this.$options) { return; } // 缓存 const cache = {}; const { computed } = this.$options(); const keys = Object.keys(computed); this.calcComputed = () => { // 须要更新的数据 const needUpdate = {}; keys.forEach(key => { const value = computed[key].call(this); // 缓存数据不等当前计算数值 if (cache[key] !== value) { cache[key] = needUpdate[key] = value; } }); // 返回须要的更新的 computed return needUpdate; }; }, attached() { // 在 attached 周期 调用一次,算出当前的computed数值 this.set(); }, methods: { // set data and set computed data // set可使用callback 和 then set(data: object, callback: Function) { const stack = []; // set时候放入数据 if (data) { stack.push(setAsync(this, data)); } if (this.calcComputed) { // 有计算属性,一样也放入 stack中,可是每次set都会调用一次,props改变也会调用 stack.push(setAsync(this, this.calcComputed())); } return Promise.all(stack).then(res => { // 全部 data以及计算属性都完成后调用callback if (callback && typeof callback === 'function') { callback.call(this); } return res; }); } } });