当前位置: 首页 > article >正文

vue3 echarts tree结构实现点击添加子节点和修改图标样式

先看这个,本篇是根据这个进行升级而来。点击进入

再看效果:

在这里插入图片描述

代码如下:

<template>
<div ref="ecs" id="ecs" style="width: 800px;height:800px; background-color:white;"></div>
</template>

<script setup>
import {onMounted, ref} from "vue";
import * as echarts from 'echarts';
import pause from "@/assets/pause.png"
import play from "@/assets/play.png"

const ecs = ref()
const myChart = ref()
const timestamp = ()=>{
  const dateTimestamp = new Date().getTime();
  return dateTimestamp+""
}
const my_data = ref(
{
    name: "根节点",
    url: "",
    id: timestamp(),
    // 根节点样式设置
    itemStyle: {
      color: "#a53626",
      borderColor: "#a53626",
    },
    label: {
        show: true,
        position: [ 15, 70],  // 用这个和box的height控制下边按钮的位置
        verticalAlign: "middle",
        fontWeight: "bold",
        formatter: ["{box|{b}}", "{bg| }"].join("\n"),
        rich: {
          box: {
            height: 50,
            color: "#0f0",
            padding: [10, -10, 10, -5],
            align: "center",
            fontWeight: "bold",
            fontSize: 16,
          },
          bg: {
            height: 30,
            color: "#fff",
            padding: [15, -15, 10, 10],
            align: "center",
            fontWeight: "bold",
            fontSize: 16,
            backgroundColor:{
              image: play,
              // image: ""
            }
          },
        },
    },
    children: [],
  }
)
const option = ref({
  tooltip: {
    trigger: "item",

    // 给tooltip绑定click事件
    triggerOn: "click",
    enterable: true,
    extraCssText: 'z-index: 99;max-width: 100px;white-space:pre-wrap',
    formatter: function(params) {
      return `<div οnclick="myDialog('${params.data.id}')">添加子节点</div>`
    },

    backgroundColor: "#dfdfdf",
    textStyle: {
      color: "black",
    },
  },
  series: [
    {
      type: "tree",
      expandAll: true,
      symbolSize: 75,
      symbol: "roundRect",
      edgeShape: "polyline",
      edgeForkPosition: "50%",
      initialTreeDepth: 10,
      orient: "vertical",
      itemStyle: {
        color: "black",
        borderColor: "black",
      },
      expandAndCollapse: true,
      animationDuration: 550,
      animationDurationUpdate: 750,
      lineStyle: {
        color: "#7b7b7b",
        width: 3,
      },
      data: [my_data.value,],
    },
  ],
});

function nodeContainsValue(node, id, i=0) {
  i =i+1
  if (node.id == id) {
    node.children.push(
      {
        name: `${i}级节点`,
        url: "",
        id: timestamp(),
        // 根节点样式设置
        itemStyle: {
          color: "#a53626",
            borderColor: "#a53626",
        },
        label: {
          show: true,
            position: [ 10, 70],
            verticalAlign: "middle",
            fontWeight: "bold",
            formatter: ["{box|{b}}", "{bg| }"].join("\n"),
            rich: {
	            box: {
	              height: 50,
	                color: "#0f0",
	                padding: [10, -10, 10, -5],
	                align: "center",
	                fontWeight: "bold",
	                fontSize: 16,
	            },
	            bg: {
	              height: 30,
	                color: "#fff",
	                padding: [15, -15, 10, 10],
	                align: "center",
	                fontWeight: "bold",
	                fontSize: 16,
	                backgroundColor:{
	                  image: play,
	                // image: ""
                }
            },
          },
        },
        children: [],
      }
    )
    return true;
  }
  if (node.children.length == 0) {
    return false;
  }
  for(let j=0; j< node.children.length; j++){
    if(nodeContainsValue(node.children[j], id, i)){
      return true
    }
  }
  return false
}
onMounted(() => {
  myChart.value = echarts.init(ecs.value);
  myChart.value.setOption(option.value);
  window.myDialog= function(id) {
    nodeContainsValue(my_data.value, id, 0)
    myChart.value.setOption(option.value);
  }
  myChart.value.on("click", (params)=>{
    let image_array = params.data.label.rich.bg.backgroundColor.image.split("/")
    let my_image = image_array[image_array.length-1]
    if (my_image == "play.png"){
      params.data.label.rich.bg.backgroundColor.image = pause
    }else {
      params.data.label.rich.bg.backgroundColor.image = play
    }
  })
})
</script>

