前言

github地址:https://github.com/yexiaochai/wxdemo

接上文继续,我们前面学习了小程序的生命周期、小程序的标签、小程序的样式,后面我们写了一个简单的loading组件,显然他是个半成品,我们在做loading组件的时候意识到一个问题:

小程序的组件事实上是标签
我们没有办法获得标签的实例,至少我暂时没有办法
所以这些前提让我们对标签的认识有很大的不同,完成小程序特有的UI库,那么就需要从标签出发
这里面关注的点从js中的实例变成了wxml中的属性

我们今天尝试做几个组件,然后先做未完成的loading,然后做消息类弹出组件,然后做日历组件,我希望在这个过程中,我们形成一套可用的体系,这里涉及了组件体系,我们可能需要整理下流程:

① 首先我们这里做的组件其实是“标签”,这个时候就要考虑引入时候的怎么处理了

② 因为写业务页面的同事(写page的同事),需要在json配置中引入需要使用的标签:

"usingComponents": {
"ui-loading": "/components/ui-loading"
}

因为不能动态插入标签,所以需要一开始就把标签放入页面wxml中:

<ui-loading is-show="{{isLoadingShow}}"></ui-loading>

③ json中的配置暂时只能拷贝,但是我们可以提供一个ui-set.wxml来动态引入一些组件,如全局使用的loading弹出类提示框

④ 像日历类组件或者平时用的比较少的弹出层组件便需要自己在页面中引入了,工作量貌似不大,后续看看情况,如何优化

⑤ 我们这里给每个组件设置一个behaviors,behaviors原则只设置一层(这里有点继承的关系),层级多了变比较复杂了,弹出层类是一个、一般类一个(用于日历类组件)

有了以上标准,我们这里先来改造我们的loading组件

⑥ 默认所有的组件初期WXSS直接设置为隐藏

改造loading

这里首先改造弹出层都要继承的behaviors behavior-layer:

 const util = require('../utils/util.js')
module.exports = Behavior({
properties: {
//重要属性,每个组件必带,定义组件是否显示
isShow: {
type: String
}
},
//这里设置弹出层必须带有一个遮盖层,所以每个弹出层都一定具有有个z-index属性
data: {
maskzIndex: util.getBiggerzIndex(),
uiIndex: util.getBiggerzIndex()
},
attached: function() {
console.log('layer')
},
methods: {
}
})

其次我们改造下我们的mask组件:

 let LayerView = require('behavior-layer')
Component({
behaviors: [LayerView],
properties: {
//只有mask的z-index属性需要被调用的弹出层动态设置
zIndex: {
type: String
}
},
data: {
},
attached: function () {
console.log('mask')
},
methods: {
onTap: function() {
this.triggerEvent('customevent', {}, {})
}
}
})

WXML不做变化,便完成了我们的代码,并且结构关系似乎更加清晰了,但是作为loading组件其实是有个问题的,比如点击遮盖层要不要关闭整个组件,像类似这种点击遮盖层要不要关闭整个组件,其实该是一个公共属性,所以我们对我们的layer、mask继续进行改造(这里具体请看github代码):

 const util = require('../utils/util.js')
module.exports = Behavior({
properties: {
//重要属性,每个组件必带,定义组件是否显示
isShow: {
type: String
}
},
//这里设置弹出层必须带有一个遮盖层,所以每个弹出层都一定具有有个z-index属性
data: {
maskzIndex: util.getBiggerzIndex(),
uiIndex: util.getBiggerzIndex(),
//默认点击遮盖层不关闭组件
clickToHide: false
},
attached: function() {
console.log('layer')
},
methods: {
}
})
 methods: {
onMaskEvent: function (e) {
console.log(e);
//如果设置了点击遮盖层关闭组件则关闭
if (this.data.clickToHide)
this.setData({
isShow: 'none'
});
}
}

这个时候,点击要不要关闭,基本就在组件里面设置一个属性即可,但是我们这个作为了内部属性,没有释放出去,这个时候我们也许发现了另外一个比较幽默的场景了:

我们因为没法获取一个标签的实例,所以我们需要在页面里面动态调用:

 onShow: function() {
let scope= this;
this.setData({
isLoadingShow: ''
});
//3秒后关闭loading
setTimeout(function () {
scope.setData({
isLoadingShow: 'none'
});
}, 3000);
},

