fix(circle): fix type not work (#3228)

fix #3168
This commit is contained in:
rex 2020-06-02 21:40:34 +08:00 committed by GitHub
parent 5ccc355e3d
commit f40a5fbe97
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 110 additions and 36 deletions

View File

@ -1,5 +1,5 @@
<demo-block title="基础用法"> <demo-block title="基础用法">
<van-circle value="{{ value }}" text="{{ value }}%" /> <van-circle type="2d" value="{{ value }}" text="{{ value }}%" />
</demo-block> </demo-block>
<demo-block title="样式定制"> <demo-block title="样式定制">

View File

@ -68,7 +68,7 @@
"less": "^3.9.0", "less": "^3.9.0",
"less-loader": "^5.0.0", "less-loader": "^5.0.0",
"lint-staged": "^10.0.0", "lint-staged": "^10.0.0",
"miniprogram-api-typings": "2.10.4", "miniprogram-api-typings": "2.11.0",
"miniprogram-ci": "^1.0.27", "miniprogram-ci": "^1.0.27",
"postcss-loader": "^3.0.0", "postcss-loader": "^3.0.0",
"prettier": "^2.0.5", "prettier": "^2.0.5",

View File

@ -89,7 +89,7 @@ Page({
| 参数 | 说明 | 类型 | 默认值 | 版本 | | 参数 | 说明 | 类型 | 默认值 | 版本 |
| --- | --- | --- | --- | --- | | --- | --- | --- | --- | --- |
| value | 目标进度 | _number_ | `0` | - | | value | 目标进度 | _number_ | `0` | - |
| type | 指定 canvas 类型,可选值为 `2d` `webgl` | _string_ | - | - | | type | 指定 canvas 类型,可选值为 `2d` | _string_ | - | - |
| size | 圆环直径,默认单位为 `px` | _number_ | `100` | - | | size | 圆环直径,默认单位为 `px` | _number_ | `100` | - |
| color | 进度条颜色,传入对象格式可以定义渐变色 | _string \| object_ | `#1989fa` | - | | color | 进度条颜色,传入对象格式可以定义渐变色 | _string \| object_ | `#1989fa` | - |
| layer-color | 轨道颜色 | _string_ | `#fff` | - | | layer-color | 轨道颜色 | _string_ | `#fff` | - |

47
packages/circle/canvas.ts Normal file
View File

@ -0,0 +1,47 @@
type CanvasContext = WechatMiniprogram.CanvasContext;
export function adaptor(
ctx: CanvasRenderingContext2D
): CanvasContext & CanvasRenderingContext2D {
// @ts-ignore
return Object.assign(ctx, {
setStrokeStyle(val) {
ctx.strokeStyle = val;
},
setLineWidth(val) {
ctx.lineWidth = val;
},
setLineCap(val) {
ctx.lineCap = val;
},
setFillStyle(val) {
ctx.fillStyle = val;
},
setFontSize(val) {
ctx.font = String(val);
},
setGlobalAlpha(val) {
ctx.globalAlpha = val;
},
setLineJoin(val) {
ctx.lineJoin = val;
},
setTextAlign(val) {
ctx.textAlign = val;
},
setMiterLimit(val) {
ctx.miterLimit = val;
},
setShadow(offsetX, offsetY, blur, color) {
ctx.shadowOffsetX = offsetX;
ctx.shadowOffsetY = offsetY;
ctx.shadowBlur = blur;
ctx.shadowColor = color;
},
setTextBaseline(val) {
ctx.textBaseline = val;
},
createCircularGradient() {},
draw() {},
});
}

View File

@ -1,6 +1,7 @@
import { VantComponent } from '../common/component'; import { VantComponent } from '../common/component';
import { isObj } from '../common/utils'; import { isObj } from '../common/utils';
import { BLUE, WHITE } from '../common/color'; import { BLUE, WHITE } from '../common/color';
import { adaptor } from './canvas';
function format(rate) { function format(rate) {
return Math.min(Math.max(rate, 0), 100); return Math.min(Math.max(rate, 0), 100);
@ -28,6 +29,9 @@ VantComponent({
size: { size: {
type: Number, type: Number,
value: 100, value: 100,
observer() {
this.drawCircle(this.currentValue);
},
}, },
fill: String, fill: String,
layerColor: { layerColor: {
@ -59,38 +63,60 @@ VantComponent({
methods: { methods: {
getContext() { getContext() {
if (!this.ctx) { const { type } = this.data;
this.ctx = wx.createCanvasContext('van-circle', this);
if (type === '') {
const ctx = wx.createCanvasContext('van-circle', this);
return Promise.resolve(ctx);
} }
return this.ctx;
const dpr = wx.getSystemInfoSync().pixelRatio;
return new Promise((resolve) => {
wx.createSelectorQuery()
.in(this)
.select('#van-circle')
.fields({ node: true, size: true })
.exec((res) => {
const canvas = res[0].node;
const ctx = canvas.getContext(type);
canvas.width = res[0].width * dpr;
canvas.height = res[0].height * dpr;
ctx.scale(dpr, dpr);
resolve(adaptor(ctx));
});
});
}, },
setHoverColor() { setHoverColor() {
const { color, size, type } = this.data; const { color, size } = this.data;
const context = type ? this.getContext(type) : this.getContext();
let hoverColor = color; let hoverColor = color;
if (isObj(color)) { this.getContext().then((context) => {
const LinearColor = context.createLinearGradient(size, 0, 0, 0); if (isObj(color)) {
Object.keys(color) const LinearColor = context.createLinearGradient(size, 0, 0, 0);
.sort((a, b) => parseFloat(a) - parseFloat(b)) Object.keys(color)
.map((key) => .sort((a, b) => parseFloat(a) - parseFloat(b))
LinearColor.addColorStop(parseFloat(key) / 100, color[key]) .map((key) =>
); LinearColor.addColorStop(parseFloat(key) / 100, color[key])
hoverColor = LinearColor; );
} hoverColor = LinearColor;
}
this.setData({ hoverColor }); this.setData({ hoverColor });
});
}, },
presetCanvas(context, strokeStyle, beginAngle, endAngle, fill) { presetCanvas(context, strokeStyle, beginAngle, endAngle, fill) {
const { strokeWidth, lineCap, clockwise, size } = this.data; const { strokeWidth, lineCap, clockwise, size, type } = this.data;
const position = size / 2; const position = size / 2;
const radius = position - strokeWidth / 2; const radius = position - strokeWidth / 2;
context.setStrokeStyle(strokeStyle); context.setStrokeStyle(strokeStyle);
context.setLineWidth(strokeWidth); context.setLineWidth(strokeWidth);
context.setLineCap(lineCap); context.setLineCap(lineCap);
context.beginPath(); context.beginPath();
context.arc(position, position, radius, beginAngle, endAngle, !clockwise); context.arc(position, position, radius, beginAngle, endAngle, !clockwise);
context.stroke(); context.stroke();
@ -118,17 +144,19 @@ VantComponent({
}, },
drawCircle(currentValue) { drawCircle(currentValue) {
const { size, type } = this.data; const { size } = this.data;
const context = type ? this.getContext(type) : this.getContext();
context.clearRect(0, 0, size, size);
this.renderLayerCircle(context);
const formatValue = format(currentValue); this.getContext().then((context) => {
if (formatValue !== 0) { context.clearRect(0, 0, size, size);
this.renderHoverCircle(context, formatValue); this.renderLayerCircle(context);
}
context.draw(); const formatValue = format(currentValue);
if (formatValue !== 0) {
this.renderHoverCircle(context, formatValue);
}
context.draw();
});
}, },
reRender() { reRender() {
@ -171,7 +199,6 @@ VantComponent({
}, },
destroyed() { destroyed() {
this.ctx = null;
this.clearInterval(); this.clearInterval();
}, },
}); });

View File

@ -1,9 +1,9 @@
<wxs src="../wxs/utils.wxs" module="utils" /> <wxs src="../wxs/utils.wxs" module="utils" />
<view class="van-circle"> <view class="van-circle">
<canvas class="van-circle__canvas" style="width: {{ utils.addUnit(size) }};height:{{ utils.addUnit(size) }}" canvas-id="van-circle"></canvas> <canvas class="van-circle__canvas" type="{{ type }}" style="width: {{ utils.addUnit(size) }};height:{{ utils.addUnit(size) }}" id="van-circle" canvas-id="van-circle"></canvas>
<view wx:if="{{ !text }}" class="van-circle__text"> <view wx:if="{{ !text }}" class="van-circle__text">
<slot></slot> <slot></slot>
</view> </view>
<cover-view wx:else class="van-circle__text">{{ text }}</cover-view> <cover-view wx:else class="van-circle__text">{{ text }}</cover-view>
</view> </view>

View File

@ -7978,10 +7978,10 @@ minimist@^1.2.0, minimist@^1.2.5:
resolved "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602" resolved "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602"
integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw== integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==
miniprogram-api-typings@2.10.4: miniprogram-api-typings@2.11.0:
version "2.10.4" version "2.11.0"
resolved "https://registry.npmjs.org/miniprogram-api-typings/-/miniprogram-api-typings-2.10.4.tgz#13165d1e77084c3ec79ec93d8c74974f77425eb0" resolved "http://registry.npm.qima-inc.com/miniprogram-api-typings/download/miniprogram-api-typings-2.11.0.tgz#c0a91ca2fea80e344509399c580cbad504a8f905"
integrity sha512-IXWi6sJFd/EH1wNR8rtOx2+hJOwZk7jKFVIshQ0aAlRaKk0P4Jack4SItzUwNURgRykvrleGgErI/jH5hsiniw== integrity sha1-wKkcov6oDjRFCTmcWAy61QSo+QU=
miniprogram-ci@^1.0.27: miniprogram-ci@^1.0.27:
version "1.0.29" version "1.0.29"