如何在原生微信小程序中实现数据双向绑定

官网:qiu8310.github.io/minapp/html

做者:Moravue

在原生小程序开发中,数据流是单向的,没法双向绑定,可是要实现双向绑定的功能仍是蛮简单的!git

下文要讲的是小程序框架 minapp 中实现双向绑定的原理,在 minapp 中,你只须要在 wxml 模板中给组件的属性名后加上 .sync 就能够实现双向绑定。下面为了解释其原理,过程可能会说的稍微复杂些,但其实 minapp 框架已经处理了那些繁杂的细节!github

首先,要使数据双向绑定,应该避免过多的数据源。 在数据从上到下天然流动的状况下,若是每一个组件中都维护它们本身的数据,而又要保持它们数据值的一致,这虽然能够作到,但实现过程并不会简单。 可是也不必说为了有一个统一的数据源就使用 mobxredux 来全局管理数据,这就有点杀鸡用牛刀的感受了。 因为双向绑定只存在于父子组件之间,而数据又是从父到子传递的,因此能够优先使用父组件中的数据为数据源, 子组件每次更新数据并不更新它本身内部的数据,而是经过事件机制触发父组件更新它的数据,而父组件更新数据后又会将更新的数据天然地传给子组件, 由此达到数据的双向流动!redux

data-stream

并非全部数据都须要双向绑定,也并非全部数据都是对外的,子组件还能够有它本身的一个内部数据。因此这就涉及到咱们要说的第二个问题:区分哪些数据须要双向绑定,哪些数据又须要子组件本身维护小程序

用过 vue 的应该都知道,在 vue 中要实现双向绑定,须要在模板中作特殊处理。好比要将父组件的 parentAttr 双向绑定到子组件的 childAttr 上,须要在父组件的模板中这样写:微信小程序

<child childAttr.sync="parentAttr" />
复制代码

可是小程序并无这样的简单的语法,小程序的 wxml 语言的属性名中甚至都不容许出现 " . " 这样的字符。回到咱们的问题上来,子组件须要知道哪些属性须要双向绑定,哪些属性须要本身维护, 给模板加个字段(syncAttrMap)专门来告诉子组件须要双向绑定的数据集合不就好了么。如,能够将上面的示例写成微信小程序支持的写法:微信

<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" />

<!-- 若是同时存在多个双向绑定和不须要双向绑定的属性时,能够写成下面这样: p1, p2 分别双向绑定到子组件的 c1, c2,而 p3 单向绑定到 c3 上 -->

<child c1="{{p1}}" c2="{{p2}}" c3="{{p3}}" syncAttrMap="c1=p1&c2=p2" />
复制代码

接着,就须要处理子组件数据更新的问题了,在子组件中有两部分数据,一部分是内部数据,另外一部分是父组件中的数据, 子组件能够经过读取属性 syncAttrMap 来获得哪些数据是内部的数据,哪些数据是父组件的数据,而且能够知道对应 的父组件中的数据的键名是什么。因为原生的组件方法 setData 不会管你是内部数据,仍是父组件中的数据,只要 你调用它去更新数据,它只会更新内部的数据。因此须要另外实现一个新的方法,来自动判断数据源,若是是内部数据, 则直接调用 setData ;若是是双向绑定中的父组件数据,则能够触发一个事件去通知父组件去更新对应的值。app

因此根据上面的描述,父组件须要有个监听函数,子组件须要有个智能的 setData 函数。不防将父组件的监听函数 命名为 onSyncAttrUpdate,将子组件的智能 setData 函数命名为 setDataSmart,则能够有以下代码:框架

// 父组件
Component({
  methods: {
    onSyncAttrUpdate(e) {
      this.setData(e.detail) // 子组件传来的须要更新的数据
    }
  }
})

复制代码
<!-- 父组件的模板 -->
<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" bind:syncAttrUpdate="onSyncAttrUpdate" />
复制代码
// 子组件
Component({
  properties: {
    childAttr: String,
    syncAttrMap: String
  },
  methods: {
    // 子组件更新数据时,只要调用此方法便可,而不是 `setData`
    setDataSmart(data) {
      // splitDataBySyncAttrMap 函数的实现过程就不说了,只是将对象拆分,你们应该都能实现
      let {parentData, innerData} = splitDataBySyncAttrMap(data, this.data.syncAttrMap)

      // 内部数据使用 setData 更新
      if (Object.keys(innerData).length) {
        this.setData(innerData) // setData 中还支持 callback 的回调,为了简化代码,这里不讨论
      }

      // 双向绑定的父组件数据触发事件让父组件本身去更新
      if (Object.keys(parentData).length) {
        this.triggerEvent('syncAttrUpdate', parentData)
      }
    }
  }
})

复制代码

到此,一个简单的双向绑定功能就完成了。可是因为子组件也有可能包含其它组件,也就是说子组件也能够是父组件,而父组件一样也 能够是子组件。因此上面的 onSyncAttrUpdate setDataSmart 函数须要在每一个组件中都实现,因此不防 定义一个公共对象 BaseComponent 来实现上面的全部功能,如:

// BaseComponent
const BaseComponent = {
  properties: {
    syncAttrMap: String
  },
  methods: {
    setDataSmart() {
      // ...
    },
    onSyncAttrUpdate() {
      // ...
    }
  }
}
复制代码

而后将 BaseComponent minin 到每一个组件的对象上去就能够了;另外小程序中还有一个特殊的组件:Page,虽然 Page 和 Component 结构是两样的, 但它也应该算是一个组件,不过它必定是父组件,不多是别的组件的子组件,因此还须要将 onSyncAttrUpdate 方法写了全部的 Page 定义中。 全部这些就是 minapp 的双向绑定的基本原理了。

等等,最后还有一件事:wxml 模板,不能让用户每次写双向绑定的时候都要写那么复杂语句吧?固然不用,minapp 在编译时,会将模板作个简单的转化:

<child childAttr.sync="parentAttr" />

<!-- 因为属性名 syncAttrMap 是固定的,因此彻底能够经过编译手段,将上面的模板转成下面这个模板 -->

<child childAttr="{{parentAttr}}" syncAttrMap="childAttr=parentAttr" />
复制代码

谢谢,文章到此结束,欢迎关注 minapp:从新定义微信小程序的开发


2019年,iKcamp原创新书《Koa与Node.js开发实战》已在京东、天猫、亚马逊、当当开售啦!

相关文章
相关标签/搜索