可以看到,标签接入到页面后,控制标签事实上是动态操作他的属性,也就是说操作页面的状态数据,页面的UI变化全部是数据触发,这样的逻辑会让界面变得更加清晰,但是作为全局类的loading这种参数,我并不想放到各个页面中,因为这样会导致很多重复代码,于是我在utils目录中新建了一个ui-util的工具类,作为一些全局类的ui公共库:

 //因为小程序页面中每个页面应该是独立的作用域
class UIUtil {
constructor(opts) {
//用于存储各种默认ui属性
this.isLoadingShow = 'none';
}
//产出页面loading需要的参数
getPageData() {
return {
isLoadingShow: this.isLoadingShow
}
}
//需要传入page实例
showLoading(page) {
this.isLoadingShow = '';
page.setData({
isLoadingShow: this.isLoadingShow
});
}
//关闭loading
hideLoading(page) {
this.isLoadingShow = 'none';
page.setData({
isLoadingShow: this.isLoadingShow
});
}
} //直接返回一个UI工具了类的实例
module.exports = new UIUtil

index.js使用上产生一点变化:

 //获取公共ui操作类实例
const uiUtil = require('../../utils/ui-util.js');
//获取应用实例
const app = getApp()
Page({
data: uiUtil.getPageData(),
onShow: function() {
let scope= this;
uiUtil.showLoading(this);
//3秒后关闭loading
setTimeout(function () {
uiUtil.hideLoading(scope);
}, 3000);
},
onLoad: function () {
}
})

这样,我们将页面里面要用于操作组件的数据全部放到了一个util类中,这样代码会变得清晰一些,组件管理也放到了一个地方,只是命名规范一定要安规则来,似乎到这里,我们的loading组件改造结束了,这里却有一个问题,我们在ui-util类中存储的事实上是页面级的数据,其中包含是组件的状态,但是真实情况我们点击遮盖层关闭组件,根本不会知会page层的数据,这个时候我们loading的显示状态搞不好是显示,而真实的组件已经关闭了,如何保证状态统一我们后面点再说,我暂时没有想到好的办法。

toast组件

我们现在先继续作toast组件,toast组件一样包含一个遮盖层,但是点击的时候可以关闭遮盖层,显示3秒后关闭,显示多久关闭的属性应该是可以配置的(作为属性传递),所以我们新增组件:

 const util = require('../utils/util.js');
let LayerView = require('behavior-layer'); Component({
behaviors: [
LayerView
],
properties: {
message: {
type: String
}
},
data: {
},
attached: function () {
console.log(this)
},
methods: {
onMaskEvent: function (e) {
console.log(e);
//如果设置了点击遮盖层关闭组件则关闭
if (this.data.clickToHide)
this.setData({
isShow: 'none'
});
}
}
})

整体代码请各位在git上面去看,这里也引起了一些问题:

① 我的组件如何居中?

② 一般来说toast消失的时候是可以定制化一个事件回调的,我们这里怎么实现?

这里我们先抛开居中问题,我们先来解决第二个问题,因为小程序中没有addEventListener这个方法,所以能够改变组件特性的方式只剩下数据操作,回顾我们这里可以引起组件隐藏的点只有:

① toast中的点击弹出层时改变显示属性

 onMaskEvent: function (e) {
console.log(e);
//如果设置了点击遮盖层关闭组件则关闭
if (this.data.clickToHide)
this.setData({
isShow: 'none'
});
}

② 然后就是页面中动态改变数据属性了:

 onShow: function() {
let scope= this;
uiUtil.showToast(this, '我是美丽可爱的toast');
//3秒后关闭loading
setTimeout(function () {
uiUtil.hideToast(scope);
}, 3000);
},

这里,我们不得不处理之前的数据同步问题了,我们应该给toast提供一个事件属性可定义的点,点击遮盖层的真正处理逻辑需要放到page层,其实认真思考下,标签就应该很纯粹,不应该与业务相关,只需要提供钩子,与业务相关的是page中的业务,这个时候大家可以看到我们代码之间的关联是多么的复杂了:

① 页面index.js依赖于index.wxml中组件的标签,并且依赖于uiUtil这个工具类

② 单单一个toast组件(标签)便依赖了mask标签,一个工具栏,还有基础的layer behavior

③ 因为不能获取实例,所以组件直接通信只能通过标签的bindevent的做法,让情况变得更加诡异

