npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

moon-hierarchy

v2.3.3

Published

### 预览地址

Downloads

115

Readme

moon-hierarchy

预览地址

https://www.fste.top/demo/html//moon-hierarchy/#/

开发目的

  • 使用 d3 开发层级数据展示图。且支持各种自定义功能。

安装 npm i moon-hierarchy -S

vue 文件中使用

  • vue2中使用

import hierarchy from 'moon-hierarchy/vue2';
import 'moon-hierarchy/moon-hierarchy.css';
components: {  hierarchy  }
  • vue3中使用

import hierarchy from 'moon-hierarchy';
import 'moon-hierarchy/moon-hierarchy.css';
components: {  hierarchy  }

Props

| 参数 | 说明 | 类型 | 可选值 | 默认值 | | ---------------- | -------------------------------------------------------------------------------------------------------- | --------------------- | ------- | ------------------------------------ | | width | svg 宽度 | Number | - | 1300 | | height | svg 高度 | Number | - | 800 | | mode | 渲染模式:水平方向 h,垂直方向 v | String | h,v | h | | layout | 布局:水平方向-左右,右左,蝴蝶,垂直->上下,下上,蝴蝶 | String | - | tb/bt/bf, lr/rl/bf | | limit | 水平模式,子节点最大展示数,多余的出收起按钮 ,-1 时全部展出 | Number | -1;1+ | 3 | | treeData | 扁平化树数据,外部修改后,会触发画布重绘。但内置新增、修改、删除方法不会触发重绘,且会修改 treeData 数据 | Array | - | [] | | treeOptions | 树数据选项 | Object | - | { id: 'id',pId: 'pId',name: 'name',} | | duration | 动画过渡时间 | Number | | 400 | | defaultOpenLevel | 默认展开层级,-1 时全部展开 | Number | -1 ,1+ | 2 | | negativeIds | 蝴蝶模型,指定负向数据对应的 id,必须是根节点的直接子节点 | Array | - | [] | | config | 配置节点连线,详情见下方说明 | Object | - | {} | | canExpendFold | 点击当前节点,展开和收缩子节点 ,传入函数,则接受当前节点数据,返回一个 boolean | Boolean, (d)=>boolean | - | true | | expendShape | 指定点击展开的元素,必须同时设置 foldShape 才起作用,可以是 id,class 或元素,默认整个节点 | string | - | - | | foldShape | 指定点击闭合的元素,必须同时设置 expendShape 才起作用,可以是 id,class 或元素 ,默认整个节点 | string | - | - |

Props.config

| 参数 | 说明 | | ---------- | -------------- | | node | 节点配置 | | arrow | 箭头配置 | | link | 连线配置 | | zoom | 缩放配置 | | customView | 自定义节点视图 |

Props.config.node 配置

| 参数 | 说明 | 默认值 | | ---------- | --------------------------------------------------------------------------------------- | ---------- | | attrs | 设置节点除 id,transform 其他的所有属性 | - | | on | 节点监听 ,详情见下方说明 | Object | | padding | 内容区域到边框的距离,详情见下 | h:15,v:10 | | nodeWidth | 布局的节点宽度,水平模式,实际的节点宽度依据内容确定。在数据 data._nodeConfig 中可查看 | h:60,v:168 | | nodeHeight | 布局的节点高度,水平模式,实际的节点高度,还会加上 padding 值 | h:16,v:68 | | separation | 节点间距 | 1.5 | | rect | 矩形配置 | - | | text | 文本配置 | - | | plus | 折叠图标配置 | - | | exShaps | 自定义图型配置, | [] |

  • node.on 节点监听

| 名称 | 说明 | 类型 | | ------------------ | ---------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------- | | clickFetchChildren | 点击后异步加载子节点。父节点上需要有_hasChildren 标记 ,返回一个数组,可以是层级结构的数据。data:当前节点源数据信息;el:当前节点对应的 d3 元素对象;svg:画布元素 | (data:treeData,el:selection,svg:d3Selection)=>object[]|object | | click | 鼠标事件,禁止冒泡; | (e:MouseEvent,d:d3Node,svg:d3Selection)=>object[] | | mouseover | 鼠标事件 | (e:MouseEvent,d:d3Node,svg:d3Selection)=>object[] | | mouseout | 鼠标事件 | (e:MouseEvent,d:d3Node,svg:d3Selection)=>object[] | | 其他事件 | 其他事件 | (e:MouseEvent,d:d3Node,svg:d3Selection)=>object[] |

