feat(CountDown): add CountDown (#2135)

This commit is contained in:
lethe 2019-10-11 20:12:22 +08:00 committed by neverland
parent a01f966dea
commit 95476ae754
13 changed files with 500 additions and 0 deletions

View File

@ -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",

View File

@ -121,6 +121,10 @@ export default [
path: '/collapse',
title: 'Collapse 折叠面板'
},
{
path: '/count-down',
title: 'CountDown 倒计时'
},
{
path: '/notice-bar',
title: 'NoticeBar 通告栏'

View 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('倒计时结束');
}
});

View File

@ -0,0 +1,3 @@
{
"navigationBarTitleText": "CountDown 倒计时"
}

View 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" />

View 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;
}

View File

@ -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;

View 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`,重设后会自动开始倒计时 |

View File

@ -0,0 +1,3 @@
{
"component": true
}

View 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;
}

View 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');
}
}
}
});

View File

@ -0,0 +1,13 @@
<view
wx:if="{{ useCustom }}"
class="van-count-down"
>
<slot/>
</view>
<view
wx:else
class="van-count-down"
>
{{ formattedTime }}
</view>

View 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);
}