从这里看起来,调用方式也着实太复杂了,而这还仅仅是一个简单的组件,这个是不是我们写法有问题呢?答案是!我的思路还是以之前做js的组件的思路,但是小程序暂时不支持动态插入标签,所以我们不应该有过多的继承关系,其中的mask是没有必要的;另一方面,每个页面要动态引入ui-utils这个莫名其妙的组件库,似乎也很别扭,所以我们这里准备进行改造,降低没有必要的复杂度

组件改造

经过思考,我们这里准备做以下优化(PS:我小程序也是上星期开始学习的,需要逐步摸索):

① 保留mask组件,但是去除toast、loading类组件与其关联,将WXML以及样式直接内联,使用空间复杂度降低代码复杂度

② 取消ui-uitil攻击类,转而实现一个page基类

我们这里先重新实现toast组件:

 //behavior-layer
const util = require('../utils/util.js')
module.exports = Behavior({
properties: {
//重要属性,每个组件必带,定义组件是否显示
isShow: {
type: String
}
},
//这里设置弹出层必须带有一个遮盖层,所以每个弹出层都一定具有有个z-index属性
data: {
maskzIndex: util.getBiggerzIndex(),
uiIndex: util.getBiggerzIndex(),
//默认点击遮盖层不关闭组件
clickToHide: true
},
attached: function() {
console.log('layer')
},
methods: {
onMaskEvent: function (e) {
this.triggerEvent('maskevent', e, {})
}
}
})
 .cm-overlay {
background: rgba(0, 0, 0, 0.5);
position: fixed;
top:;
right:;
bottom:;
left:;
} .cm-modal {
background-color: #fff;
overflow: hidden;
width: 100%;
border-radius: 8rpx;
} .cm-modal--toast {
width: auto;
margin-top: -38rpx;
background: rgba(0, 0, 0, 0.7);
color: #fff;
padding: 20rpx 30rpx;
text-align: center;
font-size: 24rpx;
white-space: nowrap;
position: fixed;
top: 50%;
left: 50%; }
.cm-modal--toast .icon-right {
display: inline-block;
margin: 10rpx 0 24rpx 10rpx;
}
.cm-modal--toast .icon-right::before {
content: "";
display: block;
width: 36rpx;
height: 16rpx;
border-bottom: 4rpx solid #fff;
border-left: 4rpx solid #fff;
-webkit-transform: rotate(-45deg);
transform: rotate(-45deg);
-webkit-box-sizing: border-box;
box-sizing: border-box;
}
 <section class="cm-modal cm-modal--toast" style="z-index: {{uiIndex}}; display: {{isShow}}; ">
{{message}}
</section>
<view class="cm-overlay" bindtap="onMaskEvent" style="z-index: {{maskzIndex}}; display: {{isShow}}" >
</view>
 const util = require('../utils/util.js');
let LayerView = require('behavior-layer');
Component({
behaviors: [
LayerView
],
properties: {
message: {
type: String
}
},
data: {
},
attached: function () {
console.log(this)
},
methods: {
}
})

页面层的使用不必变化就已经焕然一新了,这个时候我们开始做ui-util与page关系的改造,看看能不能让我们的代码变得简单,我这里的思路是设计一个公共的abstract-view出来,做所有页面的基类:

 class Page {
constructor(opts) {
//用于基础page存储各种默认ui属性
this.isLoadingShow = 'none';
this.isToastShow = 'none';
this.toastMessage = 'toast提示'; //通用方法列表配置,暂时约定用于点击
this.methodSet = [
'onToastHide', 'showToast', 'hideToast', 'showLoading', 'hideLoading'
]; //当前page对象
this.page = null;
}
initPage(pageData) {
//debugger; let _pageData = {}; //为页面动态添加操作组件的方法
Object.assign(_pageData, this.getPageFuncs(), pageData); //生成真实的页面数据
_pageData.data = {};
Object.assign(_pageData.data, this.getPageData(), pageData.data || {}); console.log(_pageData);
return _pageData;
}
//当关闭toast时触发的事件
onToastHide(e) {
this.hideToast();
}
//设置页面可能使用的方法
getPageFuncs() {
let funcs = {};
for (let i = 0, len = this.methodSet.length; i < len; i++ ) {
funcs[this.methodSet[i]] = this[this.methodSet[i]];
}
return funcs;
}
//产出页面组件需要的参数
getPageData() {
return {
isLoadingShow: this.isLoadingShow,
isToastShow: this.isToastShow,
toastMessage: this.toastMessage
}
}
showToast(message) {
this.setData({
isToastShow: '',
toastMessage: message
});
}
hideToast() {
this.setData({
isToastShow: 'none'
});
}
//需要传入page实例
showLoading() {
this.setData({
isLoadingShow: ''
});
}
//关闭loading
hideLoading() {
this.setData({
isLoadingShow: 'none'
});
}
}
//直接返回一个UI工具了类的实例
module.exports = new Page