[!NOTE]

所有的事件都不会在层级节点展开/收起按钮节点上触发

  • node.padding

    • 可以是数字,数组,函数, 函数时接受一个当前节点数据的参数,动态返回一个数组

    • type=[number,number,number,number]|number|(d:d3Node)=>{return [number,number,number,number]}
  • node.rect 默认组件

    • {
        attrs:object,
        on:object,
        show:boolean
      }
    • | 参数 | 说明 | 默认值 | | ----- | -------------------------------------------------- | ----------------------- | | attrs | 组件样式配置 | { attrs:{ } ,show:true} | | show | 是否显示,设置为 false 后可以通过 exShaps 自己指定 | true |

  • node.text 默认组件

    • {
        attrs:object,
        on:object,
        compose:object,// mode 水平模式可配置
        show:boolean
      }
    • | 参数 | 说明 | 默认值 | | ----- | -------------------------------------------------- | ------------------------------------------------ | | attrs | 组件配置 | v:{font-size:16,line-height:10} h:{font-size:10} | | show | 是否显示,设置为 false 后可以通过 exShaps 自己指定 | true |

  • node.plus 默认组件

    • {
        attrs:object,
        on:object,
        show:boolean
      }
    • | 参数 | 说明 | 默认值 | | ----- | -------------------------------------------------- | -------------- | | attrs | 组件配置 | { r: v:10;h:6} | | show | 是否显示,设置为 false 后可以通过 exShaps 自己指定 | true |

  • node.exShaps 自定义图形配置

    • 一个图形数组。

    • 图形嵌套,通过指定 children 实现

    • 具体配置如下:

exShaps = [
    {
        name: 'g',
        // 属性配置
        attrs: {
            stroke: 'rgb(153, 153, 153)',
            fill: 'rgb(234, 242, 255)',
            'stroke-width': 1
        },
        // 链式函数配置。如text
        compose: {
            text(d) {
                return d.name;
            }
        },
        // 监听事件配置。
        on: {
            click(e, d) {}
        },
        children: [
            {
                name: 'circle',
                attrs: {
                    r: 20
                }
            }
        ]
    }
];

Props.config.arrow 箭头

{
  attrs:object,
  path: string,
  show:true
}

| 参数 | 说明 | 默认值 | | ----- | ---------------- | ----------------------------------------------------------------------------------------------------------------- | | attrs | 箭头 marker 配置 | { viewBox: '0 0 10 10', refX: '10', refY: '5', markerWidth: '6', markerHeight: '6', orient: 'auto-start-reverse'} | | path | path d 配置 | d:’M 0 0 L 10 5 L 0 10 z‘ | | show | 是否显示 | true |

Props.config.link 连线配置

{
  [string]:any
}

| 参数 | 说明 | 默认值 | | ---- | ------------ | ------------------------------------------------------------------------------ | | link | 路径样式配置 | { stroke: '#D8D8D8', fill: 'none', 'stroke-opacity': '1', 'stroke-width': '1'} |

Props.config.zoom 缩放配置

{
  defaultScale:number,
  scaleRange:[number,number],
  callback:(e)=>void
}

| 参数 | 说明 | 默认值 | | ------------ | -------------------------- | ------- | | defaultScale | 默认缩放值 | 1 | | scaleRange | 可缩放范围 | [0.2,2] | | callback | 缩放回到函数,接受缩放参数 | - |

Props.config.customView 自定视图配置

| 参数 | 说明 | 默认值 | | -------- | ------------------------------------------------------------------- | ------------------------ | | width | 视图宽度 | 100 | | height | 视图高度 | 50 | | priority | 相对于布局节点,视图优先出现位置,rb 右下,rt 右上,lb 左下,lt 左上 | ['rb', 'rt', 'lb', 'lt'] | | duration | 动画过渡时间,默认 props.duration 中的值 | 400 |

