diff --git a/nezha-fronted/src/components/chart/chart/chartTopology.vue b/nezha-fronted/src/components/chart/chart/chartTopology.vue index bef1a45d8..d4acf5a7a 100644 --- a/nezha-fronted/src/components/chart/chart/chartTopology.vue +++ b/nezha-fronted/src/components/chart/chart/chartTopology.vue @@ -1,15 +1,12 @@ @@ -18,9 +15,14 @@ import * as d3 from 'd3' import chartMixin from '@/components/chart/chartMixin' import chartFormat from '@/components/chart/chartFormat' import '@svgdotjs/svg.panzoom.js' +import alertLabelMixin from '@/components/common/mixin/alertLabelMixin' +import alertLabel from '@/components/common/alert/alertLabel' export default { name: 'chartHexagonD3', - mixins: [chartMixin, chartFormat], + components: { + alertLabel + }, + mixins: [chartMixin, chartFormat, alertLabelMixin], props: { parentId: { type: String, @@ -65,8 +67,8 @@ export default { dom.append(this.ForceGraph({ nodes: this.data.nodes, links: this.data.links }, { width: dom.offsetWidth, height: dom.offsetHeight, - nodeStrength: -500, - distanceNum: 200 + nodeStrength: -50, + distanceNum: 150 })) const svg = document.getElementById('svgHex' + this.chartId) svg.setAttribute('width', '100%') @@ -106,7 +108,7 @@ export default { const icon = { asset: '', endpoint: '', - dc: '', + datacenter: '', project: '', module: '' } @@ -138,6 +140,7 @@ export default { const simulation = d3.forceSimulation(nodes) .force('link', forceLink) .force('charge', forceNode) + .alpha(1) .on('tick', ticked) const svg = d3.create('svg') @@ -169,6 +172,8 @@ export default { .attr('id', function (d) { return 'node' + d.id }) .attr('style', 'cursor:pointer') .call(drag(simulation)) + .on('mouseenter', nodeMouseover)// no i18n + .on('mouseleave', nodeMouseout)// no i18n .on('click', showChildren) render() if (W) linkCopy.attr('stroke-width', ({ index: i }) => W[i]) @@ -224,14 +229,56 @@ export default { .on('end', dragended) } - function nodeMouseover (d) { - console.log(d, 'nodeMouseover') + function nodeMouseover (event, d) { + const paramsId = d.id.split('-')[1] + d.highlight = true + if (self.alertLabelId === paramsId) { + return + } + console.log(event, d, 'nodeMouseover') + self.alertLabelObj = { + position: { + x: event.x, + y: event.y + }, + loading: true, + id: '' + } + if (self.timer) { + clearTimeout(self.timer) + self.timer = null + } + self.timer = setTimeout(() => { + self.alertLabelId = paramsId + self.alertLabelObj.id = paramsId + self.alertLabelType = d.type + self.alertLabelShow = true + }, 200) } - function nodeMouseout (d) { - console.log(d, 'nodeMouseout') + function nodeMouseout (event, d) { + // console.log(d, 'nodeMouseout') + clearTimeout(self.timer) + self.timer = null + d.highlight = false + self.alertLabelShow = false + self.alertLabelId = '' + self.alertLabelObj = { + position: { + x: event.x, + y: event.y + }, + loading: false, + id: '' + } + self.alertLabelType = d.type } - function showChildren (event, d) { + function showChildren (event, d, a) { + console.log(event, d, a, 'showChildren') + console.log(event.path[0].className.baseVal) + if (event.path[0].className.baseVal !== 'topology-button') { + return + } if (d.hasChildren) { hideChildren(d) return @@ -241,56 +288,20 @@ export default { type: d.type, id: d.id } + params.id = params.id.split('-')[1] self.$get('/stat/rel', params).then(res => { - res = { - data: { - nodes: [ - { - id: 'asset-4', - type: 'asset', - category: 'asset', - name: 'ddd', - hasChildren: false, - alert: [{ - priority: 'P1', - num: 11 - }] - }, - { - id: 'asset-5', - type: 'asset', - category: 'asset', - name: 'ddd', - hasChildren: false, - alert: [{ - priority: 'P1', - num: 11 - }] - } - ], - links: [ - { - source: 'asset-2', - target: 'asset-4' - }, { - source: 'asset-2', - target: 'asset-5' - } - ] - } - } - const parentId = 'asset-2' || d.id + const parentId = d.id + res.data.nodes = res.data.nodes.filter(item => item.id !== parentId) // 处理连线 node 重复的问题 + res.data.nodes = res.data.nodes.filter(item => item.id !== d.parentId) + res.data.links = res.data.links.filter(item => item.source !== d.parentId) + res.data.links = res.data.links.filter(item => item.target !== d.parentId) res.data.nodes.forEach(item => { - if (item.id === parentId) { - item.hasChildren = true - } else { - item.hasChildren = false - } + item.hasChildren = false if (!item.parentId) { item.parentId = parentId } - item.radius = 20 // 需要计算 分三级 - item.color = '#fa9' // 需要计算 分三级 + item.radius = self.getRadius(item) // 需要计算 分三级 + item.color = self.getColor(item) // 需要计算 分三级 }) nodes.push(...res.data.nodes) simulation.stop() @@ -323,8 +334,9 @@ export default { }) } function hideChildren (d) { + console.log(d, 'hideChildren') + // 创建一个数组 隐藏所有相关联的 节点 连线 if (d.id) { - d.hasChildren = false simulation.stop() nodes = nodes.filter(child => child.parentId !== d.id) node = node @@ -347,6 +359,7 @@ export default { simulation.nodes(nodes) simulation.force('link').links(links) simulation.alphaTarget(1).restart().tick() + d.hasChildren = false } } function render () { @@ -396,17 +409,21 @@ export default { .attr('stroke-width', 1) .attr('rx', '4') .attr('ry', '4') + .attr('class', 'topology-button') const button = node.append('svg') .attr('width', '14px') .attr('height', '14px') .attr('x', function (d) { return nodeRadius * 1 + 2 }) .attr('y', function (d) { return d.radius * 1 + 9 }) .attr('style', 'cursor:pointer') - .attr('class', function (d) { return d.hasChildren ? 'toggle-nodes' : 'hide' }) + .attr('class', function (d) { return d.hasChildren ? 'toggle-nodes topology-button' : 'hide topology-button' }) .attr('viewBox', '0 0 512 512') button.append('polygon') .attr('points', '508.3,204.302 508.3,94.372 232.104,94.372 232.104,119.163 2.128,119.163 2.128,148.02 103.282,148.02 103.282,395.751 104.073,395.751 104.073,395.879 231.416,395.879 231.416,433.091 507.612,433.091 507.612,323.161 231.416,323.161 231.416,365.65 135.572,365.65 135.572,148.02 232.104,148.02 232.104,204.302 ') .attr('fill', function (d) { return d.hasChildren ? '#666666' : '#FA901C' }) + .attr('class', 'topology-button') + // .off('click', showChildren.bind(this, '2')) + // .on('click', showChildren.bind(this, '2')) node.attr('class', function (d) { let cl = 'node' if (d.hasChildren && (d.expanded == undefined || (d.expanded != undefined && d.expanded == false))) { @@ -421,25 +438,6 @@ export default { }, 1000) return Object.assign(svg.node(), { scales: { color } }) }, - update (nodes, links) { - const force = d3.forceSimulation() - force.nodes(nodes) - force.force('link').links(links)// - }, - hexagonOver (that, e) { // 移入六边形 - this.tooltip.title = that.alias - this.tooltip.value = that.showValue - this.tooltip.show = true - this.setPosition(e) - }, - hexagonMove (e) { // 六边形内移动 - if (this.tooltip.show) { - this.setPosition(e) - } - }, - hexagonOut (that) { - this.tooltip.show = false - }, setPosition (e) { const windowWidth = window.innerWidth// 窗口宽度 const windowHeight = window.innerHeight// 窗口高度 @@ -465,6 +463,35 @@ export default { showChildren () { this.upDate() }, + getRadius (item) { + let sum = 0 + item.alert && item.alert.forEach(alert => { + sum += alert.num + }) + if (sum === 0) { + return 15 + } else if (sum <= 10) { + return 25 + } else { + return 35 + } + }, + getColor (item) { + const arr = [] + let color = '' + item.alert && item.alert.forEach(alert => { + arr.push(alert.priority) + }) + this.severityDataWeight.forEach(severity => { + if (arr.indexOf(severity.name) !== -1 && !color && severity.num > 0) { + color = severity.color + } + }) + if (!arr.length || !color) { + color = '#23bf9a' + } + return color + }, upDate () { } diff --git a/nezha-fronted/src/components/chart/panelChart.vue b/nezha-fronted/src/components/chart/panelChart.vue index fb8117dce..aaf867304 100644 --- a/nezha-fronted/src/components/chart/panelChart.vue +++ b/nezha-fronted/src/components/chart/panelChart.vue @@ -365,102 +365,102 @@ export default { } this.$get('/stat/rel', params).then(res => { this.loading = false - console.log(res) - const parentId = 'asset-1' || this.chartInfo.id - res = { - msg: 'success', - code: 200, - data: { - nodes: [{ - id: 'asset-1', - type: 'asset', - category: 'asset', - name: 'dddeee', - hasChildren: false, - alert: [{ - priority: 'P2', - num: 7 - }, { - priority: 'P3', - num: 5 - }] - }, { - id: 'endpoint-1', - type: 'endpoint', - category: 'endpoint', - name: 'ddd', - hasChildren: true, - alert: [{ - priority: 'P3', - num: 1 - }] - }, - { - id: 'asset-2', - type: 'dc', - category: 'asset', - name: 'ddd', - hasChildren: false, - alert: [{ - priority: 'P1', - num: 10 - }] - }, { - id: 'endpoint-2', - type: 'project', - category: 'endpoint', - name: 'ddd', - hasChildren: true, - alert: [{ - priority: 'P1', - num: 30 - }] - }, - { - id: 'asset-3', - type: 'module', - category: 'asset', - name: 'ddd', - hasChildren: false, - alert: [{ - priority: 'P1', - num: 10 - }] - }, { - id: 'endpoint-3', - type: 'endpoint', - category: 'endpoint', - name: 'ddd', - hasChildren: true, - alert: [{ - priority: 'P3', - num: 10 - }] - } - ], - links: [{ - source: 'asset-1', - target: 'endpoint-1' - }, - { - source: 'asset-1', - target: 'endpoint-2' - }, - { - source: 'asset-1', - target: 'endpoint-3' - }, - { - source: 'asset-1', - target: 'asset-2' - }, - { - source: 'asset-1', - target: 'asset-3' - } - ] - } - } + console.log(this.chartInfo,this.chartInfo.id) + const parentId = this.chartInfo.linkType + '-' + this.chartInfo.id + // res = { + // msg: 'success', + // code: 200, + // data: { + // nodes: [{ + // id: 'asset-1', + // type: 'asset', + // category: 'asset', + // name: 'dddeee', + // hasChildren: false, + // alert: [{ + // priority: 'P2', + // num: 7 + // }, { + // priority: 'P3', + // num: 5 + // }] + // }, { + // id: 'endpoint-1', + // type: 'endpoint', + // category: 'endpoint', + // name: 'ddd', + // hasChildren: true, + // alert: [{ + // priority: 'P3', + // num: 1 + // }] + // }, + // { + // id: 'asset-2', + // type: 'dc', + // category: 'asset', + // name: 'ddd', + // hasChildren: false, + // alert: [{ + // priority: 'P1', + // num: 10 + // }] + // }, { + // id: 'endpoint-2', + // type: 'project', + // category: 'endpoint', + // name: 'ddd', + // hasChildren: true, + // alert: [{ + // priority: 'P1', + // num: 30 + // }] + // }, + // { + // id: 'asset-3', + // type: 'module', + // category: 'asset', + // name: 'ddd', + // hasChildren: false, + // alert: [{ + // priority: 'P1', + // num: 10 + // }] + // }, { + // id: 'endpoint-3', + // type: 'endpoint', + // category: 'endpoint', + // name: 'ddd', + // hasChildren: true, + // alert: [{ + // priority: 'P3', + // num: 10 + // }] + // } + // ], + // links: [{ + // source: 'asset-1', + // target: 'endpoint-1' + // }, + // { + // source: 'asset-1', + // target: 'endpoint-2' + // }, + // { + // source: 'asset-1', + // target: 'endpoint-3' + // }, + // { + // source: 'asset-1', + // target: 'asset-2' + // }, + // { + // source: 'asset-1', + // target: 'asset-3' + // } + // ] + // } + // } res.data.nodes.forEach(item => { item.parentId = parentId if (item.id === parentId) { @@ -607,7 +607,7 @@ export default { arr.push(alert.priority) }) this.severityDataWeight.forEach(severity => { - if (arr.indexOf(severity.name) !== -1 && !color) { + if (arr.indexOf(severity.name) !== -1 && !color && severity.num > 0) { color = severity.color } }) diff --git a/nezha-fronted/src/components/common/table/settings/dcTable.vue b/nezha-fronted/src/components/common/table/settings/dcTable.vue index 61f420f09..a534b628e 100644 --- a/nezha-fronted/src/components/common/table/settings/dcTable.vue +++ b/nezha-fronted/src/components/common/table/settings/dcTable.vue @@ -107,7 +107,7 @@ {{$t('overall.duplicate')}} {{$t('overall.delete')}} - {{$t('overall.topology')}} + {{$t('overall.topology')}}