abstract-view

这里还提供了一个公共模板用于被页面include,abstract-view.wxml:

<ui-toast bindonToastHide="onToastHide" is-show="{{isToastShow}}" message="{{toastMessage}}"></ui-toast>

页面调用时候的代码发生了很大的变化:

<import src="./mod.searchbox.wxml" />
<view>
<template is="searchbox" />
</view>
<include src="../../utils/abstract-page.wxml"/>
 //获取公共ui操作类实例
const _page = require('../../utils/abstract-page.js');
//获取应用实例
const app = getApp() Page(_page.initPage({
data: {
ttt: 'ttt' },
// methods: uiUtil.getPageMethods(),
methods: {
},
onShow: function () {
let scope = this;
this.showToast('我是美丽可爱的toast');
// 3秒后关闭loading
// setTimeout(function () {
// scope.hideToast();
// }, 3000);
},
onLoad: function () {
// this.setPageMethods();
}
}))

这样我们相当于变相给page赋能了,详情请各位看github上的代码:https://github.com/yexiaochai/wxdemo,这个时候,我们要为toast组件添加关闭时候的事件回调,就变得相对简单了,事实上我们可以看到这个行为已经跟组件本身没有太多关系了:

 showToast(message, callback) {
this.toastHideCallback = null;
if (callback) this.toastHideCallback = callback;
let scope = this;
this.setData({
isToastShow: '',
toastMessage: message
}); // 3秒后关闭loading
setTimeout(function () {
scope.hideToast();
}, 3000);
}
hideToast() {
this.setData({
isToastShow: 'none'
});
if (this.toastHideCallback) this.toastHideCallback.call(this);
}
this.showToast('我是美丽可爱的toast', function () { console.log('执行回调')} );

当然这里可以做得更加人性化,比如显示时间是根据message长度动态设置的,我们这里先这样。

alert类组件

本篇篇幅已经比较长了,我们最后完成一个alert组件便结束今天的学习,明天主要实现日历等组件,alert组件一般是一个带确定框的提示弹出层,有可能有两个按钮,那个情况要稍微复杂点,我们这里依旧为其新增组件结构wxml以及wxss:

 //获取公共ui操作类实例
const _page = require('../../utils/abstract-page.js');
//获取应用实例
const app = getApp() Page(_page.initPage({
data: {
},
// methods: uiUtil.getPageMethods(),
methods: {
},
onShow: function () {
global.sss = this;
let scope = this;
this.showMessage({
message: '我是一个确定框',
ok: {
name: '确定',
callback: function () {
scope.hideMessage();
scope.showMessage('我选择了确定');
}
},
cancel: {
name: '取消',
callback: function () {
scope.hideMessage();
scope.showToast('我选择了取消');
}
}
}); },
onLoad: function () {
// this.setPageMethods();
}
}))

结语

github地址:https://github.com/yexiaochai/wxdemo

今天我们似乎找到了一个适合小程序的组件编写方式,明天我们继续完成一些组件,组件完成后我们便开始写实际业务代码了

