import { useCallback, useEffect, useRef, useState } from 'react'; import { Renderer } from '../../../renderer'; import './RoadmapRenderer.css'; import { renderResourceProgress, updateResourceProgress, type ResourceProgressType, renderTopicProgress, refreshProgressCounters, } from '../../lib/resource-progress'; import { pageProgressMessage } from '../../stores/page'; import { useToast } from '../../hooks/use-toast'; import type { RoadmapDocument } from './CreateRoadmap/CreateRoadmapModal'; import { EmptyRoadmap } from './EmptyRoadmap'; type RoadmapRendererProps = { roadmap: RoadmapDocument; }; type RoadmapNodeDetails = { nodeId: string; nodeType: string; targetGroup: SVGElement; }; export function getNodeDetails( svgElement: SVGElement ): RoadmapNodeDetails | null { const targetGroup = (svgElement?.closest('g') as SVGElement) || {}; const nodeId = targetGroup?.dataset?.nodeId; const nodeType = targetGroup?.dataset?.type; if (!nodeId || !nodeType) return null; return { nodeId, nodeType, targetGroup }; } export const allowedClickableNodeTypes = [ 'topic', 'subtopic', 'button', 'link-item', ]; export function RoadmapRenderer(props: RoadmapRendererProps) { const { roadmap } = props; const roadmapRef = useRef(null); const roadmapId = roadmap._id!; const toast = useToast(); const [hideRenderer, setHideRenderer] = useState(false); async function updateTopicStatus( topicId: string, newStatus: ResourceProgressType ) { pageProgressMessage.set('Updating progress'); updateResourceProgress( { resourceId: roadmapId, resourceType: 'roadmap', topicId, }, newStatus ) .then(() => { renderTopicProgress(topicId, newStatus); }) .catch((err) => { toast.error('Something went wrong, please try again.'); console.error(err); }) .finally(() => { pageProgressMessage.set(''); refreshProgressCounters(); }); return; } const handleSvgClick = useCallback((e: MouseEvent) => { const target = e.target as SVGElement; const { nodeId, nodeType, targetGroup } = getNodeDetails(target) || {}; if (!nodeId || !nodeType || !allowedClickableNodeTypes.includes(nodeType)) return; if (nodeType === 'button' || nodeType === 'link-item') { const link = targetGroup?.dataset?.link || ''; const isExternalLink = link.startsWith('http'); if (isExternalLink) { window.open(link, '_blank'); } else { window.location.href = link; } return; } const isCurrentStatusLearning = targetGroup?.classList.contains('learning'); const isCurrentStatusSkipped = targetGroup?.classList.contains('skipped'); if (e.shiftKey) { e.preventDefault(); updateTopicStatus( nodeId, isCurrentStatusLearning ? 'pending' : 'learning' ); return; } else if (e.altKey) { e.preventDefault(); updateTopicStatus(nodeId, isCurrentStatusSkipped ? 'pending' : 'skipped'); return; } window.dispatchEvent( new CustomEvent('roadmap.node.click', { detail: { topicId: nodeId, resourceId: roadmap?._id, resourceType: 'roadmap', isCustomResource: true, }, }) ); }, []); const handleSvgRightClick = useCallback((e: MouseEvent) => { e.preventDefault(); const target = e.target as SVGElement; const { nodeId, nodeType, targetGroup } = getNodeDetails(target) || {}; if (!nodeId || !nodeType || !allowedClickableNodeTypes.includes(nodeType)) return; if (nodeType === 'button' || nodeType === 'link-item') { return; } const isCurrentStatusDone = targetGroup?.classList.contains('done'); updateTopicStatus(nodeId, isCurrentStatusDone ? 'pending' : 'done'); }, []); useEffect(() => { if (!roadmapRef?.current) return; roadmapRef?.current?.addEventListener('click', handleSvgClick); roadmapRef?.current?.addEventListener('contextmenu', handleSvgRightClick); return () => { roadmapRef?.current?.removeEventListener('click', handleSvgClick); roadmapRef?.current?.removeEventListener( 'contextmenu', handleSvgRightClick ); }; }, []); return (
{ renderResourceProgress('roadmap', roadmapId).then(() => { if (roadmap?.nodes?.length === 0) { setHideRenderer(true); roadmapRef?.current?.classList.add('hidden'); } }); }} /> {hideRenderer && ( )}
); }