节点数据说明

  • 所有的 attrs,compose 中关于节点属性的属性配置项的值,即可以是字符串,也可以是函数

  • 函数接受一个参数,为当前节点数据。其中的 data 属性值为业务数据。

    attrs:{
        display(d){
            if(d.data....)
                return 'none'
        }
    }
{
"data": {
...业务数据
"_hasChildren": true, //异步加载子节点时,父节点中的判断标记
"_sign": 1, //不同模型下,上下,左右标记。 左、上:-1。 右,下:1
"_nodeConfig": object, //当前节点配置信息。包含节点的高度宽度形状等信息。
"_isexpend":false,//水平模式下,限制节点按钮的展开闭合状态,只有限制节点按钮才有
"_name": []  //垂直模式下,文本的多行节点信息
},
"x": 126, //节点坐标
"y": 136 //节点坐标
}

Events

| 名称 | 说明 | 参数 | | --------- | ------------------------------------------------- | --------------------------------------- | | draw-done | 画布渲染完成后事件,返回当前画布节点与内容区域节点 | {svg:d3Selection,container:d3Selection} |

Slot

| 名称 | 说明 | | ------- | -------------- | | default | 自定义视图节点 |

Methods

| 名称 | 说明 | 类型 | | ---------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------- | | getNodeById | 依据数据中唯一标识 id,对应 treeOptions 中的 id, 获取对应的数据,以及元素对象 | (id:string)=>({ data: d3Node, el: d3Selection }) | | getAllNode | 获取所有非展开收起的节点,对应的数据,以及元素对象 | (id:string)=>({ data: d3Node, el: d3Selection }[]) | | moveToCenter | 移动到中点 | ()=>void | | zoom | 缩放画布, 大于 1 的为放大,小于 1 大于 0 的为缩小。负数无效 | (scale:number)=>void | | addNode | 在 targetId 对应目标节点上新增子节点 ,childrenNode 是一个扁平树数据也可以是单个对象数据。 _sign 当目标节点为根节点且为 bf 布局时,用于指定添加的方位。会修改 treeData 数据 | (targetId,childrenNode:treeItem|treeItem[],_sign)=>void | | updateNodeByData | 更新节点对应业务数据。但不能更改组件数据,如 id,pId,children,"_"开头的属性。会修改 treeData 数据 | (dataList:treeItem|treeItem[])=>void | | removeNodeById | 依据节点 id,移除该节点以及其所有子节点。会修改 treeData 数据 | (id:string|string[])=>void | | pauseZoom | 暂停缩放功能 | ()=>void | | continueZoom | 启动缩放功能 | ()=>void | | showCustomView | 显示 slot 对应的自定义的 view 视图,e:鼠标信息,d:布局节点信息,width,height,priority,duration:参考 config.customView,优先级高于 config.customView 中的配置。 | (e, d, width, height, priority,duration)=>void | | hiddenCustomView | 隐藏 slot 对应的自定义的 view 视图, | ()=>void | | moveToNode | 展开到指定节点所在的层级,将节点移动到画布中间,并且可以触发 eventList 指定的节点事件。 | (targetNodeId:string,eventList:string[]|string)=>void | | expendAllNode | 展开所有节点 | ()=>void | | foldAllNode | 将所有节点折叠到默认层级 | ()=>void |

各个节点,图形默认的 id 和 class

| 名称 | class | id | | ------------------------------------------ | -------------------------------- | ------------------------------------------ | | 节点 | moon-hierarchy-node | 'node'+ 节点数据中的唯一标识字段对应的数据 | | 节点-根节点 | moon-hierarchy-node-root | 'node'+ 节点数据中的唯一标识字段对应的数据 | | 节点-有子节点的节点 | moon-hierarchy-node-haschildren | 'node'+ 节点数据中的唯一标识字段对应的数据 | | 节点-有子节点的节点,且展开 | moon-hierarchy-node-expend | 'node'+ 节点数据中的唯一标识字段对应的数据 | | 节点-限制展开收起的节点 | moon-hierarchy-node-limit-button | 'node'+ 节点数据中的唯一标识字段对应的数据 | | rect | moon-hierarchy-rect | - | | text | moon-hierarchy-text | - | | 展开收起图形 | moon-hierarchy-plus | - | | 连线 | moon-hierarchy-link | 'link'+"起点 id-终点 id" | | 节点鼠标悬浮,该节点对应的所有子节点间连线 | moon-hierarchy-node-hover-link | 'link'+"起点 id-终点 id" | | 用户自定义视图节点 | moon-hierarchy-custom-view | - |

