mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
[refactor] Noticebar:升级到自定义组件 (#165)
* [improvement] Tab:升级到自定义组件 * fix: 去除冗余example代码 * [refactor] 重构badge为自定义组件 (#160) * fix: 去除tab组件使用对象入参方式,修改example用例 * refactor: 重构noticebar组件 * fix: 去除tab组件冗余属性字段
This commit is contained in:
parent
a1b70a8437
commit
86152d45ce
@ -1,23 +1,28 @@
|
||||
var Zan = require('../../dist/index');
|
||||
|
||||
Page(Object.assign({}, Zan.NoticeBar, {
|
||||
Page({
|
||||
data: {
|
||||
movable: {
|
||||
bar1: {
|
||||
text: '足协杯战线连续第2年上演广州德比战,上赛季半决赛上恒大以两回合5-3的总比分淘汰富力。',
|
||||
scrollable: true,
|
||||
delay: 1000
|
||||
},
|
||||
bar2: {
|
||||
text: '足协杯战线连续第2年上演广州德比战',
|
||||
color: '#fff',
|
||||
backgroundColor: '#000'
|
||||
},
|
||||
bar3: {
|
||||
text: '足协杯战线连续第2年上演广州德比战,上赛季半决赛上恒大以两回合5-3的总比分淘汰富力。'
|
||||
},
|
||||
static1: {
|
||||
text: '足协杯战线连续第2年上演广州德比战'
|
||||
bar4: {
|
||||
text: '带icon的公告',
|
||||
leftIcon: 'https://img.yzcdn.cn/public_files/2017/8/10/6af5b7168eed548100d9041f07b7c616.png'
|
||||
},
|
||||
static2: {
|
||||
text: '足协杯战线连续第2年上演广州德比战,上赛季半决赛上恒大以两回合5-3的总比分淘汰富力。'
|
||||
bar5: {
|
||||
text: '足协杯战线连续第2年上演广州德比战,上赛季半决赛上恒大以两回合5-3的总比分淘汰富力。',
|
||||
leftIcon: 'https://img.yzcdn.cn/public_files/2017/8/10/6af5b7168eed548100d9041f07b7c616.png',
|
||||
mode: 'closeable',
|
||||
scrollable: true,
|
||||
speed: 10
|
||||
}
|
||||
},
|
||||
onShow() {
|
||||
// 滚动通告栏需要initScroll
|
||||
this.initZanNoticeBarScroll('movable');
|
||||
// initScroll的通告栏如果宽度足够显示内容将保持静止
|
||||
this.initZanNoticeBarScroll('static1');
|
||||
// 不进行initScroll的通告栏也将保持静止
|
||||
// this.initZanNoticeBarScroll('static2');
|
||||
}
|
||||
}))
|
||||
})
|
||||
|
@ -1,3 +1,6 @@
|
||||
{
|
||||
"navigationBarTitleText": "Noticebar 通告栏"
|
||||
"navigationBarTitleText": "Noticebar 通告栏",
|
||||
"usingComponents": {
|
||||
"zan-noticebar": "../../dist/noticebar/index"
|
||||
}
|
||||
}
|
||||
|
@ -1,30 +1,63 @@
|
||||
<import src="/dist/noticebar/index.wxml" />
|
||||
<view class="container">
|
||||
|
||||
<view class="doc-title zan-hairline--bottom">NOTICEBAR</view>
|
||||
|
||||
<view class="zan-panel-title">滚动通告栏</view>
|
||||
<view class="zan-panel">
|
||||
<template
|
||||
is="zan-noticebar"
|
||||
data="{{ ...movable, componentId: 'movable' }}"
|
||||
></template>
|
||||
<zan-noticebar
|
||||
text="{{ bar1.text }}"
|
||||
scrollable="{{ bar1.scrollable }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">静止通告栏1</view>
|
||||
<view class="zan-panel-title">延时滚动通告栏</view>
|
||||
<view class="zan-panel">
|
||||
<template
|
||||
is="zan-noticebar"
|
||||
data="{{ ...static1, componentId: 'static1' }}"
|
||||
></template>
|
||||
<zan-noticebar
|
||||
text="{{ bar1.text }}"
|
||||
scrollable="{{ bar1.scrollable }}"
|
||||
delay="{{ bar1.delay }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">静止通告栏2</view>
|
||||
<view class="zan-panel-title">初始速度低滚动通告栏</view>
|
||||
<view class="zan-panel">
|
||||
<template
|
||||
is="zan-noticebar"
|
||||
data="{{ ...static2, componentId: 'static2' }}"
|
||||
></template>
|
||||
<zan-noticebar
|
||||
text="{{ bar1.text }}"
|
||||
scrollable="{{ bar1.scrollable }}"
|
||||
speed="{{ bar5.speed }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">改变颜色通告栏</view>
|
||||
<view class="zan-panel">
|
||||
<zan-noticebar
|
||||
text="{{ bar2.text }}"
|
||||
color="{{ bar2.color }}"
|
||||
background-color="{{ bar2.backgroundColor }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">静止通告栏</view>
|
||||
<view class="zan-panel">
|
||||
<zan-noticebar
|
||||
text="{{ bar3.text }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">带icon公告</view>
|
||||
<view class="zan-panel">
|
||||
<zan-noticebar
|
||||
text="{{ bar4.text }}"
|
||||
left-icon="{{ bar4.leftIcon }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
<view class="zan-panel-title">可关闭公告</view>
|
||||
<view class="zan-panel">
|
||||
<zan-noticebar
|
||||
text="{{ bar5.text }}"
|
||||
mode="{{ bar5.mode }}"
|
||||
/>
|
||||
</view>
|
||||
|
||||
</view>
|
@ -1,6 +1,3 @@
|
||||
const interval = 50;
|
||||
let moduleId = 1;
|
||||
|
||||
Page({
|
||||
data: {
|
||||
tab1: {
|
||||
|
@ -1,49 +1,95 @@
|
||||
## Noticebar 通告栏
|
||||
|
||||
### 使用指南
|
||||
在 app.wxss 中引入组件库所有样式
|
||||
```css
|
||||
@import "path/to/zanui-weapp/dist/index.wxss";
|
||||
在 index.json 中引入组件
|
||||
```json
|
||||
{
|
||||
"usingComponents": {
|
||||
"zan-noticebar": "path/to/zanui-weapp/dist/noticebar/index"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
在需要使用的页面里引入组件库模板和脚本
|
||||
```html
|
||||
<import src="path/to/zanui-weapp/dist/noticebar/index.wxml" />
|
||||
|
||||
<!-- 直接使用 zan-noticebar 模板,并且直接传入设置值 -->
|
||||
<template is="zan-noticebar" data="{{ ...data, componentId: 'noticebar' }}"></template>
|
||||
```
|
||||
|
||||
// 在 Page 中混入 Noticebar 里面声明的方法
|
||||
在 index.js 中声明组件数据
|
||||
```js
|
||||
const { Noticebar, extend } = require('path/to/zanui-weapp/dist/index');
|
||||
|
||||
Page(extend({}, Noticebar, {
|
||||
// ...
|
||||
}));
|
||||
// 在 Page 中声明 Noticebar 依赖的展示数据
|
||||
Page({
|
||||
data: {
|
||||
text: 'xxx',
|
||||
scrollable: 'xxx',
|
||||
...
|
||||
}
|
||||
})
|
||||
```
|
||||
|
||||
### 代码演示
|
||||
`Noticebar` 组件支持滚动和静止两种展示方式,通过 text 传入展示文案
|
||||
|
||||
### 静止公告栏
|
||||
```html
|
||||
<template is="zan-noticebar" data="{{ text: '展示文案', componentId: 'noticebar' }}"></template>
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
/>
|
||||
```
|
||||
|
||||
**注意**
|
||||
|
||||
如果组件需要开启滚动展示,需要在 Page 的脚本中执行 initZanNoticeBarScroll 方法,来开启滚动展示
|
||||
```js
|
||||
Page(extend({}, Noticebar, {
|
||||
// ...
|
||||
onShow() {
|
||||
// 在方法中传入对应的 componentId
|
||||
this.initZanNoticeBarScroll('movable');
|
||||
}
|
||||
// ...
|
||||
}));
|
||||
### 滚动通告栏
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
scrollable="true"
|
||||
/>
|
||||
```
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 | 必须 |
|
||||
### 延时滚动通告栏
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
scrollable="true"
|
||||
delay="{{ delay }}"
|
||||
/>
|
||||
```
|
||||
|
||||
### 改变滚动通告栏滚动速度
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
scrollable="true"
|
||||
speed="{{ speed }}"
|
||||
/>
|
||||
```
|
||||
|
||||
### 自定义通告栏字体颜色和背景色
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
color="{{ color }}"
|
||||
background-color="{{ backgroundColor }}"
|
||||
/>
|
||||
```
|
||||
|
||||
### 添加左侧icon通告栏
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
left-icon="https://img.yzcdn.cn/public_files/2017/8/10/6af5b7168eed548100d9041f07b7c616.png"
|
||||
/>
|
||||
```
|
||||
|
||||
### 可关闭通告栏
|
||||
```html
|
||||
<zan-noticebar
|
||||
text="{{ text }}"
|
||||
mode="closeable"
|
||||
/>
|
||||
```
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 | 可选值 |
|
||||
|-----------|-----------|-----------|-------------|-------------|
|
||||
| text | 通告栏展示文案 | String | - | |
|
||||
| componentId | 用于区分页面多个 Noticebar 组件,在调用 initZanNoticeBarScroll 时需要传入 | String | - | |
|
||||
| mode | 通告栏模式 | String | '' | `closeable` |
|
||||
| delay | 滚动延时时间 | Number | 0 | |
|
||||
| speed | 滚动速度 | Number | 40 | |
|
||||
| scrollable | 是否可滚动 | Boolean | false | |
|
||||
| leftIcon | 左侧图标 | String | - | |
|
||||
| color | 通告栏字体颜色 | String | `#f60` | |
|
||||
| backgroundColor | 通告栏背景色 | String | `#fff7cc` |
|
||||
|
@ -1,74 +1,165 @@
|
||||
const ZanNoticeBar = {
|
||||
initZanNoticeBarScroll(componentId) {
|
||||
this.zanNoticeBarNode = this.zanNoticeBarNode || {};
|
||||
this.zanNoticeBarNode[`${componentId}`] = {
|
||||
width: undefined,
|
||||
wrapWidth: undefined,
|
||||
animation: null,
|
||||
resetAnimation: null
|
||||
};
|
||||
const VALID_MODE = ['closeable'];
|
||||
const FONT_COLOR = '#f60';
|
||||
const BG_COLOR = '#fff7cc';
|
||||
|
||||
const currentComponent = this.zanNoticeBarNode[`${componentId}`];
|
||||
wx.createSelectorQuery()
|
||||
Component({
|
||||
properties: {
|
||||
text: {
|
||||
type: String,
|
||||
value: ''
|
||||
},
|
||||
mode: {
|
||||
type: String,
|
||||
value: ''
|
||||
},
|
||||
url: {
|
||||
type: String,
|
||||
value: ''
|
||||
},
|
||||
openType: {
|
||||
type: String,
|
||||
value: 'navigate'
|
||||
},
|
||||
delay: {
|
||||
type: Number,
|
||||
value: 0
|
||||
},
|
||||
speed: {
|
||||
type: Number,
|
||||
value: 40
|
||||
},
|
||||
scrollable: {
|
||||
type: Boolean,
|
||||
value: false
|
||||
},
|
||||
leftIcon: {
|
||||
type: String,
|
||||
value: ''
|
||||
},
|
||||
color: {
|
||||
type: String,
|
||||
value: FONT_COLOR
|
||||
},
|
||||
backgroundColor: {
|
||||
type: String,
|
||||
value: BG_COLOR
|
||||
}
|
||||
},
|
||||
|
||||
data: {
|
||||
show: true,
|
||||
hasRightIcon: false,
|
||||
width: undefined,
|
||||
wrapWidth: undefined,
|
||||
elapse: undefined,
|
||||
animation: null,
|
||||
resetAnimation: null,
|
||||
timer: null
|
||||
},
|
||||
|
||||
attached() {
|
||||
const { mode } = this.data;
|
||||
if (mode && this._checkMode(mode)) {
|
||||
this.setData({
|
||||
hasRightIcon: true
|
||||
});
|
||||
}
|
||||
},
|
||||
|
||||
ready() {
|
||||
this._init();
|
||||
},
|
||||
|
||||
methods: {
|
||||
_checkMode(val) {
|
||||
const isValidMode = ~VALID_MODE.indexOf(val);
|
||||
if (!isValidMode) {
|
||||
console.warn(`mode only accept value of ${VALID_MODE}, now get ${val}.`);
|
||||
}
|
||||
return isValidMode;
|
||||
},
|
||||
|
||||
_init() {
|
||||
wx.createSelectorQuery()
|
||||
.in(this)
|
||||
.select(`#${componentId}__content`)
|
||||
.boundingClientRect((rect) => {
|
||||
.select('.zan-noticebar__content')
|
||||
.boundingClientRect(rect => {
|
||||
if (!rect || !rect.width) {
|
||||
console.warn('页面缺少 noticebar 元素');
|
||||
throw new Error('页面缺少 noticebar 元素');
|
||||
return;
|
||||
}
|
||||
this.setData({
|
||||
width: rect.width
|
||||
});
|
||||
|
||||
currentComponent.width = rect.width;
|
||||
wx
|
||||
.createSelectorQuery()
|
||||
wx.createSelectorQuery()
|
||||
.in(this)
|
||||
.select(`#${componentId}__content-wrap`)
|
||||
.select('.zan-noticebar__content-wrap')
|
||||
.boundingClientRect((rect) => {
|
||||
if (!rect || !rect.width) {
|
||||
return;
|
||||
}
|
||||
|
||||
clearTimeout(this.data[componentId].setTimeoutId)
|
||||
const wrapWidth = rect.width;
|
||||
const { width, speed, scrollable, delay } = this.data;
|
||||
|
||||
currentComponent.wrapWidth = rect.width;
|
||||
if (currentComponent.wrapWidth < currentComponent.width) {
|
||||
var mstime = currentComponent.width / 40 * 1000;
|
||||
currentComponent.animation = wx.createAnimation({
|
||||
duration: mstime,
|
||||
timingFunction: 'linear'
|
||||
if (scrollable && wrapWidth < width) {
|
||||
const elapse = width / speed * 1000;
|
||||
console.log(`delay: ${delay}`)
|
||||
const animation = wx.createAnimation({
|
||||
duration: elapse,
|
||||
timeingFunction: 'linear',
|
||||
delay
|
||||
});
|
||||
currentComponent.resetAnimation = wx.createAnimation({
|
||||
const resetAnimation = wx.createAnimation({
|
||||
duration: 0,
|
||||
timingFunction: 'linear'
|
||||
timeingFunction: 'linear'
|
||||
});
|
||||
|
||||
this.setData({
|
||||
elapse,
|
||||
wrapWidth,
|
||||
animation,
|
||||
resetAnimation
|
||||
}, () => {
|
||||
this._scroll();
|
||||
});
|
||||
this.scrollZanNoticeBar(componentId, mstime);
|
||||
}
|
||||
})
|
||||
.exec();
|
||||
.exec();
|
||||
})
|
||||
.exec();
|
||||
},
|
||||
.exec();
|
||||
},
|
||||
|
||||
scrollZanNoticeBar(componentId, mstime) {
|
||||
const currentComponent = this.zanNoticeBarNode[`${componentId}`];
|
||||
const resetAnimationData = currentComponent.resetAnimation.translateX(currentComponent.wrapWidth).step();
|
||||
this.setData({
|
||||
[`${componentId}.animationData`]: resetAnimationData.export()
|
||||
});
|
||||
const aninationData = currentComponent.animation.translateX(-mstime * 40 / 1000).step();
|
||||
setTimeout(() => {
|
||||
_scroll() {
|
||||
const { animation, resetAnimation, wrapWidth, elapse, speed } = this.data;
|
||||
const resetAnimationData = resetAnimation.translateX(wrapWidth).step();
|
||||
const animationData = animation.translateX(-elapse * speed / 1000).step();
|
||||
this.setData({
|
||||
[`${componentId}.animationData`]: aninationData.export()
|
||||
animationData: resetAnimation.export()
|
||||
});
|
||||
}, 100);
|
||||
setTimeout(() => {
|
||||
this.setData({
|
||||
animationData: animationData.export()
|
||||
})
|
||||
}, 100);
|
||||
|
||||
const setTimeoutId = setTimeout(() => {
|
||||
this.scrollZanNoticeBar(componentId, mstime);
|
||||
}, mstime);
|
||||
this.setData({
|
||||
[`${componentId}.setTimeoutId`]: setTimeoutId
|
||||
})
|
||||
const timer = setTimeout(() => {
|
||||
this._scroll();
|
||||
}, elapse);
|
||||
|
||||
this.setData({
|
||||
timer
|
||||
});
|
||||
},
|
||||
|
||||
_handleButtonClick() {
|
||||
const { timer } = this.data;
|
||||
timer && clearTimeout(timer);
|
||||
this.setData({
|
||||
show: false,
|
||||
timer: null
|
||||
});
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
module.exports = ZanNoticeBar;
|
||||
})
|
6
packages/noticebar/index.json
Normal file
6
packages/noticebar/index.json
Normal file
@ -0,0 +1,6 @@
|
||||
{
|
||||
"component": true,
|
||||
"usingComponents": {
|
||||
"zan-icon": "../icon/index"
|
||||
}
|
||||
}
|
@ -1,7 +1,43 @@
|
||||
.zan-noticebar {
|
||||
color: #f60;
|
||||
display: flex;
|
||||
padding: 9px 10px;
|
||||
font-size: 12px;
|
||||
line-height: 1.5;
|
||||
background-color: #fff7cc;
|
||||
|
||||
&--within-icon {
|
||||
position: relative;
|
||||
padding-right: 30px;
|
||||
}
|
||||
|
||||
&__left-icon {
|
||||
height: 18px;
|
||||
min-width: 20px;
|
||||
padding-top: 1px;
|
||||
box-sizing: border-box;
|
||||
|
||||
> image {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
}
|
||||
}
|
||||
|
||||
&__right-icon {
|
||||
position: absolute;
|
||||
top: 10px;
|
||||
right: 10px;
|
||||
font-size: 15px;
|
||||
line-height: 1;
|
||||
}
|
||||
|
||||
&__content-wrap {
|
||||
position: relative;
|
||||
flex: 1;
|
||||
height: 18px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
&__content {
|
||||
position: absolute;
|
||||
white-space: nowrap;
|
||||
}
|
||||
}
|
||||
|
@ -1,16 +1,31 @@
|
||||
<template name="zan-noticebar">
|
||||
<view class="zan-noticebar">
|
||||
<view
|
||||
id="{{ componentId }}__content-wrap"
|
||||
style="height: 18px; overflow: hidden; position: relative;"
|
||||
>
|
||||
<view
|
||||
animation="{{ animationData }}"
|
||||
id="{{ componentId }}__content"
|
||||
style="position: absolute; white-space: nowrap;"
|
||||
>
|
||||
{{ text }}
|
||||
</view>
|
||||
<view
|
||||
wx:if="{{ show }}"
|
||||
class="zan-noticebar {{ hasRightIcon ? 'zan-noticebar--within-icon' : '' }}"
|
||||
style="color: {{ color }};background-color: {{ backgroundColor }}"
|
||||
>
|
||||
<view wx:if="{{ leftIcon }}" class="zan-noticebar__left-icon">
|
||||
<image src="{{ leftIcon }}" />
|
||||
</view>
|
||||
<view class="zan-noticebar__content-wrap">
|
||||
<view class="zan-noticebar__content" animation="{{ animationData }}">
|
||||
{{ text }}
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<block wx:if="{{ mode }}">
|
||||
<zan-icon
|
||||
wx:if="{{ mode === 'closeable' }}"
|
||||
class="zan-noticebar__right-icon"
|
||||
type="close"
|
||||
bindtap="_handleButtonClick"
|
||||
/>
|
||||
<navigator
|
||||
wx:if="{{ mode === 'link' }}"
|
||||
url="{{ url }}"
|
||||
open-type="{{ openType }}"
|
||||
>
|
||||
<zan-icon class="zan-noticebar__right-icon" type="arrow" />
|
||||
</navigator>
|
||||
</block>
|
||||
|
||||
</view>
|
||||
|
@ -5,9 +5,13 @@
|
||||
```json
|
||||
{
|
||||
"usingComponents": {
|
||||
"zan-tab": "path/to/zanui-weapp/dist/icon/index"
|
||||
"zan-tab": "path/to/zanui-weapp/dist/tab/index"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
在 index.js 中声明组件数据
|
||||
```js
|
||||
// 在 Page 中声明 Tab 依赖的展示数据
|
||||
Page({
|
||||
data: {
|
||||
@ -36,15 +40,14 @@ Page({
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 | 必须 |
|
||||
|-----------|-----------|-----------|-------------|-------------|
|
||||
| tab.scroll | 是否开启 tab 左右滑动模式 | Boolean | - | |
|
||||
| tab.list | 可选项列表 | Array | - | |
|
||||
| tab.selectedId | 选中id | - | - | |
|
||||
| tab.height | tab高度 | Number | - | |
|
||||
| tab.fixed | 是否固定位置 | Boolean | - | |
|
||||
| componentId | 用于区分页面多个 tab 组件 | String | - | |
|
||||
| scroll | 是否开启 tab 左右滑动模式 | Boolean | - | |
|
||||
| list | 可选项列表 | Array | - | |
|
||||
| selectedId | 选中id | - | - | |
|
||||
| height | tab高度 | Number | - | |
|
||||
| fixed | 是否固定位置 | Boolean | - | |
|
||||
|
||||
|
||||
tab 组件中,tab.list 数据格式如下
|
||||
tab 组件中,list 数据格式如下
|
||||
```js
|
||||
[{
|
||||
// tab 项 id
|
||||
|
@ -26,10 +26,6 @@ Component({
|
||||
currentTab: newVal
|
||||
});
|
||||
}
|
||||
},
|
||||
componentId: {
|
||||
type: String,
|
||||
default: ''
|
||||
}
|
||||
},
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user