特殊箭头图表

在这里插入图片描述

<template>
  <div class="chart" v-resize="resizeCharts" ref="piechart"></div>
</template>

<script>
const arrow =
  "";
const legendIcons = ["circle", "circle", "circle", "image://" + arrow];
import * as echarts from "echarts";
export default {
  data() {
    return {
      data: [
        {
          name: "本年退用量",
          percentData: [50, 40, 60, 80, 30, 75], // 所占百分比
          data: [687091, 176706, 126074, 107335, 252376, 24600], // 真实数据
          color: "rgb(0, 229, 255)", // 柱子颜色
        },
        {
          name: "本周退用量",
          percentData: [5, 6, 4, 10, 3, 8],
          data: [18785, 6000, 6295, 3951, 1769, 770],
          color: "rgb(250, 193, 45)",
        },
        {
          name: "剩余量",
          percentData: [45, 54, 36, 10, 67, 17],
          data: [1159314, 128782, 382171, 228806, 380269, 39286],
          color: "rgb(186, 155, 201)",
        },
        {
          // 箭头对应的百分比和真实数据,无需传入颜色
          name: "截至11月底计划",
          percentData: [60, 50, 70, 90, 40, 85],
          data: [687091, 176706, 126074, 107335, 252376, 24600],
        },
      ],
      yAxisData: ["冀北", "唐山", "张家口", "秦皇岛", "承德", "廊坊"],
      unit: "单位:万只",
    };
  },
  methods: {
    resizeCharts() {
      this.charts.resize();
    },
    initCharts() {
      this.$nextTick(() => {
        this.charts = echarts.init(this.$refs.piechart);
        this.charts.clear();
        this.setOption();
      });
    },
    setOption() {
      const series = this.getSeries(this.data);
      const legend = this.getLegend(this.data);
      const option = this.getOption(this.data, legend, series, this.yAxisData);
      this.charts.setOption(option, true);
    },
    getSeries(data) {
      const series = [];
      for (let i = 0; i < data.length - 1; i++) {
        const tempMap = {
          name: data[i].name,
          type: "bar",
          stack: "total",
          barWidth: 14,
          itemStyle: {
            color: data[i].color,
          },
          label: {
            show: true,
            verticalAlign: "middle",
            formatter(series) {
              return data[series.seriesIndex].percentData[series.dataIndex] + "%";
            },
            offset: [0, 1],
            textStyle: {
              color: "#fff",
            },
          },
          data: data[i].percentData,
        };
        series.push(tempMap);
      }
      series.push({
        ...data[3],
        data: data[3].percentData,
        type: "bar",
        barMaxWidth: 1,
        itemStyle: {
          color: "rgba(0, 0, 0, 0)",
        },
        label: {
          show: true,
          position: "right",
          formatter(series) {
            return `\n{b|}{a|${data[series.seriesIndex].data[series.dataIndex]}}`;
          },
          rich: {
            a: {
              color: "#fff",
              height: 20,
              lineHeight: 30,
              padding: [5, 0, 0, 0],
            },
            b: {
              backgroundColor: {
                image: arrow,
              },
              height: 15,
              width: 20,
              align: "center",
              padding: [0, 0, 0],
            },
          },
        },
      });
      return series;
    },
    getLegend(data) {
      const legend = [];
      for (let i = 0; i < data.length; i++) {
        const tempMap = {
          name: data[i].name,
          icon: legendIcons[i],
        };
        legend.push(tempMap);
      }
      return legend;
    },
    getOption(data, legend, series, yAxis) {
      const option = {
        tooltip: {
          trigger: "axis",
          formatter(series) {
            let html = `<div>${series[0].name}</div>`;
            for (let i = 0; i < series.length; i++) {
              const item = series[i];
              if (i === 3) {
                item.marker = `<span style='display:inline-block;margin-left:-5px;width:20px;height:15px;background:url("${arrow}") no-repeat center / 100% 100%;'></span>`;
              }
              html += `<div>${item.marker} ${item.seriesName}: ${data[item.seriesIndex].data[item.dataIndex]}</div>`;
            }
            return html;
          },
        },
        legend: {
          show: true,
          itemWidth: 20,
          itemHeight: 15,
          data: legend,
          bottom: 10,
          textStyle: {
            color: "#fff",
          },
        },
        grid: {
          top: 40,
          left: "10",
          right: "40",
          bottom: "45",
          containLabel: true,
        },
        xAxis: {
          type: "value",
          axisLabel: {
            formatter(val) {
              return val + "%";
            },
            textStyle: {
              color: "#fff",
            },
          },
          splitLine: {
            lineStyle: {
              color: "rgba(255, 255, 255, 0.5)",
              width: 0.5,
            },
          },
        },
        yAxis: {
          type: "category",
          name: this.unit,
          nameTextStyle: {
            color: "#fff",
          },
          nameGap: 10,
          offset: 0,
          data: yAxis,
          axisLabel: {
            textStyle: {
              color: "#fff",
            },
          },
          axisTick: {
            lineStyle: {
              color: "rgba(255, 255, 255, 0.5)",
              width: 0.5,
            },
          },
          axisLine: {
            lineStyle: {
              color: "rgba(255, 255, 255, 0.5)",
              width: 0.5,
            },
          },
        },
        series,
      };
      return option;
    },
  },
  mounted() {
    this.initCharts();
  },
  beforeDestroy() {
    this.charts && this.charts.dispose();
    this.charts = null;
  },
  watch: {
    dataArr: {
      handler() {
        if (this.charts) {
          this.setOption();
        } else {
          this.initCharts();
        }
      },
    },
  },
};
</script>

<style scoped lang="less">
.chart {
  width: 100%;
  height: 100%;
  background: #001e51;
}
</style>

  • 8
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值