收藏
评论

小程序 mixin 混合, Page 选项合并

  • 模仿vue中的mixin功能

  • 避免使用 Page(Object.assign({}, require('../../mixins/index.js'), {data: {}})) 结构直接覆盖

  • data数据结构浅复制

  • Page原生结构属性顺序执行

  • 其他自定义结构属性直接覆盖

  • 最主要不破坏结构,简单使用


小程序 mixin 选项合并

微信开发工具导入项目

  新建项目 -> 导入src文件 -> 预览效果

代码

mixin.js

  const nativePage = Page
  Page = options => {
    const mixins = options.mixins
    if (Array.isArray(mixins)) {
      Reflect.deleteProperty(options, 'mixins')
      merge(mixins, options)
    }
    nativePage(options)
  }

  // 原生Page属性
  const properties = ['data', 'onLoad', 'onReady', 'onShow', 'onHide', 'onUnload', 'onPullDownRefresh', 'onReachBottom', 'onShareAppMessage', 'onPageScroll', 'onTabItemTap']

  // 合并mixins属性到Page的options中
  function merge(mixins, options) {
    mixins.reverse().forEach(mixin => {
      if (Object.prototype.toString.call(mixin).slice(8, -1) === 'Object') {
        for (let [key, value] of Object.entries(mixin)) {
          if (key === 'data') {
            options.data = Object.assign({}, value, options.data)
          } else if (properties.includes(key)) {
            let native = options[key]
            options[key] = function (...args) {
              value.call(this, ...args)
              return native && native.call(this, ...args)
            }
          } else {
            options = Object.assign({}, mixin, options)
          }
        }
      }
    })
  }

使用

在 app.js 中引入 mixin.js 文件

  require('./vendor/mixin.js')

mixins/index.js

  module.exports = {
    data: {
      indexMsg: 'index msg' 
    }
  }

mixins/init.js

  module.exports = {
    data: {
      initMsg: 'init msg'
    }
  }

在 index 页面的 index.js 中使用

  Page({
    mixins: [require('../../mixins/index.js'), require('../../mixins/init.js')],
    data: {
      motto: 'Hello World',
      userInfo: {},
      hasUserInfo: false,
      canIUse: wx.canIUse('button.open-type.getUserInfo')
    }
  })

截图预览

mixin使用前后对比

<div align=center> <img width="360" height="640" src="./screenshot/mixin_before.png"/> <img width="360" height="640" src="./screenshot/mixin_after.png"/> </div>

原理说明

  • 模仿vue中的mixin功能
  • 避免使用 Page(Object.assign({}, require('../../mixins/index.js'), {data: {}})) 结构直接覆盖
  • data数据结构浅复制
  • Page原生结构属性顺序执行
  • 其他自定义结构属性直接覆盖

如有什么问题和想法,可以加我微信(15012851811),加我请注明信息

如果觉得不错,请动动您的小拇指,star一下

收藏
登录 后发表内容