vue+antv g2+element-ui实现图表的增添拖拽

在这里插入图片描述

html

<template>
  <div class="app-container dashboard">
    <!-- 新增按钮和弹框 开始 -->
    <el-popover placement="left" width="30%" trigger="click" class="btn-add">
      <el-collapse v-model="activeNames">
        <el-collapse-item v-for="item in projectList" :key="item.key" :title="item.title" :name="item.key">
          <div v-for="row in item.list" :key="row.key">
            <el-switch v-model="row.isShow" :active-text="row.name" @change="isShow(row)"></el-switch>
          </div>
        </el-collapse-item>
      </el-collapse>
      <el-button type="primary" icon="el-icon-plus" slot="reference" circle></el-button>
    </el-popover>
    <!-- 新增按钮和弹框 结束 -->

    <div id="canvas">
      <div class="item" v-for="(item, index) in divList" :key="index" draggable="true" :id="item.key"
        @dragstart="handleDragStart($event, item)"
        @dragenter="handleDragEnter($event, item)"
        @dragover.prevent="handleDragover($event, item)"
        @drop="handleDrop($event, item)"
        @dragend="handleDragEnd($event, item)">
      </div>
    </div>
  </div>
</template>

js

<script>
// 生成图表
import dashboardChart from './chart/dashboardChart'

export default {
  name: 'dashboard',
  data() {
    return {
      divList: [],
      projectList: [],
      activeNames: [],
      dragging: null
    }
  },
  mounted() {
    this.getList()
  },
  methods: {
    // 获取已保存的图表
    getList() {
      this.projectList = [
        {
          title: '申请环节',
          key: '1',
          list: [
            { name: '累计放款VS预估值', key: 'estimate', type: 'dashboard', color: '#1890FF', value: 0.1, isShow: false },
            { name: '累计放款VS余额', key: 'totalBalance', type: 'dashboard', color: '#95DE64', value: 0.2, isShow: false }
          ]
        }
      ]
      for (const i in this.projectList) {
        const project = this.projectList[i]
        this.activeNames.push(project.key)
        // 遍历出已经选择的图表
        for (const j in project.list) {
          const one = project.list[j]
          // 把选择的图表在页面渲染出来
          if (one.isShow === true) {
            this.divList.push(one)
            this.$nextTick(() => {
              this.getChart(one)
            })
          }
        }
      }
    },
    // 在页面新增或删除图表
    isShow(row) {
      // 新增
      if (row.isShow === true) {
        this.divList.push(row)
        this.$nextTick(() => {
          this.getChart(row)
        })
      // 删除
      } else {
        for (const i in this.divList) {
          if (row.key === this.divList[i].key) {
            this.divList.splice(i, 1)
          }
        }
        this.$nextTick(() => {
          for (const j in this.divList) {
            document.getElementById(this.divList[j].key).innerHTML = ''
            this.getChart(this.divList[j])
          }
        })
      }
    },
    // 生成图表
    getChart(row) {
      if (row.type === 'dashboard') {
        const value = [
          { value: row.value }
        ] 
        dashboardChart(row.key, value, row.color, row.name)
      }
    },
    // 当元素被拖动时
    handleDragStart(e, item) {
      this.dragging = item
    },
    // 当被鼠标拖动的对象进入其容器范围内时触发此事件
    handleDragEnter(e) {
      // 为需要移动的元素设置dragstart事件
      e.dataTransfer.effectAllowed = 'move'
    },
    // 当某被拖动的对象在另一对象容器范围内拖动时触发此事件
    handleDragover(e) {
      // 首先把div变成可以放置的元素,即重写dragenter/dragover
      // 在dragenter中针对放置目标来设置!
      e.dataTransfer.dropEffect = 'move'
    },
    // 当放置被拖元素时
    handleDrop(e, item) {
      e.dataTransfer.dropEffect = 'move'
      if(item === this.dragging){
        return
      }
      const newItems = [...this.divList]
      const from = newItems.indexOf(this.dragging)
      const to = newItems.indexOf(item)
      newItems[from] = item
      newItems[to] = this.dragging
      this.divList = newItems
      document.getElementById(this.dragging.key).innerHTML = ''
      document.getElementById(item.key).innerHTML = ''
      this.$nextTick(() => {
        this.getChart(newItems[from])
        this.getChart(newItems[to])
      })
    },
    // 完成元素拖动后触发
    handleDragEnd() {
      this.dragging = null
    }
  }
}
</script>

dashboardChart

import G2 from '@antv/g2'

/**
 * 仪表盘
 * @param {*} id 容器
 * @param {*} data 数据
 * [{ value: 0.5 }]
 * @param {*} color 颜色
 * @param {*} title 标题
 */
