mirror of
https://gitee.com/vant-contrib/vant-weapp.git
synced 2025-04-06 03:58:05 +08:00
feat(Circle): add Circle component (#1984)
This commit is contained in:
parent
79c994e380
commit
407f9b51b4
@ -53,7 +53,8 @@
|
||||
"pages/datetime-picker/index",
|
||||
"pages/rate/index",
|
||||
"pages/collapse/index",
|
||||
"pages/picker/index"
|
||||
"pages/picker/index",
|
||||
"pages/circle/index"
|
||||
],
|
||||
"window": {
|
||||
"navigationBarBackgroundColor": "#f8f8f8",
|
||||
@ -111,7 +112,8 @@
|
||||
"van-rate": "./dist/rate/index",
|
||||
"van-collapse": "./dist/collapse/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"
|
||||
}
|
||||
|
@ -109,6 +109,10 @@ export default [
|
||||
groupName: '展示组件',
|
||||
icon: 'photo-o',
|
||||
list: [
|
||||
{
|
||||
path: '/circle',
|
||||
title: 'Circle 进度条'
|
||||
},
|
||||
{
|
||||
path: '/collapse',
|
||||
title: 'Collapse 折叠面板'
|
||||
|
19
example/pages/circle/index.js
Normal file
19
example/pages/circle/index.js
Normal 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))
|
||||
});
|
||||
}
|
||||
});
|
3
example/pages/circle/index.json
Normal file
3
example/pages/circle/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"navigationBarTitleText": "Circle 进度条"
|
||||
}
|
14
example/pages/circle/index.wxml
Normal file
14
example/pages/circle/index.wxml
Normal 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>
|
6
example/pages/circle/index.wxss
Normal file
6
example/pages/circle/index.wxss
Normal file
@ -0,0 +1,6 @@
|
||||
.van-circle {
|
||||
margin: 5px 10px 20px;
|
||||
}
|
||||
.van-button{
|
||||
margin-left: 10px;
|
||||
}
|
105
packages/circle/README.md
Normal file
105
packages/circle/README.md
Normal 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`,插槽内容会被原生组件覆盖 |
|
3
packages/circle/index.json
Normal file
3
packages/circle/index.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"component": true
|
||||
}
|
16
packages/circle/index.less
Normal file
16
packages/circle/index.less
Normal 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
185
packages/circle/index.ts
Normal 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();
|
||||
}
|
||||
});
|
7
packages/circle/index.wxml
Normal file
7
packages/circle/index.wxml
Normal 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>
|
@ -157,3 +157,6 @@
|
||||
|
||||
// Rate
|
||||
@rate-horizontal-padding: 2px;
|
||||
|
||||
// Circle
|
||||
@circle-text-color: @text-color;
|
||||
|
Loading…
x
Reference in New Issue
Block a user