DefaultStyle

  • 可以自行依据项目修改
    .moon-hierarchy-svg {
    .moon-hierarchy-node {
        // 默认rect样式
        .moon-hierarchy-rect {
            fill: #fff;
            stroke: rgb(216, 216, 216);
            stroke-width: 0.5;
        }
        // 默认text样式
        .moon-hierarchy-text {
            fill: rgb(51, 51, 51);
        }
        // 默认plus样式
        .moon-hierarchy-plus {
            stroke: rgb(153, 153, 153);
            fill: rgb(234, 242, 255);
            stroke-width: 1;
        }

        // 根节点样式
        &.moon-hierarchy-node-root {
            .moon-hierarchy-rect {
                fill: rgb(18, 137, 239);
            }
            .moon-hierarchy-text {
                fill: #fff;
            }
            .moon-hierarchy-plus {
                display: none;
            }
        }
        // 没有子节点的样式
        &:not(.moon-hierarchy-node-haschildren) {
            .moon-hierarchy-plus {
                display: none;
            }
        }
        // 非根节点的节点展开后样式
        &.moon-hierarchy-node-expend:not(.moon-hierarchy-node-root) {
            .moon-hierarchy-text {
                fill: rgb(18, 139, 237);
            }
        }
        // 节点展开后样式
        &.moon-hierarchy-node-expend {
            .moon-hierarchy-plus {
                line:nth-of-type(2) {
                    display: none;
                }
            }
        }
        //展开限制节点的按钮型节点样式
        &.moon-hierarchy-node-limit-button {
            .moon-hierarchy-rect {
                fill: rgb(247, 247, 247);
            }
        }
        // 非展开限制节点的按钮型节点得节点鼠标悬停样式
        &:not(.moon-hierarchy-node-limit-button):hover {
            .moon-hierarchy-rect {
                stroke: rgb(18, 137, 239);
            }
        }
    }
    .moon-hierarchy-custom-view {
        width: 100%;
        height: 100%;
        background-color: #fff;
        padding: 5px;
        border-radius: 8px;
        box-sizing: border-box;
        cursor: pointer;
    }
    .moon-hierarchy-arrow {
        fill: #128bed;
    }
    .moon-hierarchy-link {
        stroke: #d8d8d8;
        stroke-opacity: 1;
        stroke-width: 1;
    }
    .moon-hierarchy-node-hover-link {
        stroke-dasharray: 1000;
        stroke-dashoffset: 1000;
        animation: moon-hierarchy-link-run 20s linear infinite;
    }
    .moon-hierarchy-loading {
        animation: moon-hierarchy-rotate 3s linear infinite;
    }
    @keyframes moon-hierarchy-rotate {
        0% {
            transform: rotate(0deg);
        }
        100% {
            transform: rotate(360deg);
        }
    }

    @keyframes moon-hierarchy-link-run {
        from {
            stroke-dasharray: 10, 5;
        }
        to {
            stroke-dasharray: 20, 5;
        }
    }
}

