Files
cunkebao_v3/Cunkebao/components/LineChart.vue

253 lines
5.5 KiB
Vue
Raw Normal View History

2025-03-24 10:19:21 +08:00
<template>
<view class="line-chart">
<view class="chart-container">
<view class="y-axis">
<view class="axis-label" v-for="(value, index) in yAxisLabels" :key="'y-'+index">
{{ value }}
</view>
</view>
<view class="chart-body">
<view class="grid-lines">
<view class="grid-line" v-for="(value, index) in yAxisLabels" :key="'grid-'+index"></view>
</view>
<view class="line-container">
<view class="line-path">
<view class="line-segment"
v-for="(point, index) in normalizedPoints"
:key="'line-'+index"
v-if="index < normalizedPoints.length - 1"
:style="{
left: `${index * 100 / (points.length - 1)}%`,
width: `${100 / (points.length - 1)}%`,
bottom: `${point * 100}%`,
height: `${(normalizedPoints[index + 1] - point) * 100}%`,
transform: `skewX(${(normalizedPoints[index + 1] - point) > 0 ? 45 : -45}deg)`,
transformOrigin: 'bottom left'
}"
></view>
</view>
<view class="data-points">
<view class="data-point"
v-for="(point, index) in normalizedPoints"
:key="'point-'+index"
:style="{
left: `${index * 100 / (points.length - 1)}%`,
bottom: `${point * 100}%`
}"
>
<view class="point-inner"></view>
</view>
</view>
</view>
</view>
<view class="x-axis">
<view class="axis-label"
v-for="(label, index) in xAxisLabels"
:key="'x-'+index"
:style="{ left: `${index * 100 / (xAxisLabels.length - 1)}%` }"
>
{{ label }}
</view>
</view>
</view>
</view>
</template>
<script>
export default {
name: 'LineChart',
props: {
points: {
type: Array,
required: true
},
xAxisLabels: {
type: Array,
default: () => []
},
color: {
type: String,
default: '#4080ff'
},
yAxisCount: {
type: Number,
default: 6
}
},
computed: {
// 计算最大最小值
max() {
return Math.max(...this.points, 0);
},
min() {
return Math.min(...this.points, 0);
},
// 归一化数据点转换为0-1之间的值
normalizedPoints() {
const range = this.max - this.min;
if (range === 0) return this.points.map(() => 0.5);
return this.points.map(point => (point - this.min) / range);
},
// 计算Y轴标签
yAxisLabels() {
const labels = [];
const range = this.max - this.min;
const step = range / (this.yAxisCount - 1);
for (let i = 0; i < this.yAxisCount; i++) {
const value = Math.round(this.min + (step * i));
labels.unshift(value);
}
return labels;
}
}
}
</script>
<style lang="scss" scoped>
.line-chart {
width: 100%;
height: 100%;
padding: 20rpx;
.chart-container {
position: relative;
width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.y-axis {
position: absolute;
left: 0;
top: 0;
bottom: 40rpx;
width: 60rpx;
display: flex;
flex-direction: column;
justify-content: space-between;
.axis-label {
font-size: 22rpx;
color: #777;
2025-03-24 10:19:21 +08:00
text-align: right;
padding-right: 10rpx;
}
}
.chart-body {
flex: 1;
margin-left: 60rpx;
margin-bottom: 40rpx;
position: relative;
border-left: 1px solid #eeeeee;
border-bottom: 1px solid #eeeeee;
.grid-lines {
position: absolute;
left: 0;
right: 0;
top: 0;
bottom: 0;
.grid-line {
position: absolute;
left: 0;
right: 0;
border-top: 1px dashed #eeeeee;
&:nth-child(1) {
bottom: 0;
}
&:nth-child(2) {
bottom: 25%;
}
&:nth-child(3) {
bottom: 50%;
}
&:nth-child(4) {
bottom: 75%;
}
&:nth-child(5) {
bottom: 100%;
}
}
}
.line-container {
position: absolute;
left: 0;
right: 0;
top: 0;
bottom: 0;
.line-path {
position: absolute;
left: 0;
right: 0;
top: 0;
bottom: 0;
.line-segment {
position: absolute;
background-color: v-bind(color);
height: 4rpx;
}
}
.data-points {
position: absolute;
left: 0;
right: 0;
top: 0;
bottom: 0;
.data-point {
position: absolute;
transform: translate(-50%, 50%);
width: 16rpx;
height: 16rpx;
border-radius: 50%;
background-color: #fff;
border: 4rpx solid v-bind(color);
.point-inner {
position: absolute;
left: 2rpx;
top: 2rpx;
right: 2rpx;
bottom: 2rpx;
border-radius: 50%;
background-color: v-bind(color);
}
}
}
}
}
.x-axis {
height: 40rpx;
margin-left: 60rpx;
position: relative;
.axis-label {
position: absolute;
transform: translateX(-50%);
font-size: 22rpx;
color: #777;
2025-03-24 10:19:21 +08:00
text-align: center;
top: 10rpx;
}
}
}
</style>