DemuMesDataV/components/arcRingChart/index.vue

340 lines
8.2 KiB
Vue
Raw Normal View History

2018-12-12 18:46:41 +08:00
<template>
<div class="arc-ring-chart">
2019-01-16 16:56:11 +08:00
<loading v-if="!status" />
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
<div class="canvas-container">
<canvas :ref="ref" />
2018-12-12 18:46:41 +08:00
</div>
2019-01-16 16:56:11 +08:00
<label-line :label="dealAfterLabelLine" :colors="drawColors" />
2018-12-12 18:46:41 +08:00
</div>
</template>
<script>
2019-01-16 16:56:11 +08:00
import colorsMixin from '../../mixins/colorsMixin.js'
import canvasMixin from '../../mixins/canvasMixin.js'
2018-12-12 18:46:41 +08:00
export default {
name: 'ArcRingChart',
2019-01-16 16:56:11 +08:00
props: ['data', 'labelLine', 'colors'],
mixins: [colorsMixin, canvasMixin],
2018-12-12 18:46:41 +08:00
data () {
return {
ref: `concentric-arc-chart-${(new Date()).getTime()}`,
2019-01-16 16:56:11 +08:00
status: false,
2018-12-12 18:46:41 +08:00
defaultDecorationCircleRadius: 0.65,
defaultArcRadiusArea: [0.3, 0.4],
defaultArcWidthArea: [2, 10],
defaultLabelFontSize: 12,
decorationRadius: '',
radianOffset: Math.PI / -2,
totalValue: 0,
arcRadius: [],
arcRadian: [],
arcWidth: [],
2019-01-16 16:56:11 +08:00
labelLinePoints: [],
dealAfterLabelLine: []
2018-12-12 18:46:41 +08:00
}
},
watch: {
data (d) {
2019-01-16 16:56:11 +08:00
const { checkData, draw } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
checkData() && draw()
2018-12-12 18:46:41 +08:00
}
},
methods: {
2019-01-16 16:56:11 +08:00
async init () {
const { initCanvas, initColors, checkData, draw } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
await initCanvas()
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
initColors()
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
checkData() && draw()
2018-12-12 18:46:41 +08:00
},
2019-01-16 16:56:11 +08:00
checkData () {
const { data } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
this.status = false
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
if (!data || !data.series) return false
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
this.status = true
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
return true
2018-12-12 18:46:41 +08:00
},
draw () {
2019-01-16 16:56:11 +08:00
const { clearCanvas, calcLabelLineData, drawDecorationCircle } = this
clearCanvas()
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
calcLabelLineData()
2018-12-12 18:46:41 +08:00
drawDecorationCircle()
const { calcArcRadius, calcArcRadian, calcArcWidth, drawArc } = this
calcArcRadius()
calcArcRadian()
calcArcWidth()
drawArc()
const { calcLableLinePoints, drawLabelLine, drawLabelText } = this
calcLableLinePoints()
drawLabelLine()
drawLabelText()
},
2019-01-16 16:56:11 +08:00
calcLabelLineData () {
const { labelLine, deepClone, data: { series } } = this
if (!labelLine) return
const dealAfterLabelLine = this.dealAfterLabelLine = deepClone(labelLine)
if (labelLine.labels === 'inherit') dealAfterLabelLine.labels = series.map(({ title }) => title)
},
2018-12-12 18:46:41 +08:00
drawDecorationCircle () {
2019-01-16 16:56:11 +08:00
const { ctx, data: { decorationCircleRadius }, defaultDecorationCircleRadius, centerPos } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const radius = this.decorationRadius = Math.min(...centerPos) * (decorationCircleRadius || defaultDecorationCircleRadius)
2018-12-12 18:46:41 +08:00
ctx.beginPath()
ctx.strokeStyle = 'rgba(250, 250, 250, 0.2)'
ctx.lineWidth = 4
2019-01-16 16:56:11 +08:00
ctx.arc(...centerPos, radius, 0, Math.PI * 2)
2018-12-12 18:46:41 +08:00
ctx.stroke()
ctx.beginPath()
ctx.lineWidth = 1
2019-01-16 16:56:11 +08:00
ctx.arc(...centerPos, radius - 7, 0, Math.PI * 2)
2018-12-12 18:46:41 +08:00
ctx.closePath()
ctx.stroke()
},
calcArcRadius () {
2019-01-16 16:56:11 +08:00
const { data: { series, arcRadiusArea }, defaultArcRadiusArea, centerPos, randomExtend } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const fullRadius = Math.min(...centerPos)
2018-12-12 18:46:41 +08:00
const currentArcRaidusArea = arcRadiusArea || defaultArcRadiusArea
const maxRadius = fullRadius * Math.max(...currentArcRaidusArea)
const minRadius = fullRadius * Math.min(...currentArcRaidusArea)
2019-01-16 16:56:11 +08:00
this.arcRadius = series.map(t => randomExtend(minRadius, maxRadius))
2018-12-12 18:46:41 +08:00
},
calcArcRadian () {
2019-01-16 16:56:11 +08:00
const { data: { series }, multipleSum, radianOffset } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const valueSum = this.totalValue = multipleSum(...series.map(({ value }) => value))
2018-12-12 18:46:41 +08:00
let radian = radianOffset
const fullRadian = Math.PI * 2
2019-01-16 16:56:11 +08:00
const avgRadian = fullRadian / series.length
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
this.arcRadian = series.map(({ value }) => {
2018-12-12 18:46:41 +08:00
const valueRadian = valueSum === 0 ? avgRadian : value / valueSum * fullRadian
return [radian, (radian += valueRadian)]
})
},
calcArcWidth () {
2019-01-16 16:56:11 +08:00
const { data: { series, arcWidthArea }, defaultArcWidthArea, randomExtend } = this
2018-12-12 18:46:41 +08:00
const currentArea = arcWidthArea || defaultArcWidthArea
const maxWidth = Math.max(...currentArea)
const minWidth = Math.min(...currentArea)
2019-01-16 16:56:11 +08:00
this.arcWidth = series.map(t => randomExtend(minWidth, maxWidth))
2018-12-12 18:46:41 +08:00
},
drawArc () {
2019-01-16 16:56:11 +08:00
const { ctx, arcRadius, arcRadian, arcWidth, drawColors, centerPos } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const colorNum = drawColors.length
2018-12-12 18:46:41 +08:00
arcRadius.forEach((radius, i) => {
ctx.beginPath()
2019-01-16 16:56:11 +08:00
ctx.arc(...centerPos, radius, ...arcRadian[i])
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
ctx.strokeStyle = drawColors[i % colorNum]
2018-12-12 18:46:41 +08:00
ctx.lineWidth = arcWidth[i]
ctx.stroke()
})
},
calcLableLinePoints () {
2019-01-16 16:56:11 +08:00
const { arcRadian, arcRadius, centerPos: [x, y], totalValue } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const { canvas: { getCircleRadianPoint }, data: { series } } = this
2018-12-12 18:46:41 +08:00
let [leftlabelLineNum, rightLabelLineNum] = [0, 0]
const arcMiddlePoints = arcRadian.map((radian, i) => {
const middleRadian = (radian[1] - radian[0]) / 2 + radian[0]
const point = getCircleRadianPoint(x, y, arcRadius[i], middleRadian)
2019-01-16 16:56:11 +08:00
point[0] > x && (series[i].value || !totalValue) && rightLabelLineNum++
point[0] <= x && (series[i].value || !totalValue) && leftlabelLineNum++
2018-12-12 18:46:41 +08:00
return point
})
const { getYPos, decorationRadius } = this
const labelLineYArea = [y - decorationRadius + 10, y + decorationRadius - 10]
const leftYPos = getYPos(labelLineYArea, leftlabelLineNum)
const rightYPos = getYPos(labelLineYArea, rightLabelLineNum)
const offsetX = decorationRadius + 10
const leftlabelLineEndX = x - offsetX
const rightLableLineEndX = x + offsetX
const maxRadius = Math.max(...arcRadius)
const leftNearRadiusX = x - maxRadius - 8
const rightNearRadiusX = x + maxRadius + 8
this.labelLinePoints = arcMiddlePoints.map(([px, py], i) => {
2019-01-16 16:56:11 +08:00
if (!series[i].value && totalValue) return [false, false, false, false]
2018-12-12 18:46:41 +08:00
if (px > x) {
const yPos = rightYPos.shift()
return [
[px, py],
[rightNearRadiusX, py],
[rightLableLineEndX - 10, yPos],
[rightLableLineEndX, yPos]
]
} else {
const yPos = leftYPos.pop()
return [
[px, py],
[leftNearRadiusX, py],
[leftlabelLineEndX + 10, yPos],
[leftlabelLineEndX, yPos]
]
}
})
},
getYPos (area, num) {
let gap = 0
const minus = area[1] - area[0]
if (num === 1) {
return [area[0] + minus / 2]
} else if (num === 2) {
const offset = minus * 0.1
return [area[0] + offset, area[1] - offset]
} else {
gap = minus / (num - 1)
return new Array(num).fill(0).map((t, i) => area[0] + i * gap)
}
},
drawLabelLine () {
2019-01-16 16:56:11 +08:00
const { ctx, labelLinePoints, canvas: { drawPolyline }, drawColors } = this
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
const colorNum = drawColors.length
2018-12-12 18:46:41 +08:00
labelLinePoints.forEach((polyline, i) =>
2019-01-16 16:56:11 +08:00
polyline[0] && drawPolyline(ctx, polyline, 2, drawColors[i % colorNum], false, [10, 0], true))
2018-12-12 18:46:41 +08:00
},
drawLabelText () {
2019-01-16 16:56:11 +08:00
const { ctx, labelLinePoints, data: { series, labelFontSize, fixed }, totalValue, defaultLabelFontSize, centerPos: [x] } = this
2018-12-12 18:46:41 +08:00
ctx.font = `${labelFontSize || defaultLabelFontSize}px Arial`
ctx.fillStyle = '#fff'
2018-12-14 19:45:23 +08:00
let totalPercent = 0
2019-01-16 16:56:11 +08:00
const dataLast = series.length - 1
2018-12-14 19:45:23 +08:00
2019-01-16 16:56:11 +08:00
series.forEach(({ value, title }, i) => {
2018-12-12 18:46:41 +08:00
if (!value && totalValue) return
2018-12-14 19:45:23 +08:00
let currentPercent = (value / totalValue * 100).toFixed(fixed || 1)
2018-12-12 18:46:41 +08:00
2018-12-14 19:45:23 +08:00
i === dataLast && (currentPercent = (100 - totalPercent).toFixed(fixed || 1))
2018-12-12 18:46:41 +08:00
2018-12-14 19:45:23 +08:00
!totalValue && (currentPercent = 0)
2018-12-12 18:46:41 +08:00
const textPos = labelLinePoints[i][2]
const isLeft = textPos[0] < x
ctx.textAlign = isLeft ? 'end' : 'start'
ctx.textBaseline = 'bottom'
2018-12-14 19:45:23 +08:00
ctx.fillText(`${currentPercent}%`, ...textPos)
2018-12-12 18:46:41 +08:00
ctx.textBaseline = 'top'
ctx.fillText(title, ...textPos)
2018-12-14 19:45:23 +08:00
totalPercent += Number(currentPercent)
2018-12-12 18:46:41 +08:00
})
}
},
mounted () {
const { init } = this
init()
}
}
</script>
<style lang="less">
.arc-ring-chart {
position: relative;
2019-01-16 16:56:11 +08:00
display: flex;
flex-direction: column;
color: #fff;
2018-12-12 18:46:41 +08:00
2019-01-16 16:56:11 +08:00
.canvas-container {
position: relative;
flex: 1;
2018-12-12 18:46:41 +08:00
}
2019-01-16 16:56:11 +08:00
canvas {
2018-12-12 18:46:41 +08:00
width: 100%;
2019-01-16 16:56:11 +08:00
height: 100%;
2018-12-12 18:46:41 +08:00
}
}
</style>