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

代码如下:
<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],
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,
}
},
},
},
children: [],
}
)
const option = ref({
tooltip: {
trigger: "item",
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,
}
},
},
},
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>