feat: update create new project dialog
This commit is contained in:
parent
08fccdd506
commit
eabc9fa2f6
@ -152,3 +152,24 @@
|
|||||||
.tab-scroll::-webkit-scrollbar {
|
.tab-scroll::-webkit-scrollbar {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.fade-r {
|
||||||
|
--mask-gradient: linear-gradient(
|
||||||
|
to right,
|
||||||
|
white 0%,
|
||||||
|
white calc(100% - var(--fade-size)),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
|
-webkit-mask-image: var(--mask-gradient);
|
||||||
|
mask-image: var(--mask-gradient);
|
||||||
|
}
|
||||||
|
.fade-l {
|
||||||
|
--mask-gradient: linear-gradient(
|
||||||
|
to left,
|
||||||
|
white var(--fade-size),
|
||||||
|
white 100%,
|
||||||
|
transparent
|
||||||
|
);
|
||||||
|
-webkit-mask-image: var(--mask-gradient);
|
||||||
|
mask-image: var(--mask-gradient);
|
||||||
|
}
|
||||||
|
@ -9,7 +9,7 @@ import {
|
|||||||
DialogTrigger,
|
DialogTrigger,
|
||||||
} from "@/components/ui/dialog"
|
} from "@/components/ui/dialog"
|
||||||
import Image from "next/image"
|
import Image from "next/image"
|
||||||
import { useState } from "react"
|
import { useState, useCallback, useEffect, useMemo } from "react"
|
||||||
import { set, z } from "zod"
|
import { set, z } from "zod"
|
||||||
import { zodResolver } from "@hookform/resolvers/zod"
|
import { zodResolver } from "@hookform/resolvers/zod"
|
||||||
import { useForm } from "react-hook-form"
|
import { useForm } from "react-hook-form"
|
||||||
@ -34,10 +34,20 @@ import {
|
|||||||
import { useUser } from "@clerk/nextjs"
|
import { useUser } from "@clerk/nextjs"
|
||||||
import { createSandbox } from "@/lib/actions"
|
import { createSandbox } from "@/lib/actions"
|
||||||
import { useRouter } from "next/navigation"
|
import { useRouter } from "next/navigation"
|
||||||
import { Loader2 } from "lucide-react"
|
import {
|
||||||
|
Loader2,
|
||||||
|
ChevronRight,
|
||||||
|
ChevronLeft,
|
||||||
|
Search,
|
||||||
|
SlashSquare,
|
||||||
|
} from "lucide-react"
|
||||||
import { Button } from "../ui/button"
|
import { Button } from "../ui/button"
|
||||||
import { projectTemplates } from "@/lib/data"
|
import { projectTemplates } from "@/lib/data"
|
||||||
|
|
||||||
|
import useEmblaCarousel from "embla-carousel-react"
|
||||||
|
import type { EmblaCarouselType } from "embla-carousel"
|
||||||
|
import { WheelGesturesPlugin } from "embla-carousel-wheel-gestures"
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
name: z
|
name: z
|
||||||
.string()
|
.string()
|
||||||
@ -57,11 +67,20 @@ export default function NewProjectModal({
|
|||||||
open: boolean
|
open: boolean
|
||||||
setOpen: (open: boolean) => void
|
setOpen: (open: boolean) => void
|
||||||
}) {
|
}) {
|
||||||
|
const router = useRouter()
|
||||||
|
const user = useUser()
|
||||||
const [selected, setSelected] = useState("reactjs")
|
const [selected, setSelected] = useState("reactjs")
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const router = useRouter()
|
const [emblaRef, emblaApi] = useEmblaCarousel({ loop: false }, [
|
||||||
|
WheelGesturesPlugin(),
|
||||||
const user = useUser()
|
])
|
||||||
|
const {
|
||||||
|
prevBtnDisabled,
|
||||||
|
nextBtnDisabled,
|
||||||
|
onPrevButtonClick,
|
||||||
|
onNextButtonClick,
|
||||||
|
} = usePrevNextButtons(emblaApi)
|
||||||
|
const [search, setSearch] = useState("")
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof formSchema>>({
|
const form = useForm<z.infer<typeof formSchema>>({
|
||||||
resolver: zodResolver(formSchema),
|
resolver: zodResolver(formSchema),
|
||||||
@ -71,6 +90,26 @@ export default function NewProjectModal({
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const handleTemplateClick = useCallback(
|
||||||
|
({ id, index }: { id: string; index: number }) => {
|
||||||
|
setSelected(id)
|
||||||
|
emblaApi?.scrollTo(index)
|
||||||
|
},
|
||||||
|
[emblaApi]
|
||||||
|
)
|
||||||
|
const filteredTemplates = useMemo(
|
||||||
|
() =>
|
||||||
|
projectTemplates.filter(
|
||||||
|
(item) =>
|
||||||
|
item.name.toLowerCase().includes(search.toLowerCase()) ||
|
||||||
|
item.description.toLowerCase().includes(search.toLowerCase())
|
||||||
|
),
|
||||||
|
[search, projectTemplates]
|
||||||
|
)
|
||||||
|
const emptyTemplates = useMemo(
|
||||||
|
() => filteredTemplates.length === 0,
|
||||||
|
[filteredTemplates]
|
||||||
|
)
|
||||||
async function onSubmit(values: z.infer<typeof formSchema>) {
|
async function onSubmit(values: z.infer<typeof formSchema>) {
|
||||||
if (!user.isSignedIn) return
|
if (!user.isSignedIn) return
|
||||||
|
|
||||||
@ -80,7 +119,6 @@ export default function NewProjectModal({
|
|||||||
const id = await createSandbox(sandboxData)
|
const id = await createSandbox(sandboxData)
|
||||||
router.push(`/code/${id}`)
|
router.push(`/code/${id}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
open={open}
|
open={open}
|
||||||
@ -92,25 +130,86 @@ export default function NewProjectModal({
|
|||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle>Create A Sandbox</DialogTitle>
|
<DialogTitle>Create A Sandbox</DialogTitle>
|
||||||
</DialogHeader>
|
</DialogHeader>
|
||||||
<div className="grid grid-cols-2 w-full gap-2 mt-2">
|
<div className="flex flex-col gap-2 max-w-full overflow-hidden">
|
||||||
{projectTemplates.map((item) => (
|
<div className="flex items-center justify-end">
|
||||||
<button
|
<SearchInput
|
||||||
disabled={item.disabled || loading}
|
{...{
|
||||||
key={item.id}
|
value: search,
|
||||||
onClick={() => setSelected(item.id)}
|
onValueChange: setSearch,
|
||||||
className={`${
|
}}
|
||||||
selected === item.id ? "border-foreground" : "border-border"
|
/>
|
||||||
} rounded-md border bg-card text-card-foreground shadow text-left p-4 flex flex-col transition-all focus-visible:outline-none focus-visible:ring-offset-2 focus-visible:ring-offset-background focus-visible:ring-2 focus-visible:ring-ring disabled:opacity-50 disabled:cursor-not-allowed`}
|
</div>
|
||||||
|
<div className="overflow-hidden relative" ref={emblaRef}>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"grid grid-flow-col gap-x-2 min-h-[97px]",
|
||||||
|
emptyTemplates ? "auto-cols-[100%]" : "auto-cols-[200px]"
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
<div className="space-x-2 flex items-center justify-start w-full">
|
{filteredTemplates.map((item, i) => (
|
||||||
<Image alt="" src={item.icon} width={20} height={20} />
|
<button
|
||||||
<div className="font-medium">{item.name}</div>
|
disabled={item.disabled || loading}
|
||||||
</div>
|
key={item.id}
|
||||||
<div className="mt-2 text-muted-foreground text-sm">
|
onClick={handleTemplateClick.bind(null, {
|
||||||
{item.description}
|
id: item.id,
|
||||||
</div>
|
index: i,
|
||||||
</button>
|
})}
|
||||||
))}
|
className={`${
|
||||||
|
selected === item.id ? "border-foreground" : "border-border"
|
||||||
|
} rounded-md border bg-card text-card-foreground shadow text-left p-4 flex flex-col transition-all focus-visible:outline-none focus-visible:ring-offset-2 focus-visible:ring-offset-background focus-visible:ring-2 focus-visible:ring-ring disabled:opacity-50 disabled:cursor-not-allowed`}
|
||||||
|
>
|
||||||
|
<div className="space-x-2 flex items-center justify-start w-full">
|
||||||
|
<Image alt="" src={item.icon} width={20} height={20} />
|
||||||
|
<div className="font-medium">{item.name}</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-2 text-muted-foreground text-xs line-clamp-2">
|
||||||
|
{item.description}
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
{emptyTemplates && (
|
||||||
|
<div className="flex flex-col gap-2 items-center text-center justify-center text-muted-foreground text-sm">
|
||||||
|
<p>No templates found</p>
|
||||||
|
<Button size="xs" asChild>
|
||||||
|
<a
|
||||||
|
href="https://github.com/jamesmurdza/sandbox"
|
||||||
|
target="_blank"
|
||||||
|
>
|
||||||
|
Contribute
|
||||||
|
</a>
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"absolute transition-all opacity-100 duration-400 bg-gradient-to-r from-background via-background to-transparent w-14 pl-1 left-0 top-0 -translate-x-1 bottom-0 h-full flex items-center",
|
||||||
|
prevBtnDisabled && "opacity-0 pointer-events-none"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
size="smIcon"
|
||||||
|
className="rounded-full"
|
||||||
|
onClick={onPrevButtonClick}
|
||||||
|
>
|
||||||
|
<ChevronLeft className="size-5" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"absolute transition-all opacity-100 duration-400 bg-gradient-to-l from-background via-background to-transparent w-14 pl-1 right-0 top-0 translate-x-1 bottom-0 h-full flex items-center",
|
||||||
|
nextBtnDisabled && "opacity-0 pointer-events-none"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
size="smIcon"
|
||||||
|
className="rounded-full"
|
||||||
|
onClick={onNextButtonClick}
|
||||||
|
>
|
||||||
|
<ChevronRight className="size-5" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Form {...form}>
|
<Form {...form}>
|
||||||
@ -178,3 +277,68 @@ export default function NewProjectModal({
|
|||||||
</Dialog>
|
</Dialog>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function SearchInput({
|
||||||
|
value,
|
||||||
|
onValueChange,
|
||||||
|
}: {
|
||||||
|
value?: string
|
||||||
|
onValueChange?: (value: string) => void
|
||||||
|
}) {
|
||||||
|
const onSubmit = useCallback((e: React.FormEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
console.log("searching")
|
||||||
|
}, [])
|
||||||
|
return (
|
||||||
|
<form {...{ onSubmit }} className="w-40 h-8 ">
|
||||||
|
<label
|
||||||
|
htmlFor="template-search"
|
||||||
|
className="flex gap-2 rounded-sm transition-colors bg-[#2e2e2e] border border-[--s-color] [--s-color:hsl(var(--muted-foreground))] focus-within:[--s-color:#fff] h-full items-center px-2"
|
||||||
|
>
|
||||||
|
<Search className="size-4 text-[--s-color] transition-colors" />
|
||||||
|
<input
|
||||||
|
id="template-search"
|
||||||
|
type="text"
|
||||||
|
name="search"
|
||||||
|
placeholder="Search templates"
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => onValueChange?.(e.target.value)}
|
||||||
|
className="bg-transparent placeholder:text-muted-foreground text-white w-full focus:outline-none text-xs"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</form>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
const usePrevNextButtons = (emblaApi: EmblaCarouselType | undefined) => {
|
||||||
|
const [prevBtnDisabled, setPrevBtnDisabled] = useState(true)
|
||||||
|
const [nextBtnDisabled, setNextBtnDisabled] = useState(true)
|
||||||
|
|
||||||
|
const onPrevButtonClick = useCallback(() => {
|
||||||
|
if (!emblaApi) return
|
||||||
|
emblaApi.scrollPrev()
|
||||||
|
}, [emblaApi])
|
||||||
|
|
||||||
|
const onNextButtonClick = useCallback(() => {
|
||||||
|
if (!emblaApi) return
|
||||||
|
emblaApi.scrollNext()
|
||||||
|
}, [emblaApi])
|
||||||
|
|
||||||
|
const onSelect = useCallback((emblaApi: EmblaCarouselType) => {
|
||||||
|
setPrevBtnDisabled(!emblaApi.canScrollPrev())
|
||||||
|
setNextBtnDisabled(!emblaApi.canScrollNext())
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!emblaApi) return
|
||||||
|
|
||||||
|
onSelect(emblaApi)
|
||||||
|
emblaApi.on("reInit", onSelect).on("select", onSelect)
|
||||||
|
}, [emblaApi, onSelect])
|
||||||
|
|
||||||
|
return {
|
||||||
|
prevBtnDisabled,
|
||||||
|
nextBtnDisabled,
|
||||||
|
onPrevButtonClick,
|
||||||
|
onNextButtonClick,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
49
frontend/package-lock.json
generated
49
frontend/package-lock.json
generated
@ -34,6 +34,8 @@
|
|||||||
"@xterm/xterm": "^5.5.0",
|
"@xterm/xterm": "^5.5.0",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
|
"embla-carousel-react": "^8.3.0",
|
||||||
|
"embla-carousel-wheel-gestures": "^8.0.1",
|
||||||
"framer-motion": "^11.2.3",
|
"framer-motion": "^11.2.3",
|
||||||
"fs": "^0.0.1-security",
|
"fs": "^0.0.1-security",
|
||||||
"geist": "^1.3.0",
|
"geist": "^1.3.0",
|
||||||
@ -2690,6 +2692,45 @@
|
|||||||
"integrity": "sha512-w+9yAVHoHhysCa+gln7AzbO9CdjFcL/wN/5dd+XW/Msl2d/4+WisEaCF1nty0xbAKaxdaJfgLB2296U7zZB7BA==",
|
"integrity": "sha512-w+9yAVHoHhysCa+gln7AzbO9CdjFcL/wN/5dd+XW/Msl2d/4+WisEaCF1nty0xbAKaxdaJfgLB2296U7zZB7BA==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"node_modules/embla-carousel": {
|
||||||
|
"version": "8.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/embla-carousel/-/embla-carousel-8.3.0.tgz",
|
||||||
|
"integrity": "sha512-Ve8dhI4w28qBqR8J+aMtv7rLK89r1ZA5HocwFz6uMB/i5EiC7bGI7y+AM80yAVUJw3qqaZYK7clmZMUR8kM3UA=="
|
||||||
|
},
|
||||||
|
"node_modules/embla-carousel-react": {
|
||||||
|
"version": "8.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/embla-carousel-react/-/embla-carousel-react-8.3.0.tgz",
|
||||||
|
"integrity": "sha512-P1FlinFDcIvggcErRjNuVqnUR8anyo8vLMIH8Rthgofw7Nj8qTguCa2QjFAbzxAUTQTPNNjNL7yt0BGGinVdFw==",
|
||||||
|
"dependencies": {
|
||||||
|
"embla-carousel": "8.3.0",
|
||||||
|
"embla-carousel-reactive-utils": "8.3.0"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^16.8.0 || ^17.0.1 || ^18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/embla-carousel-reactive-utils": {
|
||||||
|
"version": "8.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/embla-carousel-reactive-utils/-/embla-carousel-reactive-utils-8.3.0.tgz",
|
||||||
|
"integrity": "sha512-EYdhhJ302SC4Lmkx8GRsp0sjUhEN4WyFXPOk0kGu9OXZSRMmcBlRgTvHcq8eKJE1bXWBsOi1T83B+BSSVZSmwQ==",
|
||||||
|
"peerDependencies": {
|
||||||
|
"embla-carousel": "8.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/embla-carousel-wheel-gestures": {
|
||||||
|
"version": "8.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/embla-carousel-wheel-gestures/-/embla-carousel-wheel-gestures-8.0.1.tgz",
|
||||||
|
"integrity": "sha512-LMAnruDqDmsjL6UoQD65aLotpmfO49Fsr3H0bMi7I+BH6jbv9OJiE61kN56daKsVtCQEt0SU1MrJslbhtgF3yQ==",
|
||||||
|
"dependencies": {
|
||||||
|
"wheel-gestures": "^2.2.5"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"embla-carousel": "^8.0.0 || ~8.0.0-rc03"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/emoji-regex": {
|
"node_modules/emoji-regex": {
|
||||||
"version": "9.2.2",
|
"version": "9.2.2",
|
||||||
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz",
|
||||||
@ -4553,6 +4594,14 @@
|
|||||||
"webidl-conversions": "^3.0.0"
|
"webidl-conversions": "^3.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/wheel-gestures": {
|
||||||
|
"version": "2.2.48",
|
||||||
|
"resolved": "https://registry.npmjs.org/wheel-gestures/-/wheel-gestures-2.2.48.tgz",
|
||||||
|
"integrity": "sha512-f+Gy33Oa5Z14XY9679Zze+7VFhbsQfBFXodnU2x589l4kxGM9L5Y8zETTmcMR5pWOPQyRv4Z0lNax6xCO0NSlA==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/which": {
|
"node_modules/which": {
|
||||||
"version": "2.0.2",
|
"version": "2.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
|
||||||
|
@ -35,6 +35,8 @@
|
|||||||
"@xterm/xterm": "^5.5.0",
|
"@xterm/xterm": "^5.5.0",
|
||||||
"class-variance-authority": "^0.7.0",
|
"class-variance-authority": "^0.7.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
|
"embla-carousel-react": "^8.3.0",
|
||||||
|
"embla-carousel-wheel-gestures": "^8.0.1",
|
||||||
"framer-motion": "^11.2.3",
|
"framer-motion": "^11.2.3",
|
||||||
"fs": "^0.0.1-security",
|
"fs": "^0.0.1-security",
|
||||||
"geist": "^1.3.0",
|
"geist": "^1.3.0",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user