diff --git a/nezha-fronted/src/components/chart/chart/chartTopology.vue b/nezha-fronted/src/components/chart/chart/chartTopology.vue index 9b789509b..b8ca85e3e 100644 --- a/nezha-fronted/src/components/chart/chart/chartTopology.vue +++ b/nezha-fronted/src/components/chart/chart/chartTopology.vue @@ -54,8 +54,11 @@ export default { firstInit: true, severityData: this.$store.getters.severityData, severityDataWeight: this.$store.getters.severityDataWeight, - distanceArr: [100, 150, 200, 250, 300, 350], - isDrag: false + distanceArr: [70, 100, 130, 170, 190, 210], + isDrag: false, + highlightId: '', + highlightAll: [], + radiusArr: [14, 20, 27] } }, methods: { @@ -70,7 +73,7 @@ export default { dom.append(this.ForceGraph({ nodes: this.data.nodes, links: this.data.links }, { width: dom.offsetWidth, height: dom.offsetHeight, - nodeStrength: -500, + // nodeStrength: 0, distanceNum: 150, distanceMin: -100 })) @@ -119,7 +122,7 @@ export default { module: '' } // Compute values. - const types = ['licensing', 'suit', 'resolved'] + let circleAll = '' const N = d3.map(nodes, nodeId).map(intern) const LS = d3.map(links, linkSource).map(intern) const LT = d3.map(links, linkTarget).map(intern) @@ -175,24 +178,22 @@ export default { .on('click', reset) const svg = svgBox.append('g') // 箭头 - - const defs = svg.append('defs') - - const arrowMarker = defs.append('marker') - .attr('id', 'arrow') - .attr('markerUnits', 'strokeWidth') - .attr('markerWidth', '60') - .attr('markerHeight', '60') - .attr('viewBox', '0 0 60 60') - .attr('refX', '21') - .attr('refY', '4') - .attr('orient', 'auto') - const arrowPath = 'M2,2 L8,4 L2,6 L2,2' - - arrowMarker.append('path') - .attr('d', arrowPath) - .attr('fill', '#000') + const defs = svg.append('defs') + self.radiusArr.forEach(radiusItem => { + defs.append('marker') + .attr('id', `arrow${radiusItem}`) + .attr('markerUnits', 'strokeWidth') + .attr('markerWidth', '60') + .attr('markerHeight', '60') + .attr('viewBox', '0 0 60 60') + .attr('refX', radiusItem / 2 + 8) + .attr('refY', '4') + .attr('orient', 'auto') + .append('path') + .attr('d', arrowPath) + .attr('fill', '#000') + }) const link = svg.append('g') .attr('stroke', typeof linkStroke !== 'function' ? linkStroke : null) .attr('stroke-opacity', linkStrokeOpacity) @@ -211,9 +212,13 @@ export default { .selectAll('.node') let node = nodeAll.data(nodes) .enter().append('g') - .attr('class', 'node') + .attr('class', 'node animation') .attr('id', function (d) { return 'node' + d.id }) - .attr('style', 'cursor:pointer') + .attr('style', function (d) { + let str = 'opacity:' + str += 1 + return str + ';cursor:pointer' + }) .call(drag(simulation)) .on('mouseleave', nodeMouseout) .on('click', showChildren) @@ -232,7 +237,9 @@ export default { .attr('y1', d => d.source.y) .attr('x2', d => d.target.x) .attr('y2', d => d.target.y) - .attr('marker-end', 'url(#arrow)') + .attr('marker-end', function (d) { + return `url(#arrow${d.target.radius})` + }) node .attr('transform', function (d) { if (d.id === self.parentId && self.firstInit) { @@ -274,10 +281,11 @@ export default { clearTimeout(self.timer2) self.timer2 = null } - self.timer2 = setTimeout(() => { - if (!event.active) simulation.alphaTarget(0) - simulation.stop() - }, 5000) + simulation.alphaTarget(0) + // self.timer2 = setTimeout(() => { + // if (!event.active) simulation.alphaTarget(0) + // simulation.stop() + // }, 5000) } return d3.drag() @@ -286,13 +294,61 @@ export default { .on('end', dragended) } + function highlightNode (id) { + self.highlightAll.push(id) + links.forEach(item => { + if (item.source.id === id) { + self.highlightAll.push(item.target.id) + } + if (item.target.id === id) { + self.highlightAll.push(item.source.id) + } + }) + node.transition().attr('style', function (d) { + let str = 'opacity:' + if (self.highlightAll.indexOf(d.id) !== -1) { + str += 1 + } else { + str += 0.05 + } + if (d.id === id) { + + } + return str + ';cursor:pointer' + }).duration(500) + linkCopy.transition().attr('style', function (d) { + let str = 'opacity:' + if (d.target.id === id || d.source.id === id) { + str += 1 + } else { + str += 0.05 + } + return str + '' + }).duration(500) + } + + function clearHighlight () { + self.highlightAll = [] + self.highlightId = '' + node.transition().attr('style', function (d) { + let str = 'opacity:' + str += 1 + return str + ';cursor:pointer' + }).duration(500) + linkCopy.transition().attr('style', function (d) { + let str = 'opacity:' + str += 1 + return str + '' + }).duration(500) + } + function nodeMouseover (event, d) { if (self.isDrag) { self.alertLabelShow = false return } const paramsId = d.id.split('-')[1] - d.highlight = true + highlightNode(d.id) if (self.alertLabelId === paramsId) { return } @@ -320,8 +376,8 @@ export default { function nodeMouseout (event, d) { // console.log(d, 'nodeMouseout') clearTimeout(self.timer) + clearHighlight() self.timer = null - d.highlight = false self.alertLabelShow = false self.alertLabelId = '' self.alertLabelObj = { @@ -335,6 +391,7 @@ export default { } self.alertLabelType = d.type === 'datacenter' ? 'dc' : d.type } + function showChildren (event, d, a) { window.event ? window.event.cancelBubble = true : event.stopPropagation() // console.log(event, d, a, 'showChildren') @@ -400,6 +457,7 @@ export default { simulation.restart().tick() }) } + function hideChildren (d) { // 创建一个数组 隐藏所有相关联的 节点 连线 if (d.id) { @@ -442,8 +500,9 @@ export default { simulation.restart().tick() } } + function render () { - node.append('circle') + circleAll = node.append('circle') .attr('r', function (d) { return d.radius }) .attr('fill', function (d) { return d.color }) .on('mouseenter', nodeMouseover) @@ -522,8 +581,16 @@ export default { } self.timer2 = setTimeout(() => { simulation.stop() - }, 5000) + node.attr('fx', function (d) { + d.fx = d.x + return d.fx + }).attr('fy', function (d) { + d.fy = d.y + return d.fy + }) + }, 2000) } + function reset () { // svg.transition().duration(750).call( // zoom.transform, @@ -531,11 +598,13 @@ export default { // d3.zoomTransform(svg.node()).invert([width / 2, height / 2]) // ) } + function zoomed (event) { const { transform } = event svg.attr('transform', transform) svg.attr('stroke-width', 1 / transform.k) } + setTimeout(() => { self.firstInit = false }, 1000) @@ -548,11 +617,11 @@ export default { sum += alert.num }) if (sum === 0) { - return 10 + return this.radiusArr[0] } else if (sum <= 10) { - return 15 + return this.radiusArr[1] } else { - return 25 + return this.radiusArr[2] } }, getColor (item) { diff --git a/nezha-fronted/src/components/chart/panelChart.vue b/nezha-fronted/src/components/chart/panelChart.vue index 57da0532f..587e5a342 100644 --- a/nezha-fronted/src/components/chart/panelChart.vue +++ b/nezha-fronted/src/components/chart/panelChart.vue @@ -367,6 +367,7 @@ export default { const parentId = this.chartInfo.linkType + '-' + this.chartInfo.id res.data.nodes.forEach(item => { item.parentId = [parentId] + item.highlight = true if (item.id === parentId) { item.hasChildren = true item.parentId = []