圆心动态多节点分布拓扑图,自动计算节点坐标(改进版)

描述:当前是关于Echarts图表中的 示例。
 
            // 指定图表的配置项和数据
    let [dataArr, allArr, erArr, sanArr] = [[], [], [], []];
    var items = [
        {
            name: "监管大屏",
            pointType: "cloud",
        },
        {
            name: "综合业务监管",
            pointType: "cloud",
        },
        {
            name: "传染性疾病监管",
            pointType: "cloud",
        },
        {
            name: "公共卫生监管",
            pointType: "cloud",
        },
        {
            name: "人口信息监管",
            pointType: "cloud",
        },
        {
            name: "民营医院监管",
            pointType: "cloud",
        },
        {
            name: "互联网医院监管",
            pointType: "cloud",
        },
        {
            name: "基建设备监管",
            pointType: "cloud",
        },
        {
            name: "数据中心",
            pointType: "cloud",
        },
        {
            name: "平台管理",
            pointType: "cloud",
        },
        {
            name: "平台监管大屏",
            pointType: "point",
            belong: "监管大屏",
        },
        {
            name: "综合业务监管大屏",
            pointType: "point",
            belong: "监管大屏",
        },
        {
            name: "民营机构监管大屏",
            pointType: "point",
            belong: "监管大屏",
        },
        {
            name: "公共卫生监管大屏",
            pointType: "point",
            belong: "监管大屏",
        },
        {
            name: "互联网医院监管大屏",
            pointType: "point",
            belong: "监管大屏",
        },
        {
            name: "平台监管大屏2",
            pointType: "point",
            belong: "综合业务监管",
        },
        {
            name: "综合业务监管大屏2",
            pointType: "point",
            belong: "综合业务监管",
        },
        {
            name: "民营机构监管大屏2",
            pointType: "point",
            belong: "综合业务监管",
        },
        {
            name: "公共卫生监管大屏2",
            pointType: "point",
            belong: "综合业务监管",
        },
        {
            name: "互联网医院监管大屏2",
            pointType: "point",
            belong: "综合业务监管",
        },
        {
            name: "平台监管大屏3",
            pointType: "point",
            belong: "传染性疾病监管",
        },
        {
            name: "综合业务监管大屏3",
            pointType: "point",
            belong: "传染性疾病监管",
        },
        {
            name: "民营机构监管大屏3",
            pointType: "point",
            belong: "传染性疾病监管",
        },
        {
            name: "公共卫生监管大屏3",
            pointType: "point",
            belong: "传染性疾病监管",
        },
        {
            name: "互联网医院监管大屏3",
            pointType: "point",
            belong: "传染性疾病监管",
        },
        {
            name: "平台监管大屏4",
            pointType: "point",
            belong: "公共卫生监管",
        },
        {
            name: "综合业务监管大屏4",
            pointType: "point",
            belong: "公共卫生监管",
        },
        {
            name: "民营机构监管大屏4",
            pointType: "point",
            belong: "公共卫生监管",
        },
        {
            name: "公共卫生监管大屏4",
            pointType: "point",
            belong: "公共卫生监管",
        },
        {
            name: "互联网医院监管大屏4",
            pointType: "point",
            belong: "公共卫生监管",
        },
        {
            name: "平台监管大屏5",
            pointType: "point",
            belong: "人口信息监管",
        },
        {
            name: "综合业务监管大屏5",
            pointType: "point",
            belong: "人口信息监管",
        },
        {
            name: "民营机构监管大屏5",
            pointType: "point",
            belong: "人口信息监管",
        },
        {
            name: "人口信息监管大屏5",
            pointType: "point",
            belong: "人口信息监管",
        },
        {
            name: "互联网医院监管大屏5",
            pointType: "point",
            belong: "人口信息监管",
        },
        {
            name: "平台监管大屏6",
            pointType: "point",
            belong: "民营医院监管",
        },
        {
            name: "综合业务监管大屏6",
            pointType: "point",
            belong: "民营医院监管",
        },
        {
            name: "民营机构监管大屏6",
            pointType: "point",
            belong: "民营医院监管",
        },
        {
            name: "民营医院监管大屏6",
            pointType: "point",
            belong: "民营医院监管",
        },
        {
            name: "互联网医院监管大屏6",
            pointType: "point",
            belong: "民营医院监管",
        },
        {
            name: "平台监管大屏7",
            pointType: "point",
            belong: "互联网医院监管",
        },
        {
            name: "综合业务监管大屏7",
            pointType: "point",
            belong: "互联网医院监管",
        },
        {
            name: "民营机构监管大屏7",
            pointType: "point",
            belong: "互联网医院监管",
        },
        {
            name: "民营医院监管大屏7",
            pointType: "point",
            belong: "互联网医院监管",
        },
        {
            name: "互联网医院监管大屏7",
            pointType: "point",
            belong: "互联网医院监管",
        },
        {
            name: "平台监管大屏8",
            pointType: "point",
            belong: "基建设备监管",
        },
        {
            name: "综合业务监管大屏8",
            pointType: "point",
            belong: "基建设备监管",
        },
        {
            name: "民营机构监管大屏8",
            pointType: "point",
            belong: "基建设备监管",
        },
        {
            name: "民营医院监管大屏8",
            pointType: "point",
            belong: "基建设备监管",
        },
        {
            name: "互联网医院监管大屏8",
            pointType: "point",
            belong: "基建设备监管",
        },
        {
            name: "平台监管大屏10",
            pointType: "point",
            belong: "平台管理",
        },
        {
            name: "综合业务监管大屏10",
            pointType: "point",
            belong: "平台管理",
        },
        {
            name: "民营机构监管大屏10",
            pointType: "point",
            belong: "平台管理",
        },
        {
            name: "民营医院监管大屏10",
            pointType: "point",
            belong: "平台管理",
        },
        {
            name: "互联网医院监管大屏10",
            pointType: "point",
            belong: "平台管理",
        },
    ];

    // 点
    items.forEach((el, ind) => {
        if (el.pointType === "cloud") {
            // 云
            el.symbol = "image://";
            el.symbolSize = [56, 35];
            el.label = {
                normal: {
                    show: true,
                    position: "bottom",
                    borderWidth: 1,
                    borderRadius: 12,
                    padding: [5, 10],
                    distance: 10,
                    color: "#5D77FE",
                    borderColor: "#5D77FE",
                    backgroundColor: 'rgba(93, 119, 254, 0.2)'
                },
            };
            erArr.push(el);
        } else if (el.pointType === "point") {
            // 球
            el.itemStyle = {
                borderColor: '#5D77FE',
                borderWidth: 3,
                color: "#5D77FE",
            };
            // 清除没有父节点的值
            if (el.belong && !items.some((iteP) => el.belong === iteP.name)) {
                items.splice(ind, 1);
            } else {
                sanArr.push(el);
            }
        }
        if (!el.belong) {
            el.children = [];
            items.forEach((es) => {
                if (es.belong === el.name) {
                    el.children.push(es);
                }
            });
            allArr.push(el);
        }
    });

    // 圆形分区
    function group(arr, r) {
        const newArray = [];
        const { length: arLen } = arr;
        let cloud = {}
        arr.forEach((e, ind) => {
            // 按角度均匀分布
            let ang = 0
            if (e.belong) {
                //point角度根据已经分配好的cloud的角度决定
                cloud = erArr.find(obj => {
                    return obj.name === e.belong
                });
                //多个point,在cloud的角度上递增偏移一些角度
                let index = 0
                for (let j = 0; j < cloud.children.length; j++) {
                    if (cloud.children[j].name === e.name) {
                        //是第几个子节点
                        index = j;
                        break;
                    }
                }
                //间隔角度
                var intervalDeg = 7
                let midIndex = cloud.children.length / 2
                //执行角度平均偏移
                ang = cloud.ang + intervalDeg * (index - midIndex)
            } else {
                //分配cloud角度,并记录下
                ang = 90 - (360 / arLen).toFixed(2) * (ind + 1);
                e.ang = ang
            }
            const x = (Math.cos(ang * Math.PI / 180)).toFixed(2) * r;
            const y = (Math.sin(ang * Math.PI / 180)).toFixed(2) * r;
            e.value = [x.toFixed(2), y.toFixed(2)];
            if (e.belong) {
                let rotate = '';
                const angle = (20 / arr.length).toFixed(2);
                if (x > 0 && y >= 0) {
                    rotate = (r - angle * (ind)).toFixed(2);
                } else if (x <= 0 && y > 0) {
                    rotate = (-r + angle * (arr.length - ind)).toFixed(2);
                } else if (x < 0 && y <= 0) {
                    rotate = (r - angle * (ind - arr.length * 0.5)).toFixed(2);
                } else if (x >= 0 && y < 0) {
                    rotate = (-8 + angle * (arr.length * 0.25 - ind)).toFixed(2);
                }
                e.label = {
                    show: true,
                    width: 200,
                    color: {
                        lineColor: {
                            color: "rgb(24,163,239)",
                        },
                    },
                    rotate,
                    align: x > 0 ? 'left' : 'right',
                    padding: x > 0 ? [0, 0, 0, 16] : [0, 16, 0, 0],
                };
                if (e.offset) {
                    e.label.offset = e.offset;
                }
            }
            newArray.push(e);
        });
        return newArray;
    }
    // 线配置
    function linesConfig(arr) {
        const [dataArr, targetCoord] = [[], [0, 0]];
        arr.forEach((el) => {
            if (!el.belong) {
                dataArr.push([
                    { coord: targetCoord },
                    {
                        coord: el.value,
                        lineStyle: el.lineStyle,
                        effect: {
                            symbolSize: [6, 20], // 动画的大小
                            period: 3, // 时间快慢
                            delay: 2, // 延时计算
                            ...el.effect,
                        }
                    },
                ]);
                arr.forEach((element) => {
                    if (element.belong === el.name) {
                        dataArr.push([
                            { coord: el.value },
                            {
                                coord: element.value,
                                lineStyle: element.lineStyle,
                                effect: {
                                    symbolSize: [5, 15], // 动画的大小
                                    period: 3, // 时间快慢
                                    delay: 2, // 延时计算
                                    ...element.effect,
                                },
                            },
                        ]);
                    }
                });
            }
        });
        return dataArr;
    }

    // 点分布
    erArr = group(erArr, 36);
    sanArr = group(sanArr, 50);
    allArr = [...erArr, ...sanArr];

    // 线坐标和配置
    dataArr = linesConfig(allArr);
    option = {
        xAxis: {
            show: false,
            type: "value",
            max: 50,
            min: -51,
        },
        grid: {
            top: 100,
            bottom: 100,
            left: 130,
            right: 130,
        },
        yAxis: {
            show: false,
            type: "value",
            max: 50,
            min: -50,
        },
        series: [{
            type: "graph",
            coordinateSystem: "cartesian2d",
            z: 3,
            itemStyle: {
                normal: {
                    shadowColor: "none",
                },
            },
            data: allArr,
        },
            {
                name: "",
                type: "lines",
                coordinateSystem: "cartesian2d",
                polyline: true,
                z: 1,
                lineStyle: {
                    width: 2,
                    color: '#5D77FE',
                    curveness: 0,
                },
                effect: {
                    show: true,
                    smooth: false,
                    trailLength: 0, // 拖尾的长度
                    symbol: 'pin', // 动画的类型
                },
                data: dataArr,
            },
        ],
    }