508 lines
17 KiB
TypeScript
508 lines
17 KiB
TypeScript
import React from 'react';
|
|
import { SankeyChart } from 'echarts/charts';
|
|
import { NoContent } from 'App/components/ui';
|
|
import { InfoCircleOutlined } from '@ant-design/icons';
|
|
import { sankeyTooltip, getEventPriority, getNodeName } from './sankeyUtils';
|
|
import { echarts, defaultOptions } from './init';
|
|
import { useTranslation } from 'react-i18next';
|
|
|
|
echarts.use([SankeyChart]);
|
|
|
|
interface SankeyNode {
|
|
name: string | null;
|
|
eventType?: string;
|
|
depth?: number;
|
|
id?: string | number;
|
|
}
|
|
|
|
interface SankeyLink {
|
|
source: number | string;
|
|
target: number | string;
|
|
value: number;
|
|
sessionsCount: number;
|
|
eventType?: string;
|
|
}
|
|
|
|
interface Data {
|
|
nodes: SankeyNode[];
|
|
links: SankeyLink[];
|
|
}
|
|
|
|
interface Props {
|
|
data: Data;
|
|
height?: number;
|
|
onChartClick?: (filters: any[]) => void;
|
|
isUngrouped?: boolean;
|
|
inGrid?: boolean;
|
|
startPoint: 'end' | 'start'
|
|
}
|
|
|
|
const EChartsSankey: React.FC<Props> = (props) => {
|
|
const { t } = useTranslation();
|
|
const { data, height = 240, onChartClick, isUngrouped, startPoint } = props;
|
|
const chartRef = React.useRef<HTMLDivElement>(null);
|
|
|
|
const [finalNodeCount, setFinalNodeCount] = React.useState(data.nodes.length);
|
|
|
|
React.useEffect(() => {
|
|
if (!chartRef.current || data.nodes.length === 0 || data.links.length === 0) return;
|
|
|
|
const finalNodes = data.nodes;
|
|
const finalLinks = data.links;
|
|
|
|
const chart = echarts.init(chartRef.current);
|
|
|
|
const maxDepth = 4;
|
|
const filteredNodes = finalNodes.filter((n) => (n.depth ?? 0) <= maxDepth);
|
|
const filteredLinks = finalLinks.filter((l) => {
|
|
const sourceNode = finalNodes.find((n) => n.id === l.source);
|
|
const targetNode = finalNodes.find((n) => n.id === l.target);
|
|
return (
|
|
(sourceNode?.depth ?? 0) <= maxDepth &&
|
|
(targetNode?.depth ?? 0) <= maxDepth
|
|
);
|
|
});
|
|
|
|
setFinalNodeCount(filteredNodes.length);
|
|
const nodeValues: Record<string, number> = {};
|
|
const echartNodes = filteredNodes
|
|
.map((n, i) => {
|
|
let computedName = getNodeName(n.eventType || 'Minor Paths', n.name);
|
|
if (computedName === 'Other') {
|
|
computedName = 'Others';
|
|
}
|
|
if (n.id) {
|
|
nodeValues[n.id] = 0;
|
|
} else {
|
|
nodeValues[i] = 0;
|
|
}
|
|
const itemColor =
|
|
computedName === 'Others'
|
|
? 'rgba(34,44,154,.9)'
|
|
: n.eventType === 'DROP'
|
|
? '#B5B7C8'
|
|
: '#394eff';
|
|
|
|
return {
|
|
name: computedName,
|
|
depth: n.depth,
|
|
type: n.eventType,
|
|
id: n.id,
|
|
draggable: false,
|
|
itemStyle: { color: itemColor },
|
|
};
|
|
})
|
|
.sort((a, b) => {
|
|
if (a.depth === b.depth) {
|
|
return (
|
|
getEventPriority(a.type || '') - getEventPriority(b.type || '')
|
|
);
|
|
}
|
|
return (a.depth as number) - (b.depth as number);
|
|
});
|
|
|
|
const echartLinks = filteredLinks.map((l) => ({
|
|
source: echartNodes.findIndex((n) => n.id === l.source),
|
|
target: echartNodes.findIndex((n) => n.id === l.target),
|
|
value: l.sessionsCount,
|
|
percentage: l.value,
|
|
lineStyle: { opacity: 0.1 },
|
|
}));
|
|
|
|
if (echartNodes.length === 0) return;
|
|
|
|
const startDepth = startPoint === 'end' ? Math.max(...echartNodes.map(n => n.depth ?? 0)) : 0;
|
|
const mainNodeLinks = echartNodes.filter(n => n.depth === startDepth).map(n => echartNodes.findIndex(node => node.id === n.id))
|
|
const startNodeValue = echartLinks
|
|
.filter((link) => startPoint === 'start'
|
|
? mainNodeLinks.includes(link.source)
|
|
: mainNodeLinks.includes(link.target)
|
|
)
|
|
.reduce((sum, link) => sum + link.value, 0);
|
|
|
|
Object.keys(nodeValues).forEach((nodeId) => {
|
|
const intId = parseInt(nodeId as string);
|
|
const outgoingValues = echartLinks
|
|
.filter((l) => l.source === intId)
|
|
.reduce((p, c) => p + c.value, 0);
|
|
const incomingValues = echartLinks
|
|
.filter((l) => l.target === intId)
|
|
.reduce((p, c) => p + c.value, 0);
|
|
nodeValues[nodeId] = Math.max(outgoingValues, incomingValues);
|
|
});
|
|
|
|
const option = {
|
|
...defaultOptions,
|
|
tooltip: {
|
|
trigger: 'item',
|
|
},
|
|
toolbox: {
|
|
feature: {
|
|
saveAsImage: { show: false },
|
|
},
|
|
},
|
|
series: [
|
|
{
|
|
animation: false,
|
|
layoutIterations: 0,
|
|
type: 'sankey',
|
|
data: echartNodes,
|
|
links: echartLinks,
|
|
emphasis: {
|
|
focus: 'none',
|
|
lineStyle: {
|
|
opacity: 0.5,
|
|
},
|
|
},
|
|
top: 40,
|
|
label: {
|
|
show: true,
|
|
position: 'top',
|
|
textShadowColor: 'transparent',
|
|
textBorderColor: 'transparent',
|
|
align: 'left',
|
|
overflow: 'truncate',
|
|
maxWidth: 30,
|
|
distance: 3,
|
|
offset: [-20, 0],
|
|
formatter(params: any) {
|
|
const nodeVal = params.value;
|
|
const percentage = startNodeValue
|
|
? `${((nodeVal / startNodeValue) * 100).toFixed(1)}%`
|
|
: '0%';
|
|
|
|
const maxLen = 20;
|
|
const safeName =
|
|
params.name.length > maxLen
|
|
? `${params.name.slice(
|
|
0,
|
|
maxLen / 2 - 2,
|
|
)}...${params.name.slice(-(maxLen / 2 - 2))}`
|
|
: params.name;
|
|
const nodeType = params.data.type;
|
|
|
|
const icon = getIcon(nodeType);
|
|
return (
|
|
`${icon}{header| ${safeName}}\n` +
|
|
`{body|}{percentage|${percentage}} {sessions|${nodeVal}}`
|
|
);
|
|
},
|
|
rich: {
|
|
container: {
|
|
height: 20,
|
|
width: 14,
|
|
},
|
|
header: {
|
|
fontWeight: '600',
|
|
fontSize: 12,
|
|
color: '#333',
|
|
overflow: 'truncate',
|
|
paddingBottom: '.5rem',
|
|
paddingLeft: '14px',
|
|
position: 'relative',
|
|
},
|
|
body: {
|
|
fontSize: 12,
|
|
color: '#000',
|
|
},
|
|
percentage: {
|
|
fontSize: 12,
|
|
color: '#454545',
|
|
},
|
|
sessions: {
|
|
fontSize: 12,
|
|
fontFamily: "mono, 'monospace', sans-serif",
|
|
color: '#999999',
|
|
},
|
|
clickIcon: {
|
|
backgroundColor: {
|
|
image:
|
|
'',
|
|
},
|
|
height: 20,
|
|
width: 14,
|
|
},
|
|
dropEventIcon: {
|
|
backgroundColor: {
|
|
image:
|
|
'',
|
|
},
|
|
height: 20,
|
|
width: 14,
|
|
},
|
|
groupIcon: {
|
|
backgroundColor: {
|
|
image:
|
|
'',
|
|
},
|
|
height: 20,
|
|
width: 14,
|
|
},
|
|
},
|
|
},
|
|
tooltip: {
|
|
formatter: sankeyTooltip(echartNodes, nodeValues),
|
|
},
|
|
nodeAlign: 'left',
|
|
nodeWidth: 40,
|
|
nodeGap: 40,
|
|
lineStyle: {
|
|
color: 'source',
|
|
curveness: 0.6,
|
|
opacity: 0.1,
|
|
},
|
|
itemStyle: {
|
|
color: '#394eff',
|
|
borderRadius: 7,
|
|
},
|
|
},
|
|
],
|
|
};
|
|
|
|
chart.setOption(option);
|
|
|
|
function getUpstreamNodes(nodeIdx: number, visited = new Set<number>()) {
|
|
if (visited.has(nodeIdx)) return;
|
|
visited.add(nodeIdx);
|
|
echartLinks.forEach((link) => {
|
|
if (link.target === nodeIdx && !visited.has(link.source)) {
|
|
getUpstreamNodes(link.source, visited);
|
|
}
|
|
});
|
|
return visited;
|
|
}
|
|
|
|
function getDownstreamNodes(nodeIdx: number, visited = new Set<number>()) {
|
|
if (visited.has(nodeIdx)) return;
|
|
visited.add(nodeIdx);
|
|
echartLinks.forEach((link) => {
|
|
if (link.source === nodeIdx && !visited.has(link.target)) {
|
|
getDownstreamNodes(link.target, visited);
|
|
}
|
|
});
|
|
return visited;
|
|
}
|
|
|
|
function getConnectedChain(nodeIdx: number): Set<number> {
|
|
const upstream = getUpstreamNodes(nodeIdx) || new Set<number>();
|
|
const downstream = getDownstreamNodes(nodeIdx) || new Set<number>();
|
|
return new Set<number>([...upstream, ...downstream]);
|
|
}
|
|
|
|
const originalNodes = [...echartNodes];
|
|
const originalLinks = [...echartLinks];
|
|
|
|
chart.on('mouseover', (params: any) => {
|
|
if (params.dataType === 'node') {
|
|
const hoveredIndex = params.dataIndex;
|
|
const connectedChain = getConnectedChain(hoveredIndex);
|
|
|
|
const updatedNodes = echartNodes.map((node, idx) => {
|
|
const baseOpacity = connectedChain.has(idx) ? 1 : 0.35;
|
|
const extraStyle =
|
|
idx === hoveredIndex
|
|
? { borderColor: '#000', borderWidth: 1, borderType: 'dotted' }
|
|
: {};
|
|
return {
|
|
...node,
|
|
itemStyle: {
|
|
...node.itemStyle,
|
|
opacity: baseOpacity,
|
|
...extraStyle,
|
|
},
|
|
};
|
|
});
|
|
|
|
const updatedLinks = echartLinks.map((link) => ({
|
|
...link,
|
|
lineStyle: {
|
|
...link.lineStyle,
|
|
opacity:
|
|
connectedChain.has(link.source) && connectedChain.has(link.target)
|
|
? 0.5
|
|
: 0.1,
|
|
},
|
|
}));
|
|
|
|
chart.setOption({
|
|
series: [
|
|
{
|
|
data: updatedNodes,
|
|
links: updatedLinks,
|
|
},
|
|
],
|
|
});
|
|
}
|
|
});
|
|
|
|
chart.on('mouseout', (params: any) => {
|
|
if (params.dataType === 'node') {
|
|
chart.setOption({
|
|
series: [
|
|
{
|
|
data: originalNodes,
|
|
links: originalLinks,
|
|
},
|
|
],
|
|
});
|
|
}
|
|
});
|
|
|
|
chart.on('click', (params: any) => {
|
|
if (!onChartClick) return;
|
|
const unsupported = ['other', 'drop'];
|
|
|
|
if (params.dataType === 'node') {
|
|
const node = params.data;
|
|
const filters = [];
|
|
if (node && node.type) {
|
|
const type = node.type.toLowerCase();
|
|
if (unsupported.includes(type)) {
|
|
return;
|
|
}
|
|
filters.push({
|
|
operator: 'is',
|
|
type,
|
|
value: [node.name],
|
|
isEvent: true,
|
|
});
|
|
}
|
|
onChartClick?.(filters);
|
|
} else if (params.dataType === 'edge') {
|
|
const linkIndex = params.dataIndex;
|
|
const link = filteredLinks[linkIndex];
|
|
|
|
const firstNode = data.nodes.find((n) => n.id === link.source);
|
|
const lastNode = data.nodes.find((n) => n.id === link.target);
|
|
const firstNodeType = firstNode?.eventType?.toLowerCase() ?? 'location';
|
|
const lastNodeType = lastNode?.eventType?.toLowerCase() ?? 'location';
|
|
if (
|
|
unsupported.includes(firstNodeType) ||
|
|
unsupported.includes(lastNodeType)
|
|
) {
|
|
return;
|
|
}
|
|
const filters = [];
|
|
if (firstNode) {
|
|
filters.push({
|
|
operator: 'is',
|
|
type: firstNodeType,
|
|
value: [firstNode.name],
|
|
isEvent: true,
|
|
});
|
|
}
|
|
|
|
if (lastNode) {
|
|
filters.push({
|
|
operator: 'is',
|
|
type: lastNodeType,
|
|
value: [lastNode.name],
|
|
isEvent: true,
|
|
});
|
|
}
|
|
|
|
onChartClick?.(filters);
|
|
}
|
|
});
|
|
|
|
const ro = new ResizeObserver(() => chart.resize());
|
|
ro.observe(chartRef.current);
|
|
|
|
return () => {
|
|
chart.dispose();
|
|
ro.disconnect();
|
|
};
|
|
}, [data, height, onChartClick]);
|
|
|
|
if (data.nodes.length === 0 || data.links.length === 0) {
|
|
return (
|
|
<NoContent
|
|
style={{ minHeight: height }}
|
|
title={
|
|
<div className="flex items-center relative">
|
|
<InfoCircleOutlined className="hidden md:inline-block mr-1" />
|
|
Set a start or end point to visualize the journey. If set, try
|
|
adjusting filters.
|
|
</div>
|
|
}
|
|
show={true}
|
|
/>
|
|
);
|
|
}
|
|
let containerStyle: React.CSSProperties;
|
|
if (isUngrouped) {
|
|
const dynamicMinHeight = finalNodeCount * 15;
|
|
containerStyle = {
|
|
width: '100%',
|
|
minHeight: Math.max(550, dynamicMinHeight),
|
|
height: '100%',
|
|
overflowY: 'auto',
|
|
};
|
|
} else {
|
|
containerStyle = {
|
|
width: '100%',
|
|
height,
|
|
};
|
|
}
|
|
|
|
return (
|
|
<div
|
|
style={{
|
|
maxHeight: 620,
|
|
overflow: 'auto',
|
|
maxWidth: 1240,
|
|
minHeight: 240,
|
|
}}
|
|
>
|
|
<div ref={chartRef} style={containerStyle} className="min-w-[600px]" />
|
|
</div>
|
|
);
|
|
};
|
|
|
|
function getIcon(type: string) {
|
|
if (type === 'LOCATION') {
|
|
return '{locationIcon|}';
|
|
}
|
|
if (type === 'INPUT') {
|
|
return '{inputIcon|}';
|
|
}
|
|
if (type === 'CUSTOM_EVENT') {
|
|
return '{customEventIcon|}';
|
|
}
|
|
if (type === 'CLICK') {
|
|
return '{clickIcon|}';
|
|
}
|
|
if (type === 'DROP') {
|
|
return '{dropEventIcon|}';
|
|
}
|
|
if (type === 'OTHER') {
|
|
return '{groupIcon|}';
|
|
}
|
|
return '';
|
|
}
|
|
|
|
export default EChartsSankey;
|