computer-scienceangular-roadmapbackend-roadmapblockchain-roadmapdba-roadmapdeveloper-roadmapdevops-roadmapfrontend-roadmapgo-roadmaphactoberfestjava-roadmapjavascript-roadmapnodejs-roadmappython-roadmapqa-roadmapreact-roadmaproadmapstudy-planvue-roadmapweb3-roadmap
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
242 lines
6.5 KiB
242 lines
6.5 KiB
import { useEffect, useRef, useState } from 'react'; |
|
import { wireframeJSONToSVG } from 'roadmap-renderer'; |
|
import '../FrameRenderer/FrameRenderer.css'; |
|
import { useOutsideClick } from '../../hooks/use-outside-click'; |
|
import { useKeydown } from '../../hooks/use-keydown'; |
|
import { httpGet } from '../../lib/http'; |
|
import type { ResourceType } from '../../lib/resource-progress'; |
|
import { topicSelectorAll } from '../../lib/resource-progress'; |
|
import { deleteUrlParam, getUrlParams } from '../../lib/browser'; |
|
import { useAuth } from '../../hooks/use-auth'; |
|
import { ModalLoader } from './ModalLoader.tsx'; |
|
import { UserProgressModalHeader } from './UserProgressModalHeader'; |
|
import { X } from 'lucide-react'; |
|
import type { PageType } from '../CommandMenu/CommandMenu.tsx'; |
|
import type { AllowedRoadmapRenderer } from '../../lib/roadmap.ts'; |
|
import { renderFlowJSON } from '../../../editor/renderer/renderer.ts'; |
|
|
|
export type ProgressMapProps = { |
|
userId?: string; |
|
resourceId: string; |
|
resourceType: ResourceType; |
|
onClose?: () => void; |
|
isCustomResource?: boolean; |
|
renderer?: AllowedRoadmapRenderer; |
|
}; |
|
|
|
export type UserProgressResponse = { |
|
user: { |
|
_id: string; |
|
name: string; |
|
}; |
|
progress: { |
|
total: number; |
|
done: string[]; |
|
learning: string[]; |
|
skipped: string[]; |
|
}; |
|
}; |
|
|
|
export function UserProgressModal(props: ProgressMapProps) { |
|
const { |
|
resourceId, |
|
resourceType, |
|
userId: propUserId, |
|
onClose: onModalClose, |
|
renderer = 'balsamiq', |
|
} = props; |
|
|
|
const { s: userId = propUserId } = getUrlParams(); |
|
if (!userId) { |
|
return null; |
|
} |
|
|
|
const resourceSvgEl = useRef<HTMLDivElement>(null); |
|
const popupBodyEl = useRef<HTMLDivElement>(null); |
|
const currentUser = useAuth(); |
|
|
|
const [showModal, setShowModal] = useState(!!userId); |
|
const [resourceSvg, setResourceSvg] = useState<SVGElement | null>(null); |
|
const [progressResponse, setProgressResponse] = |
|
useState<UserProgressResponse>(); |
|
|
|
const [isLoading, setIsLoading] = useState(true); |
|
const [error, setError] = useState(''); |
|
|
|
let resourceJsonUrl = import.meta.env.DEV |
|
? 'http://localhost:3000' |
|
: 'https://roadmap.sh'; |
|
if (resourceType === 'roadmap') { |
|
resourceJsonUrl += `/${resourceId}.json`; |
|
} else { |
|
resourceJsonUrl += `/best-practices/${resourceId}.json`; |
|
} |
|
|
|
async function getUserProgress( |
|
userId: string, |
|
resourceType: string, |
|
resourceId: string, |
|
): Promise<UserProgressResponse | undefined> { |
|
const { error, response } = await httpGet<UserProgressResponse>( |
|
`${ |
|
import.meta.env.PUBLIC_API_URL |
|
}/v1-get-user-progress/${userId}?resourceType=${resourceType}&resourceId=${resourceId}`, |
|
); |
|
|
|
if (error || !response) { |
|
throw error || new Error('Something went wrong. Please try again!'); |
|
} |
|
|
|
return response; |
|
} |
|
|
|
async function getRoadmapSVG( |
|
jsonUrl: string, |
|
renderer: AllowedRoadmapRenderer = 'balsamiq', |
|
): Promise<SVGElement | undefined> { |
|
const { error, response: roadmapJson } = await httpGet(jsonUrl); |
|
if (error || !roadmapJson) { |
|
throw error || new Error('Something went wrong. Please try again!'); |
|
} |
|
|
|
return renderer === 'editor' |
|
? await renderFlowJSON(roadmapJson as any) |
|
: await wireframeJSONToSVG(roadmapJson, { |
|
fontURL: '/fonts/balsamiq.woff2', |
|
}); |
|
} |
|
|
|
function onClose() { |
|
deleteUrlParam('s'); |
|
setError(''); |
|
setShowModal(false); |
|
|
|
if (onModalClose) { |
|
onModalClose(); |
|
} else { |
|
window.location.reload(); |
|
} |
|
} |
|
|
|
useKeydown('Escape', () => { |
|
onClose(); |
|
}); |
|
|
|
useOutsideClick(popupBodyEl, () => { |
|
onClose(); |
|
}); |
|
|
|
useEffect(() => { |
|
if (!resourceJsonUrl || !resourceId || !resourceType || !userId) { |
|
return; |
|
} |
|
|
|
setIsLoading(true); |
|
setError(''); |
|
|
|
Promise.all([ |
|
getRoadmapSVG(resourceJsonUrl, renderer), |
|
getUserProgress(userId, resourceType, resourceId), |
|
]) |
|
.then(([svg, user]) => { |
|
if (!user || !svg) { |
|
return; |
|
} |
|
|
|
const { progress } = user; |
|
const { done, learning, skipped } = progress || { |
|
done: [], |
|
learning: [], |
|
skipped: [], |
|
}; |
|
|
|
done?.forEach((topicId: string) => { |
|
topicSelectorAll(topicId, svg).forEach((el) => { |
|
el.classList.add('done'); |
|
}); |
|
}); |
|
|
|
learning?.forEach((topicId: string) => { |
|
topicSelectorAll(topicId, svg).forEach((el) => { |
|
el.classList.add('learning'); |
|
}); |
|
}); |
|
|
|
skipped?.forEach((topicId: string) => { |
|
topicSelectorAll(topicId, svg).forEach((el) => { |
|
el.classList.add('skipped'); |
|
}); |
|
}); |
|
|
|
svg.querySelectorAll('.clickable-group').forEach((el) => { |
|
el.classList.remove('clickable-group'); |
|
}); |
|
|
|
svg.querySelectorAll('[data-group-id]').forEach((el) => { |
|
el.removeAttribute('data-group-id'); |
|
}); |
|
|
|
setResourceSvg(svg); |
|
setProgressResponse(user); |
|
}) |
|
.catch((err) => { |
|
setError(err?.message || 'Something went wrong. Please try again!'); |
|
}) |
|
.finally(() => { |
|
setIsLoading(false); |
|
}); |
|
}, [userId]); |
|
|
|
if (currentUser?.id === userId) { |
|
deleteUrlParam('s'); |
|
return null; |
|
} |
|
|
|
if (!showModal) { |
|
return null; |
|
} |
|
|
|
if (isLoading || error) { |
|
return ( |
|
<ModalLoader |
|
text={'Loading user progress..'} |
|
isLoading={isLoading} |
|
error={error} |
|
/> |
|
); |
|
} |
|
|
|
return ( |
|
<div |
|
id={'user-progress-modal'} |
|
className="fixed left-0 right-0 top-0 z-[100] h-full items-center justify-center overflow-y-auto overflow-x-hidden overscroll-contain bg-black/50" |
|
> |
|
<div className="relative mx-auto h-full w-full max-w-4xl p-4 md:h-auto"> |
|
<div |
|
ref={popupBodyEl} |
|
className={`popup-body relative rounded-lg bg-white pt-[1px] shadow`} |
|
> |
|
<UserProgressModalHeader |
|
isLoading={isLoading} |
|
progressResponse={progressResponse} |
|
/> |
|
|
|
<div |
|
ref={resourceSvgEl} |
|
className="px-4 pb-2" |
|
dangerouslySetInnerHTML={{ __html: resourceSvg?.outerHTML || '' }} |
|
></div> |
|
|
|
<button |
|
type="button" |
|
className={`absolute right-2.5 top-3 ml-auto inline-flex items-center rounded-lg bg-gray-100 bg-transparent p-1.5 text-sm text-gray-400 hover:text-gray-900 lg:hidden`} |
|
onClick={onClose} |
|
> |
|
<X className="h-4 w-4" /> |
|
<span className="sr-only">Close modal</span> |
|
</button> |
|
</div> |
|
</div> |
|
</div> |
|
); |
|
}
|
|
|