<style >
html, body{
  height: 100%;
  background-color: pink;
}
</style>

另附formatter里的{a},{b},{c},{d}代表什么:点击可参考

1、折线图、区域图、柱状图、条形图、K线图:
{a}:系列名称(series.name)
{b}:类目值(通常是 x 轴的标签)
{c}:数值(series.data.value)
{d}:无
2、散点图、气泡图:
{a}:系列名称(series.name)
{b}:数据名称(通常是 x 轴的标签)
{c}:数值数组(series.data.value)
{d}:无
3、饼图、雷达图:
{a}:系列名称(series.name)
{b}:数据项名称(series.data.name)
{c}:数值(series.data.value)
{d}:百分比(series.data.percent)
4、地图:
{a}:系列名称(series.name)
{b}:区域名称(region name)
{c}:合并数值(merged value)
{d}:无

下边这个改动点在这, οnclick=‘myDialog(${JSON.stringify(params)})’>添加子节点 // 使用单引号这种方式是可以的

<template>
  <div ref="ecs" id="ecs" style="width: 800px;height:800px; background-color:white;"></div>
</template>

<script setup>
import {onMounted, ref} from "vue";
import * as echarts from 'echarts';
// import pause from "@/assets/pause.png"
// import play from "@/assets/play.png"

const ecs = ref()
const myChart = ref()
const my_params = ref()
const timestamp = ()=>{
  const dateTimestamp = new Date().getTime();
  return dateTimestamp+""
}
const my_data = ref(
    {
      name: "根节点",
      url: "",
      id: timestamp(),
      // 根节点样式设置
      itemStyle: {
        color: "#a53626",
        borderColor: "#a53626",
      },
      label: {
        show: true,
        position: [ 15, 70],  // 用这个和box的height控制下边按钮的位置
        verticalAlign: "middle",
        fontWeight: "bold",
        formatter: ["{box|{b}}", "{bg| }"].join("\n"),
        rich: {
          box: {
            height: 50,
            color: "#0f0",
            padding: [10, -10, 10, -5],
            align: "center",
            fontWeight: "bold",
            fontSize: 16,
          },
          bg: {
            height: 30,
            color: "#fff",
            padding: [15, -15, 10, 10],
            align: "center",
            fontWeight: "bold",
            fontSize: 16,
            backgroundColor:{
              // image: play,
              image: ""
            }
          },
        },
      },
      children: [],
    }
)
const option = ref({
  tooltip: {
    trigger: "item",

    // 给tooltip绑定click事件
    triggerOn: "click",
    enterable: true,
    extraCssText: 'z-index: 99;max-width: 100px;white-space:pre-wrap',
    formatter: function(params) {
      console.log("before params", params)
      console.log("typeof params", typeof params);
      // my_params.value = params  // 通过定义变量的方式也可以传递params
      return `<div onclick="myDialog('${params.data.id}')">添加子节点</div>`
      // return `<div onclick="myDialog('${JSON.stringify(params)}')">添加子节点</div>`  // 使用双引号这种方式不行
      // return `<div onclick='myDialog(${JSON.stringify(params)})'>添加子节点</div>`   // 使用单引号这种方式是可以的
    },

    backgroundColor: "#dfdfdf",
    textStyle: {
      color: "black",
    },
  },
  series: [
    {
      type: "tree",
      expandAll: true,
      symbolSize: 75,
      symbol: "roundRect",
      edgeShape: "polyline",
      edgeForkPosition: "50%",
      initialTreeDepth: 10,
      orient: "vertical",
      itemStyle: {
        color: "black",
        borderColor: "black",
      },
      expandAndCollapse: true,
      animationDuration: 550,
      animationDurationUpdate: 750,
      lineStyle: {
        color: "#7b7b7b",
        width: 3,
      },
      data: [my_data.value,],
    },
  ],
});

