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.
188 lines
6.5 KiB
188 lines
6.5 KiB
import { |
|
ArrowUpRight, |
|
Ban, |
|
CircleFadingPlus, |
|
Cog, |
|
Telescope, |
|
Wand, |
|
} from 'lucide-react'; |
|
import type { FormEvent } from 'react'; |
|
import { getOpenAIKey, isLoggedIn } from '../../lib/jwt'; |
|
import { showLoginPopup } from '../../lib/popup'; |
|
import { cn } from '../../lib/classname.ts'; |
|
import { useState } from 'react'; |
|
import { OpenAISettings } from './OpenAISettings.tsx'; |
|
|
|
type RoadmapSearchProps = { |
|
roadmapTerm: string; |
|
setRoadmapTerm: (topic: string) => void; |
|
handleSubmit: (e: FormEvent<HTMLFormElement>) => void; |
|
loadAIRoadmapLimit: () => void; |
|
onLoadTerm: (topic: string) => void; |
|
limit: number; |
|
limitUsed: number; |
|
}; |
|
|
|
export function RoadmapSearch(props: RoadmapSearchProps) { |
|
const { |
|
roadmapTerm, |
|
setRoadmapTerm, |
|
handleSubmit, |
|
limit = 0, |
|
limitUsed = 0, |
|
onLoadTerm, |
|
loadAIRoadmapLimit, |
|
} = props; |
|
|
|
const canGenerateMore = limitUsed < limit; |
|
const [isConfiguring, setIsConfiguring] = useState(false); |
|
const openAPIKey = getOpenAIKey(); |
|
|
|
const randomTerms = ['OAuth', 'APIs', 'UX Design', 'gRPC']; |
|
|
|
return ( |
|
<div className="flex flex-grow flex-col items-center justify-center px-4 py-6 sm:px-6"> |
|
{isConfiguring && ( |
|
<OpenAISettings |
|
onClose={() => { |
|
setIsConfiguring(false); |
|
loadAIRoadmapLimit(); |
|
}} |
|
/> |
|
)} |
|
<div className="flex flex-col gap-0 text-center sm:gap-2"> |
|
<h1 className="relative text-2xl font-medium sm:text-3xl"> |
|
<span className="hidden sm:inline">Generate roadmaps with AI</span> |
|
<span className="inline sm:hidden">AI Roadmap Generator</span> |
|
</h1> |
|
<p className="text-base text-gray-500 sm:text-lg"> |
|
<span className="hidden sm:inline"> |
|
Enter a topic and let the AI generate a roadmap for you |
|
</span> |
|
<span className="inline sm:hidden"> |
|
Enter a topic to generate a roadmap |
|
</span> |
|
</p> |
|
</div> |
|
<div className="my-3 flex w-full max-w-[600px] flex-col items-center gap-3 sm:my-5"> |
|
<form |
|
onSubmit={(e) => { |
|
if (limit > 0 && canGenerateMore) { |
|
handleSubmit(e); |
|
} else { |
|
e.preventDefault(); |
|
} |
|
}} |
|
className="flex w-full flex-col gap-2 sm:flex-row" |
|
> |
|
<input |
|
autoFocus |
|
type="text" |
|
placeholder="Enter a topic to generate a roadmap for" |
|
className="w-full rounded-md border border-gray-400 px-3 py-2.5 transition-colors focus:border-black focus:outline-none" |
|
value={roadmapTerm} |
|
onInput={(e) => |
|
setRoadmapTerm((e.target as HTMLInputElement).value) |
|
} |
|
/> |
|
<button |
|
className={cn( |
|
'flex min-w-[154px] flex-shrink-0 items-center justify-center gap-2 rounded-md bg-black px-4 py-2 text-white', |
|
'disabled:cursor-not-allowed disabled:opacity-50', |
|
)} |
|
disabled={!limit || !roadmapTerm || limitUsed >= limit} |
|
> |
|
{(!limit || canGenerateMore) && ( |
|
<> |
|
<Wand size={20} /> |
|
Generate |
|
</> |
|
)} |
|
|
|
{limit > 0 && !canGenerateMore && ( |
|
<span className="flex items-center text-base"> |
|
<Ban size={15} className="mr-2" /> |
|
Limit reached |
|
</span> |
|
)} |
|
</button> |
|
</form> |
|
<div className="flex flex-row items-center justify-center gap-2 flex-wrap"> |
|
{randomTerms.map((term) => ( |
|
<button |
|
key={term} |
|
disabled={!limit || !canGenerateMore} |
|
type="button" |
|
onClick={() => { |
|
onLoadTerm(term); |
|
}} |
|
className="flex items-center gap-1.5 rounded-full border px-2 py-0.5 text-sm transition-colors hover:border-black hover:bg-gray-100 disabled:cursor-not-allowed disabled:opacity-50" |
|
> |
|
{term} <ArrowUpRight size={17} /> |
|
</button> |
|
))} |
|
<a |
|
href="/ai/explore" |
|
className="flex items-center gap-1.5 rounded-full border border-black bg-gray-700 px-2 py-0.5 text-sm text-white transition-colors hover:border-black hover:bg-black" |
|
> |
|
Explore AI Roadmaps <Telescope size={17} /> |
|
</a> |
|
</div> |
|
</div> |
|
<div className="mt-12 flex flex-col items-center gap-4"> |
|
<p className="text-gray-500 text-center"> |
|
You have generated{' '} |
|
<span |
|
className={cn( |
|
'inline-block min-w-[50px] rounded-xl border px-1.5 text-center text-sm tabular-nums text-gray-800', |
|
{ |
|
'animate-pulse border-zinc-300 bg-zinc-300 text-zinc-300': |
|
!limit, |
|
}, |
|
)} |
|
> |
|
{limitUsed} of {limit} |
|
</span>{' '} |
|
roadmaps. |
|
</p> |
|
<p className="flex min-h-[46px] sm:min-h-[26px] items-center text-sm"> |
|
{limit > 0 && !isLoggedIn() && ( |
|
<button |
|
onClick={showLoginPopup} |
|
className="rounded-xl border border-current px-2 py-0.5 text-sm text-blue-500 transition-colors hover:bg-blue-400 hover:text-white" |
|
> |
|
Generate more by{' '} |
|
<span className="font-semibold"> |
|
signing up (free and takes 2 seconds) |
|
</span>{' '} |
|
or <span className="font-semibold">logging in</span> |
|
</button> |
|
)} |
|
</p> |
|
<p className="-mt-[75px] sm:-mt-[45px] flex min-h-[46px] sm:min-h-[26px] items-center text-sm"> |
|
{limit > 0 && isLoggedIn() && !openAPIKey && ( |
|
<button |
|
onClick={() => setIsConfiguring(true)} |
|
className="rounded-xl border border-current px-2 py-0.5 text-sm text-blue-500 transition-colors hover:bg-blue-400 hover:text-white" |
|
> |
|
By-pass all limits by{' '} |
|
<span className="font-semibold"> |
|
adding your own OpenAI API key |
|
</span> |
|
</button> |
|
)} |
|
|
|
{limit > 0 && isLoggedIn() && openAPIKey && ( |
|
<button |
|
onClick={() => setIsConfiguring(true)} |
|
className="flex flex-row items-center gap-1 rounded-xl border border-current px-2 py-0.5 text-sm text-blue-500 transition-colors hover:bg-blue-400 hover:text-white" |
|
> |
|
<Cog size={15} /> |
|
Configure OpenAI key |
|
</button> |
|
)} |
|
</p> |
|
</div> |
|
</div> |
|
); |
|
}
|
|
|