2023-08-30 14:21:51 +08:00
|
|
|
import { Graph, Extensions, extend } from '@antv/g6';
|
|
|
|
const ExtGraph = extend(Graph, {
|
|
|
|
nodes: {
|
|
|
|
'sphere-node': Extensions.SphereNode,
|
|
|
|
},
|
|
|
|
behaviors: {
|
|
|
|
'orbit-canvas-3d': Extensions.OrbitCanvas3D,
|
|
|
|
'zoom-canvas-3d': Extensions.ZoomCanvas3D,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
const container = document.getElementById('container');
|
|
|
|
const width = container.scrollWidth;
|
|
|
|
const height = container.scrollHeight || 500;
|
|
|
|
|
|
|
|
fetch('https://gw.alipayobjects.com/os/antvdemo/assets/data/relations.json')
|
|
|
|
.then((res) => res.json())
|
|
|
|
.then((data) => {
|
|
|
|
const degrees = {};
|
|
|
|
data.edges.forEach((edge) => {
|
|
|
|
const { source, target } = edge;
|
|
|
|
degrees[source] = degrees[source] || 0;
|
|
|
|
degrees[target] = degrees[target] || 0;
|
|
|
|
degrees[source]++;
|
|
|
|
degrees[target]++;
|
|
|
|
});
|
|
|
|
|
|
|
|
const graph = new ExtGraph({
|
|
|
|
container: 'container',
|
|
|
|
width,
|
|
|
|
height,
|
2023-08-31 18:16:13 +08:00
|
|
|
transforms: ['transform-v4-data'],
|
2023-08-30 14:21:51 +08:00
|
|
|
layout: {
|
|
|
|
type: 'force',
|
|
|
|
preventOverlap: true,
|
|
|
|
linkDistance: 100,
|
|
|
|
},
|
|
|
|
autoFit: 'view',
|
|
|
|
modes: {
|
|
|
|
default: ['zoom-canvas', 'drag-canvas', 'drag-node', 'brush-select', 'click-select', 'hover-activate'],
|
|
|
|
},
|
|
|
|
theme: {
|
|
|
|
type: 'spec',
|
|
|
|
specification: {
|
|
|
|
node: {
|
|
|
|
dataTypeField: 'cluster',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
edge: (innerModel) => {
|
|
|
|
return {
|
|
|
|
...innerModel,
|
|
|
|
data: {
|
|
|
|
...innerModel.data,
|
|
|
|
keyShape: {
|
|
|
|
lineWidth: 0.5,
|
|
|
|
},
|
|
|
|
haloShape: {},
|
|
|
|
animates: {
|
|
|
|
buildIn: [
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
duration: 500,
|
|
|
|
delay: 1000,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
buildOut: [
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 200,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
update: [
|
|
|
|
{
|
|
|
|
fields: ['lineWidth'],
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
shapeId: 'haloShape',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
},
|
|
|
|
node: (innerModel) => {
|
|
|
|
const degree = degrees[innerModel.id] || 0;
|
|
|
|
let labelLod = 4;
|
|
|
|
if (degree > 20) labelLod = -1;
|
|
|
|
else if (degree > 15) labelLod = 0;
|
|
|
|
else if (degree > 10) labelLod = 1;
|
|
|
|
else if (degree > 6) labelLod = 2;
|
|
|
|
else if (degree > 3) labelLod = 3;
|
|
|
|
|
|
|
|
const badgeShapes = {};
|
|
|
|
|
|
|
|
if (degree > 20) {
|
|
|
|
badgeShapes[0] = {
|
|
|
|
text: '核心人员',
|
|
|
|
position: 'rightBottom',
|
|
|
|
lod: labelLod - 2,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (degree > 15) {
|
|
|
|
badgeShapes[1] = {
|
|
|
|
text: 'A',
|
|
|
|
position: 'right',
|
|
|
|
lod: labelLod - 1,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (degree > 10) {
|
|
|
|
badgeShapes[2] = {
|
|
|
|
text: 'B',
|
|
|
|
position: 'rightTop',
|
|
|
|
lod: labelLod - 1,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...innerModel,
|
|
|
|
data: {
|
|
|
|
...innerModel.data,
|
|
|
|
lodStrategy: {
|
|
|
|
levels: [
|
|
|
|
{ zoomRange: [0, 0.8] }, // -2
|
|
|
|
{ zoomRange: [0.8, 0.9] }, // -1
|
|
|
|
{ zoomRange: [0.9, 1], primary: true }, // 0
|
|
|
|
{ zoomRange: [1, 1.1] }, // 1
|
|
|
|
{ zoomRange: [1.1, 0.2] }, // 2
|
|
|
|
{ zoomRange: [1.2, 1.3] }, // 3
|
|
|
|
{ zoomRange: [1.3, 1.4] }, // 4
|
|
|
|
{ zoomRange: [1.4, 1.5] }, // 5
|
|
|
|
{ zoomRange: [1.5, Infinity] }, // 6
|
|
|
|
],
|
|
|
|
animateCfg: {
|
|
|
|
duration: 500,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
animates: {
|
|
|
|
buildIn: [
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 1000,
|
|
|
|
delay: 500 + Math.random() * 500,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
buildOut: [
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 200,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
hide: [
|
|
|
|
{
|
|
|
|
fields: ['size'],
|
|
|
|
duration: 200,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 200,
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 200,
|
|
|
|
shapeId: 'labelShape',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
show: [
|
|
|
|
{
|
|
|
|
fields: ['size'],
|
|
|
|
duration: 200,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['opacity'],
|
|
|
|
duration: 200,
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
order: 0,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
update: [
|
|
|
|
{
|
|
|
|
fields: ['fill', 'r'],
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['lineWidth'],
|
|
|
|
shapeId: 'keyShape',
|
|
|
|
duration: 100,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['fontSize'],
|
|
|
|
shapeId: 'iconShape',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: ['opacity'], // 'r' error, 'lineWidth' has no effect
|
|
|
|
shapeId: 'haloShape',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
haloShape: {},
|
|
|
|
// animate in shapes, unrelated to each other, excuted parallely
|
|
|
|
keyShape: {
|
|
|
|
r: innerModel.data.size ? innerModel.data.size / 2 : 15,
|
|
|
|
},
|
|
|
|
iconShape: {
|
|
|
|
img: 'https://gw.alipayobjects.com/zos/basement_prod/012bcf4f-423b-4922-8c24-32a89f8c41ce.svg',
|
|
|
|
fill: '#fff',
|
|
|
|
lod: labelLod - 1,
|
|
|
|
fontSize: innerModel.data.size ? innerModel.data.size / 3 + 5 : 13,
|
|
|
|
},
|
|
|
|
labelShape: {
|
|
|
|
text: innerModel.id,
|
|
|
|
opacity: 0.8,
|
|
|
|
maxWidth: '150%',
|
|
|
|
lod: labelLod,
|
|
|
|
},
|
|
|
|
labelBackgroundShape: {
|
|
|
|
lod: labelLod,
|
|
|
|
},
|
|
|
|
badgeShapes,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
},
|
|
|
|
data,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
const btnContainer = document.createElement('div');
|
|
|
|
btnContainer.style.position = 'absolute';
|
|
|
|
container.appendChild(btnContainer);
|
|
|
|
btnContainer.style.zIndex = 10;
|
|
|
|
const tip = document.createElement('span');
|
|
|
|
tip.innerHTML = '👉 Zoom Canvas to See Level of Details:';
|
|
|
|
btnContainer.appendChild(tip);
|
|
|
|
|
|
|
|
if (typeof window !== 'undefined')
|
|
|
|
window.onresize = () => {
|
|
|
|
if (!graph || graph.destroyed) return;
|
|
|
|
if (!container || !container.scrollWidth || !container.scrollHeight) return;
|
|
|
|
graph.setSize([container.scrollWidth, container.scrollHeight - 160]);
|
|
|
|
};
|