function draw(id, data, color, title) {
  var Shape = G2.Shape;
  // 自定义Shape 部分
  Shape.registerShape('point', 'pointer', {
    drawShape: function drawShape(cfg, group) {
      var center = this.parsePoint({ // 获取极坐标系下画布中心点
        x: 0,
        y: 0
      });
      // 绘制指针
      group.addShape('line', {
        attrs: {
          x1: center.x,
          y1: center.y,
          x2: cfg.x,
          y2: cfg.y,
          stroke: cfg.color,
          lineWidth: 3,
          lineCap: 'round'
        }
      });
      return group.addShape('circle', {
        attrs: {
          x: center.x,
          y: center.y,
          r: 8,
          stroke: cfg.color,
          lineWidth: 2.5,
          fill: '#fff'
        }
      });
    }
  });

  var chart = new G2.Chart({
    container: id,
    forceFit: true,
    height: 250,
    padding: 'auto'
  });
  chart.source(data);

  chart.coord('polar', {
    startAngle: -10 / 8 * Math.PI,
    endAngle: 2 / 8 * Math.PI,
    radius: 0.75
  });
  chart.scale('value', {
    min: 0,
    max: 1,
    nice: false,
    ticks: [0.1, 0.3, 0.5, 0.7, 0.9]
  });

  chart.axis('1', false);
  chart.axis('value', {
    zIndex: 2,
    line: null,
    label: {
      offset: -15,
      formatter: function formatter(val) {
        if (val === '0.1') {
          return '10%';
        } else if (val === '0.3') {
          return '30%';
        } else if (val === '0.5') {
          return '50%';
        } else if (val === '0.7') {
          return '70%';
        } else if (val === '0.9') {
          return '90%';
        }
      },
      textStyle: {
        fontSize: 14,
        textAlign: 'center'
      }
    },
    tickLine: null,
    grid: null
  });
  chart.legend(false);
  chart.point().position('value*1').shape('pointer').color(color).active(false);

  // 绘制仪表盘刻度线
  chart.guide().line({
    start: [0.1, 0.95],
    end: [0.1, 0.85],
    lineStyle: {
      stroke: color, // 线的颜色
      lineDash: null, // 虚线的设置
      lineWidth: 2
    }
  });

  chart.guide().line({
    start: [0.3, 0.95],
    end: [0.3, 0.85],
    lineStyle: {
      stroke: color, // 线的颜色
      lineDash: null, // 虚线的设置
      lineWidth: 2
    }
  });

  chart.guide().line({
    start: [0.5, 0.95],
    end: [0.5, 0.85],
    lineStyle: {
      stroke: color, // 线的颜色
      lineDash: null, // 虚线的设置
      lineWidth: 2
    }
  });

  chart.guide().line({
    start: [0.7, 0.95],
    end: [0.7, 0.85],
    lineStyle: {
      stroke: color, // 线的颜色
      lineDash: null, // 虚线的设置
      lineWidth: 2
    }
  });

  chart.guide().line({
    start: [0.9, 0.95],
    end: [0.9, 0.85],
    lineStyle: {
      stroke: color, // 线的颜色
      lineDash: null, // 虚线的设置
      lineWidth: 2
    }
  });

  // 绘制仪表盘背景
  chart.guide().arc({
    zIndex: 0,
    top: false,
    start: [0, 0.965],
    end: [1, 0.965],
    style: { // 底灰色
      stroke: '#F5F5F5',
      lineWidth: 10
    }
  });
  // 绘制指标
  chart.guide().arc({
    zIndex: 1,
    start: [0, 0.965],
    end: [data[0].value, 0.965],
    style: {
      stroke: color,
      lineWidth: 10
    }
  });
  // 绘制指标数字
  chart.guide().html({
    position: ['50%', '95%'],
    html: '<div style="width: 300px;text-align: center;">' + '<p style="font-size: 14px; color: #545454;margin: 0;margin-top: 10px;">' + title + '</p>' + '<p style="font-size: 20px;color: #545454;margin: 0;">' + ((data[0].value) * 100).toFixed(2) + '%</p>' + '</div>'
  });

  chart.render();
  return chart
}

export default draw

css

<style lang="scss" scoped>
.dashboard {
  .btn-add {
    position: fixed;
    right: 20px;
    z-index: 100;
  }
  #canvas {
    width: 100%;
    .item {
      display: inline-block;
      margin-left: 20px;
      margin-bottom: 20px;
      width: 30%;
      border-radius: 4px;
      box-shadow: 0 2px 4px rgba(0, 0, 0, .12), 0 0 6px rgba(0, 0, 0, .04);
    }
  }
}
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值