1
0
mirror of https://github.com/kamranahmedse/developer-roadmap.git synced 2025-09-03 06:12:53 +02:00
This commit is contained in:
Arik Chakma
2025-06-24 23:48:59 +06:00
parent ae681a58b8
commit b0c3b1505c
3 changed files with 160 additions and 35 deletions

View File

@@ -7,7 +7,7 @@ import { useToast } from '../../hooks/use-toast';
import { queryClient } from '../../stores/query-client'; import { queryClient } from '../../stores/query-client';
import { AITutorLayout } from '../AITutor/AITutorLayout'; import { AITutorLayout } from '../AITutor/AITutorLayout';
import { UpgradeAccountModal } from '../Billing/UpgradeAccountModal'; import { UpgradeAccountModal } from '../Billing/UpgradeAccountModal';
import { aiRoadmapOptions } from '../../queries/ai-roadmap'; import { aiRoadmapOptions, generateAIRoadmap } from '../../queries/ai-roadmap';
import { GenerateAIRoadmap } from './GenerateAIRoadmap'; import { GenerateAIRoadmap } from './GenerateAIRoadmap';
import { AIRoadmapContent } from './AIRoadmapContent'; import { AIRoadmapContent } from './AIRoadmapContent';
import { AIRoadmapChat } from './AIRoadmapChat'; import { AIRoadmapChat } from './AIRoadmapChat';
@@ -23,6 +23,7 @@ export function AIRoadmap(props: AIRoadmapProps) {
const toast = useToast(); const toast = useToast();
const [showUpgradeModal, setShowUpgradeModal] = useState(false); const [showUpgradeModal, setShowUpgradeModal] = useState(false);
const [isRegenerating, setIsRegenerating] = useState(false); const [isRegenerating, setIsRegenerating] = useState(false);
const [regeneratedSvgHtml, setRegeneratedSvgHtml] = useState('');
// only fetch the guide if the guideSlug is provided // only fetch the guide if the guideSlug is provided
// otherwise we are still generating the guide // otherwise we are still generating the guide
@@ -45,9 +46,27 @@ export function AIRoadmap(props: AIRoadmapProps) {
return { return {
...old, ...old,
data: '', data: '',
svg: null, svgHtml: '',
}; };
}); });
await generateAIRoadmap({
roadmapSlug: aiRoadmap?.slug || '',
term: aiRoadmap?.term || '',
prompt,
isForce: true,
onStreamingChange: setIsRegenerating,
onRoadmapSvgChange: (svg) => {
setRegeneratedSvgHtml(svg.outerHTML);
},
onError: (error) => {
toast.error(error);
},
onFinish: () => {
setIsRegenerating(false);
queryClient.invalidateQueries(aiRoadmapOptions(roadmapSlug));
},
});
}; };
return ( return (
@@ -62,8 +81,10 @@ export function AIRoadmap(props: AIRoadmapProps) {
<div className="grow overflow-y-auto p-4 pt-0"> <div className="grow overflow-y-auto p-4 pt-0">
{roadmapSlug && ( {roadmapSlug && (
<AIRoadmapContent <AIRoadmapContent
svgHtml={aiRoadmap?.svgHtml || ''} svgHtml={regeneratedSvgHtml || aiRoadmap?.svgHtml || ''}
isLoading={isLoadingBySlug || isRegenerating} isLoading={isLoadingBySlug || isRegenerating}
onRegenerate={handleRegenerate}
roadmapSlug={roadmapSlug}
/> />
)} )}
{!roadmapSlug && ( {!roadmapSlug && (

View File

@@ -1,13 +1,16 @@
import { cn } from '../../lib/classname'; import { cn } from '../../lib/classname';
import { AIRoadmapRegenerate } from './AIRoadmapRegenerate';
import { LoadingChip } from '../LoadingChip'; import { LoadingChip } from '../LoadingChip';
type AIRoadmapContentProps = { type AIRoadmapContentProps = {
isLoading?: boolean; isLoading?: boolean;
svgHtml: string; svgHtml: string;
onRegenerate?: (prompt?: string) => void;
roadmapSlug?: string;
}; };
export function AIRoadmapContent(props: AIRoadmapContentProps) { export function AIRoadmapContent(props: AIRoadmapContentProps) {
const { isLoading, svgHtml } = props; const { isLoading, svgHtml, onRegenerate, roadmapSlug } = props;
return ( return (
<div <div
@@ -27,6 +30,15 @@ export function AIRoadmapContent(props: AIRoadmapContentProps) {
<LoadingChip message="Please wait..." /> <LoadingChip message="Please wait..." />
</div> </div>
)} )}
{onRegenerate && !isLoading && roadmapSlug && (
<div className="absolute top-4 right-4">
<AIRoadmapRegenerate
onRegenerate={onRegenerate}
roadmapSlug={roadmapSlug}
/>
</div>
)}
</div> </div>
); );
} }

View File

@@ -1,4 +1,11 @@
import { PenSquare, RefreshCcw, SettingsIcon } from 'lucide-react'; import {
Loader2Icon,
PenSquare,
RefreshCcw,
SaveIcon,
SettingsIcon,
type LucideIcon,
} from 'lucide-react';
import { useRef, useState } from 'react'; import { useRef, useState } from 'react';
import { useOutsideClick } from '../../hooks/use-outside-click'; import { useOutsideClick } from '../../hooks/use-outside-click';
import { cn } from '../../lib/classname'; import { cn } from '../../lib/classname';
@@ -10,6 +17,8 @@ import { queryClient } from '../../stores/query-client';
import { httpPost } from '../../lib/query-http'; import { httpPost } from '../../lib/query-http';
import { aiRoadmapOptions } from '../../queries/ai-roadmap'; import { aiRoadmapOptions } from '../../queries/ai-roadmap';
import { UpdatePreferences } from '../GenerateGuide/UpdatePreferences'; import { UpdatePreferences } from '../GenerateGuide/UpdatePreferences';
import { generateAIRoadmapFromText } from '@roadmapsh/editor';
import { useToast } from '../../hooks/use-toast';
type AIRoadmapRegenerateProps = { type AIRoadmapRegenerateProps = {
onRegenerate: (prompt?: string) => void; onRegenerate: (prompt?: string) => void;
@@ -19,6 +28,7 @@ type AIRoadmapRegenerateProps = {
export function AIRoadmapRegenerate(props: AIRoadmapRegenerateProps) { export function AIRoadmapRegenerate(props: AIRoadmapRegenerateProps) {
const { onRegenerate, roadmapSlug } = props; const { onRegenerate, roadmapSlug } = props;
const toast = useToast();
const [isDropdownVisible, setIsDropdownVisible] = useState(false); const [isDropdownVisible, setIsDropdownVisible] = useState(false);
const [showUpgradeModal, setShowUpgradeModal] = useState(false); const [showUpgradeModal, setShowUpgradeModal] = useState(false);
const [showPromptModal, setShowPromptModal] = useState(false); const [showPromptModal, setShowPromptModal] = useState(false);
@@ -63,6 +73,65 @@ export function AIRoadmapRegenerate(props: AIRoadmapRegenerateProps) {
queryClient, queryClient,
); );
const handleSaveAIRoadmap = async () => {
const { nodes, edges } = generateAIRoadmapFromText(aiRoadmap?.data || '');
return httpPost<{
roadmapId: string;
roadmapSlug: string;
}>(`/v1-save-ai-roadmap/${aiRoadmap?._id}`, {
title: aiRoadmap?.term,
nodes: nodes.map((node) => ({
...node,
// To reset the width and height of the node
// so that it can be calculated based on the content in the editor
width: undefined,
height: undefined,
style: {
...node.style,
width: undefined,
height: undefined,
},
measured: {
width: undefined,
height: undefined,
},
})),
edges,
});
};
const { mutate: saveAIRoadmap, isPending: isSavingAIRoadmap } = useMutation(
{
mutationFn: handleSaveAIRoadmap,
onSuccess: (data) => {
if (!data?.roadmapId) {
toast.error('Something went wrong');
return;
}
window.location.href = `/r/${data?.roadmapSlug}`;
},
},
queryClient,
);
const { mutate: editAIRoadmap, isPending: isEditingAIRoadmap } = useMutation(
{
mutationFn: handleSaveAIRoadmap,
onSuccess: (data) => {
if (!data?.roadmapId) {
toast.error('Something went wrong');
return;
}
window.open(
`${import.meta.env.PUBLIC_EDITOR_APP_URL}/${data?.roadmapId}`,
'_blank',
);
},
},
queryClient,
);
return ( return (
<> <>
{showUpgradeModal && ( {showUpgradeModal && (
@@ -108,51 +177,74 @@ export function AIRoadmapRegenerate(props: AIRoadmapRegenerateProps) {
</button> </button>
{isDropdownVisible && ( {isDropdownVisible && (
<div className="absolute top-full right-0 min-w-[190px] translate-y-1 overflow-hidden rounded-md border border-gray-200 bg-white shadow-md"> <div className="absolute top-full right-0 min-w-[190px] translate-y-1 overflow-hidden rounded-md border border-gray-200 bg-white shadow-md">
<button <ActionButton
onClick={() => { onClick={() => {
setIsDropdownVisible(false); setIsDropdownVisible(false);
setShowUpdatePreferencesModal(true); setShowUpdatePreferencesModal(true);
}} }}
className="flex w-full items-center gap-2.5 px-3 py-2 text-left text-sm text-gray-600 hover:bg-gray-100" icon={SettingsIcon}
> label="Update Preferences"
<SettingsIcon />
size={16} <ActionButton
className="text-gray-400"
strokeWidth={2.5}
/>
Update Preferences
</button>
<button
onClick={() => { onClick={() => {
setIsDropdownVisible(false); setIsDropdownVisible(false);
onRegenerate(); onRegenerate();
}} }}
className="flex w-full items-center gap-2.5 px-3 py-2 text-left text-sm text-gray-600 hover:bg-gray-100" icon={RefreshCcw}
> label="Regenerate"
<RefreshCcw />
size={16} <ActionButton
className="text-gray-400"
strokeWidth={2.5}
/>
Regenerate
</button>
<button
onClick={() => { onClick={() => {
setIsDropdownVisible(false); setIsDropdownVisible(false);
setShowPromptModal(true); setShowPromptModal(true);
}} }}
className="flex w-full items-center gap-2.5 px-3 py-2 text-left text-sm text-gray-600 hover:bg-gray-100" icon={PenSquare}
> label="Modify Prompt"
<PenSquare />
size={16}
className="text-gray-400" <ActionButton
strokeWidth={2.5} onClick={saveAIRoadmap}
/> icon={SaveIcon}
Modify Prompt label="Start Learning"
</button> isLoading={isSavingAIRoadmap}
/>
<ActionButton
onClick={editAIRoadmap}
icon={PenSquare}
label="Edit in Editor"
isLoading={isEditingAIRoadmap}
/>
</div> </div>
)} )}
</div> </div>
</> </>
); );
} }
type ActionButtonProps = {
onClick: () => void;
isLoading?: boolean;
icon: LucideIcon;
label: string;
};
function ActionButton(props: ActionButtonProps) {
const { onClick, isLoading, icon: Icon, label } = props;
return (
<button
className="flex w-full items-center gap-2.5 px-3 py-2 text-left text-sm text-gray-600 hover:bg-gray-100"
onClick={onClick}
disabled={isLoading}
>
{isLoading ? (
<Loader2Icon className="animate-spin" size={16} strokeWidth={2.5} />
) : (
<Icon size={16} className="text-gray-400" strokeWidth={2.5} />
)}
{label}
</button>
);
}