feat(Circle): add Circle component (#1984)

This commit is contained in:
ShuaiKang Zhang 2019-09-09 14:44:33 +08:00 committed by neverland
parent 79c994e380
commit 407f9b51b4
12 changed files with 369 additions and 2 deletions

View File

@ -53,7 +53,8 @@
"pages/datetime-picker/index", "pages/datetime-picker/index",
"pages/rate/index", "pages/rate/index",
"pages/collapse/index", "pages/collapse/index",
"pages/picker/index" "pages/picker/index",
"pages/circle/index"
], ],
"window": { "window": {
"navigationBarBackgroundColor": "#f8f8f8", "navigationBarBackgroundColor": "#f8f8f8",
@ -111,7 +112,8 @@
"van-rate": "./dist/rate/index", "van-rate": "./dist/rate/index",
"van-collapse": "./dist/collapse/index", "van-collapse": "./dist/collapse/index",
"van-collapse-item": "./dist/collapse-item/index", "van-collapse-item": "./dist/collapse-item/index",
"van-picker": "./dist/picker/index" "van-picker": "./dist/picker/index",
"van-circle": "./dist/circle/index"
}, },
"sitemapLocation": "sitemap.json" "sitemapLocation": "sitemap.json"
} }

View File

@ -109,6 +109,10 @@ export default [
groupName: '展示组件', groupName: '展示组件',
icon: 'photo-o', icon: 'photo-o',
list: [ list: [
{
path: '/circle',
title: 'Circle 进度条'
},
{ {
path: '/collapse', path: '/collapse',
title: 'Collapse 折叠面板' title: 'Collapse 折叠面板'

View File

@ -0,0 +1,19 @@
import Page from '../../common/page';
const format = rate => Math.min(Math.max(rate, 0), 100);
Page({
data: {
value: 25,
gradientColor: {
'0%': '#ffd01e',
'100%': '#ee0a24'
}
},
run(e) {
const step = parseFloat(e.currentTarget.dataset.step);
this.setData({
value: format((this.data.value += step))
});
}
});

View File

@ -0,0 +1,3 @@
{
"navigationBarTitleText": "Circle 进度条"
}

View File

@ -0,0 +1,14 @@
<demo-block title="基础用法">
<van-circle value="{{ value }}" text="{{ value }}%" />
</demo-block>
<demo-block title="样式定制">
<van-circle value="{{ value }}" stroke-width="{{ 6 }}" text="宽度定制" />
<van-circle value="{{ value }}" layer-color="#eee" color="#ee0a24" text="颜色定制" />
<van-circle value="{{ value }}" color="{{ gradientColor }}" text="渐变色" />
<van-circle value="{{ value }}" color="#07c160" clockwise="{{ false }}" text="逆时针" />
<van-circle value="{{ value }}" size="120" text="大小定制" />
</demo-block>
<van-button type="primary" size="small" data-step="10" bind:tap="run">增加</van-button>
<van-button type="danger" size="small" data-step="-10" bind:tap="run">减少</van-button>

View File

@ -0,0 +1,6 @@
.van-circle {
margin: 5px 10px 20px;
}
.van-button{
margin-left: 10px;
}

105
packages/circle/README.md Normal file
View File

@ -0,0 +1,105 @@
# Circle 环形进度条
### 引入
`app.json``index.json`中引入组件,默认为`ES6`版本,`ES5`引入方式参见[快速上手](#/quickstart)
```json
"usingComponents": {
"van-circle": "path/to/vant-weapp/dist/circle/index"
}
```
## 代码演示
### 基础用法
`value`属性表示进度条的目标进度。
```html
<van-circle :value="{{ 30 }}" :text="text" />
```
### 宽度定制
通过`stroke-width`属性来控制进度条宽度
```html
<van-circle :value="{{ value }}" :stroke-width="6" text="宽度定制" />
```
### 颜色定制
通过`color`属性来控制进度条颜色,`layer-color`属性来控制轨道颜色
```html
<van-circle
value="{{ value }}"
layer-color="#eeeeee"
color="#ee0a24"
text="颜色定制"
/>
```
### 渐变色
`color`属性支持传入对象格式来定义渐变色
```html
<van-circle value="{{ value }}" color="{{ gradientColor }}" text="渐变色" />
```
```javascript
Page({
data: {
value: 25,
gradientColor: {
'0%': '#ffd01e',
'100%': '#ee0a24'
}
}
});
```
### 逆时针方向
`clockwise`设置为`false`,进度会从逆时针方向开始
```html
<van-circle
value="{{ value }}"
color="#07c160"
clockwise="{{ false }}"
text="逆时针"
/>
```
### 大小定制
通过`size`属性设置圆环直径
```html
<van-circle value="{{ value }}" size="120" text="大小定制" />
```
## API
### Props
| 参数 | 说明 | 类型 | 默认值 | 版本 |
| ------------ | -------------------------------------- | ------------------ | --------- | ---- |
| value | 目标进度 | *number* | `0` | - |
| size | 圆环直径,默认单位为 `px` | *string \| number* | `100` | - |
| color | 进度条颜色,传入对象格式可以定义渐变色 | *string \| object* | `#1989fa` | - |
| layer-color | 轨道颜色 | *string* | `#fff` | - |
| fill | 填充颜色 | *string* | - | - |
| speed | 动画速度(单位为 value/s | *number* | `50` | - |
| text | 文字 | *string* | - | - |
| stroke-width | 进度条宽度 | *number* | `4` | - |
| clockwise | 是否顺时针增加 | *boolean* | `true` | - |
### Slots
| 名称 | 说明 |
| ------- | -------------- |
| default | 自定义文字内容,如果设置了`fill`,插槽内容会被原生组件覆盖 |

View File

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

View File

@ -0,0 +1,16 @@
@import '../common/style/var.less';
.van-circle {
position: relative;
display: inline-block;
text-align: center;
&__text {
position: absolute;
color: @circle-text-color;
top: 50%;
left: 0;
width: 100%;
transform: translateY(-50%);
}
}

185
packages/circle/index.ts Normal file
View File

@ -0,0 +1,185 @@
import { VantComponent } from '../common/component';
import { isObj } from '../common/utils';
import { BLUE, WHITE } from '../common/color';
function format(rate) {
return Math.min(Math.max(rate, 0), 100);
}
const PERIMETER = 2 * Math.PI;
const BEGIN_ANGLE = -Math.PI / 2;
const STEP = 1;
VantComponent({
props: {
text: String,
lineCap: {
type: String,
value: 'round'
},
value: {
type: Number,
value: 0,
observer: 'reRender'
},
speed: {
type: Number,
value: 50
},
size: {
type: Number,
value: 100,
observer: 'setStyle'
},
fill: String,
layerColor: {
type: String,
value: WHITE
},
color: {
type: [String, Object],
value: BLUE,
observer: 'setHoverColor'
},
strokeWidth: {
type: Number,
value: 4
},
clockwise: {
type: Boolean,
value: true
}
},
data: {
style: 'width: 100px; height: 100px;',
hoverColor: BLUE
},
methods: {
getContext() {
return (
this.ctx || (this.ctx = wx.createCanvasContext('van-circle', this))
);
},
setHoverColor() {
const context = this.getContext();
const { color, size } = this.data;
let hoverColor = color;
if (isObj(color)) {
const LinearColor = context.createLinearGradient(size, 0, 0, 0);
Object.keys(color)
.sort((a, b) => parseFloat(a) - parseFloat(b))
.map(key => {
LinearColor.addColorStop(parseFloat(key) / 100, color[key]);
});
hoverColor = LinearColor;
}
this.set({
hoverColor
});
},
setStyle() {
const { size } = this.data;
const style = `width: ${size}px; height: ${size}px;`;
this.set({
style
});
},
presetCanvas(context, strokeStyle, beginAngle, endAngle, fill) {
const { strokeWidth, lineCap, clockwise, size } = this.data;
const position = size / 2;
const radius = position - strokeWidth / 2;
context.setStrokeStyle(strokeStyle);
context.setLineWidth(strokeWidth);
context.setLineCap(lineCap);
context.beginPath();
context.arc(position, position, radius, beginAngle, endAngle, !clockwise);
context.stroke();
if (fill) {
context.setFillStyle(fill);
context.fill();
}
},
renderLayerCircle(context) {
const { layerColor, fill } = this.data;
this.presetCanvas(context, layerColor, 0, PERIMETER, fill);
},
renderHoverCircle(context, formatValue) {
const { clockwise, hoverColor } = this.data;
// 结束角度
const progress = PERIMETER * (formatValue / 100);
const endAngle = clockwise
? BEGIN_ANGLE + progress
: 3 * Math.PI - (BEGIN_ANGLE + progress);
this.presetCanvas(context, hoverColor, BEGIN_ANGLE, endAngle);
},
drawCircle(currentValue) {
const context = this.getContext();
const { size } = this.data;
context.clearRect(0, 0, size, size);
this.renderLayerCircle(context);
const formatValue = format(currentValue);
if (formatValue !== 0) {
this.renderHoverCircle(context, formatValue);
}
context.draw();
},
reRender() {
// tofector 动画暂时没有想到好的解决方案
const { value, speed } = this.data;
if (speed <= 0 || speed > 1000) {
this.drawCircle(value);
return;
}
this.clearInterval();
this.currentValue = this.currentValue || 0;
this.interval = setInterval(() => {
if (this.currentValue !== value) {
if (this.currentValue < value) {
this.currentValue += STEP;
} else {
this.currentValue -= STEP;
}
this.drawCircle(this.currentValue);
} else {
this.clearInterval();
}
}, 1000 / speed);
},
clearInterval() {
if (this.interval) {
clearInterval(this.interval);
this.interval = null;
}
}
},
created() {
const { value } = this.data;
this.currentValue = value;
this.drawCircle(value);
},
destroyed() {
this.ctx = null;
this.clearInterval();
}
});

View File

@ -0,0 +1,7 @@
<view class="van-circle">
<canvas class="van-circle__canvas" style="{{ style }}" canvas-id="van-circle"></canvas>
<view wx:if="{{ !text }}" class="van-circle__text">
<slot></slot>
</view>
<cover-view wx:else class="van-circle__text">{{ text }}</cover-view>
</view>

View File

@ -157,3 +157,6 @@
// Rate // Rate
@rate-horizontal-padding: 2px; @rate-horizontal-padding: 2px;
// Circle
@circle-text-color: @text-color;