function nodeContainsValue(node, id, i=0) {
  i =i+1
  if (node.id == id) {
    node.children.push(
        {
          name: `第${i}级节点`,
          url: "",
          id: timestamp(),
          // 根节点样式设置
          itemStyle: {
            color: "#a53626",
            borderColor: "#a53626",
          },
          label: {
            show: true,
            position: [ 10, 70],
            verticalAlign: "middle",
            fontWeight: "bold",
            formatter: ["{box|{b}}", "{bg| }"].join("\n"),
            rich: {
              box: {
                height: 50,
                color: "#0f0",
                padding: [10, -10, 10, -5],
                align: "center",
                fontWeight: "bold",
                fontSize: 16,
              },
              bg: {
                height: 30,
                color: "#fff",
                padding: [15, -15, 10, 10],
                align: "center",
                fontWeight: "bold",
                fontSize: 16,
                backgroundColor:{
                  // image: play,
                  image: ""
                }
              },
            },
          },
          children: [],
        }
    )
    return true;
  }
  if (node.children.length == 0) {
    return false;
  }
  for(let j=0; j< node.children.length; j++){
    if(nodeContainsValue(node.children[j], id, i)){
      return true
    }
  }
  return false
}
onMounted(() => {
  myChart.value = echarts.init(ecs.value);
  myChart.value.setOption(option.value);
  window.myDialog= function(id) {
  // window.myDialog= function(params) {
    console.log("my_params.value", my_params.value)
    // console.log("params====>", params);
    nodeContainsValue(my_data.value, id, 0)   // 不通过递归直接在params中push children试试可以不  已经试验过了不可以
    // nodeContainsValue(my_data.value, params.data.id, 0)
    myChart.value.setOption(option.value);
  }
  myChart.value.on("click", (params)=>{
    let image_array = params.data.label.rich.bg.backgroundColor.image.split("/")
    let my_image = image_array[image_array.length-1]
    if (my_image == "play.png"){
      params.data.label.rich.bg.backgroundColor.image = pause
    }else {
      params.data.label.rich.bg.backgroundColor.image = play
    }
  })
})
</script>

<style >
html, body{
  height: 100%;
  background-color: pink;
}
</style>

http://www.kler.cn/a/441599.html

相关文章:

  • 基于streamlit搭简易前端页面
  • 二、STM32MP257安全启动流程简介
  • oracle多次替换字符,批量替换,循环替换------------gxl
  • 实战攻防中针对JS路径的泄露和Webpack漏洞的初探
  • 无人机故障安全模式设计逻辑与技术!
  • SQL -- 条件分支
  • OpenCV及基本用法
  • 原理 | dubbo [与 springboot 整合时服务导出的触发]
  • Java全栈项目 - 学生宿舍管理系统
  • 加载文件到docker中的mysql上
  • Linux高性能服务器编程 | 读书笔记 | 6. 高性能服务器程序框架
  • 【报错解决】pip install volcengine-python-sdk无法安装包
  • 【行政区编码对应表及生态等级数据的制作】-python
  • centos下安装ffmpeg
  • Python爬虫之代理的设置
  • linux0.11源码分析第一弹——bootset.s内容
  • (2024.12)记录——Ubuntu20.04安装opencv库
  • 【JVM】JVM基础教程(四)
  • ubuntu20.04安装qt creator
  • Leetcode经典题8--H指数