Demo

  <template>
    <div>
        <div class="pannel">
            <div class="button-group">
                <button @click="$refs.hierarchy.moveToCenter()">移动到中心</button>
                <button @click="$refs.hierarchy.zoom(1.5)">放大</button>
                <button @click="$refs.hierarchy.zoom(0.5)">缩小</button>
                <button @click="$refs.hierarchy.pauseZoom()">暂停缩放</button>
                <button @click="$refs.hierarchy.continueZoom()">恢复缩放</button>
                <button @click="$refs.hierarchy.moveToNode('qyfxsbpggl', ['click', 'contextmenu'])">
                    移动到指定节点,并触发contextmenu
                </button>
                <button @click="$refs.hierarchy.expendAllNode()">展开全部节点</button>
                <button @click="$refs.hierarchy.foldAllNode()">折叠全部节点</button>
            </div>
            <div style="margin-top: 10px">
                <input type="radio" id="h" value="h" v-model="mode" />
                <label for="h">水平模式</label>
                <input type="radio" id="v" value="v" v-model="mode" />
                <label for="v">垂直模式</label>
            </div>
            <div class="item" v-if="mode == 'h'">
                <div><input type="radio" id="h-lr" value="lr" v-model="layout" /> <label for="h-lr">左-右布局</label></div>
                <div><input type="radio" id="h-rl" value="rl" v-model="layout" /> <label for="h-rl">右-左布局</label></div>
                <div><input type="radio" id="h-bf" value="bf" v-model="layout" /> <label for="h-bf">蝴蝶布局</label></div>
            </div>
            <div class="item" v-if="mode == 'v'">
                <div><input type="radio" id="tb" value="tb" v-model="layout" /> <label for="tb">上-下布局</label></div>
                <div><input type="radio" id="bt" value="bt" v-model="layout" /> <label for="bt">下-上布局</label></div>
                <div><input type="radio" id="v-bf" value="bf" v-model="layout" /> <label for="v-bf">蝴蝶布局</label></div>
            </div>
        </div>
        <div class="document">
            <!-- <router-link to="/md-view" target="_blank">文档</router-link> -->
            <a href="https://github.com/luna-lee/moon-hierarchy" target="_blank">github地址</a>
        </div>

        <hierarchy
            ref="hierarchy"
            class="moon-hierarchy"
            :mode="mode"
            :treeData="treeData"
            :treeOptions="treeOptions"
            :layout="layout"
            :negativeIds="['qydak', 'root1', 'root2', 'root3']"
            :config="config"
            :width="width"
            :height="height"
            expendShape=".moon-hierarchy-plus"
            foldShape=".moon-hierarchy-plus"
            @draw-done="onDrawDone"
        >
            <div>
                <ul>
                    <li class="contentmenu-item" @click="onAdd">新增子节点</li>
                    <li class="contentmenu-item" @click="onRemove">删除节点</li>
                    <li class="contentmenu-item" @click="onUpdate">更新数据</li>
                </ul>
            </div>
        </hierarchy>
    </div>