微信小程序开发04-打造自己的UI库的更多相关文章

  1. 微信小程序开发 [04] 模板和模块化

    1.模板 如果相同的wxml代码可能在不同的页面重复使用,ctrl+c配合ctrl+v的方式,后期维护起来未免也太麻烦了.微信提供了"模板",可以在模板中定义代码片段,然后在不同的 ...

  2. 微信小程序开发05-日历组件的实现

    接上文:微信小程序开发04-打造自己的UI库 github地址:https://github.com/yexiaochai/wxdemo 我们这里继续实现我们的日历组件,这个日历组件稍微有点特殊,算是 ...

  3. 微信小程序开发教程,大多数人都搞错的八个问题

    小程序目前被炒得沸沸扬扬,无数媒体和企业借机获取阅读流量. 这再次证明一点,微信想让什么火,真的就能让什么火.这种能力真是全中国再也没有人有了,政府也没有. 但四处传的消息很多是失真的,废话不说,先列 ...

  4. 微信小程序开发心得

    微信小程序也已出来有一段时间了,最近写了几款微信小程序项目,今天来说说感受. 首先开发一款微信小程序,最主要的就是针对于公司来运营的,因为,在申请appid(微信小程序ID号)时候,需要填写相关的公司 ...

  5. 【微信小程序开发•系列文章六】生命周期和路由

    这篇文章理论的知识比较多一些,都是个人观点,描述有失妥当的地方希望读者指出. [微信小程序开发•系列文章一]入门 [微信小程序开发•系列文章二]视图层 [微信小程序开发•系列文章三]数据层 [微信小程 ...

  6. 微信小程序开发日记——高仿知乎日报(下)

    本人对知乎日报是情有独钟,看我的博客和github就知道了,写了几个不同技术类型的知乎日报APP 要做微信小程序首先要对html,css,js有一定的基础,还有对微信小程序的API也要非常熟悉 我将该 ...

  7. 微信小程序开发日记——高仿知乎日报(中)

    本人对知乎日报是情有独钟,看我的博客和github就知道了,写了几个不同技术类型的知乎日报APP要做微信小程序首先要对html,css,js有一定的基础,还有对微信小程序的API也要非常熟悉 我将该教 ...

  8. 微信小程序开发日记——高仿知乎日报(上)

    本人对知乎日报是情有独钟,看我的博客和github就知道了,写了几个不同技术类型的知乎日报APP 要做微信小程序首先要对html,css,js有一定的基础,还有对微信小程序的API也要非常熟悉 我将该 ...

  9. 微信小程序开发工具测评

    1月9日微信小程序正式上线.很多企业都希望能在这个.但是在技术开发的问题上,却不知道该如何下手.经过一些程序员不辞辛苦连夜测试,终于从十余款工具呕心沥血筛选出四款比较靠谱实用的微信小程序开发工具.接下 ...

随机推荐

  1. 【bzoj 3309 】 DZY Loves Math

    Description 对于正整数n,定义f(n)为n所含质因子的最大幂指数.例如f(1960)=f(2^3 * 5^1 * 7^2)=3, f(10007)=1, f(1)=0.给定正整数a,b,求 ...

  2. bzoj 4832 抵制克苏恩 概率期望dp

    考试时又翻车了..... 一定要及时调整自己的思路!!! 随从最多有7个,只有三种,所以把每一种随从多开一维 so:f[i][j][k][l]为到第i次攻击前,场上有j个1血,k个2血,l个3血随从的 ...

  3. BZOJ_2738_矩阵乘法_整体二分

    BZOJ_2738_矩阵乘法_整体二分 Description 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. Input 第一行两个数N,Q,表示矩阵大小和询问组数: 接下 ...

  4. BZOJ_1858_[Scoi2010]序列操作_线段树

    BZOJ_1858_[Scoi2010]序列操作_线段树 Description lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询 ...

  5. VIJOS-P1635 城市连接

    嘿嘿嘿,逆向spfa,貌似不难... #include <cstdio> #include <algorithm> #include <cmath> #includ ...

  6. vue iview UPload,但文件上传是,clearFiles的使用方法

    <template> <div> <button @click="clearUploadedImage">重新上传</button> ...

  7. 轻量化卷积神经网络MobileNet论文详解(V1&V2)

    本文是 Google 团队在 MobileNet 基础上提出的 MobileNetV2,其同样是一个轻量化卷积神经网络.目标主要是在提升现有算法的精度的同时也提升速度,以便加速深度网络在移动端的应用.

  8. Django设置查看原生SQL语句

    LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'handlers': { 'console':{ 'level':'DEBU ...

  9. 微服务架构 - 离线部署k8s平台并部署测试实例

    一般在公司部署或者真实环境部署k8s平台,很有可能是内网环境,也即意味着是无法连接互联网的环境,这时就需要离线部署k8s平台.在此整理离线部署k8s的步骤,分享给大家,有什么不足之处,欢迎指正. 1. ...

  10. 使用Http-Repl工具测试ASP.NET Core 2.2中的Web Api项目

    今天,Visual Studio中没有内置工具来测试WEB API.使用浏览器,只能测试http GET请求.您需要使用Postman,SoapUI,Fiddler或Swagger等第三方工具来执行W ...