openreplay/frontend/app/components/Charts/SankeyChart.tsx
2025-05-06 16:47:53 +02:00

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;