mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-05 19:41:45 +08:00
feat(CountDown): add CountDown (#2135)
This commit is contained in:
parent
a01f966dea
commit
95476ae754
@ -6,6 +6,7 @@
|
||||
"pages/card/index",
|
||||
"pages/cell/index",
|
||||
"pages/col/index",
|
||||
"pages/count-down/index",
|
||||
"pages/dialog/index",
|
||||
"pages/field/index",
|
||||
"pages/icon/index",
|
||||
@ -61,6 +62,7 @@
|
||||
"van-checkbox": "./dist/checkbox/index",
|
||||
"van-checkbox-group": "./dist/checkbox-group/index",
|
||||
"van-col": "./dist/col/index",
|
||||
"van-count-down": "./dist/count-down/index",
|
||||
"van-dialog": "./dist/dialog/index",
|
||||
"van-divider": "./dist/divider/index",
|
||||
"van-field": "./dist/field/index",
|
||||
|
@ -121,6 +121,10 @@ export default [
|
||||
path: '/collapse',
|
||||
title: 'Collapse 折叠面板'
|
||||
},
|
||||
{
|
||||
path: '/count-down',
|
||||
title: 'CountDown 倒计时'
|
||||
},
|
||||
{
|
||||
path: '/notice-bar',
|
||||
title: 'NoticeBar 通告栏'
|
||||
|
34
example/pages/count-down/index.js
Normal file
34
example/pages/count-down/index.js
Normal file
@ -0,0 +1,34 @@
|
||||
import Page from '../../common/page';
|
||||
import Toast from '../../dist/toast/toast';
|
||||
|
||||
Page({
|
||||
data: {
|
||||
time: 30 * 60 * 60 * 1000,
|
||||
timeData: {}
|
||||
},
|
||||
|
||||
onChange(e) {
|
||||
this.setData({
|
||||
timeData: e.detail
|
||||
});
|
||||
},
|
||||
|
||||
start() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.start();
|
||||
},
|
||||
|
||||
pause() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.pause();
|
||||
},
|
||||
|
||||
reset() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.reset();
|
||||
},
|
||||
|
||||
finished() {
|
||||
Toast('倒计时结束');
|
||||
}
|
||||
});
|
3
example/pages/count-down/index.json
Normal file
3
example/pages/count-down/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"navigationBarTitleText": "CountDown 倒计时"
|
||||
}
|
49
example/pages/count-down/index.wxml
Normal file
49
example/pages/count-down/index.wxml
Normal file
@ -0,0 +1,49 @@
|
||||
<demo-block title="基础用法">
|
||||
<van-count-down time="{{ time }}" />
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="自定义格式">
|
||||
<van-count-down
|
||||
time="{{ time }}"
|
||||
format="DD 天 HH 时 mm 分 ss 秒"
|
||||
/>
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="毫秒级渲染">
|
||||
<van-count-down
|
||||
millisecond
|
||||
time="{{ time }}"
|
||||
format="HH:mm:ss:SSS"
|
||||
/>
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="自定义样式">
|
||||
<van-count-down
|
||||
useCustom
|
||||
time="{{ time }}"
|
||||
bind:change="onChange"
|
||||
>
|
||||
<text class="item">{{ timeData.hours }}</text>
|
||||
<text class="item">{{ timeData.minutes }}</text>
|
||||
<text class="item">{{ timeData.seconds }}</text>
|
||||
</van-count-down>
|
||||
</demo-block>
|
||||
|
||||
<demo-block title="手动控制">
|
||||
<van-count-down
|
||||
class="control-count-down"
|
||||
millisecond
|
||||
time="{{ 3000 }}"
|
||||
auto-start="{{ false }}"
|
||||
format="ss:SSS"
|
||||
bind:finish="finished"
|
||||
/>
|
||||
|
||||
<van-grid clickable column-num="3">
|
||||
<van-grid-item text="开始" icon="play-circle-o" bindclick="start" />
|
||||
<van-grid-item text="暂停" icon="pause-circle-o" bindclick="pause" />
|
||||
<van-grid-item text="重置" icon="replay" bind:click="reset" />
|
||||
</van-grid>
|
||||
</demo-block>
|
||||
|
||||
<van-toast id="van-toast" />
|
15
example/pages/count-down/index.wxss
Normal file
15
example/pages/count-down/index.wxss
Normal file
@ -0,0 +1,15 @@
|
||||
.van-count-down {
|
||||
margin: 0 16px 10px;
|
||||
}
|
||||
|
||||
.item {
|
||||
display: inline-block;
|
||||
width: 22px;
|
||||
margin-right: 5px;
|
||||
color: #fff;
|
||||
font-size: 12px;
|
||||
text-align: center;
|
||||
background-color: #1989fa;
|
||||
border-radius: 2px;
|
||||
}
|
||||
|
@ -88,6 +88,11 @@
|
||||
@collapse-item-content-background-color: @white;
|
||||
@collapse-item-title-disabled-color: @gray;
|
||||
|
||||
// CountDown
|
||||
@count-down-text-color: @text-color;
|
||||
@count-down-font-size: @font-size-md;
|
||||
@count-down-line-height: 20px;
|
||||
|
||||
// Info
|
||||
@info-size: 16px;
|
||||
@info-color: @white;
|
||||
|
180
packages/count-down/README.md
Normal file
180
packages/count-down/README.md
Normal file
@ -0,0 +1,180 @@
|
||||
# CountDown 倒计时
|
||||
|
||||
### 引入
|
||||
|
||||
在`app.json`或`index.json`中引入组件,详细介绍见[快速上手](#/quickstart#yin-ru-zu-jian)
|
||||
|
||||
```json
|
||||
"usingComponents": {
|
||||
"van-count-down": "path/to/vant-weapp/dist/count-down/index"
|
||||
}
|
||||
```
|
||||
|
||||
## 代码演示
|
||||
|
||||
### 基本用法
|
||||
|
||||
`time`属性表示倒计时总时长,单位为毫秒
|
||||
|
||||
```html
|
||||
<van-count-down time="{{ time }}" />
|
||||
```
|
||||
|
||||
```js
|
||||
Page({
|
||||
data: {
|
||||
time: 30 * 60 * 60 * 1000
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
### 自定义格式
|
||||
|
||||
通过`format`属性设置倒计时文本的内容
|
||||
|
||||
```html
|
||||
<van-count-down
|
||||
time="{{ time }}"
|
||||
format="DD 天 HH 时 mm 分 ss 秒"
|
||||
/>
|
||||
```
|
||||
|
||||
### 毫秒级渲染
|
||||
|
||||
倒计时默认每秒渲染一次,设置`millisecond`属性可以开启毫秒级渲染
|
||||
|
||||
```html
|
||||
<van-count-down
|
||||
millisecond
|
||||
time="{{ time }}"
|
||||
format="HH:mm:ss:SSS"
|
||||
/>
|
||||
```
|
||||
|
||||
### 自定义样式
|
||||
|
||||
通过`bind:change`事件获取`timeData`对象,格式见下方表格
|
||||
|
||||
```html
|
||||
<van-count-down
|
||||
useCustom
|
||||
time="{{ time }}"
|
||||
bind:change="onChange"
|
||||
>
|
||||
<text class="item">{{ timeData.hours }}</text>
|
||||
<text class="item">{{ timeData.minutes }}</text>
|
||||
<text class="item">{{ timeData.seconds }}</text>
|
||||
</van-count-down>
|
||||
```
|
||||
|
||||
```js
|
||||
|
||||
Page({
|
||||
data: {
|
||||
time: 30 * 60 * 60 * 1000,
|
||||
timeData: {}
|
||||
},
|
||||
|
||||
onChange(e) {
|
||||
this.setData({
|
||||
timeData: e.detail
|
||||
});
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
```css
|
||||
.item {
|
||||
display: inline-block;
|
||||
width: 22px;
|
||||
margin-right: 5px;
|
||||
color: #fff;
|
||||
font-size: 12px;
|
||||
text-align: center;
|
||||
background-color: #1989fa;
|
||||
border-radius: 2px;
|
||||
}
|
||||
```
|
||||
|
||||
### 手动控制
|
||||
|
||||
通过 `selectComponent` 选择器获取到组件实例后,可以调用`start`、`pause`、`reset`方法
|
||||
|
||||
```html
|
||||
<van-count-down
|
||||
class="control-count-down"
|
||||
millisecond
|
||||
time="{{ 3000 }}"
|
||||
auto-start="{{ false }}"
|
||||
format="ss:SSS"
|
||||
bind:finish="finished"
|
||||
/>
|
||||
|
||||
<van-grid clickable column-num="3">
|
||||
<van-grid-item text="开始" icon="play-circle-o" bindclick="start" />
|
||||
<van-grid-item text="暂停" icon="pause-circle-o" bindclick="pause" />
|
||||
<van-grid-item text="重置" icon="replay" bindclick="reset" />
|
||||
</van-grid>
|
||||
```
|
||||
|
||||
```js
|
||||
Page({
|
||||
start() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.start();
|
||||
},
|
||||
|
||||
pause() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.pause();
|
||||
},
|
||||
|
||||
reset() {
|
||||
const countDown = this.selectComponent('.control-count-down');
|
||||
countDown.reset();
|
||||
},
|
||||
|
||||
finished() {
|
||||
Toast('倒计时结束');
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
## API
|
||||
|
||||
### Props
|
||||
|
||||
| 参数 | 说明 | 类型 | 默认值 | 版本 |
|
||||
|------|------|------|------|------|
|
||||
| time | 倒计时时长,单位毫秒 | *number* | - | - |
|
||||
| format | 时间格式,DD-日,HH-时,mm-分,ss-秒,SSS-毫秒 | *string* | `HH:mm:ss` | - |
|
||||
| auto-start | 是否自动开始倒计时 | *boolean* | `true` | - |
|
||||
| millisecond | 是否开启毫秒级渲染 | *boolean* | `false` | - |
|
||||
| useCustom | 是否自定义样式 | *boolean* | `false` | - |
|
||||
|
||||
### Events
|
||||
|
||||
| 事件名 | 说明 | 回调参数 |
|
||||
|------|------|------|
|
||||
| change | 时间变化时触发 | timeData |
|
||||
| finish | 倒计时结束时触发 | - |
|
||||
|
||||
### timeData 格式
|
||||
|
||||
| 名称 | 说明 | 类型 |
|
||||
|------|------|------|
|
||||
| days | 剩余天数 | *number* |
|
||||
| hours | 剩余小时 | *number* |
|
||||
| minutes | 剩余分钟 | *number* |
|
||||
| seconds | 剩余秒数 | *number* |
|
||||
| milliseconds | 剩余毫秒 | *number* |
|
||||
|
||||
### 方法
|
||||
|
||||
通过 selectComponent 可以获取到 CountDown 实例并调用实例方法
|
||||
|
||||
| 方法名 | 参数 | 返回值 | 介绍 |
|
||||
|------|------|------|------|
|
||||
| start | - | - | 开始倒计时 |
|
||||
| pause | - | - | 暂停倒计时 |
|
||||
| reset | - | - | 重设倒计时,若`auto-start`为`true`,重设后会自动开始倒计时 |
|
3
packages/count-down/index.json
Normal file
3
packages/count-down/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"component": true
|
||||
}
|
7
packages/count-down/index.less
Normal file
7
packages/count-down/index.less
Normal file
@ -0,0 +1,7 @@
|
||||
@import '../common/style/var.less';
|
||||
|
||||
.van-count-down {
|
||||
color: @count-down-text-color;
|
||||
font-size: @count-down-font-size;
|
||||
line-height: @count-down-line-height;
|
||||
}
|
112
packages/count-down/index.ts
Normal file
112
packages/count-down/index.ts
Normal file
@ -0,0 +1,112 @@
|
||||
import { VantComponent } from '../common/component';
|
||||
import { isSameSecond, parseFormat, parseTimeData } from './utils';
|
||||
|
||||
VantComponent({
|
||||
props: {
|
||||
time: {
|
||||
type: Number,
|
||||
observer: 'reset',
|
||||
},
|
||||
format: {
|
||||
type: String,
|
||||
value: 'HH:mm:ss'
|
||||
},
|
||||
useCustom: {
|
||||
type: Boolean,
|
||||
value: false,
|
||||
},
|
||||
autoStart: {
|
||||
type: Boolean,
|
||||
value: true,
|
||||
},
|
||||
millisecond: {
|
||||
type: Boolean,
|
||||
value: false,
|
||||
},
|
||||
},
|
||||
|
||||
data: {
|
||||
timeData: parseTimeData(0),
|
||||
formattedTime: '0'
|
||||
},
|
||||
|
||||
methods: {
|
||||
// 开始
|
||||
start() {
|
||||
if (this.counting) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.counting = true;
|
||||
this.endTime = Date.now() + this.remain;
|
||||
this.tick();
|
||||
},
|
||||
|
||||
// 暂停
|
||||
pause() {
|
||||
this.counting = false;
|
||||
clearTimeout(this.tid);
|
||||
},
|
||||
|
||||
// 重置
|
||||
reset() {
|
||||
this.pause();
|
||||
this.remain = this.data.time;
|
||||
this.setRemain(this.remain);
|
||||
|
||||
if (this.data.autoStart) {
|
||||
this.start();
|
||||
}
|
||||
},
|
||||
|
||||
tick() {
|
||||
if (this.data.millisecond) {
|
||||
this.microTick();
|
||||
} else {
|
||||
this.macroTick();
|
||||
}
|
||||
},
|
||||
|
||||
microTick() {
|
||||
this.tid = setTimeout(() => {
|
||||
this.setRemain(this.getRemain());
|
||||
|
||||
if (this.remain !== 0) {
|
||||
this.microTick();
|
||||
}
|
||||
}, 100);
|
||||
},
|
||||
|
||||
macroTick() {
|
||||
this.tid = setTimeout(() => {
|
||||
const remain = this.getRemain();
|
||||
|
||||
if (!isSameSecond(remain, this.remain) || remain === 0) {
|
||||
this.setRemain(remain);
|
||||
}
|
||||
|
||||
if (this.remain !== 0) {
|
||||
this.macroTick();
|
||||
}
|
||||
}, 1000);
|
||||
},
|
||||
|
||||
getRemain() {
|
||||
return Math.max(this.endTime - Date.now(), 0);
|
||||
},
|
||||
|
||||
setRemain(remain) {
|
||||
this.remain = remain;
|
||||
const timeData = parseTimeData(remain);
|
||||
this.$emit('change', timeData);
|
||||
this.setData({
|
||||
formattedTime: parseFormat(this.data.format, timeData)
|
||||
});
|
||||
|
||||
if (remain === 0) {
|
||||
this.pause();
|
||||
this.$emit('finish');
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
13
packages/count-down/index.wxml
Normal file
13
packages/count-down/index.wxml
Normal file
@ -0,0 +1,13 @@
|
||||
<view
|
||||
wx:if="{{ useCustom }}"
|
||||
class="van-count-down"
|
||||
>
|
||||
<slot/>
|
||||
</view>
|
||||
|
||||
<view
|
||||
wx:else
|
||||
class="van-count-down"
|
||||
>
|
||||
{{ formattedTime }}
|
||||
</view>
|
73
packages/count-down/utils.ts
Normal file
73
packages/count-down/utils.ts
Normal file
@ -0,0 +1,73 @@
|
||||
function padZero(num: number | string, targetLength = 2): string {
|
||||
let str = num + '';
|
||||
|
||||
while (str.length < targetLength) {
|
||||
str = '0' + str;
|
||||
}
|
||||
|
||||
return str;
|
||||
}
|
||||
|
||||
export type TimeData = {
|
||||
days: number;
|
||||
hours: number;
|
||||
minutes: number;
|
||||
seconds: number;
|
||||
milliseconds: number;
|
||||
};
|
||||
|
||||
const SECOND = 1000;
|
||||
const MINUTE = 60 * SECOND;
|
||||
const HOUR = 60 * MINUTE;
|
||||
const DAY = 24 * HOUR;
|
||||
|
||||
export function parseTimeData(time: number): TimeData {
|
||||
const days = Math.floor(time / DAY);
|
||||
const hours = Math.floor((time % DAY) / HOUR);
|
||||
const minutes = Math.floor((time % HOUR) / MINUTE);
|
||||
const seconds = Math.floor((time % MINUTE) / SECOND);
|
||||
const milliseconds = Math.floor(time % SECOND);
|
||||
|
||||
return {
|
||||
days,
|
||||
hours,
|
||||
minutes,
|
||||
seconds,
|
||||
milliseconds
|
||||
};
|
||||
}
|
||||
|
||||
export function parseFormat(format: string, timeData: TimeData): string {
|
||||
const { days } = timeData;
|
||||
let { hours, minutes, seconds, milliseconds } = timeData;
|
||||
|
||||
if (format.indexOf('DD') === -1) {
|
||||
hours += days * 24;
|
||||
} else {
|
||||
format = format.replace('DD', padZero(days));
|
||||
}
|
||||
|
||||
if (format.indexOf('HH') === -1) {
|
||||
minutes += hours * 60;
|
||||
} else {
|
||||
format = format.replace('HH', padZero(hours));
|
||||
}
|
||||
|
||||
if (format.indexOf('mm') === -1) {
|
||||
seconds += minutes * 60;
|
||||
} else {
|
||||
format = format.replace('mm', padZero(minutes));
|
||||
}
|
||||
|
||||
if (format.indexOf('ss') === -1) {
|
||||
milliseconds += seconds * 1000;
|
||||
} else {
|
||||
format = format.replace('ss', padZero(seconds));
|
||||
}
|
||||
|
||||
return format.replace('SSS', padZero(milliseconds, 3));
|
||||
}
|
||||
|
||||
export function isSameSecond(time1: number, time2: number): boolean {
|
||||
return Math.floor(time1 / 1000) === Math.floor(time2 / 1000);
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user