</template>
<script>
//import hierarchy from 'moon-hierarchy/vue2';
import hierarchy from 'moon-hierarchy';
import 'moon-hierarchy/moon-hierarchy.css';
export default {
    inheritAttrs: false,
    name: '',
    props: {},
    components: {
        hierarchy
    },
    created() {
        this.setWidthHeight();
        window.addEventListener('resize', () => {
            // 窗口大小改变时执行的操作
            this.setWidthHeight();
        });
    },
    mounted() {
        let _this = this;
        fetch('https://www.fste.top/files/d3335980e04011ed91b4f7437d34c747/dataTree.js')
            .then((response) => {
                return response.text();
            })
            .then((data) => {
                _this.treeData = eval(data);
            });
    },
    data() {
        return {
            mode: 'v',
            layout: 'bf',
            treeData: [],
            currentNode: {},
            treeOptions: { id: 'code', pId: 'pcode' },
            width: 0,
            height: 0
        };
    },
    watch: {},
    computed: {
        config() {
            return {
                node: {
                    on: {
                        clickFetchChildren: (data, node, svg) => {
                            console.log(data, node, svg);
                            return new Promise((r) => {
                                setTimeout(() => {
                                    r([
                                        {
                                            id: '32323',
                                            name: '金融贷款余额test',
                                            code: '980eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '3233',
                                            name: '金融贷款余额test',
                                            code: '980444eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '323243333',
                                            name: '金融贷款余额test',
                                            code: '1980eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '323323',
                                            name: '金融贷款余额test',
                                            code: '2980eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '323223',
                                            name: '金融贷款余额test',
                                            code: '3980eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '323123',
                                            name: '金融贷款余额test',
                                            code: '480eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '323232',
                                            name: '金融贷款余额test',
                                            code: '94580eccec9a23237b49e488c10f8fa70f9c2d'
                                        },
                                        {
                                            id: '3333',
                                            name: 'lv-2',
                                            code: '94580eccec9a23237b49e488c10f8fa70f9c2d11',
                                            pcode: '94580eccec9a23237b49e488c10f8fa70f9c2d'
                                        }
                                    ]);
                                }, 2000);
                            });
                        },
                        click: (e, d, el, svg) => {
                            svg.selectAll('.active-node').classed('active-node', false);
                            el.classed('active-node', true);
                            this.$emit('node-click', d.data);
                            this.$refs.hierarchy.hiddenCustomView();
                        },
                        contextmenu: (e, d, node, svg) => {
                            e.preventDefault();
                            this.currentNode = d.data;
                            this.$refs.hierarchy.showCustomView(e, d);
                        }
                    },
                    exShaps: this.mode == 'h' ? this.exShaps() : [],
                    plus: {
                        artts: {},
                        show: this.mode != 'h'
                    }
                },
                customView: {
                    width: 120,
                    height: 110,
                    duration: 400
                },
                arrow: {
                    // show: false
                }
            };
        }
    },
    methods: {
        setWidthHeight() {
            this.width = (window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth) - 10;
            this.height = (window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight) - 30;
        },
        addNew() {
            this.$refs.hierarchy.addNode('root', [
                {
                    id: 'new' + new Date().getTime(),
                    name: '企业信息' + 'new' + new Date().getTime(),
                    code: 'new' + new Date().getTime(),
                    modelType: '',
                    domainId: '',
                    pcode: 'root'
                }
            ]);
        },
        onDrawDone({ svg, container }) {
            svg.on('click', () => {
                console.log('这是画布');
                this.$refs.hierarchy.hiddenCustomView();
            });
            this.svg = svg;
            this.container = container;
        },
        onAdd() {
            this.$refs.hierarchy.addNode(
                this.currentNode[this.treeOptions.id],
                [
                    {
                        id: 'new' + new Date().getTime(),
                        name: '企业信息' + 'new' + new Date().getTime(),
                        code: 'new' + new Date().getTime(),
                        modelType: '',
                        domainId: '',
                        pcode: this.currentNode[this.treeOptions.id]
                    }
                ],
                -1
            );
        },
        onRemove() {
            this.$refs.hierarchy.removeNodeById(this.currentNode[this.treeOptions.id]);
        },
        onUpdate() {
            this.$refs.hierarchy.updateNodeByData({ ...this.currentNode, name: 'hello', children: [] });
        },
        exShaps() {
            let plusCircleWidth = 15;
            function isNonEmptyArray(arr) {
                return arr && arr.length;
            }
            return [
                {
                    name: 'text',
                    attrs: {
                        fill: (d) => {
                            if (d.data.children?.length) return 'red';
                        },
                        'font-size': 19,
                        transform: (d) => {
                            return d.data._sign == 1
                                ? `translate(${d.data._nodeConfig.nodeWidth},${d.data._nodeConfig.nodeHeight / 2 + 5})`
                                : `translate(-20,${d.data._nodeConfig.nodeHeight / 2 + 5})`;
                        }
                    },
                    compose: {
                        text(d) {
                            if (typeof d.data._isexpend == 'boolean') {
                                return d.data._isexpend ? '🤩' : '🤓';
                            }
                            return d.data?.children?.length ? '😝' : '😃';
                        }
                    }
                },

                {
                    name: 'g',
                    on: {
                        click: (e) => {
                            this.$refs.hierarchy.hiddenCustomView();
                        }
                    },
                    attrs: {
                        class: 'moon-hierarchy-plus',
                        display: (d) => {
                            if (
                                (!isNonEmptyArray(d.data.children) && !isNonEmptyArray(d.data._children)) ||
                                d.data.track.length == 1
                            ) {
                                return 'none';
                            }
                        },
                        transform: (d) =>
                            `translate(${
                                d.data._sign == 1 ? d.data._nodeConfig.nodeWidth + 2 + plusCircleWidth / 2 : -plusCircleWidth
                            },${d.data._nodeConfig.nodeHeight / 2 + 1})`
                    },
                    children: [
                        {
                            name: 'circle',
                            attrs: {
                                class: 'moon-hierarchy-plus-circle',
                                r: plusCircleWidth / 2
                            }
                        },
                        {
                            name: 'line',
                            attrs: {
                                x1: -plusCircleWidth / 4,
                                y1: '0',
                                x2: plusCircleWidth / 4,
                                y2: '0'
                            }
                        },
                        /*  {
                            name: 'text',
                            attrs: {
                                display: (d) => {
                                    if (d.data?.children?.length) {
                                        return 'none';
                                    }
                                },
                                x: -4,
                                y: 5
                            },
                            compose: {
                                text(d) {
                                    return 22 + d.data?._children?.length;
                                }
                            }
                        }, */
                        {
                            name: 'line',
                            attrs: {
                                display: (d) => {
                                    if (d.data?.children?.length) {
                                        return 'none';
                                    }
                                },
                                x1: '0',
                                y1: -plusCircleWidth / 4,
                                x2: '0',
                                y2: plusCircleWidth / 4
                            }
                        },
                        {
                            name: 'circle',
                            attrs: {
                                r: plusCircleWidth / 2,
                                'fill-opacity': 0,
                                'stroke-width': '0.5'
                            }
                        }
                    ]
                }
            ];
        }
    }
};
</script>
<style lang="scss" scoped>
.moon-hierarchy {
    background: #edf0fd;
    ul {
        margin: 0;
        padding: 0;
        list-style: none;
    }
    li {
        display: list-item;
        text-align: -webkit-match-parent;
        unicode-bidi: isolate;
    }
    .contentmenu-item {
        font-size: 14px;
        position: relative;
        white-space: nowrap;
        overflow: hidden;
        color: #606266;
        height: 34px;
        line-height: 34px;
        box-sizing: border-box;
        cursor: pointer;
        &:hover {
            background-color: #f5f7fa;
        }
    }
    ::v-deep(.moon-hierarchy-node-root) {
        .moon-hierarchy-rect {
            fill: #003bc1;
        }
    }
    ::v-deep(.moon-hierarchy-link) {
        stroke: #1961f5;
        stroke-opacity: 1;
        stroke-width: 1.3;
    }
    ::v-deep(.moon-hierarchy-node) {
        &.moon-hierarchy-node-expend:not(.moon-hierarchy-node-root):not(.active-node) {
            .moon-hierarchy-text {
                fill: rgb(51, 51, 51);
            }
        }
        &.deep-1-node:not(.active-node) {
            .moon-hierarchy-rect {
                fill: #0044fe !important;
            }
            .moon-hierarchy-text {
                fill: #fff !important;
            }
        }
        &.active-node {
            &:not(.moon-hierarchy-node-root) {
                .moon-hierarchy-rect {
                    fill: #003bc1;
                }
                .moon-hierarchy-text {
                    fill: #fff;
                }
            }
        }
        .moon-hierarchy-plus {
            stroke: #1961f5;
            &:hover {
                circle {
                    fill: #5984f1;
                }
                line {
                    stroke: #fff;
                }
            }
        }
    }
    .contentmenu-item {
        font-size: 14px;
        padding: 0 20px;
        position: relative;
        white-space: nowrap;
        overflow: hidden;
        color: #606266;
        height: 34px;
        line-height: 34px;
        box-sizing: border-box;
        cursor: pointer;
        &:hover {
            background-color: #f5f7fa;
        }
    }
}
.button-group {
    display: flex;
    gap: 10px;
}
.pannel {
    left: 0;
    top: 0;
    padding: 10px;
    box-sizing: border-box;
    position: absolute;
    width: 100%;
    background-color: rgba(250, 250, 250, 0.5);
    label {
        cursor: pointer;
    }
    .item {
        margin-top: 10px;
        display: flex;
        flex-direction: column;
        align-items: flex-start;
        border-top: 1px solid;
        div {
            padding: 5px;
        }
    }
}
.document {
    position: absolute;
    right: 10px;
    display: flex;
    gap: 20px;
}
.contextmenu {
    background-color: antiquewhite;
    width: 100%;
    height: 100%;
    background-color: #fff;
    padding: 5px;
    border-radius: 8px;
    box-sizing: border-box;
    box-shadow: 0 6px 16px 0 rgba(0, 0, 0, 0.08), 0 3px 6px -4px rgba(0, 0, 0, 0.12), 0 9px 28px 8px rgba(0, 0, 0, 0.05);
}
</style>