Spaces:
				
			
			
	
			
			
					
		Running
		
	
	
	
			
			
	
	
	
	
		
		
					
		Running
		
	initial commit
Browse filesThis view is limited to 50 files because it contains too many changes.  
							See raw diff
- Dockerfile +19 -0
- README.md +22 -36
- app/(public)/layout.tsx +15 -0
- app/(public)/page.tsx +44 -0
- app/(public)/projects/page.tsx +33 -0
- app/api/ask-ai/route.ts +406 -0
- app/api/auth/route.ts +94 -0
- app/api/me/projects/[namespace]/[repoId]/route.ts +162 -0
- app/api/me/projects/route.ts +126 -0
- app/api/me/route.ts +25 -0
- app/auth/callback/page.tsx +27 -0
- app/auth/page.tsx +36 -0
- app/globals.css +0 -26
- app/layout.tsx +81 -12
- app/page.tsx +0 -103
- app/projects/[namespace]/[repoId]/page.tsx +39 -0
- app/projects/new/page.tsx +5 -0
- assets/globals.css +146 -0
- assets/logo.svg +316 -0
- assets/space.svg +7 -0
- components.json +21 -0
- components/contexts/app-context.tsx +57 -0
- components/contexts/user-context.tsx +8 -0
- components/editor/ask-ai/index.tsx +376 -0
- components/editor/ask-ai/re-imagine.tsx +140 -0
- components/editor/ask-ai/settings.tsx +212 -0
- components/editor/deploy-button/index.tsx +171 -0
- components/editor/footer/index.tsx +118 -0
- components/editor/header/index.tsx +69 -0
- components/editor/history/index.tsx +69 -0
- components/editor/index.tsx +311 -0
- components/editor/preview/index.tsx +78 -0
- components/editor/save-button/index.tsx +76 -0
- components/invite-friends/index.tsx +85 -0
- components/loading/index.tsx +41 -0
- components/login-modal/index.tsx +61 -0
- components/magic-ui/grid-pattern.tsx +69 -0
- components/my-projects/index.tsx +41 -0
- components/my-projects/project-card.tsx +74 -0
- components/pro-modal/index.tsx +93 -0
- components/providers/tanstack-query-provider.tsx +19 -0
- components/public/navigation/index.tsx +156 -0
- components/space/ask-ai/index.tsx +41 -0
- components/ui/avatar.tsx +53 -0
- components/ui/button.tsx +67 -0
- components/ui/dialog.tsx +143 -0
- components/ui/dropdown-menu.tsx +257 -0
- components/ui/input.tsx +21 -0
- components/ui/popover.tsx +48 -0
- components/ui/select.tsx +185 -0
    	
        Dockerfile
    ADDED
    
    | @@ -0,0 +1,19 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            FROM node:20-alpine
         | 
| 2 | 
            +
            USER root
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            USER 1000
         | 
| 5 | 
            +
            WORKDIR /usr/src/app
         | 
| 6 | 
            +
            # Copy package.json and package-lock.json to the container
         | 
| 7 | 
            +
            COPY --chown=1000 package.json package-lock.json ./
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            # Copy the rest of the application files to the container
         | 
| 10 | 
            +
            COPY --chown=1000 . .
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            RUN npm install
         | 
| 13 | 
            +
            RUN npm run build
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            # Expose the application port (assuming your app runs on port 3000)
         | 
| 16 | 
            +
            EXPOSE 3000
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            # Start the application
         | 
| 19 | 
            +
            CMD ["npm", "start"]
         | 
    	
        README.md
    CHANGED
    
    | @@ -1,36 +1,22 @@ | |
| 1 | 
            -
             | 
| 2 | 
            -
             | 
| 3 | 
            -
             | 
| 4 | 
            -
             | 
| 5 | 
            -
             | 
| 6 | 
            -
             | 
| 7 | 
            -
             | 
| 8 | 
            -
             | 
| 9 | 
            -
             | 
| 10 | 
            -
             | 
| 11 | 
            -
             | 
| 12 | 
            -
             | 
| 13 | 
            -
             | 
| 14 | 
            -
             | 
| 15 | 
            -
             | 
| 16 | 
            -
             | 
| 17 | 
            -
             | 
| 18 | 
            -
             | 
| 19 | 
            -
             | 
| 20 | 
            -
             | 
| 21 | 
            -
             | 
| 22 | 
            -
             | 
| 23 | 
            -
            ## Learn More
         | 
| 24 | 
            -
             | 
| 25 | 
            -
            To learn more about Next.js, take a look at the following resources:
         | 
| 26 | 
            -
             | 
| 27 | 
            -
            - [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API.
         | 
| 28 | 
            -
            - [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial.
         | 
| 29 | 
            -
             | 
| 30 | 
            -
            You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome!
         | 
| 31 | 
            -
             | 
| 32 | 
            -
            ## Deploy on Vercel
         | 
| 33 | 
            -
             | 
| 34 | 
            -
            The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js.
         | 
| 35 | 
            -
             | 
| 36 | 
            -
            Check out our [Next.js deployment documentation](https://nextjs.org/docs/app/building-your-application/deploying) for more details.
         | 
|  | |
| 1 | 
            +
            ---
         | 
| 2 | 
            +
            title: DeepSite v2
         | 
| 3 | 
            +
            emoji: 🐳
         | 
| 4 | 
            +
            colorFrom: blue
         | 
| 5 | 
            +
            colorTo: blue
         | 
| 6 | 
            +
            sdk: docker
         | 
| 7 | 
            +
            pinned: true
         | 
| 8 | 
            +
            app_port: 5173
         | 
| 9 | 
            +
            license: mit
         | 
| 10 | 
            +
            short_description: Generate any application with DeepSeek
         | 
| 11 | 
            +
            models:
         | 
| 12 | 
            +
              - deepseek-ai/DeepSeek-V3-0324
         | 
| 13 | 
            +
              - deepseek-ai/DeepSeek-R1-0528
         | 
| 14 | 
            +
            ---
         | 
| 15 | 
            +
             | 
| 16 | 
            +
            # DeepSite 🐳
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            DeepSite is a coding platform powered by DeepSeek AI, designed to make coding smarter and more efficient. Tailored for developers, data scientists, and AI engineers, it integrates generative AI into your coding projects to enhance creativity and productivity.
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            ## How to use it locally
         | 
| 21 | 
            +
             | 
| 22 | 
            +
            Follow [this discussion](https://huggingface.co/spaces/enzostvs/deepsite/discussions/74)
         | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
    	
        app/(public)/layout.tsx
    ADDED
    
    | @@ -0,0 +1,15 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import Navigation from "@/components/public/navigation";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            export default async function PublicLayout({
         | 
| 4 | 
            +
              children,
         | 
| 5 | 
            +
            }: Readonly<{
         | 
| 6 | 
            +
              children: React.ReactNode;
         | 
| 7 | 
            +
            }>) {
         | 
| 8 | 
            +
              return (
         | 
| 9 | 
            +
                <div className="min-h-screen bg-black z-1 relative">
         | 
| 10 | 
            +
                  <div className="background__noisy" />
         | 
| 11 | 
            +
                  <Navigation />
         | 
| 12 | 
            +
                  {children}
         | 
| 13 | 
            +
                </div>
         | 
| 14 | 
            +
              );
         | 
| 15 | 
            +
            }
         | 
    	
        app/(public)/page.tsx
    ADDED
    
    | @@ -0,0 +1,44 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { AskAi } from "@/components/space/ask-ai";
         | 
| 2 | 
            +
            import { redirect } from "next/navigation";
         | 
| 3 | 
            +
            export default function Home() {
         | 
| 4 | 
            +
              redirect("/projects/new");
         | 
| 5 | 
            +
              return (
         | 
| 6 | 
            +
                <>
         | 
| 7 | 
            +
                  <header className="container mx-auto pt-20 px-6 relative flex flex-col items-center justify-center text-center">
         | 
| 8 | 
            +
                    <div className="rounded-full border border-neutral-100/10 bg-neutral-100/5 text-xs text-neutral-300 px-3 py-1 max-w-max mx-auto mb-2">
         | 
| 9 | 
            +
                      ✨ DeepSite Public Beta
         | 
| 10 | 
            +
                    </div>
         | 
| 11 | 
            +
                    <h1 className="text-8xl font-semibold text-white font-mono max-w-4xl">
         | 
| 12 | 
            +
                      Code your website with AI in seconds
         | 
| 13 | 
            +
                    </h1>
         | 
| 14 | 
            +
                    <p className="text-2xl text-neutral-300/80 mt-4 text-center max-w-2xl">
         | 
| 15 | 
            +
                      Vibe Coding has never been so easy.
         | 
| 16 | 
            +
                    </p>
         | 
| 17 | 
            +
                    <div className="mt-14 max-w-2xl w-full mx-auto">
         | 
| 18 | 
            +
                      <AskAi />
         | 
| 19 | 
            +
                    </div>
         | 
| 20 | 
            +
                    <div className="absolute inset-0 pointer-events-none -z-[1]">
         | 
| 21 | 
            +
                      <div className="w-full h-full bg-gradient-to-r from-purple-500 to-pink-500 opacity-10 blur-3xl rounded-full" />
         | 
| 22 | 
            +
                      <div className="w-2/3 h-3/4 bg-gradient-to-r from-blue-500 to-teal-500 opacity-24 blur-3xl absolute -top-20 right-10 transform rotate-12" />
         | 
| 23 | 
            +
                      <div className="w-1/2 h-1/2 bg-gradient-to-r from-amber-500 to-rose-500 opacity-20 blur-3xl absolute bottom-0 left-10 rounded-3xl" />
         | 
| 24 | 
            +
                      <div className="w-48 h-48 bg-gradient-to-r from-cyan-500 to-indigo-500 opacity-20 blur-3xl absolute top-1/3 right-1/3 rounded-lg transform -rotate-15" />
         | 
| 25 | 
            +
                    </div>
         | 
| 26 | 
            +
                  </header>
         | 
| 27 | 
            +
                  <div id="community" className="h-screen flex items-center justify-center">
         | 
| 28 | 
            +
                    <h1 className="text-7xl font-extrabold text-white font-mono">
         | 
| 29 | 
            +
                      Community Driven
         | 
| 30 | 
            +
                    </h1>
         | 
| 31 | 
            +
                  </div>
         | 
| 32 | 
            +
                  <div id="deploy" className="h-screen flex items-center justify-center">
         | 
| 33 | 
            +
                    <h1 className="text-7xl font-extrabold text-white font-mono">
         | 
| 34 | 
            +
                      Deploy your website in seconds
         | 
| 35 | 
            +
                    </h1>
         | 
| 36 | 
            +
                  </div>
         | 
| 37 | 
            +
                  <div id="features" className="h-screen flex items-center justify-center">
         | 
| 38 | 
            +
                    <h1 className="text-7xl font-extrabold text-white font-mono">
         | 
| 39 | 
            +
                      Features that make you smile
         | 
| 40 | 
            +
                    </h1>
         | 
| 41 | 
            +
                  </div>
         | 
| 42 | 
            +
                </>
         | 
| 43 | 
            +
              );
         | 
| 44 | 
            +
            }
         | 
    	
        app/(public)/projects/page.tsx
    ADDED
    
    | @@ -0,0 +1,33 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { cookies } from "next/headers";
         | 
| 2 | 
            +
            import { redirect } from "next/navigation";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import { apiServer } from "@/lib/api";
         | 
| 5 | 
            +
            import MY_TOKEN_KEY from "@/lib/get-cookie-name";
         | 
| 6 | 
            +
            import { MyProjects } from "@/components/my-projects";
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            async function getMyProjects() {
         | 
| 9 | 
            +
              const cookieStore = await cookies();
         | 
| 10 | 
            +
              const token = cookieStore.get(MY_TOKEN_KEY())?.value;
         | 
| 11 | 
            +
              if (!token) return { redirectUrl: true, projects: [] };
         | 
| 12 | 
            +
              try {
         | 
| 13 | 
            +
                const { data } = await apiServer.get("/me/projects", {
         | 
| 14 | 
            +
                  headers: {
         | 
| 15 | 
            +
                    Authorization: `Bearer ${token}`,
         | 
| 16 | 
            +
                  },
         | 
| 17 | 
            +
                });
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                return {
         | 
| 20 | 
            +
                  projects: data.projects,
         | 
| 21 | 
            +
                };
         | 
| 22 | 
            +
              } catch {
         | 
| 23 | 
            +
                return { projects: [] };
         | 
| 24 | 
            +
              }
         | 
| 25 | 
            +
            }
         | 
| 26 | 
            +
            export default async function ProjectsPage() {
         | 
| 27 | 
            +
              const { redirectUrl, projects } = await getMyProjects();
         | 
| 28 | 
            +
              if (redirectUrl) {
         | 
| 29 | 
            +
                redirect("/");
         | 
| 30 | 
            +
              }
         | 
| 31 | 
            +
             | 
| 32 | 
            +
              return <MyProjects projects={projects} />;
         | 
| 33 | 
            +
            }
         | 
    	
        app/api/ask-ai/route.ts
    ADDED
    
    | @@ -0,0 +1,406 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 2 | 
            +
            import type { NextRequest } from "next/server";
         | 
| 3 | 
            +
            import { NextResponse } from "next/server";
         | 
| 4 | 
            +
            import { headers } from "next/headers";
         | 
| 5 | 
            +
            import { InferenceClient } from "@huggingface/inference";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { MODELS, PROVIDERS } from "@/lib/providers";
         | 
| 8 | 
            +
            import {
         | 
| 9 | 
            +
              DIVIDER,
         | 
| 10 | 
            +
              FOLLOW_UP_SYSTEM_PROMPT,
         | 
| 11 | 
            +
              INITIAL_SYSTEM_PROMPT,
         | 
| 12 | 
            +
              MAX_REQUESTS_PER_IP,
         | 
| 13 | 
            +
              REPLACE_END,
         | 
| 14 | 
            +
              SEARCH_START,
         | 
| 15 | 
            +
            } from "@/lib/prompts";
         | 
| 16 | 
            +
            import MY_TOKEN_KEY from "@/lib/get-cookie-name";
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            const ipAddresses = new Map();
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            export async function POST(request: NextRequest) {
         | 
| 21 | 
            +
              const authHeaders = await headers();
         | 
| 22 | 
            +
              const userToken = request.cookies.get(MY_TOKEN_KEY())?.value;
         | 
| 23 | 
            +
             | 
| 24 | 
            +
              const body = await request.json();
         | 
| 25 | 
            +
              const { prompt, provider, model, redesignMarkdown } = body;
         | 
| 26 | 
            +
             | 
| 27 | 
            +
              if (!model || (!prompt && !redesignMarkdown)) {
         | 
| 28 | 
            +
                return NextResponse.json(
         | 
| 29 | 
            +
                  { ok: false, error: "Missing required fields" },
         | 
| 30 | 
            +
                  { status: 400 }
         | 
| 31 | 
            +
                );
         | 
| 32 | 
            +
              }
         | 
| 33 | 
            +
             | 
| 34 | 
            +
              const selectedModel = MODELS.find(
         | 
| 35 | 
            +
                (m) => m.value === model || m.label === model
         | 
| 36 | 
            +
              );
         | 
| 37 | 
            +
              if (!selectedModel) {
         | 
| 38 | 
            +
                return NextResponse.json(
         | 
| 39 | 
            +
                  { ok: false, error: "Invalid model selected" },
         | 
| 40 | 
            +
                  { status: 400 }
         | 
| 41 | 
            +
                );
         | 
| 42 | 
            +
              }
         | 
| 43 | 
            +
             | 
| 44 | 
            +
              if (!selectedModel.providers.includes(provider) && provider !== "auto") {
         | 
| 45 | 
            +
                return NextResponse.json(
         | 
| 46 | 
            +
                  {
         | 
| 47 | 
            +
                    ok: false,
         | 
| 48 | 
            +
                    error: `The selected model does not support the ${provider} provider.`,
         | 
| 49 | 
            +
                    openSelectProvider: true,
         | 
| 50 | 
            +
                  },
         | 
| 51 | 
            +
                  { status: 400 }
         | 
| 52 | 
            +
                );
         | 
| 53 | 
            +
              }
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              let token = userToken;
         | 
| 56 | 
            +
              let billTo: string | null = null;
         | 
| 57 | 
            +
             | 
| 58 | 
            +
              /**
         | 
| 59 | 
            +
               * Handle local usage token, this bypass the need for a user token
         | 
| 60 | 
            +
               * and allows local testing without authentication.
         | 
| 61 | 
            +
               * This is useful for development and testing purposes.
         | 
| 62 | 
            +
               */
         | 
| 63 | 
            +
              if (process.env.HF_TOKEN && process.env.HF_TOKEN.length > 0) {
         | 
| 64 | 
            +
                token = process.env.HF_TOKEN;
         | 
| 65 | 
            +
              }
         | 
| 66 | 
            +
             | 
| 67 | 
            +
              const ip =
         | 
| 68 | 
            +
                authHeaders.get("x-forwarded-for")?.split(",")[0].trim() ||
         | 
| 69 | 
            +
                authHeaders.get("x-real-ip") ||
         | 
| 70 | 
            +
                "0.0.0.0";
         | 
| 71 | 
            +
             | 
| 72 | 
            +
              if (!token) {
         | 
| 73 | 
            +
                ipAddresses.set(ip, (ipAddresses.get(ip) || 0) + 1);
         | 
| 74 | 
            +
                if (ipAddresses.get(ip) > MAX_REQUESTS_PER_IP) {
         | 
| 75 | 
            +
                  return NextResponse.json(
         | 
| 76 | 
            +
                    {
         | 
| 77 | 
            +
                      ok: false,
         | 
| 78 | 
            +
                      openLogin: true,
         | 
| 79 | 
            +
                      message: "Log In to continue using the service",
         | 
| 80 | 
            +
                    },
         | 
| 81 | 
            +
                    { status: 429 }
         | 
| 82 | 
            +
                  );
         | 
| 83 | 
            +
                }
         | 
| 84 | 
            +
             | 
| 85 | 
            +
                token = process.env.DEFAULT_HF_TOKEN as string;
         | 
| 86 | 
            +
                billTo = "huggingface";
         | 
| 87 | 
            +
              }
         | 
| 88 | 
            +
             | 
| 89 | 
            +
              const DEFAULT_PROVIDER = PROVIDERS.novita;
         | 
| 90 | 
            +
              const selectedProvider =
         | 
| 91 | 
            +
                provider === "auto"
         | 
| 92 | 
            +
                  ? PROVIDERS[selectedModel.autoProvider as keyof typeof PROVIDERS]
         | 
| 93 | 
            +
                  : PROVIDERS[provider as keyof typeof PROVIDERS] ?? DEFAULT_PROVIDER;
         | 
| 94 | 
            +
             | 
| 95 | 
            +
              try {
         | 
| 96 | 
            +
                // Create a stream response
         | 
| 97 | 
            +
                const encoder = new TextEncoder();
         | 
| 98 | 
            +
                const stream = new TransformStream();
         | 
| 99 | 
            +
                const writer = stream.writable.getWriter();
         | 
| 100 | 
            +
             | 
| 101 | 
            +
                // Start the response
         | 
| 102 | 
            +
                const response = new NextResponse(stream.readable, {
         | 
| 103 | 
            +
                  headers: {
         | 
| 104 | 
            +
                    "Content-Type": "text/plain; charset=utf-8",
         | 
| 105 | 
            +
                    "Cache-Control": "no-cache",
         | 
| 106 | 
            +
                    Connection: "keep-alive",
         | 
| 107 | 
            +
                  },
         | 
| 108 | 
            +
                });
         | 
| 109 | 
            +
             | 
| 110 | 
            +
                // Process in background
         | 
| 111 | 
            +
                (async () => {
         | 
| 112 | 
            +
                  let completeResponse = "";
         | 
| 113 | 
            +
                  try {
         | 
| 114 | 
            +
                    const client = new InferenceClient(token);
         | 
| 115 | 
            +
                    const chatCompletion = client.chatCompletionStream(
         | 
| 116 | 
            +
                      {
         | 
| 117 | 
            +
                        model: selectedModel.value,
         | 
| 118 | 
            +
                        provider: selectedProvider.id as any,
         | 
| 119 | 
            +
                        messages: [
         | 
| 120 | 
            +
                          {
         | 
| 121 | 
            +
                            role: "system",
         | 
| 122 | 
            +
                            content: INITIAL_SYSTEM_PROMPT,
         | 
| 123 | 
            +
                          },
         | 
| 124 | 
            +
                          {
         | 
| 125 | 
            +
                            role: "user",
         | 
| 126 | 
            +
                            content: redesignMarkdown
         | 
| 127 | 
            +
                              ? `Here is my current design as a markdown:\n\n${redesignMarkdown}\n\nNow, please create a new design based on this markdown.`
         | 
| 128 | 
            +
                              : prompt,
         | 
| 129 | 
            +
                          },
         | 
| 130 | 
            +
                        ],
         | 
| 131 | 
            +
                        max_tokens: selectedProvider.max_tokens,
         | 
| 132 | 
            +
                      },
         | 
| 133 | 
            +
                      billTo ? { billTo } : {}
         | 
| 134 | 
            +
                    );
         | 
| 135 | 
            +
             | 
| 136 | 
            +
                    while (true) {
         | 
| 137 | 
            +
                      const { done, value } = await chatCompletion.next();
         | 
| 138 | 
            +
                      if (done) {
         | 
| 139 | 
            +
                        break;
         | 
| 140 | 
            +
                      }
         | 
| 141 | 
            +
             | 
| 142 | 
            +
                      const chunk = value.choices[0]?.delta?.content;
         | 
| 143 | 
            +
                      if (chunk) {
         | 
| 144 | 
            +
                        let newChunk = chunk;
         | 
| 145 | 
            +
                        if (!selectedModel?.isThinker) {
         | 
| 146 | 
            +
                          if (provider !== "sambanova") {
         | 
| 147 | 
            +
                            await writer.write(encoder.encode(chunk));
         | 
| 148 | 
            +
                            completeResponse += chunk;
         | 
| 149 | 
            +
             | 
| 150 | 
            +
                            if (completeResponse.includes("</html>")) {
         | 
| 151 | 
            +
                              break;
         | 
| 152 | 
            +
                            }
         | 
| 153 | 
            +
                          } else {
         | 
| 154 | 
            +
                            if (chunk.includes("</html>")) {
         | 
| 155 | 
            +
                              newChunk = newChunk.replace(/<\/html>[\s\S]*/, "</html>");
         | 
| 156 | 
            +
                            }
         | 
| 157 | 
            +
                            completeResponse += newChunk;
         | 
| 158 | 
            +
                            await writer.write(encoder.encode(newChunk));
         | 
| 159 | 
            +
                            if (newChunk.includes("</html>")) {
         | 
| 160 | 
            +
                              break;
         | 
| 161 | 
            +
                            }
         | 
| 162 | 
            +
                          }
         | 
| 163 | 
            +
                        } else {
         | 
| 164 | 
            +
                          const lastThinkTagIndex =
         | 
| 165 | 
            +
                            completeResponse.lastIndexOf("</think>");
         | 
| 166 | 
            +
                          completeResponse += newChunk;
         | 
| 167 | 
            +
                          await writer.write(encoder.encode(newChunk));
         | 
| 168 | 
            +
                          if (lastThinkTagIndex !== -1) {
         | 
| 169 | 
            +
                            const afterLastThinkTag = completeResponse.slice(
         | 
| 170 | 
            +
                              lastThinkTagIndex + "</think>".length
         | 
| 171 | 
            +
                            );
         | 
| 172 | 
            +
                            if (afterLastThinkTag.includes("</html>")) {
         | 
| 173 | 
            +
                              break;
         | 
| 174 | 
            +
                            }
         | 
| 175 | 
            +
                          }
         | 
| 176 | 
            +
                        }
         | 
| 177 | 
            +
                      }
         | 
| 178 | 
            +
                    }
         | 
| 179 | 
            +
                  } catch (error: any) {
         | 
| 180 | 
            +
                    if (error.message?.includes("exceeded your monthly included credits")) {
         | 
| 181 | 
            +
                      await writer.write(
         | 
| 182 | 
            +
                        encoder.encode(
         | 
| 183 | 
            +
                          JSON.stringify({
         | 
| 184 | 
            +
                            ok: false,
         | 
| 185 | 
            +
                            openProModal: true,
         | 
| 186 | 
            +
                            message: error.message,
         | 
| 187 | 
            +
                          })
         | 
| 188 | 
            +
                        )
         | 
| 189 | 
            +
                      );
         | 
| 190 | 
            +
                    } else {
         | 
| 191 | 
            +
                      await writer.write(
         | 
| 192 | 
            +
                        encoder.encode(
         | 
| 193 | 
            +
                          JSON.stringify({
         | 
| 194 | 
            +
                            ok: false,
         | 
| 195 | 
            +
                            message:
         | 
| 196 | 
            +
                              error.message ||
         | 
| 197 | 
            +
                              "An error occurred while processing your request.",
         | 
| 198 | 
            +
                          })
         | 
| 199 | 
            +
                        )
         | 
| 200 | 
            +
                      );
         | 
| 201 | 
            +
                    }
         | 
| 202 | 
            +
                  } finally {
         | 
| 203 | 
            +
                    await writer.close();
         | 
| 204 | 
            +
                  }
         | 
| 205 | 
            +
                })();
         | 
| 206 | 
            +
             | 
| 207 | 
            +
                return response;
         | 
| 208 | 
            +
              } catch (error: any) {
         | 
| 209 | 
            +
                return NextResponse.json(
         | 
| 210 | 
            +
                  {
         | 
| 211 | 
            +
                    ok: false,
         | 
| 212 | 
            +
                    openSelectProvider: true,
         | 
| 213 | 
            +
                    message:
         | 
| 214 | 
            +
                      error?.message || "An error occurred while processing your request.",
         | 
| 215 | 
            +
                  },
         | 
| 216 | 
            +
                  { status: 500 }
         | 
| 217 | 
            +
                );
         | 
| 218 | 
            +
              }
         | 
| 219 | 
            +
            }
         | 
| 220 | 
            +
             | 
| 221 | 
            +
            export async function PUT(request: NextRequest) {
         | 
| 222 | 
            +
              const authHeaders = await headers();
         | 
| 223 | 
            +
              const userToken = request.cookies.get(MY_TOKEN_KEY())?.value;
         | 
| 224 | 
            +
             | 
| 225 | 
            +
              const body = await request.json();
         | 
| 226 | 
            +
              const { prompt, html, previousPrompt, provider } = body;
         | 
| 227 | 
            +
             | 
| 228 | 
            +
              if (!prompt || !html) {
         | 
| 229 | 
            +
                return NextResponse.json(
         | 
| 230 | 
            +
                  { ok: false, error: "Missing required fields" },
         | 
| 231 | 
            +
                  { status: 400 }
         | 
| 232 | 
            +
                );
         | 
| 233 | 
            +
              }
         | 
| 234 | 
            +
             | 
| 235 | 
            +
              const selectedModel = MODELS[0];
         | 
| 236 | 
            +
             | 
| 237 | 
            +
              let token = userToken;
         | 
| 238 | 
            +
              let billTo: string | null = null;
         | 
| 239 | 
            +
             | 
| 240 | 
            +
              /**
         | 
| 241 | 
            +
               * Handle local usage token, this bypass the need for a user token
         | 
| 242 | 
            +
               * and allows local testing without authentication.
         | 
| 243 | 
            +
               * This is useful for development and testing purposes.
         | 
| 244 | 
            +
               */
         | 
| 245 | 
            +
              if (process.env.HF_TOKEN && process.env.HF_TOKEN.length > 0) {
         | 
| 246 | 
            +
                token = process.env.HF_TOKEN;
         | 
| 247 | 
            +
              }
         | 
| 248 | 
            +
             | 
| 249 | 
            +
              const ip =
         | 
| 250 | 
            +
                authHeaders.get("x-forwarded-for")?.split(",")[0].trim() ||
         | 
| 251 | 
            +
                authHeaders.get("x-real-ip") ||
         | 
| 252 | 
            +
                "0.0.0.0";
         | 
| 253 | 
            +
             | 
| 254 | 
            +
              if (!token) {
         | 
| 255 | 
            +
                ipAddresses.set(ip, (ipAddresses.get(ip) || 0) + 1);
         | 
| 256 | 
            +
                if (ipAddresses.get(ip) > MAX_REQUESTS_PER_IP) {
         | 
| 257 | 
            +
                  return NextResponse.json(
         | 
| 258 | 
            +
                    {
         | 
| 259 | 
            +
                      ok: false,
         | 
| 260 | 
            +
                      openLogin: true,
         | 
| 261 | 
            +
                      message: "Log In to continue using the service",
         | 
| 262 | 
            +
                    },
         | 
| 263 | 
            +
                    { status: 429 }
         | 
| 264 | 
            +
                  );
         | 
| 265 | 
            +
                }
         | 
| 266 | 
            +
             | 
| 267 | 
            +
                token = process.env.DEFAULT_HF_TOKEN as string;
         | 
| 268 | 
            +
                billTo = "huggingface";
         | 
| 269 | 
            +
              }
         | 
| 270 | 
            +
             | 
| 271 | 
            +
              const client = new InferenceClient(token);
         | 
| 272 | 
            +
             | 
| 273 | 
            +
              const DEFAULT_PROVIDER = PROVIDERS.novita;
         | 
| 274 | 
            +
              const selectedProvider =
         | 
| 275 | 
            +
                provider === "auto"
         | 
| 276 | 
            +
                  ? PROVIDERS[selectedModel.autoProvider as keyof typeof PROVIDERS]
         | 
| 277 | 
            +
                  : PROVIDERS[provider as keyof typeof PROVIDERS] ?? DEFAULT_PROVIDER;
         | 
| 278 | 
            +
             | 
| 279 | 
            +
              try {
         | 
| 280 | 
            +
                const response = await client.chatCompletion(
         | 
| 281 | 
            +
                  {
         | 
| 282 | 
            +
                    model: selectedModel.value,
         | 
| 283 | 
            +
                    provider: selectedProvider.id as any,
         | 
| 284 | 
            +
                    messages: [
         | 
| 285 | 
            +
                      {
         | 
| 286 | 
            +
                        role: "system",
         | 
| 287 | 
            +
                        content: FOLLOW_UP_SYSTEM_PROMPT,
         | 
| 288 | 
            +
                      },
         | 
| 289 | 
            +
                      {
         | 
| 290 | 
            +
                        role: "user",
         | 
| 291 | 
            +
                        content: previousPrompt
         | 
| 292 | 
            +
                          ? previousPrompt
         | 
| 293 | 
            +
                          : "You are modifying the HTML file based on the user's request.",
         | 
| 294 | 
            +
                      },
         | 
| 295 | 
            +
                      {
         | 
| 296 | 
            +
                        role: "assistant",
         | 
| 297 | 
            +
                        content: `The current code is: \n\`\`\`html\n${html}\n\`\`\``,
         | 
| 298 | 
            +
                      },
         | 
| 299 | 
            +
                      {
         | 
| 300 | 
            +
                        role: "user",
         | 
| 301 | 
            +
                        content: prompt,
         | 
| 302 | 
            +
                      },
         | 
| 303 | 
            +
                    ],
         | 
| 304 | 
            +
                    ...(selectedProvider.id !== "sambanova"
         | 
| 305 | 
            +
                      ? {
         | 
| 306 | 
            +
                          max_tokens: selectedProvider.max_tokens,
         | 
| 307 | 
            +
                        }
         | 
| 308 | 
            +
                      : {}),
         | 
| 309 | 
            +
                  },
         | 
| 310 | 
            +
                  billTo ? { billTo } : {}
         | 
| 311 | 
            +
                );
         | 
| 312 | 
            +
             | 
| 313 | 
            +
                const chunk = response.choices[0]?.message?.content;
         | 
| 314 | 
            +
                if (!chunk) {
         | 
| 315 | 
            +
                  return NextResponse.json(
         | 
| 316 | 
            +
                    { ok: false, message: "No content returned from the model" },
         | 
| 317 | 
            +
                    { status: 400 }
         | 
| 318 | 
            +
                  );
         | 
| 319 | 
            +
                }
         | 
| 320 | 
            +
             | 
| 321 | 
            +
                if (chunk) {
         | 
| 322 | 
            +
                  const updatedLines: number[][] = [];
         | 
| 323 | 
            +
                  let newHtml = html;
         | 
| 324 | 
            +
                  let position = 0;
         | 
| 325 | 
            +
                  let moreBlocks = true;
         | 
| 326 | 
            +
             | 
| 327 | 
            +
                  while (moreBlocks) {
         | 
| 328 | 
            +
                    const searchStartIndex = chunk.indexOf(SEARCH_START, position);
         | 
| 329 | 
            +
                    if (searchStartIndex === -1) {
         | 
| 330 | 
            +
                      moreBlocks = false;
         | 
| 331 | 
            +
                      continue;
         | 
| 332 | 
            +
                    }
         | 
| 333 | 
            +
             | 
| 334 | 
            +
                    const dividerIndex = chunk.indexOf(DIVIDER, searchStartIndex);
         | 
| 335 | 
            +
                    if (dividerIndex === -1) {
         | 
| 336 | 
            +
                      moreBlocks = false;
         | 
| 337 | 
            +
                      continue;
         | 
| 338 | 
            +
                    }
         | 
| 339 | 
            +
             | 
| 340 | 
            +
                    const replaceEndIndex = chunk.indexOf(REPLACE_END, dividerIndex);
         | 
| 341 | 
            +
                    if (replaceEndIndex === -1) {
         | 
| 342 | 
            +
                      moreBlocks = false;
         | 
| 343 | 
            +
                      continue;
         | 
| 344 | 
            +
                    }
         | 
| 345 | 
            +
             | 
| 346 | 
            +
                    const searchBlock = chunk.substring(
         | 
| 347 | 
            +
                      searchStartIndex + SEARCH_START.length,
         | 
| 348 | 
            +
                      dividerIndex
         | 
| 349 | 
            +
                    );
         | 
| 350 | 
            +
                    const replaceBlock = chunk.substring(
         | 
| 351 | 
            +
                      dividerIndex + DIVIDER.length,
         | 
| 352 | 
            +
                      replaceEndIndex
         | 
| 353 | 
            +
                    );
         | 
| 354 | 
            +
             | 
| 355 | 
            +
                    if (searchBlock.trim() === "") {
         | 
| 356 | 
            +
                      newHtml = `${replaceBlock}\n${newHtml}`;
         | 
| 357 | 
            +
                      updatedLines.push([1, replaceBlock.split("\n").length]);
         | 
| 358 | 
            +
                    } else {
         | 
| 359 | 
            +
                      const blockPosition = newHtml.indexOf(searchBlock);
         | 
| 360 | 
            +
                      if (blockPosition !== -1) {
         | 
| 361 | 
            +
                        const beforeText = newHtml.substring(0, blockPosition);
         | 
| 362 | 
            +
                        const startLineNumber = beforeText.split("\n").length;
         | 
| 363 | 
            +
                        const replaceLines = replaceBlock.split("\n").length;
         | 
| 364 | 
            +
                        const endLineNumber = startLineNumber + replaceLines - 1;
         | 
| 365 | 
            +
             | 
| 366 | 
            +
                        updatedLines.push([startLineNumber, endLineNumber]);
         | 
| 367 | 
            +
                        newHtml = newHtml.replace(searchBlock, replaceBlock);
         | 
| 368 | 
            +
                      }
         | 
| 369 | 
            +
                    }
         | 
| 370 | 
            +
             | 
| 371 | 
            +
                    position = replaceEndIndex + REPLACE_END.length;
         | 
| 372 | 
            +
                  }
         | 
| 373 | 
            +
             | 
| 374 | 
            +
                  return NextResponse.json({
         | 
| 375 | 
            +
                    ok: true,
         | 
| 376 | 
            +
                    html: newHtml,
         | 
| 377 | 
            +
                    updatedLines,
         | 
| 378 | 
            +
                  });
         | 
| 379 | 
            +
                } else {
         | 
| 380 | 
            +
                  return NextResponse.json(
         | 
| 381 | 
            +
                    { ok: false, message: "No content returned from the model" },
         | 
| 382 | 
            +
                    { status: 400 }
         | 
| 383 | 
            +
                  );
         | 
| 384 | 
            +
                }
         | 
| 385 | 
            +
              } catch (error: any) {
         | 
| 386 | 
            +
                if (error.message?.includes("exceeded your monthly included credits")) {
         | 
| 387 | 
            +
                  return NextResponse.json(
         | 
| 388 | 
            +
                    {
         | 
| 389 | 
            +
                      ok: false,
         | 
| 390 | 
            +
                      openProModal: true,
         | 
| 391 | 
            +
                      message: error.message,
         | 
| 392 | 
            +
                    },
         | 
| 393 | 
            +
                    { status: 402 }
         | 
| 394 | 
            +
                  );
         | 
| 395 | 
            +
                }
         | 
| 396 | 
            +
                return NextResponse.json(
         | 
| 397 | 
            +
                  {
         | 
| 398 | 
            +
                    ok: false,
         | 
| 399 | 
            +
                    openSelectProvider: true,
         | 
| 400 | 
            +
                    message:
         | 
| 401 | 
            +
                      error.message || "An error occurred while processing your request.",
         | 
| 402 | 
            +
                  },
         | 
| 403 | 
            +
                  { status: 500 }
         | 
| 404 | 
            +
                );
         | 
| 405 | 
            +
              }
         | 
| 406 | 
            +
            }
         | 
    	
        app/api/auth/route.ts
    ADDED
    
    | @@ -0,0 +1,94 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { NextRequest, NextResponse } from "next/server";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            export async function GET() {
         | 
| 4 | 
            +
              const redirect_uri = process.env.REDIRECT_URI;
         | 
| 5 | 
            +
              const loginRedirectUrl = `https://huggingface.co/oauth/authorize?client_id=${process.env.HUGGINGFACE_CLIENT_ID}&redirect_uri=${redirect_uri}&response_type=code&scope=openid%20profile%20write-repos%20manage-repos%20inference-api&prompt=consent&state=1234567890`;
         | 
| 6 | 
            +
             | 
| 7 | 
            +
              return NextResponse.json(
         | 
| 8 | 
            +
                {
         | 
| 9 | 
            +
                  redirect: loginRedirectUrl,
         | 
| 10 | 
            +
                },
         | 
| 11 | 
            +
                {
         | 
| 12 | 
            +
                  status: 200,
         | 
| 13 | 
            +
                  headers: {
         | 
| 14 | 
            +
                    "Content-Type": "application/json",
         | 
| 15 | 
            +
                  },
         | 
| 16 | 
            +
                }
         | 
| 17 | 
            +
              );
         | 
| 18 | 
            +
            }
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            export async function POST(req: NextRequest) {
         | 
| 21 | 
            +
              const body = await req.json();
         | 
| 22 | 
            +
              const { code } = body;
         | 
| 23 | 
            +
             | 
| 24 | 
            +
              if (!code) {
         | 
| 25 | 
            +
                return NextResponse.json(
         | 
| 26 | 
            +
                  { error: "Code is required" },
         | 
| 27 | 
            +
                  {
         | 
| 28 | 
            +
                    status: 400,
         | 
| 29 | 
            +
                    headers: {
         | 
| 30 | 
            +
                      "Content-Type": "application/json",
         | 
| 31 | 
            +
                    },
         | 
| 32 | 
            +
                  }
         | 
| 33 | 
            +
                );
         | 
| 34 | 
            +
              }
         | 
| 35 | 
            +
             | 
| 36 | 
            +
              const Authorization = `Basic ${Buffer.from(
         | 
| 37 | 
            +
                `${process.env.HUGGINGFACE_CLIENT_ID}:${process.env.HUGGINGFACE_CLIENT_SECRET}`
         | 
| 38 | 
            +
              ).toString("base64")}`;
         | 
| 39 | 
            +
             | 
| 40 | 
            +
              const redirect_uri = process.env.REDIRECT_URI as string;
         | 
| 41 | 
            +
              const request_auth = await fetch("https://huggingface.co/oauth/token", {
         | 
| 42 | 
            +
                method: "POST",
         | 
| 43 | 
            +
                headers: {
         | 
| 44 | 
            +
                  "Content-Type": "application/x-www-form-urlencoded",
         | 
| 45 | 
            +
                  Authorization,
         | 
| 46 | 
            +
                },
         | 
| 47 | 
            +
                body: new URLSearchParams({
         | 
| 48 | 
            +
                  grant_type: "authorization_code",
         | 
| 49 | 
            +
                  code,
         | 
| 50 | 
            +
                  redirect_uri,
         | 
| 51 | 
            +
                }),
         | 
| 52 | 
            +
              });
         | 
| 53 | 
            +
             | 
| 54 | 
            +
              const response = await request_auth.json();
         | 
| 55 | 
            +
              if (!response.access_token) {
         | 
| 56 | 
            +
                return NextResponse.json(
         | 
| 57 | 
            +
                  { error: "Failed to retrieve access token" },
         | 
| 58 | 
            +
                  {
         | 
| 59 | 
            +
                    status: 400,
         | 
| 60 | 
            +
                    headers: {
         | 
| 61 | 
            +
                      "Content-Type": "application/json",
         | 
| 62 | 
            +
                    },
         | 
| 63 | 
            +
                  }
         | 
| 64 | 
            +
                );
         | 
| 65 | 
            +
              }
         | 
| 66 | 
            +
             | 
| 67 | 
            +
              const userResponse = await fetch("https://huggingface.co/api/whoami-v2", {
         | 
| 68 | 
            +
                headers: {
         | 
| 69 | 
            +
                  Authorization: `Bearer ${response.access_token}`,
         | 
| 70 | 
            +
                },
         | 
| 71 | 
            +
              });
         | 
| 72 | 
            +
             | 
| 73 | 
            +
              if (!userResponse.ok) {
         | 
| 74 | 
            +
                return NextResponse.json(
         | 
| 75 | 
            +
                  { user: null, errCode: userResponse.status },
         | 
| 76 | 
            +
                  { status: userResponse.status }
         | 
| 77 | 
            +
                );
         | 
| 78 | 
            +
              }
         | 
| 79 | 
            +
              const user = await userResponse.json();
         | 
| 80 | 
            +
             | 
| 81 | 
            +
              return NextResponse.json(
         | 
| 82 | 
            +
                {
         | 
| 83 | 
            +
                  access_token: response.access_token,
         | 
| 84 | 
            +
                  expires_in: response.expires_in,
         | 
| 85 | 
            +
                  user,
         | 
| 86 | 
            +
                },
         | 
| 87 | 
            +
                {
         | 
| 88 | 
            +
                  status: 200,
         | 
| 89 | 
            +
                  headers: {
         | 
| 90 | 
            +
                    "Content-Type": "application/json",
         | 
| 91 | 
            +
                  },
         | 
| 92 | 
            +
                }
         | 
| 93 | 
            +
              );
         | 
| 94 | 
            +
            }
         | 
    	
        app/api/me/projects/[namespace]/[repoId]/route.ts
    ADDED
    
    | @@ -0,0 +1,162 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { NextRequest, NextResponse } from "next/server";
         | 
| 2 | 
            +
            import { RepoDesignation, spaceInfo, uploadFile } from "@huggingface/hub";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import { isAuthenticated } from "@/lib/auth";
         | 
| 5 | 
            +
            import Project from "@/models/Project";
         | 
| 6 | 
            +
            import dbConnect from "@/lib/mongodb";
         | 
| 7 | 
            +
            import { getPTag } from "@/lib/utils";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            export async function GET(
         | 
| 10 | 
            +
              req: NextRequest,
         | 
| 11 | 
            +
              { params }: { params: Promise<{ namespace: string; repoId: string }> }
         | 
| 12 | 
            +
            ) {
         | 
| 13 | 
            +
              const user = await isAuthenticated();
         | 
| 14 | 
            +
             | 
| 15 | 
            +
              if (user instanceof NextResponse || !user) {
         | 
| 16 | 
            +
                return NextResponse.json({ message: "Unauthorized" }, { status: 401 });
         | 
| 17 | 
            +
              }
         | 
| 18 | 
            +
             | 
| 19 | 
            +
              await dbConnect();
         | 
| 20 | 
            +
              const param = await params;
         | 
| 21 | 
            +
              const { namespace, repoId } = param;
         | 
| 22 | 
            +
             | 
| 23 | 
            +
              const project = await Project.findOne({
         | 
| 24 | 
            +
                user_id: user.id,
         | 
| 25 | 
            +
                space_id: `${namespace}/${repoId}`,
         | 
| 26 | 
            +
              }).lean();
         | 
| 27 | 
            +
              if (!project) {
         | 
| 28 | 
            +
                return NextResponse.json(
         | 
| 29 | 
            +
                  {
         | 
| 30 | 
            +
                    ok: false,
         | 
| 31 | 
            +
                    error: "Project not found",
         | 
| 32 | 
            +
                  },
         | 
| 33 | 
            +
                  { status: 404 }
         | 
| 34 | 
            +
                );
         | 
| 35 | 
            +
              }
         | 
| 36 | 
            +
              const space_url = `https://huggingface.co/spaces/${namespace}/${repoId}/raw/main/index.html`;
         | 
| 37 | 
            +
              try {
         | 
| 38 | 
            +
                const space = await spaceInfo({
         | 
| 39 | 
            +
                  name: namespace + "/" + repoId,
         | 
| 40 | 
            +
                  accessToken: user.token as string,
         | 
| 41 | 
            +
                  additionalFields: ["author"],
         | 
| 42 | 
            +
                });
         | 
| 43 | 
            +
             | 
| 44 | 
            +
                if (!space || space.sdk !== "static" || space.private) {
         | 
| 45 | 
            +
                  return NextResponse.json(
         | 
| 46 | 
            +
                    {
         | 
| 47 | 
            +
                      ok: false,
         | 
| 48 | 
            +
                      error: "Space is not a static space or is private",
         | 
| 49 | 
            +
                    },
         | 
| 50 | 
            +
                    { status: 404 }
         | 
| 51 | 
            +
                  );
         | 
| 52 | 
            +
                }
         | 
| 53 | 
            +
                if (space.author !== user.name) {
         | 
| 54 | 
            +
                  return NextResponse.json(
         | 
| 55 | 
            +
                    {
         | 
| 56 | 
            +
                      ok: false,
         | 
| 57 | 
            +
                      error: "Space does not belong to the authenticated user",
         | 
| 58 | 
            +
                    },
         | 
| 59 | 
            +
                    { status: 403 }
         | 
| 60 | 
            +
                  );
         | 
| 61 | 
            +
                }
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                const response = await fetch(space_url);
         | 
| 64 | 
            +
                if (!response.ok) {
         | 
| 65 | 
            +
                  return NextResponse.json(
         | 
| 66 | 
            +
                    {
         | 
| 67 | 
            +
                      ok: false,
         | 
| 68 | 
            +
                      error: "Failed to fetch space HTML",
         | 
| 69 | 
            +
                    },
         | 
| 70 | 
            +
                    { status: 404 }
         | 
| 71 | 
            +
                  );
         | 
| 72 | 
            +
                }
         | 
| 73 | 
            +
                let html = await response.text();
         | 
| 74 | 
            +
                // remove the last p tag including this url https://enzostvs-deepsite.hf.space
         | 
| 75 | 
            +
                html = html.replace(getPTag(namespace + "/" + repoId), "");
         | 
| 76 | 
            +
             | 
| 77 | 
            +
                return NextResponse.json(
         | 
| 78 | 
            +
                  {
         | 
| 79 | 
            +
                    project: {
         | 
| 80 | 
            +
                      ...project,
         | 
| 81 | 
            +
                      html,
         | 
| 82 | 
            +
                    },
         | 
| 83 | 
            +
                    ok: true,
         | 
| 84 | 
            +
                  },
         | 
| 85 | 
            +
                  { status: 200 }
         | 
| 86 | 
            +
                );
         | 
| 87 | 
            +
             | 
| 88 | 
            +
                // eslint-disable-next-line @typescript-eslint/no-explicit-any
         | 
| 89 | 
            +
              } catch (error: any) {
         | 
| 90 | 
            +
                if (error.statusCode === 404) {
         | 
| 91 | 
            +
                  await Project.deleteOne({
         | 
| 92 | 
            +
                    user_id: user.id,
         | 
| 93 | 
            +
                    space_id: `${namespace}/${repoId}`,
         | 
| 94 | 
            +
                  });
         | 
| 95 | 
            +
                  return NextResponse.json(
         | 
| 96 | 
            +
                    { error: "Space not found", ok: false },
         | 
| 97 | 
            +
                    { status: 404 }
         | 
| 98 | 
            +
                  );
         | 
| 99 | 
            +
                }
         | 
| 100 | 
            +
                return NextResponse.json(
         | 
| 101 | 
            +
                  { error: error.message, ok: false },
         | 
| 102 | 
            +
                  { status: 500 }
         | 
| 103 | 
            +
                );
         | 
| 104 | 
            +
              }
         | 
| 105 | 
            +
            }
         | 
| 106 | 
            +
             | 
| 107 | 
            +
            export async function PUT(
         | 
| 108 | 
            +
              req: NextRequest,
         | 
| 109 | 
            +
              { params }: { params: Promise<{ namespace: string; repoId: string }> }
         | 
| 110 | 
            +
            ) {
         | 
| 111 | 
            +
              const user = await isAuthenticated();
         | 
| 112 | 
            +
             | 
| 113 | 
            +
              if (user instanceof NextResponse || !user) {
         | 
| 114 | 
            +
                return NextResponse.json({ message: "Unauthorized" }, { status: 401 });
         | 
| 115 | 
            +
              }
         | 
| 116 | 
            +
             | 
| 117 | 
            +
              await dbConnect();
         | 
| 118 | 
            +
              const param = await params;
         | 
| 119 | 
            +
              const { namespace, repoId } = param;
         | 
| 120 | 
            +
              const { html, prompts } = await req.json();
         | 
| 121 | 
            +
             | 
| 122 | 
            +
              const project = await Project.findOne({
         | 
| 123 | 
            +
                user_id: user.id,
         | 
| 124 | 
            +
                space_id: `${namespace}/${repoId}`,
         | 
| 125 | 
            +
              }).lean();
         | 
| 126 | 
            +
              if (!project) {
         | 
| 127 | 
            +
                return NextResponse.json(
         | 
| 128 | 
            +
                  {
         | 
| 129 | 
            +
                    ok: false,
         | 
| 130 | 
            +
                    error: "Project not found",
         | 
| 131 | 
            +
                  },
         | 
| 132 | 
            +
                  { status: 404 }
         | 
| 133 | 
            +
                );
         | 
| 134 | 
            +
              }
         | 
| 135 | 
            +
             | 
| 136 | 
            +
              const repo: RepoDesignation = {
         | 
| 137 | 
            +
                type: "space",
         | 
| 138 | 
            +
                name: `${namespace}/${repoId}`,
         | 
| 139 | 
            +
              };
         | 
| 140 | 
            +
             | 
| 141 | 
            +
              const newHtml = html.replace(/<\/body>/, `${getPTag(repo.name)}</body>`);
         | 
| 142 | 
            +
              const file = new File([newHtml], "index.html", { type: "text/html" });
         | 
| 143 | 
            +
              await uploadFile({
         | 
| 144 | 
            +
                repo,
         | 
| 145 | 
            +
                file,
         | 
| 146 | 
            +
                accessToken: user.token as string,
         | 
| 147 | 
            +
                commitTitle: `${prompts[prompts.length - 1]} - Follow Up Deployment`,
         | 
| 148 | 
            +
              });
         | 
| 149 | 
            +
             | 
| 150 | 
            +
              await Project.updateOne(
         | 
| 151 | 
            +
                { user_id: user.id, space_id: `${namespace}/${repoId}` },
         | 
| 152 | 
            +
                {
         | 
| 153 | 
            +
                  $set: {
         | 
| 154 | 
            +
                    prompts: [
         | 
| 155 | 
            +
                      ...(project && "prompts" in project ? project.prompts : []),
         | 
| 156 | 
            +
                      ...prompts,
         | 
| 157 | 
            +
                    ],
         | 
| 158 | 
            +
                  },
         | 
| 159 | 
            +
                }
         | 
| 160 | 
            +
              );
         | 
| 161 | 
            +
              return NextResponse.json({ ok: true }, { status: 200 });
         | 
| 162 | 
            +
            }
         | 
    	
        app/api/me/projects/route.ts
    ADDED
    
    | @@ -0,0 +1,126 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { NextRequest, NextResponse } from "next/server";
         | 
| 2 | 
            +
            import { createRepo, RepoDesignation, uploadFiles } from "@huggingface/hub";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import { isAuthenticated } from "@/lib/auth";
         | 
| 5 | 
            +
            import Project from "@/models/Project";
         | 
| 6 | 
            +
            import dbConnect from "@/lib/mongodb";
         | 
| 7 | 
            +
            import { COLORS, getPTag } from "@/lib/utils";
         | 
| 8 | 
            +
            // import type user
         | 
| 9 | 
            +
            export async function GET() {
         | 
| 10 | 
            +
              const user = await isAuthenticated();
         | 
| 11 | 
            +
             | 
| 12 | 
            +
              if (user instanceof NextResponse || !user) {
         | 
| 13 | 
            +
                return NextResponse.json({ message: "Unauthorized" }, { status: 401 });
         | 
| 14 | 
            +
              }
         | 
| 15 | 
            +
             | 
| 16 | 
            +
              await dbConnect();
         | 
| 17 | 
            +
             | 
| 18 | 
            +
              const projects = await Project.find({
         | 
| 19 | 
            +
                user_id: user?.id,
         | 
| 20 | 
            +
              })
         | 
| 21 | 
            +
                .sort({ _createdAt: -1 })
         | 
| 22 | 
            +
                .limit(100)
         | 
| 23 | 
            +
                .lean();
         | 
| 24 | 
            +
              if (!projects) {
         | 
| 25 | 
            +
                return NextResponse.json(
         | 
| 26 | 
            +
                  {
         | 
| 27 | 
            +
                    ok: false,
         | 
| 28 | 
            +
                    projects: [],
         | 
| 29 | 
            +
                  },
         | 
| 30 | 
            +
                  { status: 404 }
         | 
| 31 | 
            +
                );
         | 
| 32 | 
            +
              }
         | 
| 33 | 
            +
              return NextResponse.json(
         | 
| 34 | 
            +
                {
         | 
| 35 | 
            +
                  ok: true,
         | 
| 36 | 
            +
                  projects,
         | 
| 37 | 
            +
                },
         | 
| 38 | 
            +
                { status: 200 }
         | 
| 39 | 
            +
              );
         | 
| 40 | 
            +
            }
         | 
| 41 | 
            +
             | 
| 42 | 
            +
            /**
         | 
| 43 | 
            +
             * This API route creates a new project in Hugging Face Spaces.
         | 
| 44 | 
            +
             * It requires an Authorization header with a valid token and a JSON body with the project details.
         | 
| 45 | 
            +
             */
         | 
| 46 | 
            +
            export async function POST(request: NextRequest) {
         | 
| 47 | 
            +
              const user = await isAuthenticated();
         | 
| 48 | 
            +
             | 
| 49 | 
            +
              if (user instanceof NextResponse || !user) {
         | 
| 50 | 
            +
                return NextResponse.json({ message: "Unauthorized" }, { status: 401 });
         | 
| 51 | 
            +
              }
         | 
| 52 | 
            +
             | 
| 53 | 
            +
              const { title, html, prompts } = await request.json();
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              if (!title || !html) {
         | 
| 56 | 
            +
                return NextResponse.json(
         | 
| 57 | 
            +
                  { message: "Title and HTML content are required.", ok: false },
         | 
| 58 | 
            +
                  { status: 400 }
         | 
| 59 | 
            +
                );
         | 
| 60 | 
            +
              }
         | 
| 61 | 
            +
             | 
| 62 | 
            +
              await dbConnect();
         | 
| 63 | 
            +
             | 
| 64 | 
            +
              try {
         | 
| 65 | 
            +
                let readme = "";
         | 
| 66 | 
            +
                let newHtml = html;
         | 
| 67 | 
            +
             | 
| 68 | 
            +
                const newTitle = title
         | 
| 69 | 
            +
                  .toLowerCase()
         | 
| 70 | 
            +
                  .replace(/[^a-z0-9]+/g, "-")
         | 
| 71 | 
            +
                  .split("-")
         | 
| 72 | 
            +
                  .filter(Boolean)
         | 
| 73 | 
            +
                  .join("-")
         | 
| 74 | 
            +
                  .slice(0, 96);
         | 
| 75 | 
            +
             | 
| 76 | 
            +
                const repo: RepoDesignation = {
         | 
| 77 | 
            +
                  type: "space",
         | 
| 78 | 
            +
                  name: `${user.name}/${newTitle}`,
         | 
| 79 | 
            +
                };
         | 
| 80 | 
            +
             | 
| 81 | 
            +
                const { repoUrl } = await createRepo({
         | 
| 82 | 
            +
                  repo,
         | 
| 83 | 
            +
                  accessToken: user.token as string,
         | 
| 84 | 
            +
                });
         | 
| 85 | 
            +
                const colorFrom = COLORS[Math.floor(Math.random() * COLORS.length)];
         | 
| 86 | 
            +
                const colorTo = COLORS[Math.floor(Math.random() * COLORS.length)];
         | 
| 87 | 
            +
                readme = `---
         | 
| 88 | 
            +
            title: ${newTitle}
         | 
| 89 | 
            +
            emoji: 🐳
         | 
| 90 | 
            +
            colorFrom: ${colorFrom}
         | 
| 91 | 
            +
            colorTo: ${colorTo}
         | 
| 92 | 
            +
            sdk: static
         | 
| 93 | 
            +
            pinned: false
         | 
| 94 | 
            +
            tags:
         | 
| 95 | 
            +
              - deepsite
         | 
| 96 | 
            +
            ---
         | 
| 97 | 
            +
             | 
| 98 | 
            +
            Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference`;
         | 
| 99 | 
            +
             | 
| 100 | 
            +
                newHtml = html.replace(/<\/body>/, `${getPTag(repo.name)}</body>`);
         | 
| 101 | 
            +
                const file = new File([newHtml], "index.html", { type: "text/html" });
         | 
| 102 | 
            +
                const readmeFile = new File([readme], "README.md", {
         | 
| 103 | 
            +
                  type: "text/markdown",
         | 
| 104 | 
            +
                });
         | 
| 105 | 
            +
                const files = [file, readmeFile];
         | 
| 106 | 
            +
                await uploadFiles({
         | 
| 107 | 
            +
                  repo,
         | 
| 108 | 
            +
                  files,
         | 
| 109 | 
            +
                  accessToken: user.token as string,
         | 
| 110 | 
            +
                  commitTitle: `${prompts[prompts.length - 1]} - Initial Deployment`,
         | 
| 111 | 
            +
                });
         | 
| 112 | 
            +
                const path = repoUrl.split("/").slice(-2).join("/");
         | 
| 113 | 
            +
                const project = await Project.create({
         | 
| 114 | 
            +
                  user_id: user.id,
         | 
| 115 | 
            +
                  space_id: path,
         | 
| 116 | 
            +
                  prompts,
         | 
| 117 | 
            +
                });
         | 
| 118 | 
            +
                return NextResponse.json({ project, path, ok: true }, { status: 201 });
         | 
| 119 | 
            +
                // eslint-disable-next-line @typescript-eslint/no-explicit-any
         | 
| 120 | 
            +
              } catch (err: any) {
         | 
| 121 | 
            +
                return NextResponse.json(
         | 
| 122 | 
            +
                  { error: err.message, ok: false },
         | 
| 123 | 
            +
                  { status: 500 }
         | 
| 124 | 
            +
                );
         | 
| 125 | 
            +
              }
         | 
| 126 | 
            +
            }
         | 
    	
        app/api/me/route.ts
    ADDED
    
    | @@ -0,0 +1,25 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { headers } from "next/headers";
         | 
| 2 | 
            +
            import { NextResponse } from "next/server";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            export async function GET() {
         | 
| 5 | 
            +
              const authHeaders = await headers();
         | 
| 6 | 
            +
              const token = authHeaders.get("Authorization");
         | 
| 7 | 
            +
              if (!token) {
         | 
| 8 | 
            +
                return NextResponse.json({ user: null, errCode: 401 }, { status: 401 });
         | 
| 9 | 
            +
              }
         | 
| 10 | 
            +
             | 
| 11 | 
            +
              const userResponse = await fetch("https://huggingface.co/api/whoami-v2", {
         | 
| 12 | 
            +
                headers: {
         | 
| 13 | 
            +
                  Authorization: `${token}`,
         | 
| 14 | 
            +
                },
         | 
| 15 | 
            +
              });
         | 
| 16 | 
            +
             | 
| 17 | 
            +
              if (!userResponse.ok) {
         | 
| 18 | 
            +
                return NextResponse.json(
         | 
| 19 | 
            +
                  { user: null, errCode: userResponse.status },
         | 
| 20 | 
            +
                  { status: userResponse.status }
         | 
| 21 | 
            +
                );
         | 
| 22 | 
            +
              }
         | 
| 23 | 
            +
              const user = await userResponse.json();
         | 
| 24 | 
            +
              return NextResponse.json({ user, errCode: null }, { status: 200 });
         | 
| 25 | 
            +
            }
         | 
    	
        app/auth/callback/page.tsx
    ADDED
    
    | @@ -0,0 +1,27 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { use } from "react";
         | 
| 4 | 
            +
            import { useBroadcastChannel } from "@/lib/useBroadcastChannel";
         | 
| 5 | 
            +
            import { useMount } from "react-use";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            export default function AuthCallback({
         | 
| 8 | 
            +
              searchParams,
         | 
| 9 | 
            +
            }: {
         | 
| 10 | 
            +
              searchParams: Promise<{ code: string }>;
         | 
| 11 | 
            +
            }) {
         | 
| 12 | 
            +
              const { code } = use(searchParams);
         | 
| 13 | 
            +
             | 
| 14 | 
            +
              const { postMessage } = useBroadcastChannel("auth", () => {});
         | 
| 15 | 
            +
              useMount(() => {
         | 
| 16 | 
            +
                if (code) {
         | 
| 17 | 
            +
                  postMessage({
         | 
| 18 | 
            +
                    code: code,
         | 
| 19 | 
            +
                    type: "user-oauth",
         | 
| 20 | 
            +
                  });
         | 
| 21 | 
            +
                  window.close();
         | 
| 22 | 
            +
                  return;
         | 
| 23 | 
            +
                }
         | 
| 24 | 
            +
              });
         | 
| 25 | 
            +
             | 
| 26 | 
            +
              return <div>Login in progress ...</div>;
         | 
| 27 | 
            +
            }
         | 
    	
        app/auth/page.tsx
    ADDED
    
    | @@ -0,0 +1,36 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { redirect } from "next/navigation";
         | 
| 2 | 
            +
            import { Metadata } from "next";
         | 
| 3 | 
            +
            import { apiServer } from "@/lib/api";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            async function getLoginUrl() {
         | 
| 6 | 
            +
              try {
         | 
| 7 | 
            +
                const res = await apiServer.get("/auth");
         | 
| 8 | 
            +
                return res.data;
         | 
| 9 | 
            +
              } catch (error) {
         | 
| 10 | 
            +
                return error;
         | 
| 11 | 
            +
              }
         | 
| 12 | 
            +
            }
         | 
| 13 | 
            +
             | 
| 14 | 
            +
            export const revalidate = 1;
         | 
| 15 | 
            +
             | 
| 16 | 
            +
            export const metadata: Metadata = {
         | 
| 17 | 
            +
              robots: "noindex, nofollow",
         | 
| 18 | 
            +
            };
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            export default async function Auth() {
         | 
| 21 | 
            +
              const login = await getLoginUrl();
         | 
| 22 | 
            +
              if (login?.redirect) {
         | 
| 23 | 
            +
                redirect(login.redirect);
         | 
| 24 | 
            +
              }
         | 
| 25 | 
            +
             | 
| 26 | 
            +
              return (
         | 
| 27 | 
            +
                <div className="p-4">
         | 
| 28 | 
            +
                  <div className="border bg-red-500/10 border-red-500/20 text-red-500 px-5 py-3 rounded-lg">
         | 
| 29 | 
            +
                    <h1 className="text-xl font-bold">Error</h1>
         | 
| 30 | 
            +
                    <p className="text-sm">
         | 
| 31 | 
            +
                      An error occurred while trying to log in. Please try again later.
         | 
| 32 | 
            +
                    </p>
         | 
| 33 | 
            +
                  </div>
         | 
| 34 | 
            +
                </div>
         | 
| 35 | 
            +
              );
         | 
| 36 | 
            +
            }
         | 
    	
        app/globals.css
    DELETED
    
    | @@ -1,26 +0,0 @@ | |
| 1 | 
            -
            @import "tailwindcss";
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            :root {
         | 
| 4 | 
            -
              --background: #ffffff;
         | 
| 5 | 
            -
              --foreground: #171717;
         | 
| 6 | 
            -
            }
         | 
| 7 | 
            -
             | 
| 8 | 
            -
            @theme inline {
         | 
| 9 | 
            -
              --color-background: var(--background);
         | 
| 10 | 
            -
              --color-foreground: var(--foreground);
         | 
| 11 | 
            -
              --font-sans: var(--font-geist-sans);
         | 
| 12 | 
            -
              --font-mono: var(--font-geist-mono);
         | 
| 13 | 
            -
            }
         | 
| 14 | 
            -
             | 
| 15 | 
            -
            @media (prefers-color-scheme: dark) {
         | 
| 16 | 
            -
              :root {
         | 
| 17 | 
            -
                --background: #0a0a0a;
         | 
| 18 | 
            -
                --foreground: #ededed;
         | 
| 19 | 
            -
              }
         | 
| 20 | 
            -
            }
         | 
| 21 | 
            -
             | 
| 22 | 
            -
            body {
         | 
| 23 | 
            -
              background: var(--background);
         | 
| 24 | 
            -
              color: var(--foreground);
         | 
| 25 | 
            -
              font-family: Arial, Helvetica, sans-serif;
         | 
| 26 | 
            -
            }
         | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
    	
        app/layout.tsx
    CHANGED
    
    | @@ -1,33 +1,102 @@ | |
| 1 | 
            -
             | 
| 2 | 
            -
            import {  | 
| 3 | 
            -
            import " | 
|  | |
| 4 |  | 
| 5 | 
            -
             | 
| 6 | 
            -
             | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
| 7 | 
             
              subsets: ["latin"],
         | 
| 8 | 
             
            });
         | 
| 9 |  | 
| 10 | 
            -
            const  | 
| 11 | 
            -
              variable: "--font- | 
| 12 | 
             
              subsets: ["latin"],
         | 
|  | |
| 13 | 
             
            });
         | 
| 14 |  | 
| 15 | 
             
            export const metadata: Metadata = {
         | 
| 16 | 
            -
              title: " | 
| 17 | 
            -
              description: | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
| 18 | 
             
            };
         | 
| 19 |  | 
| 20 | 
            -
             | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
| 21 | 
             
              children,
         | 
| 22 | 
             
            }: Readonly<{
         | 
| 23 | 
             
              children: React.ReactNode;
         | 
| 24 | 
             
            }>) {
         | 
|  | |
| 25 | 
             
              return (
         | 
| 26 | 
             
                <html lang="en">
         | 
| 27 | 
             
                  <body
         | 
| 28 | 
            -
                    className={`${ | 
| 29 | 
             
                  >
         | 
| 30 | 
            -
                     | 
|  | |
|  | |
|  | |
| 31 | 
             
                  </body>
         | 
| 32 | 
             
                </html>
         | 
| 33 | 
             
              );
         | 
|  | |
| 1 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 2 | 
            +
            import type { Metadata, Viewport } from "next";
         | 
| 3 | 
            +
            import { Inter, PT_Sans } from "next/font/google";
         | 
| 4 | 
            +
            import { cookies } from "next/headers";
         | 
| 5 |  | 
| 6 | 
            +
            import TanstackProvider from "@/components/providers/tanstack-query-provider";
         | 
| 7 | 
            +
            import "@/assets/globals.css";
         | 
| 8 | 
            +
            import { Toaster } from "@/components/ui/sonner";
         | 
| 9 | 
            +
            import MY_TOKEN_KEY from "@/lib/get-cookie-name";
         | 
| 10 | 
            +
            import { apiServer } from "@/lib/api";
         | 
| 11 | 
            +
            import AppContext from "@/components/contexts/app-context";
         | 
| 12 | 
            +
             | 
| 13 | 
            +
            const inter = Inter({
         | 
| 14 | 
            +
              variable: "--font-inter-sans",
         | 
| 15 | 
             
              subsets: ["latin"],
         | 
| 16 | 
             
            });
         | 
| 17 |  | 
| 18 | 
            +
            const ptSans = PT_Sans({
         | 
| 19 | 
            +
              variable: "--font-ptSans-mono",
         | 
| 20 | 
             
              subsets: ["latin"],
         | 
| 21 | 
            +
              weight: ["400", "700"],
         | 
| 22 | 
             
            });
         | 
| 23 |  | 
| 24 | 
             
            export const metadata: Metadata = {
         | 
| 25 | 
            +
              title: "DeepSite | Build with AI ✨",
         | 
| 26 | 
            +
              description:
         | 
| 27 | 
            +
                "DeepSite is a web development tool that helps you build websites with AI, no code required. Let's deploy your website with DeepSite and enjoy the magic of AI.",
         | 
| 28 | 
            +
              openGraph: {
         | 
| 29 | 
            +
                title: "DeepSite | Build with AI ✨",
         | 
| 30 | 
            +
                description:
         | 
| 31 | 
            +
                  "DeepSite is a web development tool that helps you build websites with AI, no code required. Let's deploy your website with DeepSite and enjoy the magic of AI.",
         | 
| 32 | 
            +
                url: "https://deepsite.hf.co",
         | 
| 33 | 
            +
                siteName: "DeepSite",
         | 
| 34 | 
            +
                images: [
         | 
| 35 | 
            +
                  {
         | 
| 36 | 
            +
                    url: "https://deepsite.hf.co/banner.png",
         | 
| 37 | 
            +
                    width: 1200,
         | 
| 38 | 
            +
                    height: 630,
         | 
| 39 | 
            +
                    alt: "DeepSite Open Graph Image",
         | 
| 40 | 
            +
                  },
         | 
| 41 | 
            +
                ],
         | 
| 42 | 
            +
              },
         | 
| 43 | 
            +
              twitter: {
         | 
| 44 | 
            +
                card: "summary_large_image",
         | 
| 45 | 
            +
                title: "DeepSite | Build with AI ✨",
         | 
| 46 | 
            +
                description:
         | 
| 47 | 
            +
                  "DeepSite is a web development tool that helps you build websites with AI, no code required. Let's deploy your website with DeepSite and enjoy the magic of AI.",
         | 
| 48 | 
            +
                images: ["https://deepsite.hf.co/banner.png"],
         | 
| 49 | 
            +
              },
         | 
| 50 | 
            +
              appleWebApp: {
         | 
| 51 | 
            +
                capable: true,
         | 
| 52 | 
            +
                title: "DeepSite",
         | 
| 53 | 
            +
                statusBarStyle: "black-translucent",
         | 
| 54 | 
            +
              },
         | 
| 55 | 
            +
              themeColor: "#000000",
         | 
| 56 | 
            +
              icons: {
         | 
| 57 | 
            +
                icon: "/logo.svg",
         | 
| 58 | 
            +
                shortcut: "/logo.svg",
         | 
| 59 | 
            +
                apple: "/logo.svg",
         | 
| 60 | 
            +
              },
         | 
| 61 | 
            +
            };
         | 
| 62 | 
            +
             | 
| 63 | 
            +
            export const viewport: Viewport = {
         | 
| 64 | 
            +
              initialScale: 1,
         | 
| 65 | 
            +
              maximumScale: 1,
         | 
| 66 | 
            +
              themeColor: "#000000",
         | 
| 67 | 
             
            };
         | 
| 68 |  | 
| 69 | 
            +
            async function getMe() {
         | 
| 70 | 
            +
              const cookieStore = await cookies();
         | 
| 71 | 
            +
              const token = cookieStore.get(MY_TOKEN_KEY())?.value;
         | 
| 72 | 
            +
              if (!token) return { user: null, errCode: null };
         | 
| 73 | 
            +
              try {
         | 
| 74 | 
            +
                const res = await apiServer.get("/me", {
         | 
| 75 | 
            +
                  headers: {
         | 
| 76 | 
            +
                    Authorization: `Bearer ${token}`,
         | 
| 77 | 
            +
                  },
         | 
| 78 | 
            +
                });
         | 
| 79 | 
            +
                return { user: res.data.user, errCode: null };
         | 
| 80 | 
            +
              } catch (err: any) {
         | 
| 81 | 
            +
                return { user: null, errCode: err.status };
         | 
| 82 | 
            +
              }
         | 
| 83 | 
            +
            }
         | 
| 84 | 
            +
             | 
| 85 | 
            +
            export default async function RootLayout({
         | 
| 86 | 
             
              children,
         | 
| 87 | 
             
            }: Readonly<{
         | 
| 88 | 
             
              children: React.ReactNode;
         | 
| 89 | 
             
            }>) {
         | 
| 90 | 
            +
              const data = await getMe();
         | 
| 91 | 
             
              return (
         | 
| 92 | 
             
                <html lang="en">
         | 
| 93 | 
             
                  <body
         | 
| 94 | 
            +
                    className={`${inter.variable} ${ptSans.variable} antialiased bg-black dark`}
         | 
| 95 | 
             
                  >
         | 
| 96 | 
            +
                    <Toaster richColors position="bottom-center" />
         | 
| 97 | 
            +
                    <TanstackProvider>
         | 
| 98 | 
            +
                      <AppContext me={data}>{children}</AppContext>
         | 
| 99 | 
            +
                    </TanstackProvider>
         | 
| 100 | 
             
                  </body>
         | 
| 101 | 
             
                </html>
         | 
| 102 | 
             
              );
         | 
    	
        app/page.tsx
    DELETED
    
    | @@ -1,103 +0,0 @@ | |
| 1 | 
            -
            import Image from "next/image";
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            export default function Home() {
         | 
| 4 | 
            -
              return (
         | 
| 5 | 
            -
                <div className="grid grid-rows-[20px_1fr_20px] items-center justify-items-center min-h-screen p-8 pb-20 gap-16 sm:p-20 font-[family-name:var(--font-geist-sans)]">
         | 
| 6 | 
            -
                  <main className="flex flex-col gap-[32px] row-start-2 items-center sm:items-start">
         | 
| 7 | 
            -
                    <Image
         | 
| 8 | 
            -
                      className="dark:invert"
         | 
| 9 | 
            -
                      src="/next.svg"
         | 
| 10 | 
            -
                      alt="Next.js logo"
         | 
| 11 | 
            -
                      width={180}
         | 
| 12 | 
            -
                      height={38}
         | 
| 13 | 
            -
                      priority
         | 
| 14 | 
            -
                    />
         | 
| 15 | 
            -
                    <ol className="list-inside list-decimal text-sm/6 text-center sm:text-left font-[family-name:var(--font-geist-mono)]">
         | 
| 16 | 
            -
                      <li className="mb-2 tracking-[-.01em]">
         | 
| 17 | 
            -
                        Get started by editing{" "}
         | 
| 18 | 
            -
                        <code className="bg-black/[.05] dark:bg-white/[.06] px-1 py-0.5 rounded font-[family-name:var(--font-geist-mono)] font-semibold">
         | 
| 19 | 
            -
                          app/page.tsx
         | 
| 20 | 
            -
                        </code>
         | 
| 21 | 
            -
                        .
         | 
| 22 | 
            -
                      </li>
         | 
| 23 | 
            -
                      <li className="tracking-[-.01em]">
         | 
| 24 | 
            -
                        Save and see your changes instantly.
         | 
| 25 | 
            -
                      </li>
         | 
| 26 | 
            -
                    </ol>
         | 
| 27 | 
            -
             | 
| 28 | 
            -
                    <div className="flex gap-4 items-center flex-col sm:flex-row">
         | 
| 29 | 
            -
                      <a
         | 
| 30 | 
            -
                        className="rounded-full border border-solid border-transparent transition-colors flex items-center justify-center bg-foreground text-background gap-2 hover:bg-[#383838] dark:hover:bg-[#ccc] font-medium text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5 sm:w-auto"
         | 
| 31 | 
            -
                        href="https://vercel.com/new?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
         | 
| 32 | 
            -
                        target="_blank"
         | 
| 33 | 
            -
                        rel="noopener noreferrer"
         | 
| 34 | 
            -
                      >
         | 
| 35 | 
            -
                        <Image
         | 
| 36 | 
            -
                          className="dark:invert"
         | 
| 37 | 
            -
                          src="/vercel.svg"
         | 
| 38 | 
            -
                          alt="Vercel logomark"
         | 
| 39 | 
            -
                          width={20}
         | 
| 40 | 
            -
                          height={20}
         | 
| 41 | 
            -
                        />
         | 
| 42 | 
            -
                        Deploy now
         | 
| 43 | 
            -
                      </a>
         | 
| 44 | 
            -
                      <a
         | 
| 45 | 
            -
                        className="rounded-full border border-solid border-black/[.08] dark:border-white/[.145] transition-colors flex items-center justify-center hover:bg-[#f2f2f2] dark:hover:bg-[#1a1a1a] hover:border-transparent font-medium text-sm sm:text-base h-10 sm:h-12 px-4 sm:px-5 w-full sm:w-auto md:w-[158px]"
         | 
| 46 | 
            -
                        href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
         | 
| 47 | 
            -
                        target="_blank"
         | 
| 48 | 
            -
                        rel="noopener noreferrer"
         | 
| 49 | 
            -
                      >
         | 
| 50 | 
            -
                        Read our docs
         | 
| 51 | 
            -
                      </a>
         | 
| 52 | 
            -
                    </div>
         | 
| 53 | 
            -
                  </main>
         | 
| 54 | 
            -
                  <footer className="row-start-3 flex gap-[24px] flex-wrap items-center justify-center">
         | 
| 55 | 
            -
                    <a
         | 
| 56 | 
            -
                      className="flex items-center gap-2 hover:underline hover:underline-offset-4"
         | 
| 57 | 
            -
                      href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
         | 
| 58 | 
            -
                      target="_blank"
         | 
| 59 | 
            -
                      rel="noopener noreferrer"
         | 
| 60 | 
            -
                    >
         | 
| 61 | 
            -
                      <Image
         | 
| 62 | 
            -
                        aria-hidden
         | 
| 63 | 
            -
                        src="/file.svg"
         | 
| 64 | 
            -
                        alt="File icon"
         | 
| 65 | 
            -
                        width={16}
         | 
| 66 | 
            -
                        height={16}
         | 
| 67 | 
            -
                      />
         | 
| 68 | 
            -
                      Learn
         | 
| 69 | 
            -
                    </a>
         | 
| 70 | 
            -
                    <a
         | 
| 71 | 
            -
                      className="flex items-center gap-2 hover:underline hover:underline-offset-4"
         | 
| 72 | 
            -
                      href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
         | 
| 73 | 
            -
                      target="_blank"
         | 
| 74 | 
            -
                      rel="noopener noreferrer"
         | 
| 75 | 
            -
                    >
         | 
| 76 | 
            -
                      <Image
         | 
| 77 | 
            -
                        aria-hidden
         | 
| 78 | 
            -
                        src="/window.svg"
         | 
| 79 | 
            -
                        alt="Window icon"
         | 
| 80 | 
            -
                        width={16}
         | 
| 81 | 
            -
                        height={16}
         | 
| 82 | 
            -
                      />
         | 
| 83 | 
            -
                      Examples
         | 
| 84 | 
            -
                    </a>
         | 
| 85 | 
            -
                    <a
         | 
| 86 | 
            -
                      className="flex items-center gap-2 hover:underline hover:underline-offset-4"
         | 
| 87 | 
            -
                      href="https://nextjs.org?utm_source=create-next-app&utm_medium=appdir-template-tw&utm_campaign=create-next-app"
         | 
| 88 | 
            -
                      target="_blank"
         | 
| 89 | 
            -
                      rel="noopener noreferrer"
         | 
| 90 | 
            -
                    >
         | 
| 91 | 
            -
                      <Image
         | 
| 92 | 
            -
                        aria-hidden
         | 
| 93 | 
            -
                        src="/globe.svg"
         | 
| 94 | 
            -
                        alt="Globe icon"
         | 
| 95 | 
            -
                        width={16}
         | 
| 96 | 
            -
                        height={16}
         | 
| 97 | 
            -
                      />
         | 
| 98 | 
            -
                      Go to nextjs.org →
         | 
| 99 | 
            -
                    </a>
         | 
| 100 | 
            -
                  </footer>
         | 
| 101 | 
            -
                </div>
         | 
| 102 | 
            -
              );
         | 
| 103 | 
            -
            }
         | 
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
    	
        app/projects/[namespace]/[repoId]/page.tsx
    ADDED
    
    | @@ -0,0 +1,39 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { cookies } from "next/headers";
         | 
| 2 | 
            +
            import { redirect } from "next/navigation";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import { apiServer } from "@/lib/api";
         | 
| 5 | 
            +
            import MY_TOKEN_KEY from "@/lib/get-cookie-name";
         | 
| 6 | 
            +
            import { AppEditor } from "@/components/editor";
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            async function getProject(namespace: string, repoId: string) {
         | 
| 9 | 
            +
              const cookieStore = await cookies();
         | 
| 10 | 
            +
              const token = cookieStore.get(MY_TOKEN_KEY())?.value;
         | 
| 11 | 
            +
              if (!token) return {};
         | 
| 12 | 
            +
              try {
         | 
| 13 | 
            +
                const { data } = await apiServer.get(
         | 
| 14 | 
            +
                  `/me/projects/${namespace}/${repoId}`,
         | 
| 15 | 
            +
                  {
         | 
| 16 | 
            +
                    headers: {
         | 
| 17 | 
            +
                      Authorization: `Bearer ${token}`,
         | 
| 18 | 
            +
                    },
         | 
| 19 | 
            +
                  }
         | 
| 20 | 
            +
                );
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                return data.project;
         | 
| 23 | 
            +
              } catch {
         | 
| 24 | 
            +
                return {};
         | 
| 25 | 
            +
              }
         | 
| 26 | 
            +
            }
         | 
| 27 | 
            +
             | 
| 28 | 
            +
            export default async function ProjectNamespacePage({
         | 
| 29 | 
            +
              params,
         | 
| 30 | 
            +
            }: {
         | 
| 31 | 
            +
              params: Promise<{ namespace: string; repoId: string }>;
         | 
| 32 | 
            +
            }) {
         | 
| 33 | 
            +
              const { namespace, repoId } = await params;
         | 
| 34 | 
            +
              const project = await getProject(namespace, repoId);
         | 
| 35 | 
            +
              if (!project?.html) {
         | 
| 36 | 
            +
                redirect("/projects");
         | 
| 37 | 
            +
              }
         | 
| 38 | 
            +
              return <AppEditor project={project} />;
         | 
| 39 | 
            +
            }
         | 
    	
        app/projects/new/page.tsx
    ADDED
    
    | @@ -0,0 +1,5 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { AppEditor } from "@/components/editor";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            export default function ProjectsNewPage() {
         | 
| 4 | 
            +
              return <AppEditor />;
         | 
| 5 | 
            +
            }
         | 
    	
        assets/globals.css
    ADDED
    
    | @@ -0,0 +1,146 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            @import "tailwindcss";
         | 
| 2 | 
            +
            @import "tw-animate-css";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            @custom-variant dark (&:is(.dark *));
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            @theme inline {
         | 
| 7 | 
            +
              --color-background: var(--background);
         | 
| 8 | 
            +
              --color-foreground: var(--foreground);
         | 
| 9 | 
            +
              --font-sans: var(--font-inter-sans);
         | 
| 10 | 
            +
              --font-mono: var(--font-ptSans-mono);
         | 
| 11 | 
            +
              --color-sidebar-ring: var(--sidebar-ring);
         | 
| 12 | 
            +
              --color-sidebar-border: var(--sidebar-border);
         | 
| 13 | 
            +
              --color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
         | 
| 14 | 
            +
              --color-sidebar-accent: var(--sidebar-accent);
         | 
| 15 | 
            +
              --color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
         | 
| 16 | 
            +
              --color-sidebar-primary: var(--sidebar-primary);
         | 
| 17 | 
            +
              --color-sidebar-foreground: var(--sidebar-foreground);
         | 
| 18 | 
            +
              --color-sidebar: var(--sidebar);
         | 
| 19 | 
            +
              --color-chart-5: var(--chart-5);
         | 
| 20 | 
            +
              --color-chart-4: var(--chart-4);
         | 
| 21 | 
            +
              --color-chart-3: var(--chart-3);
         | 
| 22 | 
            +
              --color-chart-2: var(--chart-2);
         | 
| 23 | 
            +
              --color-chart-1: var(--chart-1);
         | 
| 24 | 
            +
              --color-ring: var(--ring);
         | 
| 25 | 
            +
              --color-input: var(--input);
         | 
| 26 | 
            +
              --color-border: var(--border);
         | 
| 27 | 
            +
              --color-destructive: var(--destructive);
         | 
| 28 | 
            +
              --color-accent-foreground: var(--accent-foreground);
         | 
| 29 | 
            +
              --color-accent: var(--accent);
         | 
| 30 | 
            +
              --color-muted-foreground: var(--muted-foreground);
         | 
| 31 | 
            +
              --color-muted: var(--muted);
         | 
| 32 | 
            +
              --color-secondary-foreground: var(--secondary-foreground);
         | 
| 33 | 
            +
              --color-secondary: var(--secondary);
         | 
| 34 | 
            +
              --color-primary-foreground: var(--primary-foreground);
         | 
| 35 | 
            +
              --color-primary: var(--primary);
         | 
| 36 | 
            +
              --color-popover-foreground: var(--popover-foreground);
         | 
| 37 | 
            +
              --color-popover: var(--popover);
         | 
| 38 | 
            +
              --color-card-foreground: var(--card-foreground);
         | 
| 39 | 
            +
              --color-card: var(--card);
         | 
| 40 | 
            +
              --radius-sm: calc(var(--radius) - 4px);
         | 
| 41 | 
            +
              --radius-md: calc(var(--radius) - 2px);
         | 
| 42 | 
            +
              --radius-lg: var(--radius);
         | 
| 43 | 
            +
              --radius-xl: calc(var(--radius) + 4px);
         | 
| 44 | 
            +
            }
         | 
| 45 | 
            +
             | 
| 46 | 
            +
            :root {
         | 
| 47 | 
            +
              --radius: 0.625rem;
         | 
| 48 | 
            +
              --background: oklch(1 0 0);
         | 
| 49 | 
            +
              --foreground: oklch(0.145 0 0);
         | 
| 50 | 
            +
              --card: oklch(1 0 0);
         | 
| 51 | 
            +
              --card-foreground: oklch(0.145 0 0);
         | 
| 52 | 
            +
              --popover: oklch(1 0 0);
         | 
| 53 | 
            +
              --popover-foreground: oklch(0.145 0 0);
         | 
| 54 | 
            +
              --primary: oklch(0.205 0 0);
         | 
| 55 | 
            +
              --primary-foreground: oklch(0.985 0 0);
         | 
| 56 | 
            +
              --secondary: oklch(0.97 0 0);
         | 
| 57 | 
            +
              --secondary-foreground: oklch(0.205 0 0);
         | 
| 58 | 
            +
              --muted: oklch(0.97 0 0);
         | 
| 59 | 
            +
              --muted-foreground: oklch(0.556 0 0);
         | 
| 60 | 
            +
              --accent: oklch(0.97 0 0);
         | 
| 61 | 
            +
              --accent-foreground: oklch(0.205 0 0);
         | 
| 62 | 
            +
              --destructive: oklch(0.577 0.245 27.325);
         | 
| 63 | 
            +
              --border: oklch(0.922 0 0);
         | 
| 64 | 
            +
              --input: oklch(0.922 0 0);
         | 
| 65 | 
            +
              --ring: oklch(0.708 0 0);
         | 
| 66 | 
            +
              --chart-1: oklch(0.646 0.222 41.116);
         | 
| 67 | 
            +
              --chart-2: oklch(0.6 0.118 184.704);
         | 
| 68 | 
            +
              --chart-3: oklch(0.398 0.07 227.392);
         | 
| 69 | 
            +
              --chart-4: oklch(0.828 0.189 84.429);
         | 
| 70 | 
            +
              --chart-5: oklch(0.769 0.188 70.08);
         | 
| 71 | 
            +
              --sidebar: oklch(0.985 0 0);
         | 
| 72 | 
            +
              --sidebar-foreground: oklch(0.145 0 0);
         | 
| 73 | 
            +
              --sidebar-primary: oklch(0.205 0 0);
         | 
| 74 | 
            +
              --sidebar-primary-foreground: oklch(0.985 0 0);
         | 
| 75 | 
            +
              --sidebar-accent: oklch(0.97 0 0);
         | 
| 76 | 
            +
              --sidebar-accent-foreground: oklch(0.205 0 0);
         | 
| 77 | 
            +
              --sidebar-border: oklch(0.922 0 0);
         | 
| 78 | 
            +
              --sidebar-ring: oklch(0.708 0 0);
         | 
| 79 | 
            +
            }
         | 
| 80 | 
            +
             | 
| 81 | 
            +
            .dark {
         | 
| 82 | 
            +
              --background: oklch(0.145 0 0);
         | 
| 83 | 
            +
              --foreground: oklch(0.985 0 0);
         | 
| 84 | 
            +
              --card: oklch(0.205 0 0);
         | 
| 85 | 
            +
              --card-foreground: oklch(0.985 0 0);
         | 
| 86 | 
            +
              --popover: oklch(0.205 0 0);
         | 
| 87 | 
            +
              --popover-foreground: oklch(0.985 0 0);
         | 
| 88 | 
            +
              --primary: oklch(0.922 0 0);
         | 
| 89 | 
            +
              --primary-foreground: oklch(0.205 0 0);
         | 
| 90 | 
            +
              --secondary: oklch(0.269 0 0);
         | 
| 91 | 
            +
              --secondary-foreground: oklch(0.985 0 0);
         | 
| 92 | 
            +
              --muted: oklch(0.269 0 0);
         | 
| 93 | 
            +
              --muted-foreground: oklch(0.708 0 0);
         | 
| 94 | 
            +
              --accent: oklch(0.269 0 0);
         | 
| 95 | 
            +
              --accent-foreground: oklch(0.985 0 0);
         | 
| 96 | 
            +
              --destructive: oklch(0.704 0.191 22.216);
         | 
| 97 | 
            +
              --border: oklch(1 0 0 / 10%);
         | 
| 98 | 
            +
              --input: oklch(1 0 0 / 15%);
         | 
| 99 | 
            +
              --ring: oklch(0.556 0 0);
         | 
| 100 | 
            +
              --chart-1: oklch(0.488 0.243 264.376);
         | 
| 101 | 
            +
              --chart-2: oklch(0.696 0.17 162.48);
         | 
| 102 | 
            +
              --chart-3: oklch(0.769 0.188 70.08);
         | 
| 103 | 
            +
              --chart-4: oklch(0.627 0.265 303.9);
         | 
| 104 | 
            +
              --chart-5: oklch(0.645 0.246 16.439);
         | 
| 105 | 
            +
              --sidebar: oklch(0.205 0 0);
         | 
| 106 | 
            +
              --sidebar-foreground: oklch(0.985 0 0);
         | 
| 107 | 
            +
              --sidebar-primary: oklch(0.488 0.243 264.376);
         | 
| 108 | 
            +
              --sidebar-primary-foreground: oklch(0.985 0 0);
         | 
| 109 | 
            +
              --sidebar-accent: oklch(0.269 0 0);
         | 
| 110 | 
            +
              --sidebar-accent-foreground: oklch(0.985 0 0);
         | 
| 111 | 
            +
              --sidebar-border: oklch(1 0 0 / 10%);
         | 
| 112 | 
            +
              --sidebar-ring: oklch(0.556 0 0);
         | 
| 113 | 
            +
            }
         | 
| 114 | 
            +
             | 
| 115 | 
            +
            @layer base {
         | 
| 116 | 
            +
              * {
         | 
| 117 | 
            +
                @apply border-border outline-ring/50;
         | 
| 118 | 
            +
              }
         | 
| 119 | 
            +
              body {
         | 
| 120 | 
            +
                @apply bg-background text-foreground;
         | 
| 121 | 
            +
              }
         | 
| 122 | 
            +
              html {
         | 
| 123 | 
            +
                @apply scroll-smooth;
         | 
| 124 | 
            +
              }
         | 
| 125 | 
            +
            }
         | 
| 126 | 
            +
             | 
| 127 | 
            +
            .background__noisy {
         | 
| 128 | 
            +
              @apply bg-blend-normal pointer-events-none opacity-90;
         | 
| 129 | 
            +
              background-size: 25ww auto;
         | 
| 130 | 
            +
              background-image: url("/background_noisy.webp");
         | 
| 131 | 
            +
              @apply fixed w-screen h-screen -z-1 top-0 left-0;
         | 
| 132 | 
            +
            }
         | 
| 133 | 
            +
             | 
| 134 | 
            +
            .monaco-editor .margin {
         | 
| 135 | 
            +
              @apply !bg-neutral-900;
         | 
| 136 | 
            +
            }
         | 
| 137 | 
            +
            .monaco-editor .monaco-editor-background {
         | 
| 138 | 
            +
              @apply !bg-neutral-900;
         | 
| 139 | 
            +
            }
         | 
| 140 | 
            +
            .monaco-editor .line-numbers {
         | 
| 141 | 
            +
              @apply !text-neutral-500;
         | 
| 142 | 
            +
            }
         | 
| 143 | 
            +
             | 
| 144 | 
            +
            .matched-line {
         | 
| 145 | 
            +
              @apply bg-sky-500/30;
         | 
| 146 | 
            +
            }
         | 
    	
        assets/logo.svg
    ADDED
    
    |  | 
    	
        assets/space.svg
    ADDED
    
    |  | 
    	
        components.json
    ADDED
    
    | @@ -0,0 +1,21 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            {
         | 
| 2 | 
            +
              "$schema": "https://ui.shadcn.com/schema.json",
         | 
| 3 | 
            +
              "style": "new-york",
         | 
| 4 | 
            +
              "rsc": true,
         | 
| 5 | 
            +
              "tsx": true,
         | 
| 6 | 
            +
              "tailwind": {
         | 
| 7 | 
            +
                "config": "",
         | 
| 8 | 
            +
                "css": "app/globals.css",
         | 
| 9 | 
            +
                "baseColor": "neutral",
         | 
| 10 | 
            +
                "cssVariables": true,
         | 
| 11 | 
            +
                "prefix": ""
         | 
| 12 | 
            +
              },
         | 
| 13 | 
            +
              "aliases": {
         | 
| 14 | 
            +
                "components": "@/components",
         | 
| 15 | 
            +
                "utils": "@/lib/utils",
         | 
| 16 | 
            +
                "ui": "@/components/ui",
         | 
| 17 | 
            +
                "lib": "@/lib",
         | 
| 18 | 
            +
                "hooks": "@/hooks"
         | 
| 19 | 
            +
              },
         | 
| 20 | 
            +
              "iconLibrary": "lucide"
         | 
| 21 | 
            +
            }
         | 
    	
        components/contexts/app-context.tsx
    ADDED
    
    | @@ -0,0 +1,57 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 2 | 
            +
            "use client";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 5 | 
            +
            import { usePathname, useRouter } from "next/navigation";
         | 
| 6 | 
            +
            import { useMount } from "react-use";
         | 
| 7 | 
            +
            import { UserContext } from "@/components/contexts/user-context";
         | 
| 8 | 
            +
            import { User } from "@/types";
         | 
| 9 | 
            +
            import { toast } from "sonner";
         | 
| 10 | 
            +
            import { useBroadcastChannel } from "@/lib/useBroadcastChannel";
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            export default function AppContext({
         | 
| 13 | 
            +
              children,
         | 
| 14 | 
            +
              me: initialData,
         | 
| 15 | 
            +
            }: {
         | 
| 16 | 
            +
              children: React.ReactNode;
         | 
| 17 | 
            +
              me?: {
         | 
| 18 | 
            +
                user: User | null;
         | 
| 19 | 
            +
                errCode: number | null;
         | 
| 20 | 
            +
              };
         | 
| 21 | 
            +
            }) {
         | 
| 22 | 
            +
              const { loginFromCode, user, logout, loading, errCode } =
         | 
| 23 | 
            +
                useUser(initialData);
         | 
| 24 | 
            +
              const pathname = usePathname();
         | 
| 25 | 
            +
              const router = useRouter();
         | 
| 26 | 
            +
             | 
| 27 | 
            +
              useMount(() => {
         | 
| 28 | 
            +
                if (!initialData?.user && !user) {
         | 
| 29 | 
            +
                  if ([401, 403].includes(errCode as number)) {
         | 
| 30 | 
            +
                    logout();
         | 
| 31 | 
            +
                  } else if (pathname.includes("/spaces")) {
         | 
| 32 | 
            +
                    if (errCode) {
         | 
| 33 | 
            +
                      toast.error("An error occured while trying to log in");
         | 
| 34 | 
            +
                    }
         | 
| 35 | 
            +
                    // If we did not manage to log in (probs because api is down), we simply redirect to the home page
         | 
| 36 | 
            +
                    router.push("/");
         | 
| 37 | 
            +
                  }
         | 
| 38 | 
            +
                }
         | 
| 39 | 
            +
              });
         | 
| 40 | 
            +
             | 
| 41 | 
            +
              const events: any = {};
         | 
| 42 | 
            +
             | 
| 43 | 
            +
              useBroadcastChannel("auth", (message) => {
         | 
| 44 | 
            +
                if (pathname.includes("/auth/callback")) return;
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                if (!message.code) return;
         | 
| 47 | 
            +
                if (message.type === "user-oauth" && message?.code && !events.code) {
         | 
| 48 | 
            +
                  loginFromCode(message.code);
         | 
| 49 | 
            +
                }
         | 
| 50 | 
            +
              });
         | 
| 51 | 
            +
             | 
| 52 | 
            +
              return (
         | 
| 53 | 
            +
                <UserContext value={{ user, loading, logout } as any}>
         | 
| 54 | 
            +
                  {children}
         | 
| 55 | 
            +
                </UserContext>
         | 
| 56 | 
            +
              );
         | 
| 57 | 
            +
            }
         | 
    	
        components/contexts/user-context.tsx
    ADDED
    
    | @@ -0,0 +1,8 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { createContext } from "react";
         | 
| 4 | 
            +
            import { User } from "@/types";
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            export const UserContext = createContext({
         | 
| 7 | 
            +
              user: undefined as User | undefined,
         | 
| 8 | 
            +
            });
         | 
    	
        components/editor/ask-ai/index.tsx
    ADDED
    
    | @@ -0,0 +1,376 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 3 | 
            +
            import { useState, useRef } from "react";
         | 
| 4 | 
            +
            import classNames from "classnames";
         | 
| 5 | 
            +
            import { toast } from "sonner";
         | 
| 6 | 
            +
            import { useLocalStorage, useUpdateEffect } from "react-use";
         | 
| 7 | 
            +
            import { ArrowUp, ChevronDown } from "lucide-react";
         | 
| 8 | 
            +
            import { FaStopCircle } from "react-icons/fa";
         | 
| 9 | 
            +
             | 
| 10 | 
            +
            import { defaultHTML } from "@/lib/consts";
         | 
| 11 | 
            +
            import ProModal from "@/components/pro-modal";
         | 
| 12 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 13 | 
            +
            import { MODELS } from "@/lib/providers";
         | 
| 14 | 
            +
            import { HtmlHistory } from "@/types";
         | 
| 15 | 
            +
            import { InviteFriends } from "@/components/invite-friends";
         | 
| 16 | 
            +
            import { Settings } from "@/components/editor/ask-ai/settings";
         | 
| 17 | 
            +
            import { LoginModal } from "@/components/login-modal";
         | 
| 18 | 
            +
            import { ReImagine } from "@/components/editor/ask-ai/re-imagine";
         | 
| 19 | 
            +
            import Loading from "@/components/loading";
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            export function AskAI({
         | 
| 22 | 
            +
              html,
         | 
| 23 | 
            +
              setHtml,
         | 
| 24 | 
            +
              onScrollToBottom,
         | 
| 25 | 
            +
              isAiWorking,
         | 
| 26 | 
            +
              setisAiWorking,
         | 
| 27 | 
            +
              onNewPrompt,
         | 
| 28 | 
            +
              onSuccess,
         | 
| 29 | 
            +
            }: {
         | 
| 30 | 
            +
              html: string;
         | 
| 31 | 
            +
              setHtml: (html: string) => void;
         | 
| 32 | 
            +
              onScrollToBottom: () => void;
         | 
| 33 | 
            +
              isAiWorking: boolean;
         | 
| 34 | 
            +
              onNewPrompt: (prompt: string) => void;
         | 
| 35 | 
            +
              htmlHistory?: HtmlHistory[];
         | 
| 36 | 
            +
              setisAiWorking: React.Dispatch<React.SetStateAction<boolean>>;
         | 
| 37 | 
            +
              onSuccess: (h: string, p: string, n?: number[][]) => void;
         | 
| 38 | 
            +
            }) {
         | 
| 39 | 
            +
              const refThink = useRef<HTMLDivElement | null>(null);
         | 
| 40 | 
            +
              const audio = useRef<HTMLAudioElement | null>(null);
         | 
| 41 | 
            +
             | 
| 42 | 
            +
              const [open, setOpen] = useState(false);
         | 
| 43 | 
            +
              const [prompt, setPrompt] = useState("");
         | 
| 44 | 
            +
              const [hasAsked, setHasAsked] = useState(false);
         | 
| 45 | 
            +
              const [previousPrompt, setPreviousPrompt] = useState("");
         | 
| 46 | 
            +
              const [provider, setProvider] = useLocalStorage("provider", "auto");
         | 
| 47 | 
            +
              const [model, setModel] = useLocalStorage("model", MODELS[0].value);
         | 
| 48 | 
            +
              const [openProvider, setOpenProvider] = useState(false);
         | 
| 49 | 
            +
              const [providerError, setProviderError] = useState("");
         | 
| 50 | 
            +
              const [openProModal, setOpenProModal] = useState(false);
         | 
| 51 | 
            +
              const [think, setThink] = useState<string | undefined>(undefined);
         | 
| 52 | 
            +
              const [openThink, setOpenThink] = useState(false);
         | 
| 53 | 
            +
              const [isThinking, setIsThinking] = useState(true);
         | 
| 54 | 
            +
              const [controller, setController] = useState<AbortController | null>(null);
         | 
| 55 | 
            +
             | 
| 56 | 
            +
              const callAi = async (redesignMarkdown?: string) => {
         | 
| 57 | 
            +
                if (isAiWorking) return;
         | 
| 58 | 
            +
                if (!redesignMarkdown && !prompt.trim()) return;
         | 
| 59 | 
            +
                setisAiWorking(true);
         | 
| 60 | 
            +
                setProviderError("");
         | 
| 61 | 
            +
                setThink("");
         | 
| 62 | 
            +
                setOpenThink(false);
         | 
| 63 | 
            +
                setIsThinking(true);
         | 
| 64 | 
            +
             | 
| 65 | 
            +
                let contentResponse = "";
         | 
| 66 | 
            +
                let thinkResponse = "";
         | 
| 67 | 
            +
                let lastRenderTime = 0;
         | 
| 68 | 
            +
             | 
| 69 | 
            +
                const isFollowUp = html !== defaultHTML;
         | 
| 70 | 
            +
                const abortController = new AbortController();
         | 
| 71 | 
            +
                setController(abortController);
         | 
| 72 | 
            +
                try {
         | 
| 73 | 
            +
                  onNewPrompt(prompt);
         | 
| 74 | 
            +
                  if (isFollowUp && !redesignMarkdown) {
         | 
| 75 | 
            +
                    // TODO use @/lib/api instead of fetch directly (if possible)
         | 
| 76 | 
            +
                    const request = await fetch("/api/ask-ai", {
         | 
| 77 | 
            +
                      method: "PUT",
         | 
| 78 | 
            +
                      body: JSON.stringify({
         | 
| 79 | 
            +
                        prompt,
         | 
| 80 | 
            +
                        provider,
         | 
| 81 | 
            +
                        previousPrompt,
         | 
| 82 | 
            +
                        model,
         | 
| 83 | 
            +
                        html,
         | 
| 84 | 
            +
                      }),
         | 
| 85 | 
            +
                      headers: {
         | 
| 86 | 
            +
                        "Content-Type": "application/json",
         | 
| 87 | 
            +
                        "x-forwarded-for": window.location.hostname,
         | 
| 88 | 
            +
                        "x-real-ip": window.location.hostname,
         | 
| 89 | 
            +
                      },
         | 
| 90 | 
            +
                      signal: abortController.signal,
         | 
| 91 | 
            +
                    });
         | 
| 92 | 
            +
                    if (request && request.body) {
         | 
| 93 | 
            +
                      const res = await request.json();
         | 
| 94 | 
            +
                      if (!request.ok) {
         | 
| 95 | 
            +
                        if (res.openLogin) {
         | 
| 96 | 
            +
                          setOpen(true);
         | 
| 97 | 
            +
                        } else if (res.openSelectProvider) {
         | 
| 98 | 
            +
                          setOpenProvider(true);
         | 
| 99 | 
            +
                          setProviderError(res.message);
         | 
| 100 | 
            +
                        } else if (res.openProModal) {
         | 
| 101 | 
            +
                          setOpenProModal(true);
         | 
| 102 | 
            +
                        } else {
         | 
| 103 | 
            +
                          toast.error(res.message);
         | 
| 104 | 
            +
                        }
         | 
| 105 | 
            +
                        setisAiWorking(false);
         | 
| 106 | 
            +
                        return;
         | 
| 107 | 
            +
                      }
         | 
| 108 | 
            +
                      setHtml(res.html);
         | 
| 109 | 
            +
                      toast.success("AI responded successfully");
         | 
| 110 | 
            +
                      setPreviousPrompt(prompt);
         | 
| 111 | 
            +
                      setPrompt("");
         | 
| 112 | 
            +
                      setisAiWorking(false);
         | 
| 113 | 
            +
                      onSuccess(res.html, prompt, res.updatedLines);
         | 
| 114 | 
            +
                      if (audio.current) audio.current.play();
         | 
| 115 | 
            +
                    }
         | 
| 116 | 
            +
                  } else {
         | 
| 117 | 
            +
                    const request = await fetch("/api/ask-ai", {
         | 
| 118 | 
            +
                      method: "POST",
         | 
| 119 | 
            +
                      body: JSON.stringify({
         | 
| 120 | 
            +
                        prompt,
         | 
| 121 | 
            +
                        provider,
         | 
| 122 | 
            +
                        model,
         | 
| 123 | 
            +
                        redesignMarkdown,
         | 
| 124 | 
            +
                      }),
         | 
| 125 | 
            +
                      headers: {
         | 
| 126 | 
            +
                        "Content-Type": "application/json",
         | 
| 127 | 
            +
                        "x-forwarded-for": window.location.hostname,
         | 
| 128 | 
            +
                        "x-real-ip": window.location.hostname,
         | 
| 129 | 
            +
                      },
         | 
| 130 | 
            +
                      signal: abortController.signal,
         | 
| 131 | 
            +
                    });
         | 
| 132 | 
            +
                    if (request && request.body) {
         | 
| 133 | 
            +
                      if (!request.ok) {
         | 
| 134 | 
            +
                        const res = await request.json();
         | 
| 135 | 
            +
                        if (res.openLogin) {
         | 
| 136 | 
            +
                          setOpen(true);
         | 
| 137 | 
            +
                        } else if (res.openSelectProvider) {
         | 
| 138 | 
            +
                          setOpenProvider(true);
         | 
| 139 | 
            +
                          setProviderError(res.message);
         | 
| 140 | 
            +
                        } else if (res.openProModal) {
         | 
| 141 | 
            +
                          setOpenProModal(true);
         | 
| 142 | 
            +
                        } else {
         | 
| 143 | 
            +
                          toast.error(res.message);
         | 
| 144 | 
            +
                        }
         | 
| 145 | 
            +
                        setisAiWorking(false);
         | 
| 146 | 
            +
                        return;
         | 
| 147 | 
            +
                      }
         | 
| 148 | 
            +
                      const reader = request.body.getReader();
         | 
| 149 | 
            +
                      const decoder = new TextDecoder("utf-8");
         | 
| 150 | 
            +
                      const selectedModel = MODELS.find(
         | 
| 151 | 
            +
                        (m: { value: string }) => m.value === model
         | 
| 152 | 
            +
                      );
         | 
| 153 | 
            +
                      let contentThink: string | undefined = undefined;
         | 
| 154 | 
            +
                      const read = async () => {
         | 
| 155 | 
            +
                        const { done, value } = await reader.read();
         | 
| 156 | 
            +
                        if (done) {
         | 
| 157 | 
            +
                          toast.success("AI responded successfully");
         | 
| 158 | 
            +
                          setPreviousPrompt(prompt);
         | 
| 159 | 
            +
                          setPrompt("");
         | 
| 160 | 
            +
                          setisAiWorking(false);
         | 
| 161 | 
            +
                          setHasAsked(true);
         | 
| 162 | 
            +
                          setModel(MODELS[0].value);
         | 
| 163 | 
            +
                          if (audio.current) audio.current.play();
         | 
| 164 | 
            +
             | 
| 165 | 
            +
                          // Now we have the complete HTML including </html>, so set it to be sure
         | 
| 166 | 
            +
                          const finalDoc = contentResponse.match(
         | 
| 167 | 
            +
                            /<!DOCTYPE html>[\s\S]*<\/html>/
         | 
| 168 | 
            +
                          )?.[0];
         | 
| 169 | 
            +
                          if (finalDoc) {
         | 
| 170 | 
            +
                            setHtml(finalDoc);
         | 
| 171 | 
            +
                          }
         | 
| 172 | 
            +
                          onSuccess(finalDoc ?? contentResponse, prompt);
         | 
| 173 | 
            +
             | 
| 174 | 
            +
                          return;
         | 
| 175 | 
            +
                        }
         | 
| 176 | 
            +
             | 
| 177 | 
            +
                        const chunk = decoder.decode(value, { stream: true });
         | 
| 178 | 
            +
                        thinkResponse += chunk;
         | 
| 179 | 
            +
                        if (selectedModel?.isThinker) {
         | 
| 180 | 
            +
                          const thinkMatch = thinkResponse.match(/<think>[\s\S]*/)?.[0];
         | 
| 181 | 
            +
                          if (thinkMatch && !thinkResponse?.includes("</think>")) {
         | 
| 182 | 
            +
                            if ((contentThink?.length ?? 0) < 3) {
         | 
| 183 | 
            +
                              setOpenThink(true);
         | 
| 184 | 
            +
                            }
         | 
| 185 | 
            +
                            setThink(thinkMatch.replace("<think>", "").trim());
         | 
| 186 | 
            +
                            contentThink += chunk;
         | 
| 187 | 
            +
                            return read();
         | 
| 188 | 
            +
                          }
         | 
| 189 | 
            +
                        }
         | 
| 190 | 
            +
             | 
| 191 | 
            +
                        contentResponse += chunk;
         | 
| 192 | 
            +
             | 
| 193 | 
            +
                        const newHtml = contentResponse.match(
         | 
| 194 | 
            +
                          /<!DOCTYPE html>[\s\S]*/
         | 
| 195 | 
            +
                        )?.[0];
         | 
| 196 | 
            +
                        if (newHtml) {
         | 
| 197 | 
            +
                          setIsThinking(false);
         | 
| 198 | 
            +
                          let partialDoc = newHtml;
         | 
| 199 | 
            +
                          if (
         | 
| 200 | 
            +
                            partialDoc.includes("<head>") &&
         | 
| 201 | 
            +
                            !partialDoc.includes("</head>")
         | 
| 202 | 
            +
                          ) {
         | 
| 203 | 
            +
                            partialDoc += "\n</head>";
         | 
| 204 | 
            +
                          }
         | 
| 205 | 
            +
                          if (
         | 
| 206 | 
            +
                            partialDoc.includes("<body") &&
         | 
| 207 | 
            +
                            !partialDoc.includes("</body>")
         | 
| 208 | 
            +
                          ) {
         | 
| 209 | 
            +
                            partialDoc += "\n</body>";
         | 
| 210 | 
            +
                          }
         | 
| 211 | 
            +
                          if (!partialDoc.includes("</html>")) {
         | 
| 212 | 
            +
                            partialDoc += "\n</html>";
         | 
| 213 | 
            +
                          }
         | 
| 214 | 
            +
             | 
| 215 | 
            +
                          // Throttle the re-renders to avoid flashing/flicker
         | 
| 216 | 
            +
                          const now = Date.now();
         | 
| 217 | 
            +
                          if (now - lastRenderTime > 300) {
         | 
| 218 | 
            +
                            setHtml(partialDoc);
         | 
| 219 | 
            +
                            lastRenderTime = now;
         | 
| 220 | 
            +
                          }
         | 
| 221 | 
            +
             | 
| 222 | 
            +
                          if (partialDoc.length > 200) {
         | 
| 223 | 
            +
                            onScrollToBottom();
         | 
| 224 | 
            +
                          }
         | 
| 225 | 
            +
                        }
         | 
| 226 | 
            +
                        read();
         | 
| 227 | 
            +
                      };
         | 
| 228 | 
            +
             | 
| 229 | 
            +
                      read();
         | 
| 230 | 
            +
                    }
         | 
| 231 | 
            +
                  }
         | 
| 232 | 
            +
                } catch (error: any) {
         | 
| 233 | 
            +
                  setisAiWorking(false);
         | 
| 234 | 
            +
                  toast.error(error.message);
         | 
| 235 | 
            +
                  if (error.openLogin) {
         | 
| 236 | 
            +
                    setOpen(true);
         | 
| 237 | 
            +
                  }
         | 
| 238 | 
            +
                }
         | 
| 239 | 
            +
              };
         | 
| 240 | 
            +
             | 
| 241 | 
            +
              const stopController = () => {
         | 
| 242 | 
            +
                if (controller) {
         | 
| 243 | 
            +
                  controller.abort();
         | 
| 244 | 
            +
                  setController(null);
         | 
| 245 | 
            +
                  setisAiWorking(false);
         | 
| 246 | 
            +
                  setThink("");
         | 
| 247 | 
            +
                  setOpenThink(false);
         | 
| 248 | 
            +
                  setIsThinking(false);
         | 
| 249 | 
            +
                }
         | 
| 250 | 
            +
              };
         | 
| 251 | 
            +
             | 
| 252 | 
            +
              useUpdateEffect(() => {
         | 
| 253 | 
            +
                if (refThink.current) {
         | 
| 254 | 
            +
                  refThink.current.scrollTop = refThink.current.scrollHeight;
         | 
| 255 | 
            +
                }
         | 
| 256 | 
            +
              }, [think]);
         | 
| 257 | 
            +
             | 
| 258 | 
            +
              useUpdateEffect(() => {
         | 
| 259 | 
            +
                if (!isThinking) {
         | 
| 260 | 
            +
                  setOpenThink(false);
         | 
| 261 | 
            +
                }
         | 
| 262 | 
            +
              }, [isThinking]);
         | 
| 263 | 
            +
             | 
| 264 | 
            +
              return (
         | 
| 265 | 
            +
                <>
         | 
| 266 | 
            +
                  <div className="bg-neutral-800 border border-neutral-700 rounded-2xl ring-[4px] focus-within:ring-neutral-500/30 focus-within:border-neutral-600 ring-transparent z-10 absolute bottom-3 left-3 w-[calc(100%-20px)] group">
         | 
| 267 | 
            +
                    {think && (
         | 
| 268 | 
            +
                      <div className="w-full border-b border-neutral-700 relative overflow-hidden">
         | 
| 269 | 
            +
                        <header
         | 
| 270 | 
            +
                          className="flex items-center justify-between px-5 py-2.5 group hover:bg-neutral-600/20 transition-colors duration-200 cursor-pointer"
         | 
| 271 | 
            +
                          onClick={() => {
         | 
| 272 | 
            +
                            setOpenThink(!openThink);
         | 
| 273 | 
            +
                          }}
         | 
| 274 | 
            +
                        >
         | 
| 275 | 
            +
                          <p className="text-sm font-medium text-neutral-300 group-hover:text-neutral-200 transition-colors duration-200">
         | 
| 276 | 
            +
                            {isThinking ? "DeepSite is thinking..." : "DeepSite's plan"}
         | 
| 277 | 
            +
                          </p>
         | 
| 278 | 
            +
                          <ChevronDown
         | 
| 279 | 
            +
                            className={classNames(
         | 
| 280 | 
            +
                              "size-4 text-neutral-400 group-hover:text-neutral-300 transition-all duration-200",
         | 
| 281 | 
            +
                              {
         | 
| 282 | 
            +
                                "rotate-180": openThink,
         | 
| 283 | 
            +
                              }
         | 
| 284 | 
            +
                            )}
         | 
| 285 | 
            +
                          />
         | 
| 286 | 
            +
                        </header>
         | 
| 287 | 
            +
                        <main
         | 
| 288 | 
            +
                          ref={refThink}
         | 
| 289 | 
            +
                          className={classNames(
         | 
| 290 | 
            +
                            "overflow-y-auto transition-all duration-200 ease-in-out",
         | 
| 291 | 
            +
                            {
         | 
| 292 | 
            +
                              "max-h-[0px]": !openThink,
         | 
| 293 | 
            +
                              "min-h-[250px] max-h-[250px] border-t border-neutral-700":
         | 
| 294 | 
            +
                                openThink,
         | 
| 295 | 
            +
                            }
         | 
| 296 | 
            +
                          )}
         | 
| 297 | 
            +
                        >
         | 
| 298 | 
            +
                          <p className="text-[13px] text-neutral-400 whitespace-pre-line px-5 pb-4 pt-3">
         | 
| 299 | 
            +
                            {think}
         | 
| 300 | 
            +
                          </p>
         | 
| 301 | 
            +
                        </main>
         | 
| 302 | 
            +
                      </div>
         | 
| 303 | 
            +
                    )}
         | 
| 304 | 
            +
                    <div className="w-full relative flex items-center justify-between">
         | 
| 305 | 
            +
                      {isAiWorking && (
         | 
| 306 | 
            +
                        <div className="absolute bg-neutral-800 rounded-lg bottom-0 left-4 w-[calc(100%-30px)] h-full z-1 flex items-center justify-between max-lg:text-sm">
         | 
| 307 | 
            +
                          <div className="flex items-center justify-start gap-2">
         | 
| 308 | 
            +
                            <Loading overlay={false} className="!size-4" />
         | 
| 309 | 
            +
                            <p className="text-neutral-400 text-sm">
         | 
| 310 | 
            +
                              AI is {isThinking ? "thinking" : "coding"}...{" "}
         | 
| 311 | 
            +
                            </p>
         | 
| 312 | 
            +
                          </div>
         | 
| 313 | 
            +
                          <div
         | 
| 314 | 
            +
                            className="text-xs text-neutral-400 px-1 py-0.5 rounded-md border border-neutral-600 flex items-center justify-center gap-1.5 bg-neutral-800 hover:brightness-110 transition-all duration-200 cursor-pointer"
         | 
| 315 | 
            +
                            onClick={stopController}
         | 
| 316 | 
            +
                          >
         | 
| 317 | 
            +
                            <FaStopCircle />
         | 
| 318 | 
            +
                            Stop generation
         | 
| 319 | 
            +
                          </div>
         | 
| 320 | 
            +
                        </div>
         | 
| 321 | 
            +
                      )}
         | 
| 322 | 
            +
                      <input
         | 
| 323 | 
            +
                        type="text"
         | 
| 324 | 
            +
                        disabled={isAiWorking}
         | 
| 325 | 
            +
                        className="w-full bg-transparent text-sm outline-none text-white placeholder:text-neutral-400 p-4"
         | 
| 326 | 
            +
                        placeholder={
         | 
| 327 | 
            +
                          hasAsked ? "Ask DeepSite for edits" : "Ask DeepSite anything..."
         | 
| 328 | 
            +
                        }
         | 
| 329 | 
            +
                        value={prompt}
         | 
| 330 | 
            +
                        onChange={(e) => setPrompt(e.target.value)}
         | 
| 331 | 
            +
                        onKeyDown={(e) => {
         | 
| 332 | 
            +
                          if (e.key === "Enter" && !e.shiftKey) {
         | 
| 333 | 
            +
                            callAi();
         | 
| 334 | 
            +
                          }
         | 
| 335 | 
            +
                        }}
         | 
| 336 | 
            +
                      />
         | 
| 337 | 
            +
                    </div>
         | 
| 338 | 
            +
                    <div className="flex items-center justify-between gap-2 px-4 pb-3">
         | 
| 339 | 
            +
                      <div className="flex-1 flex items-center justify-start gap-1.5">
         | 
| 340 | 
            +
                        <ReImagine onRedesign={(md) => callAi(md)} />
         | 
| 341 | 
            +
                        <InviteFriends />
         | 
| 342 | 
            +
                      </div>
         | 
| 343 | 
            +
                      <div className="flex items-center justify-end gap-2">
         | 
| 344 | 
            +
                        <Settings
         | 
| 345 | 
            +
                          provider={provider as string}
         | 
| 346 | 
            +
                          model={model as string}
         | 
| 347 | 
            +
                          onChange={setProvider}
         | 
| 348 | 
            +
                          onModelChange={setModel}
         | 
| 349 | 
            +
                          open={openProvider}
         | 
| 350 | 
            +
                          error={providerError}
         | 
| 351 | 
            +
                          isFollowUp={html !== defaultHTML}
         | 
| 352 | 
            +
                          onClose={setOpenProvider}
         | 
| 353 | 
            +
                        />
         | 
| 354 | 
            +
                        <Button
         | 
| 355 | 
            +
                          size="iconXs"
         | 
| 356 | 
            +
                          disabled={isAiWorking || !prompt.trim()}
         | 
| 357 | 
            +
                          onClick={() => callAi()}
         | 
| 358 | 
            +
                        >
         | 
| 359 | 
            +
                          <ArrowUp className="size-4" />
         | 
| 360 | 
            +
                        </Button>
         | 
| 361 | 
            +
                      </div>
         | 
| 362 | 
            +
                    </div>
         | 
| 363 | 
            +
                    <LoginModal open={open} onClose={() => setOpen(false)} html={html} />
         | 
| 364 | 
            +
                    <ProModal
         | 
| 365 | 
            +
                      html={html}
         | 
| 366 | 
            +
                      open={openProModal}
         | 
| 367 | 
            +
                      onClose={() => setOpenProModal(false)}
         | 
| 368 | 
            +
                    />
         | 
| 369 | 
            +
                  </div>
         | 
| 370 | 
            +
                  <audio ref={audio} id="audio" className="hidden">
         | 
| 371 | 
            +
                    <source src="/success.mp3" type="audio/mpeg" />
         | 
| 372 | 
            +
                    Your browser does not support the audio element.
         | 
| 373 | 
            +
                  </audio>
         | 
| 374 | 
            +
                </>
         | 
| 375 | 
            +
              );
         | 
| 376 | 
            +
            }
         | 
    	
        components/editor/ask-ai/re-imagine.tsx
    ADDED
    
    | @@ -0,0 +1,140 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { useState } from "react";
         | 
| 2 | 
            +
            import { Paintbrush } from "lucide-react";
         | 
| 3 | 
            +
            import { toast } from "sonner";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 6 | 
            +
            import {
         | 
| 7 | 
            +
              Popover,
         | 
| 8 | 
            +
              PopoverContent,
         | 
| 9 | 
            +
              PopoverTrigger,
         | 
| 10 | 
            +
            } from "@/components/ui/popover";
         | 
| 11 | 
            +
            import { Input } from "@/components/ui/input";
         | 
| 12 | 
            +
            import Loading from "@/components/loading";
         | 
| 13 | 
            +
             | 
| 14 | 
            +
            export function ReImagine({
         | 
| 15 | 
            +
              onRedesign,
         | 
| 16 | 
            +
            }: {
         | 
| 17 | 
            +
              onRedesign: (md: string) => void;
         | 
| 18 | 
            +
            }) {
         | 
| 19 | 
            +
              const [url, setUrl] = useState<string>("");
         | 
| 20 | 
            +
              const [open, setOpen] = useState(false);
         | 
| 21 | 
            +
              const [isLoading, setIsLoading] = useState(false);
         | 
| 22 | 
            +
             | 
| 23 | 
            +
              const checkIfUrlIsValid = (url: string) => {
         | 
| 24 | 
            +
                const urlPattern = new RegExp(
         | 
| 25 | 
            +
                  /^(https?:\/\/)?([\da-z.-]+)\.([a-z.]{2,6})([/\w .-]*)*\/?$/,
         | 
| 26 | 
            +
                  "i"
         | 
| 27 | 
            +
                );
         | 
| 28 | 
            +
                return urlPattern.test(url);
         | 
| 29 | 
            +
              };
         | 
| 30 | 
            +
             | 
| 31 | 
            +
              const handleClick = async () => {
         | 
| 32 | 
            +
                if (!url) {
         | 
| 33 | 
            +
                  toast.error("Please enter a URL.");
         | 
| 34 | 
            +
                  return;
         | 
| 35 | 
            +
                }
         | 
| 36 | 
            +
                if (!checkIfUrlIsValid(url)) {
         | 
| 37 | 
            +
                  toast.error("Please enter a valid URL.");
         | 
| 38 | 
            +
                  return;
         | 
| 39 | 
            +
                }
         | 
| 40 | 
            +
                // Here you would typically handle the re-design logic
         | 
| 41 | 
            +
                setIsLoading(true);
         | 
| 42 | 
            +
                const request = await fetch("/api/re-design", {
         | 
| 43 | 
            +
                  method: "POST",
         | 
| 44 | 
            +
                  body: JSON.stringify({ url }),
         | 
| 45 | 
            +
                  headers: {
         | 
| 46 | 
            +
                    "Content-Type": "application/json",
         | 
| 47 | 
            +
                  },
         | 
| 48 | 
            +
                });
         | 
| 49 | 
            +
                const response = await request.json();
         | 
| 50 | 
            +
                if (response.ok) {
         | 
| 51 | 
            +
                  setOpen(false);
         | 
| 52 | 
            +
                  setUrl("");
         | 
| 53 | 
            +
                  onRedesign(response.markdown);
         | 
| 54 | 
            +
                  toast.success("DeepSite is redesigning your site! Let him cook... 🔥");
         | 
| 55 | 
            +
                } else {
         | 
| 56 | 
            +
                  toast.error(response.message || "Failed to redesign the site.");
         | 
| 57 | 
            +
                }
         | 
| 58 | 
            +
                setIsLoading(false);
         | 
| 59 | 
            +
              };
         | 
| 60 | 
            +
             | 
| 61 | 
            +
              return (
         | 
| 62 | 
            +
                <Popover open={open} onOpenChange={setOpen}>
         | 
| 63 | 
            +
                  <form>
         | 
| 64 | 
            +
                    <PopoverTrigger asChild>
         | 
| 65 | 
            +
                      <Button
         | 
| 66 | 
            +
                        size="iconXs"
         | 
| 67 | 
            +
                        variant="outline"
         | 
| 68 | 
            +
                        className="!border-neutral-600 !text-neutral-400 !hover:!border-neutral-500 hover:!text-neutral-300"
         | 
| 69 | 
            +
                      >
         | 
| 70 | 
            +
                        <Paintbrush className="size-4" />
         | 
| 71 | 
            +
                      </Button>
         | 
| 72 | 
            +
                    </PopoverTrigger>
         | 
| 73 | 
            +
                    <PopoverContent
         | 
| 74 | 
            +
                      align="start"
         | 
| 75 | 
            +
                      className="!rounded-2xl !p-0 !bg-white !border-neutral-100 min-w-xs text-center overflow-hidden"
         | 
| 76 | 
            +
                    >
         | 
| 77 | 
            +
                      <header className="bg-neutral-50 p-6 border-b border-neutral-200/60">
         | 
| 78 | 
            +
                        <div className="flex items-center justify-center -space-x-4 mb-3">
         | 
| 79 | 
            +
                          <div className="size-9 rounded-full bg-pink-200 shadow-2xs flex items-center justify-center text-xl opacity-50">
         | 
| 80 | 
            +
                            🎨
         | 
| 81 | 
            +
                          </div>
         | 
| 82 | 
            +
                          <div className="size-11 rounded-full bg-amber-200 shadow-2xl flex items-center justify-center text-2xl z-2">
         | 
| 83 | 
            +
                            🥳
         | 
| 84 | 
            +
                          </div>
         | 
| 85 | 
            +
                          <div className="size-9 rounded-full bg-sky-200 shadow-2xs flex items-center justify-center text-xl opacity-50">
         | 
| 86 | 
            +
                            💎
         | 
| 87 | 
            +
                          </div>
         | 
| 88 | 
            +
                        </div>
         | 
| 89 | 
            +
                        <p className="text-xl font-semibold text-neutral-950">
         | 
| 90 | 
            +
                          Redesign your Site!
         | 
| 91 | 
            +
                        </p>
         | 
| 92 | 
            +
                        <p className="text-sm text-neutral-500 mt-1.5">
         | 
| 93 | 
            +
                          Try our new Redesign feature to give your site a fresh look.
         | 
| 94 | 
            +
                        </p>
         | 
| 95 | 
            +
                      </header>
         | 
| 96 | 
            +
                      <main className="space-y-4 p-6">
         | 
| 97 | 
            +
                        <div>
         | 
| 98 | 
            +
                          <p className="text-sm text-neutral-700 mb-2">
         | 
| 99 | 
            +
                            Enter your website URL to get started:
         | 
| 100 | 
            +
                          </p>
         | 
| 101 | 
            +
                          <Input
         | 
| 102 | 
            +
                            type="text"
         | 
| 103 | 
            +
                            placeholder="https://example.com"
         | 
| 104 | 
            +
                            value={url}
         | 
| 105 | 
            +
                            onChange={(e) => setUrl(e.target.value)}
         | 
| 106 | 
            +
                            onBlur={(e) => {
         | 
| 107 | 
            +
                              const inputUrl = e.target.value.trim();
         | 
| 108 | 
            +
                              if (!inputUrl) {
         | 
| 109 | 
            +
                                setUrl("");
         | 
| 110 | 
            +
                                return;
         | 
| 111 | 
            +
                              }
         | 
| 112 | 
            +
                              if (!checkIfUrlIsValid(inputUrl)) {
         | 
| 113 | 
            +
                                toast.error("Please enter a valid URL.");
         | 
| 114 | 
            +
                                return;
         | 
| 115 | 
            +
                              }
         | 
| 116 | 
            +
                              setUrl(inputUrl);
         | 
| 117 | 
            +
                            }}
         | 
| 118 | 
            +
                            className="!bg-white !border-neutral-300 !text-neutral-800 !placeholder:text-neutral-400 selection:!bg-blue-100"
         | 
| 119 | 
            +
                          />
         | 
| 120 | 
            +
                        </div>
         | 
| 121 | 
            +
                        <div>
         | 
| 122 | 
            +
                          <p className="text-sm text-neutral-700 mb-2">
         | 
| 123 | 
            +
                            Then, let's redesign it!
         | 
| 124 | 
            +
                          </p>
         | 
| 125 | 
            +
                          <Button
         | 
| 126 | 
            +
                            variant="black"
         | 
| 127 | 
            +
                            onClick={handleClick}
         | 
| 128 | 
            +
                            className="relative w-full"
         | 
| 129 | 
            +
                            disabled={isLoading}
         | 
| 130 | 
            +
                          >
         | 
| 131 | 
            +
                            Redesign <Paintbrush className="size-4" />
         | 
| 132 | 
            +
                            {isLoading && <Loading className="ml-2 size-4 animate-spin" />}
         | 
| 133 | 
            +
                          </Button>
         | 
| 134 | 
            +
                        </div>
         | 
| 135 | 
            +
                      </main>
         | 
| 136 | 
            +
                    </PopoverContent>
         | 
| 137 | 
            +
                  </form>
         | 
| 138 | 
            +
                </Popover>
         | 
| 139 | 
            +
              );
         | 
| 140 | 
            +
            }
         | 
    	
        components/editor/ask-ai/settings.tsx
    ADDED
    
    | @@ -0,0 +1,212 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import classNames from "classnames";
         | 
| 2 | 
            +
            import { PiGearSixFill } from "react-icons/pi";
         | 
| 3 | 
            +
            import { RiCheckboxCircleFill } from "react-icons/ri";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            import {
         | 
| 6 | 
            +
              Popover,
         | 
| 7 | 
            +
              PopoverContent,
         | 
| 8 | 
            +
              PopoverTrigger,
         | 
| 9 | 
            +
            } from "@/components/ui/popover";
         | 
| 10 | 
            +
            import { PROVIDERS, MODELS } from "@/lib/providers";
         | 
| 11 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 12 | 
            +
            import {
         | 
| 13 | 
            +
              Select,
         | 
| 14 | 
            +
              SelectContent,
         | 
| 15 | 
            +
              SelectGroup,
         | 
| 16 | 
            +
              SelectItem,
         | 
| 17 | 
            +
              SelectLabel,
         | 
| 18 | 
            +
              SelectTrigger,
         | 
| 19 | 
            +
              SelectValue,
         | 
| 20 | 
            +
            } from "@/components/ui/select";
         | 
| 21 | 
            +
            import { useMemo } from "react";
         | 
| 22 | 
            +
            import { useUpdateEffect } from "react-use";
         | 
| 23 | 
            +
            import Image from "next/image";
         | 
| 24 | 
            +
             | 
| 25 | 
            +
            export function Settings({
         | 
| 26 | 
            +
              open,
         | 
| 27 | 
            +
              onClose,
         | 
| 28 | 
            +
              provider,
         | 
| 29 | 
            +
              model,
         | 
| 30 | 
            +
              error,
         | 
| 31 | 
            +
              isFollowUp = false,
         | 
| 32 | 
            +
              onChange,
         | 
| 33 | 
            +
              onModelChange,
         | 
| 34 | 
            +
            }: {
         | 
| 35 | 
            +
              open: boolean;
         | 
| 36 | 
            +
              provider: string;
         | 
| 37 | 
            +
              model: string;
         | 
| 38 | 
            +
              error?: string;
         | 
| 39 | 
            +
              isFollowUp?: boolean;
         | 
| 40 | 
            +
              onClose: React.Dispatch<React.SetStateAction<boolean>>;
         | 
| 41 | 
            +
              onChange: (provider: string) => void;
         | 
| 42 | 
            +
              onModelChange: (model: string) => void;
         | 
| 43 | 
            +
            }) {
         | 
| 44 | 
            +
              const modelAvailableProviders = useMemo(() => {
         | 
| 45 | 
            +
                const availableProviders = MODELS.find(
         | 
| 46 | 
            +
                  (m: { value: string }) => m.value === model
         | 
| 47 | 
            +
                )?.providers;
         | 
| 48 | 
            +
                if (!availableProviders) return Object.keys(PROVIDERS);
         | 
| 49 | 
            +
                return Object.keys(PROVIDERS).filter((id) =>
         | 
| 50 | 
            +
                  availableProviders.includes(id)
         | 
| 51 | 
            +
                );
         | 
| 52 | 
            +
              }, [model]);
         | 
| 53 | 
            +
             | 
| 54 | 
            +
              useUpdateEffect(() => {
         | 
| 55 | 
            +
                if (provider !== "auto" && !modelAvailableProviders.includes(provider)) {
         | 
| 56 | 
            +
                  onChange("auto");
         | 
| 57 | 
            +
                }
         | 
| 58 | 
            +
              }, [model, provider]);
         | 
| 59 | 
            +
             | 
| 60 | 
            +
              return (
         | 
| 61 | 
            +
                <div className="">
         | 
| 62 | 
            +
                  <Popover open={open} onOpenChange={onClose}>
         | 
| 63 | 
            +
                    <PopoverTrigger asChild>
         | 
| 64 | 
            +
                      <Button variant="black" size="sm">
         | 
| 65 | 
            +
                        <PiGearSixFill className="size-4" />
         | 
| 66 | 
            +
                        Settings
         | 
| 67 | 
            +
                      </Button>
         | 
| 68 | 
            +
                    </PopoverTrigger>
         | 
| 69 | 
            +
                    <PopoverContent
         | 
| 70 | 
            +
                      className="!rounded-2xl p-0 !w-96 overflow-hidden !bg-neutral-900"
         | 
| 71 | 
            +
                      align="center"
         | 
| 72 | 
            +
                    >
         | 
| 73 | 
            +
                      <header className="flex items-center justify-center text-sm px-4 py-3 border-b gap-2 bg-neutral-950 border-neutral-800 font-semibold text-neutral-200">
         | 
| 74 | 
            +
                        Customize Settings
         | 
| 75 | 
            +
                      </header>
         | 
| 76 | 
            +
                      <main className="px-4 pt-5 pb-6 space-y-5">
         | 
| 77 | 
            +
                        {/* <a
         | 
| 78 | 
            +
                          href="https://huggingface.co/spaces/enzostvs/deepsite/discussions/74"
         | 
| 79 | 
            +
                          target="_blank"
         | 
| 80 | 
            +
                          className="w-full flex items-center justify-between text-neutral-300 bg-neutral-300/15 border border-neutral-300/15 pl-4 p-1.5 rounded-full text-sm font-medium hover:brightness-95"
         | 
| 81 | 
            +
                        >
         | 
| 82 | 
            +
                          How to use it locally?
         | 
| 83 | 
            +
                          <Button size="xs">See guide</Button>
         | 
| 84 | 
            +
                        </a> */}
         | 
| 85 | 
            +
                        {error !== "" && (
         | 
| 86 | 
            +
                          <p className="text-red-500 text-sm font-medium mb-2 flex items-center justify-between bg-red-500/10 p-2 rounded-md">
         | 
| 87 | 
            +
                            {error}
         | 
| 88 | 
            +
                          </p>
         | 
| 89 | 
            +
                        )}
         | 
| 90 | 
            +
                        <label className="block">
         | 
| 91 | 
            +
                          <p className="text-neutral-300 text-sm mb-2.5">
         | 
| 92 | 
            +
                            Choose a DeepSeek model
         | 
| 93 | 
            +
                          </p>
         | 
| 94 | 
            +
                          <Select defaultValue={model} onValueChange={onModelChange}>
         | 
| 95 | 
            +
                            <SelectTrigger className="w-full">
         | 
| 96 | 
            +
                              <SelectValue placeholder="Select a DeepSeek model" />
         | 
| 97 | 
            +
                            </SelectTrigger>
         | 
| 98 | 
            +
                            <SelectContent>
         | 
| 99 | 
            +
                              <SelectGroup>
         | 
| 100 | 
            +
                                <SelectLabel>DeepSeek models</SelectLabel>
         | 
| 101 | 
            +
                                {MODELS.map(
         | 
| 102 | 
            +
                                  ({
         | 
| 103 | 
            +
                                    value,
         | 
| 104 | 
            +
                                    label,
         | 
| 105 | 
            +
                                    isNew = false,
         | 
| 106 | 
            +
                                    isThinker = false,
         | 
| 107 | 
            +
                                  }: {
         | 
| 108 | 
            +
                                    value: string;
         | 
| 109 | 
            +
                                    label: string;
         | 
| 110 | 
            +
                                    isNew?: boolean;
         | 
| 111 | 
            +
                                    isThinker?: boolean;
         | 
| 112 | 
            +
                                  }) => (
         | 
| 113 | 
            +
                                    <SelectItem
         | 
| 114 | 
            +
                                      key={value}
         | 
| 115 | 
            +
                                      value={value}
         | 
| 116 | 
            +
                                      className=""
         | 
| 117 | 
            +
                                      disabled={isThinker && isFollowUp}
         | 
| 118 | 
            +
                                    >
         | 
| 119 | 
            +
                                      {label}
         | 
| 120 | 
            +
                                      {isNew && (
         | 
| 121 | 
            +
                                        <span className="text-xs bg-gradient-to-br from-sky-400 to-sky-600 text-white rounded-full px-1.5 py-0.5">
         | 
| 122 | 
            +
                                          New
         | 
| 123 | 
            +
                                        </span>
         | 
| 124 | 
            +
                                      )}
         | 
| 125 | 
            +
                                    </SelectItem>
         | 
| 126 | 
            +
                                  )
         | 
| 127 | 
            +
                                )}
         | 
| 128 | 
            +
                              </SelectGroup>
         | 
| 129 | 
            +
                            </SelectContent>
         | 
| 130 | 
            +
                          </Select>
         | 
| 131 | 
            +
                        </label>
         | 
| 132 | 
            +
                        {isFollowUp && (
         | 
| 133 | 
            +
                          <div className="bg-amber-500/10 border-amber-500/10 p-3 text-xs text-amber-500 border rounded-lg">
         | 
| 134 | 
            +
                            Note: You can't use a Thinker model for follow-up requests.
         | 
| 135 | 
            +
                            We automatically switch to the default model for you.
         | 
| 136 | 
            +
                          </div>
         | 
| 137 | 
            +
                        )}
         | 
| 138 | 
            +
                        <div className="flex flex-col gap-3">
         | 
| 139 | 
            +
                          <div className="flex items-center justify-between">
         | 
| 140 | 
            +
                            <div>
         | 
| 141 | 
            +
                              <p className="text-neutral-300 text-sm mb-1.5">
         | 
| 142 | 
            +
                                Use auto-provider
         | 
| 143 | 
            +
                              </p>
         | 
| 144 | 
            +
                              <p className="text-xs text-neutral-400/70">
         | 
| 145 | 
            +
                                We'll automatically select the best provider for you
         | 
| 146 | 
            +
                                based on your prompt.
         | 
| 147 | 
            +
                              </p>
         | 
| 148 | 
            +
                            </div>
         | 
| 149 | 
            +
                            <div
         | 
| 150 | 
            +
                              className={classNames(
         | 
| 151 | 
            +
                                "bg-neutral-700 rounded-full min-w-10 w-10 h-6 flex items-center justify-between p-1 cursor-pointer transition-all duration-200",
         | 
| 152 | 
            +
                                {
         | 
| 153 | 
            +
                                  "!bg-sky-500": provider === "auto",
         | 
| 154 | 
            +
                                }
         | 
| 155 | 
            +
                              )}
         | 
| 156 | 
            +
                              onClick={() => {
         | 
| 157 | 
            +
                                const foundModel = MODELS.find(
         | 
| 158 | 
            +
                                  (m: { value: string }) => m.value === model
         | 
| 159 | 
            +
                                );
         | 
| 160 | 
            +
                                if (provider === "auto" && foundModel?.autoProvider) {
         | 
| 161 | 
            +
                                  onChange(foundModel.autoProvider);
         | 
| 162 | 
            +
                                } else {
         | 
| 163 | 
            +
                                  onChange("auto");
         | 
| 164 | 
            +
                                }
         | 
| 165 | 
            +
                              }}
         | 
| 166 | 
            +
                            >
         | 
| 167 | 
            +
                              <div
         | 
| 168 | 
            +
                                className={classNames(
         | 
| 169 | 
            +
                                  "w-4 h-4 rounded-full shadow-md transition-all duration-200 bg-neutral-200",
         | 
| 170 | 
            +
                                  {
         | 
| 171 | 
            +
                                    "translate-x-4": provider === "auto",
         | 
| 172 | 
            +
                                  }
         | 
| 173 | 
            +
                                )}
         | 
| 174 | 
            +
                              />
         | 
| 175 | 
            +
                            </div>
         | 
| 176 | 
            +
                          </div>
         | 
| 177 | 
            +
                          <label className="block">
         | 
| 178 | 
            +
                            <p className="text-neutral-300 text-sm mb-2">
         | 
| 179 | 
            +
                              Inference Provider
         | 
| 180 | 
            +
                            </p>
         | 
| 181 | 
            +
                            <div className="grid grid-cols-2 gap-1.5">
         | 
| 182 | 
            +
                              {modelAvailableProviders.map((id: string) => (
         | 
| 183 | 
            +
                                <Button
         | 
| 184 | 
            +
                                  key={id}
         | 
| 185 | 
            +
                                  variant={id === provider ? "default" : "secondary"}
         | 
| 186 | 
            +
                                  size="sm"
         | 
| 187 | 
            +
                                  onClick={() => {
         | 
| 188 | 
            +
                                    onChange(id);
         | 
| 189 | 
            +
                                  }}
         | 
| 190 | 
            +
                                >
         | 
| 191 | 
            +
                                  <Image
         | 
| 192 | 
            +
                                    src={`/providers/${id}.svg`}
         | 
| 193 | 
            +
                                    alt={PROVIDERS[id as keyof typeof PROVIDERS].name}
         | 
| 194 | 
            +
                                    className="size-5 mr-2"
         | 
| 195 | 
            +
                                    width={20}
         | 
| 196 | 
            +
                                    height={20}
         | 
| 197 | 
            +
                                  />
         | 
| 198 | 
            +
                                  {PROVIDERS[id as keyof typeof PROVIDERS].name}
         | 
| 199 | 
            +
                                  {id === provider && (
         | 
| 200 | 
            +
                                    <RiCheckboxCircleFill className="ml-2 size-4 text-blue-500" />
         | 
| 201 | 
            +
                                  )}
         | 
| 202 | 
            +
                                </Button>
         | 
| 203 | 
            +
                              ))}
         | 
| 204 | 
            +
                            </div>
         | 
| 205 | 
            +
                          </label>
         | 
| 206 | 
            +
                        </div>
         | 
| 207 | 
            +
                      </main>
         | 
| 208 | 
            +
                    </PopoverContent>
         | 
| 209 | 
            +
                  </Popover>
         | 
| 210 | 
            +
                </div>
         | 
| 211 | 
            +
              );
         | 
| 212 | 
            +
            }
         | 
    	
        components/editor/deploy-button/index.tsx
    ADDED
    
    | @@ -0,0 +1,171 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 2 | 
            +
            import { useState } from "react";
         | 
| 3 | 
            +
            import { toast } from "sonner";
         | 
| 4 | 
            +
            import Image from "next/image";
         | 
| 5 | 
            +
            import { useRouter } from "next/navigation";
         | 
| 6 | 
            +
            import { MdSave } from "react-icons/md";
         | 
| 7 | 
            +
            import { Rocket } from "lucide-react";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            import SpaceIcon from "@/assets/space.svg";
         | 
| 10 | 
            +
            import Loading from "@/components/loading";
         | 
| 11 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 12 | 
            +
            import {
         | 
| 13 | 
            +
              Popover,
         | 
| 14 | 
            +
              PopoverContent,
         | 
| 15 | 
            +
              PopoverTrigger,
         | 
| 16 | 
            +
            } from "@/components/ui/popover";
         | 
| 17 | 
            +
            import { Input } from "@/components/ui/input";
         | 
| 18 | 
            +
            import { api } from "@/lib/api";
         | 
| 19 | 
            +
            import { LoginModal } from "@/components/login-modal";
         | 
| 20 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 21 | 
            +
             | 
| 22 | 
            +
            export function DeployButton({
         | 
| 23 | 
            +
              html,
         | 
| 24 | 
            +
              prompts,
         | 
| 25 | 
            +
            }: {
         | 
| 26 | 
            +
              html: string;
         | 
| 27 | 
            +
              prompts: string[];
         | 
| 28 | 
            +
            }) {
         | 
| 29 | 
            +
              const router = useRouter();
         | 
| 30 | 
            +
              const { user } = useUser();
         | 
| 31 | 
            +
              const [loading, setLoading] = useState(false);
         | 
| 32 | 
            +
              const [open, setOpen] = useState(false);
         | 
| 33 | 
            +
             | 
| 34 | 
            +
              const [config, setConfig] = useState({
         | 
| 35 | 
            +
                title: "",
         | 
| 36 | 
            +
              });
         | 
| 37 | 
            +
             | 
| 38 | 
            +
              const createSpace = async () => {
         | 
| 39 | 
            +
                if (!config.title) {
         | 
| 40 | 
            +
                  toast.error("Please enter a title for your space.");
         | 
| 41 | 
            +
                  return;
         | 
| 42 | 
            +
                }
         | 
| 43 | 
            +
                setLoading(true);
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                try {
         | 
| 46 | 
            +
                  const res = await api.post("/me/projects", {
         | 
| 47 | 
            +
                    title: config.title,
         | 
| 48 | 
            +
                    html,
         | 
| 49 | 
            +
                    prompts,
         | 
| 50 | 
            +
                  });
         | 
| 51 | 
            +
                  if (res.data.ok) {
         | 
| 52 | 
            +
                    router.push(`/projects/${res.data.path}?deploy=true`);
         | 
| 53 | 
            +
                  } else {
         | 
| 54 | 
            +
                    toast.error(res?.data?.error || "Failed to create space");
         | 
| 55 | 
            +
                  }
         | 
| 56 | 
            +
                } catch (err: any) {
         | 
| 57 | 
            +
                  toast.error(err.response?.data?.error || err.message);
         | 
| 58 | 
            +
                } finally {
         | 
| 59 | 
            +
                  setLoading(false);
         | 
| 60 | 
            +
                }
         | 
| 61 | 
            +
              };
         | 
| 62 | 
            +
             | 
| 63 | 
            +
              return (
         | 
| 64 | 
            +
                <div className="flex items-center justify-end gap-5">
         | 
| 65 | 
            +
                  <div className="relative flex items-center justify-end">
         | 
| 66 | 
            +
                    {user?.id ? (
         | 
| 67 | 
            +
                      <Popover>
         | 
| 68 | 
            +
                        <PopoverTrigger asChild>
         | 
| 69 | 
            +
                          <div>
         | 
| 70 | 
            +
                            <Button variant="sky" className="max-lg:hidden !px-4">
         | 
| 71 | 
            +
                              <MdSave className="size-4" />
         | 
| 72 | 
            +
                              Save your Project
         | 
| 73 | 
            +
                            </Button>
         | 
| 74 | 
            +
                            <Button variant="sky" size="sm" className="lg:hidden">
         | 
| 75 | 
            +
                              Save
         | 
| 76 | 
            +
                            </Button>
         | 
| 77 | 
            +
                          </div>
         | 
| 78 | 
            +
                        </PopoverTrigger>
         | 
| 79 | 
            +
                        <PopoverContent
         | 
| 80 | 
            +
                          className="!rounded-2xl !p-0 !bg-white !border-neutral-200 min-w-xs text-center overflow-hidden"
         | 
| 81 | 
            +
                          align="end"
         | 
| 82 | 
            +
                        >
         | 
| 83 | 
            +
                          <header className="bg-neutral-50 p-6 border-b border-neutral-200/60">
         | 
| 84 | 
            +
                            <div className="flex items-center justify-center -space-x-4 mb-3">
         | 
| 85 | 
            +
                              <div className="size-9 rounded-full bg-amber-200 shadow-2xs flex items-center justify-center text-xl opacity-50">
         | 
| 86 | 
            +
                                🚀
         | 
| 87 | 
            +
                              </div>
         | 
| 88 | 
            +
                              <div className="size-11 rounded-full bg-red-200 shadow-2xl flex items-center justify-center z-2">
         | 
| 89 | 
            +
                                <Image
         | 
| 90 | 
            +
                                  src={SpaceIcon}
         | 
| 91 | 
            +
                                  alt="Space Icon"
         | 
| 92 | 
            +
                                  className="size-7"
         | 
| 93 | 
            +
                                />
         | 
| 94 | 
            +
                              </div>
         | 
| 95 | 
            +
                              <div className="size-9 rounded-full bg-sky-200 shadow-2xs flex items-center justify-center text-xl opacity-50">
         | 
| 96 | 
            +
                                👻
         | 
| 97 | 
            +
                              </div>
         | 
| 98 | 
            +
                            </div>
         | 
| 99 | 
            +
                            <p className="text-xl font-semibold text-neutral-950">
         | 
| 100 | 
            +
                              Deploy your Project!
         | 
| 101 | 
            +
                            </p>
         | 
| 102 | 
            +
                            <p className="text-sm text-neutral-500 mt-1.5">
         | 
| 103 | 
            +
                              Deploy your project to a space on the Hub. Spaces are a way to
         | 
| 104 | 
            +
                              share your project with the world.
         | 
| 105 | 
            +
                            </p>
         | 
| 106 | 
            +
                          </header>
         | 
| 107 | 
            +
                          <main className="space-y-4 p-6">
         | 
| 108 | 
            +
                            <div>
         | 
| 109 | 
            +
                              <p className="text-sm text-neutral-700 mb-2">
         | 
| 110 | 
            +
                                Choose a title for your space:
         | 
| 111 | 
            +
                              </p>
         | 
| 112 | 
            +
                              <Input
         | 
| 113 | 
            +
                                type="text"
         | 
| 114 | 
            +
                                placeholder="My Awesome Website"
         | 
| 115 | 
            +
                                value={config.title}
         | 
| 116 | 
            +
                                onChange={(e) =>
         | 
| 117 | 
            +
                                  setConfig({ ...config, title: e.target.value })
         | 
| 118 | 
            +
                                }
         | 
| 119 | 
            +
                                className="!bg-white !border-neutral-300 !text-neutral-800 !placeholder:text-neutral-400 selection:!bg-blue-100"
         | 
| 120 | 
            +
                              />
         | 
| 121 | 
            +
                            </div>
         | 
| 122 | 
            +
                            <div>
         | 
| 123 | 
            +
                              <p className="text-sm text-neutral-700 mb-2">
         | 
| 124 | 
            +
                                Then, let's deploy it!
         | 
| 125 | 
            +
                              </p>
         | 
| 126 | 
            +
                              <Button
         | 
| 127 | 
            +
                                variant="black"
         | 
| 128 | 
            +
                                onClick={createSpace}
         | 
| 129 | 
            +
                                className="relative w-full"
         | 
| 130 | 
            +
                                disabled={loading}
         | 
| 131 | 
            +
                              >
         | 
| 132 | 
            +
                                Deploy <Rocket className="size-4" />
         | 
| 133 | 
            +
                                {loading && (
         | 
| 134 | 
            +
                                  <Loading className="ml-2 size-4 animate-spin" />
         | 
| 135 | 
            +
                                )}
         | 
| 136 | 
            +
                              </Button>
         | 
| 137 | 
            +
                            </div>
         | 
| 138 | 
            +
                          </main>
         | 
| 139 | 
            +
                        </PopoverContent>
         | 
| 140 | 
            +
                      </Popover>
         | 
| 141 | 
            +
                    ) : (
         | 
| 142 | 
            +
                      <>
         | 
| 143 | 
            +
                        <Button
         | 
| 144 | 
            +
                          variant="sky"
         | 
| 145 | 
            +
                          className="max-lg:hidden !px-4"
         | 
| 146 | 
            +
                          onClick={() => setOpen(true)}
         | 
| 147 | 
            +
                        >
         | 
| 148 | 
            +
                          <MdSave className="size-4" />
         | 
| 149 | 
            +
                          Save your Project
         | 
| 150 | 
            +
                        </Button>
         | 
| 151 | 
            +
                        <Button
         | 
| 152 | 
            +
                          variant="sky"
         | 
| 153 | 
            +
                          size="sm"
         | 
| 154 | 
            +
                          className="lg:hidden"
         | 
| 155 | 
            +
                          onClick={() => setOpen(true)}
         | 
| 156 | 
            +
                        >
         | 
| 157 | 
            +
                          Save
         | 
| 158 | 
            +
                        </Button>
         | 
| 159 | 
            +
                      </>
         | 
| 160 | 
            +
                    )}
         | 
| 161 | 
            +
                    <LoginModal
         | 
| 162 | 
            +
                      open={open}
         | 
| 163 | 
            +
                      onClose={() => setOpen(false)}
         | 
| 164 | 
            +
                      html={html}
         | 
| 165 | 
            +
                      title="Log In to save your Project"
         | 
| 166 | 
            +
                      description="Log In through your Hugging Face account to save your project and increase your monthly free limit."
         | 
| 167 | 
            +
                    />
         | 
| 168 | 
            +
                  </div>
         | 
| 169 | 
            +
                </div>
         | 
| 170 | 
            +
              );
         | 
| 171 | 
            +
            }
         | 
    	
        components/editor/footer/index.tsx
    ADDED
    
    | @@ -0,0 +1,118 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import classNames from "classnames";
         | 
| 2 | 
            +
            import { FaMobileAlt } from "react-icons/fa";
         | 
| 3 | 
            +
            import { RefreshCcw, SparkleIcon } from "lucide-react";
         | 
| 4 | 
            +
            import { FaLaptopCode } from "react-icons/fa6";
         | 
| 5 | 
            +
            import { HtmlHistory } from "@/types";
         | 
| 6 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 7 | 
            +
            import { MdAdd } from "react-icons/md";
         | 
| 8 | 
            +
            import { History } from "@/components/editor/history";
         | 
| 9 | 
            +
            import { UserMenu } from "@/components/user-menu";
         | 
| 10 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            const DEVICES = [
         | 
| 13 | 
            +
              {
         | 
| 14 | 
            +
                name: "desktop",
         | 
| 15 | 
            +
                icon: FaLaptopCode,
         | 
| 16 | 
            +
              },
         | 
| 17 | 
            +
              {
         | 
| 18 | 
            +
                name: "mobile",
         | 
| 19 | 
            +
                icon: FaMobileAlt,
         | 
| 20 | 
            +
              },
         | 
| 21 | 
            +
            ];
         | 
| 22 | 
            +
             | 
| 23 | 
            +
            export function Footer({
         | 
| 24 | 
            +
              onReset,
         | 
| 25 | 
            +
              htmlHistory,
         | 
| 26 | 
            +
              setHtml,
         | 
| 27 | 
            +
              device,
         | 
| 28 | 
            +
              setDevice,
         | 
| 29 | 
            +
              iframeRef,
         | 
| 30 | 
            +
            }: {
         | 
| 31 | 
            +
              onReset: () => void;
         | 
| 32 | 
            +
              htmlHistory?: HtmlHistory[];
         | 
| 33 | 
            +
              device: "desktop" | "mobile";
         | 
| 34 | 
            +
              setHtml: (html: string) => void;
         | 
| 35 | 
            +
              iframeRef?: React.RefObject<HTMLIFrameElement | null>;
         | 
| 36 | 
            +
              setDevice: React.Dispatch<React.SetStateAction<"desktop" | "mobile">>;
         | 
| 37 | 
            +
            }) {
         | 
| 38 | 
            +
              const { user } = useUser();
         | 
| 39 | 
            +
             | 
| 40 | 
            +
              const handleRefreshIframe = () => {
         | 
| 41 | 
            +
                if (iframeRef?.current) {
         | 
| 42 | 
            +
                  const iframe = iframeRef.current;
         | 
| 43 | 
            +
                  const content = iframe.srcdoc;
         | 
| 44 | 
            +
                  iframe.srcdoc = "";
         | 
| 45 | 
            +
                  setTimeout(() => {
         | 
| 46 | 
            +
                    iframe.srcdoc = content;
         | 
| 47 | 
            +
                  }, 10);
         | 
| 48 | 
            +
                }
         | 
| 49 | 
            +
              };
         | 
| 50 | 
            +
             | 
| 51 | 
            +
              return (
         | 
| 52 | 
            +
                <footer className="border-t bg-slate-200 border-slate-300 dark:bg-neutral-950 dark:border-neutral-800 px-3 py-2 flex items-center justify-between sticky bottom-0 z-20">
         | 
| 53 | 
            +
                  <div className="flex items-center gap-2">
         | 
| 54 | 
            +
                    {user &&
         | 
| 55 | 
            +
                      (user?.isLocalUse ? (
         | 
| 56 | 
            +
                        <>
         | 
| 57 | 
            +
                          <div className="max-w-max bg-amber-500/10 rounded-full px-3 py-1 text-amber-500 border border-amber-500/20 text-sm font-semibold">
         | 
| 58 | 
            +
                            Local Usage
         | 
| 59 | 
            +
                          </div>
         | 
| 60 | 
            +
                        </>
         | 
| 61 | 
            +
                      ) : (
         | 
| 62 | 
            +
                        <UserMenu />
         | 
| 63 | 
            +
                      ))}
         | 
| 64 | 
            +
                    {user && <p className="text-neutral-700">|</p>}
         | 
| 65 | 
            +
                    <Button size="sm" variant="secondary" onClick={onReset}>
         | 
| 66 | 
            +
                      <MdAdd className="text-sm" />
         | 
| 67 | 
            +
                      <span>New Project</span>
         | 
| 68 | 
            +
                    </Button>
         | 
| 69 | 
            +
                    {htmlHistory && htmlHistory.length > 0 && (
         | 
| 70 | 
            +
                      <>
         | 
| 71 | 
            +
                        <p className="text-neutral-700">|</p>
         | 
| 72 | 
            +
                        <History history={htmlHistory} setHtml={setHtml} />
         | 
| 73 | 
            +
                      </>
         | 
| 74 | 
            +
                    )}
         | 
| 75 | 
            +
                  </div>
         | 
| 76 | 
            +
                  <div className="flex justify-end items-center gap-2.5">
         | 
| 77 | 
            +
                    <a
         | 
| 78 | 
            +
                      href="https://huggingface.co/spaces/victor/deepsite-gallery"
         | 
| 79 | 
            +
                      target="_blank"
         | 
| 80 | 
            +
                    >
         | 
| 81 | 
            +
                      <Button size="sm" variant="ghost">
         | 
| 82 | 
            +
                        <SparkleIcon className="size-3.5" />
         | 
| 83 | 
            +
                        <span className="max-lg:hidden">DeepSite Gallery</span>
         | 
| 84 | 
            +
                      </Button>
         | 
| 85 | 
            +
                    </a>
         | 
| 86 | 
            +
                    <Button size="sm" variant="default" onClick={handleRefreshIframe}>
         | 
| 87 | 
            +
                      <RefreshCcw className="size-3.5" />
         | 
| 88 | 
            +
                      <span className="max-lg:hidden">Refresh Preview</span>
         | 
| 89 | 
            +
                    </Button>
         | 
| 90 | 
            +
                    <div className="flex items-center rounded-full p-0.5 bg-neutral-700/70 relative overflow-hidden z-0 max-lg:hidden gap-0.5">
         | 
| 91 | 
            +
                      <div
         | 
| 92 | 
            +
                        className={classNames(
         | 
| 93 | 
            +
                          "absolute left-0.5 top-0.5 rounded-full bg-white size-7 -z-[1] transition-all duration-200",
         | 
| 94 | 
            +
                          {
         | 
| 95 | 
            +
                            "translate-x-[calc(100%+2px)]": device === "mobile",
         | 
| 96 | 
            +
                          }
         | 
| 97 | 
            +
                        )}
         | 
| 98 | 
            +
                      />
         | 
| 99 | 
            +
                      {DEVICES.map((deviceItem) => (
         | 
| 100 | 
            +
                        <button
         | 
| 101 | 
            +
                          key={deviceItem.name}
         | 
| 102 | 
            +
                          className={classNames(
         | 
| 103 | 
            +
                            "rounded-full text-neutral-300 size-7 flex items-center justify-center cursor-pointer",
         | 
| 104 | 
            +
                            {
         | 
| 105 | 
            +
                              "!text-black": device === deviceItem.name,
         | 
| 106 | 
            +
                              "hover:bg-neutral-800": device !== deviceItem.name,
         | 
| 107 | 
            +
                            }
         | 
| 108 | 
            +
                          )}
         | 
| 109 | 
            +
                          onClick={() => setDevice(deviceItem.name as "desktop" | "mobile")}
         | 
| 110 | 
            +
                        >
         | 
| 111 | 
            +
                          <deviceItem.icon className="text-sm" />
         | 
| 112 | 
            +
                        </button>
         | 
| 113 | 
            +
                      ))}
         | 
| 114 | 
            +
                    </div>
         | 
| 115 | 
            +
                  </div>
         | 
| 116 | 
            +
                </footer>
         | 
| 117 | 
            +
              );
         | 
| 118 | 
            +
            }
         | 
    	
        components/editor/header/index.tsx
    ADDED
    
    | @@ -0,0 +1,69 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { ReactNode } from "react";
         | 
| 2 | 
            +
            import { Eye, MessageCircleCode } from "lucide-react";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            import Logo from "@/assets/logo.svg";
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 7 | 
            +
            import classNames from "classnames";
         | 
| 8 | 
            +
            import Image from "next/image";
         | 
| 9 | 
            +
             | 
| 10 | 
            +
            const TABS = [
         | 
| 11 | 
            +
              {
         | 
| 12 | 
            +
                value: "chat",
         | 
| 13 | 
            +
                label: "Chat",
         | 
| 14 | 
            +
                icon: MessageCircleCode,
         | 
| 15 | 
            +
              },
         | 
| 16 | 
            +
              {
         | 
| 17 | 
            +
                value: "preview",
         | 
| 18 | 
            +
                label: "Preview",
         | 
| 19 | 
            +
                icon: Eye,
         | 
| 20 | 
            +
              },
         | 
| 21 | 
            +
            ];
         | 
| 22 | 
            +
             | 
| 23 | 
            +
            export function Header({
         | 
| 24 | 
            +
              tab,
         | 
| 25 | 
            +
              onNewTab,
         | 
| 26 | 
            +
              children,
         | 
| 27 | 
            +
            }: {
         | 
| 28 | 
            +
              tab: string;
         | 
| 29 | 
            +
              onNewTab: (tab: string) => void;
         | 
| 30 | 
            +
              children?: ReactNode;
         | 
| 31 | 
            +
            }) {
         | 
| 32 | 
            +
              return (
         | 
| 33 | 
            +
                <header className="border-b bg-slate-200 border-slate-300 dark:bg-neutral-950 dark:border-neutral-800 px-3 lg:px-6 py-2 grid grid-cols-3 sticky top-0 z-20">
         | 
| 34 | 
            +
                  <div className="flex items-center justify-start gap-3">
         | 
| 35 | 
            +
                    <h1 className="text-neutral-900 dark:text-white text-lg lg:text-xl font-bold flex items-center justify-start">
         | 
| 36 | 
            +
                      <Image
         | 
| 37 | 
            +
                        src={Logo}
         | 
| 38 | 
            +
                        alt="DeepSite Logo"
         | 
| 39 | 
            +
                        className="size-6 lg:size-8 mr-2 invert-100 dark:invert-0"
         | 
| 40 | 
            +
                      />
         | 
| 41 | 
            +
                      <p className="max-md:hidden flex items-center justify-start">
         | 
| 42 | 
            +
                        DeepSite
         | 
| 43 | 
            +
                        <span className="font-mono bg-gradient-to-br from-sky-500 to-emerald-500 text-neutral-950 rounded-full text-xs ml-2 px-1.5 py-0.5">
         | 
| 44 | 
            +
                          {" "}
         | 
| 45 | 
            +
                          v2
         | 
| 46 | 
            +
                        </span>
         | 
| 47 | 
            +
                      </p>
         | 
| 48 | 
            +
                    </h1>
         | 
| 49 | 
            +
                  </div>
         | 
| 50 | 
            +
                  <div className="flex items-center justify-center gap-1">
         | 
| 51 | 
            +
                    {TABS.map((item) => (
         | 
| 52 | 
            +
                      <Button
         | 
| 53 | 
            +
                        key={item.value}
         | 
| 54 | 
            +
                        variant={tab === item.value ? "secondary" : "ghost"}
         | 
| 55 | 
            +
                        className={classNames("", {
         | 
| 56 | 
            +
                          "opacity-60": tab !== item.value,
         | 
| 57 | 
            +
                        })}
         | 
| 58 | 
            +
                        size="sm"
         | 
| 59 | 
            +
                        onClick={() => onNewTab(item.value)}
         | 
| 60 | 
            +
                      >
         | 
| 61 | 
            +
                        <item.icon className="size-4" />
         | 
| 62 | 
            +
                        <span className="hidden md:inline">{item.label}</span>
         | 
| 63 | 
            +
                      </Button>
         | 
| 64 | 
            +
                    ))}
         | 
| 65 | 
            +
                  </div>
         | 
| 66 | 
            +
                  <div className="flex items-center justify-end gap-3">{children}</div>
         | 
| 67 | 
            +
                </header>
         | 
| 68 | 
            +
              );
         | 
| 69 | 
            +
            }
         | 
    	
        components/editor/history/index.tsx
    ADDED
    
    | @@ -0,0 +1,69 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { HtmlHistory } from "@/types";
         | 
| 2 | 
            +
            import {
         | 
| 3 | 
            +
              Popover,
         | 
| 4 | 
            +
              PopoverContent,
         | 
| 5 | 
            +
              PopoverTrigger,
         | 
| 6 | 
            +
            } from "@/components/ui/popover";
         | 
| 7 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            export function History({
         | 
| 10 | 
            +
              history,
         | 
| 11 | 
            +
              setHtml,
         | 
| 12 | 
            +
            }: {
         | 
| 13 | 
            +
              history: HtmlHistory[];
         | 
| 14 | 
            +
              setHtml: (html: string) => void;
         | 
| 15 | 
            +
            }) {
         | 
| 16 | 
            +
              return (
         | 
| 17 | 
            +
                <Popover>
         | 
| 18 | 
            +
                  <PopoverTrigger asChild>
         | 
| 19 | 
            +
                    <Button variant="ghost" size="sm" className="max-lg:hidden">
         | 
| 20 | 
            +
                      {history?.length} edit{history.length !== 1 ? "s" : ""}
         | 
| 21 | 
            +
                    </Button>
         | 
| 22 | 
            +
                  </PopoverTrigger>
         | 
| 23 | 
            +
                  <PopoverContent
         | 
| 24 | 
            +
                    className="!rounded-2xl !p-0 overflow-hidden !bg-neutral-900"
         | 
| 25 | 
            +
                    align="start"
         | 
| 26 | 
            +
                  >
         | 
| 27 | 
            +
                    <header className="text-sm px-4 py-3 border-b gap-2 bg-neutral-950 border-neutral-800 font-semibold text-neutral-200">
         | 
| 28 | 
            +
                      History
         | 
| 29 | 
            +
                    </header>
         | 
| 30 | 
            +
                    <main className="px-4 space-y-3">
         | 
| 31 | 
            +
                      <ul className="max-h-[250px] overflow-y-auto">
         | 
| 32 | 
            +
                        {history?.map((item, index) => (
         | 
| 33 | 
            +
                          <li
         | 
| 34 | 
            +
                            key={index}
         | 
| 35 | 
            +
                            className="text-gray-300 text-xs py-2 border-b border-gray-800 last:border-0 flex items-center justify-between gap-2"
         | 
| 36 | 
            +
                          >
         | 
| 37 | 
            +
                            <div className="">
         | 
| 38 | 
            +
                              <span className="line-clamp-1">{item.prompt}</span>
         | 
| 39 | 
            +
                              <span className="text-gray-500 text-[10px]">
         | 
| 40 | 
            +
                                {new Date(item.createdAt).toLocaleDateString("en-US", {
         | 
| 41 | 
            +
                                  month: "2-digit",
         | 
| 42 | 
            +
                                  day: "2-digit",
         | 
| 43 | 
            +
                                  year: "2-digit",
         | 
| 44 | 
            +
                                }) +
         | 
| 45 | 
            +
                                  " " +
         | 
| 46 | 
            +
                                  new Date(item.createdAt).toLocaleTimeString("en-US", {
         | 
| 47 | 
            +
                                    hour: "2-digit",
         | 
| 48 | 
            +
                                    minute: "2-digit",
         | 
| 49 | 
            +
                                    second: "2-digit",
         | 
| 50 | 
            +
                                    hour12: false,
         | 
| 51 | 
            +
                                  })}
         | 
| 52 | 
            +
                              </span>
         | 
| 53 | 
            +
                            </div>
         | 
| 54 | 
            +
                            <button
         | 
| 55 | 
            +
                              className="bg-pink-500 text-white text-xs font-medium rounded-md px-2 py-1 transition-all duration-100 hover:bg-pink-600 cursor-pointer"
         | 
| 56 | 
            +
                              onClick={() => {
         | 
| 57 | 
            +
                                setHtml(item.html);
         | 
| 58 | 
            +
                              }}
         | 
| 59 | 
            +
                            >
         | 
| 60 | 
            +
                              Select
         | 
| 61 | 
            +
                            </button>
         | 
| 62 | 
            +
                          </li>
         | 
| 63 | 
            +
                        ))}
         | 
| 64 | 
            +
                      </ul>
         | 
| 65 | 
            +
                    </main>
         | 
| 66 | 
            +
                  </PopoverContent>
         | 
| 67 | 
            +
                </Popover>
         | 
| 68 | 
            +
              );
         | 
| 69 | 
            +
            }
         | 
    	
        components/editor/index.tsx
    ADDED
    
    | @@ -0,0 +1,311 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
            import { useRef, useState } from "react";
         | 
| 3 | 
            +
            import { toast } from "sonner";
         | 
| 4 | 
            +
            import { editor } from "monaco-editor";
         | 
| 5 | 
            +
            import Editor from "@monaco-editor/react";
         | 
| 6 | 
            +
            import { CopyIcon } from "lucide-react";
         | 
| 7 | 
            +
            import {
         | 
| 8 | 
            +
              useCopyToClipboard,
         | 
| 9 | 
            +
              useEvent,
         | 
| 10 | 
            +
              useLocalStorage,
         | 
| 11 | 
            +
              useMount,
         | 
| 12 | 
            +
              useUnmount,
         | 
| 13 | 
            +
              useUpdateEffect,
         | 
| 14 | 
            +
            } from "react-use";
         | 
| 15 | 
            +
            import classNames from "classnames";
         | 
| 16 | 
            +
            import { useRouter, useSearchParams } from "next/navigation";
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            import { Header } from "@/components/editor/header";
         | 
| 19 | 
            +
            import { Footer } from "@/components/editor/footer";
         | 
| 20 | 
            +
            import { defaultHTML } from "@/lib/consts";
         | 
| 21 | 
            +
            import { Preview } from "@/components/editor/preview";
         | 
| 22 | 
            +
            import { useEditor } from "@/hooks/useEditor";
         | 
| 23 | 
            +
            import { AskAI } from "@/components/editor/ask-ai";
         | 
| 24 | 
            +
            import { DeployButton } from "./deploy-button";
         | 
| 25 | 
            +
            import { Project } from "@/types";
         | 
| 26 | 
            +
            import { SaveButton } from "./save-button";
         | 
| 27 | 
            +
             | 
| 28 | 
            +
            export const AppEditor = ({ project }: { project?: Project | null }) => {
         | 
| 29 | 
            +
              const [htmlStorage, , removeHtmlStorage] = useLocalStorage("html_content");
         | 
| 30 | 
            +
              const [, copyToClipboard] = useCopyToClipboard();
         | 
| 31 | 
            +
              const { html, setHtml, htmlHistory, setHtmlHistory, prompts, setPrompts } =
         | 
| 32 | 
            +
                useEditor(project?.html ?? (htmlStorage as string) ?? defaultHTML);
         | 
| 33 | 
            +
              // get query params from URL
         | 
| 34 | 
            +
              const searchParams = useSearchParams();
         | 
| 35 | 
            +
              const router = useRouter();
         | 
| 36 | 
            +
              const deploy = searchParams.get("deploy") === "true";
         | 
| 37 | 
            +
             | 
| 38 | 
            +
              const iframeRef = useRef<HTMLIFrameElement | null>(null);
         | 
| 39 | 
            +
              const preview = useRef<HTMLDivElement>(null);
         | 
| 40 | 
            +
              const editor = useRef<HTMLDivElement>(null);
         | 
| 41 | 
            +
              const editorRef = useRef<editor.IStandaloneCodeEditor | null>(null);
         | 
| 42 | 
            +
              const resizer = useRef<HTMLDivElement>(null);
         | 
| 43 | 
            +
              // eslint-disable-next-line @typescript-eslint/no-explicit-any
         | 
| 44 | 
            +
              const monacoRef = useRef<any>(null);
         | 
| 45 | 
            +
             | 
| 46 | 
            +
              const [currentTab, setCurrentTab] = useState("chat");
         | 
| 47 | 
            +
              const [device, setDevice] = useState<"desktop" | "mobile">("desktop");
         | 
| 48 | 
            +
              const [isResizing, setIsResizing] = useState(false);
         | 
| 49 | 
            +
              const [isAiWorking, setIsAiWorking] = useState(false);
         | 
| 50 | 
            +
             | 
| 51 | 
            +
              /**
         | 
| 52 | 
            +
               * Resets the layout based on screen size
         | 
| 53 | 
            +
               * - For desktop: Sets editor to 1/3 width and preview to 2/3
         | 
| 54 | 
            +
               * - For mobile: Removes inline styles to let CSS handle it
         | 
| 55 | 
            +
               */
         | 
| 56 | 
            +
              const resetLayout = () => {
         | 
| 57 | 
            +
                if (!editor.current || !preview.current) return;
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                // lg breakpoint is 1024px based on useBreakpoint definition and Tailwind defaults
         | 
| 60 | 
            +
                if (window.innerWidth >= 1024) {
         | 
| 61 | 
            +
                  // Set initial 1/3 - 2/3 sizes for large screens, accounting for resizer width
         | 
| 62 | 
            +
                  const resizerWidth = resizer.current?.offsetWidth ?? 8; // w-2 = 0.5rem = 8px
         | 
| 63 | 
            +
                  const availableWidth = window.innerWidth - resizerWidth;
         | 
| 64 | 
            +
                  const initialEditorWidth = availableWidth / 3; // Editor takes 1/3 of space
         | 
| 65 | 
            +
                  const initialPreviewWidth = availableWidth - initialEditorWidth; // Preview takes 2/3
         | 
| 66 | 
            +
                  editor.current.style.width = `${initialEditorWidth}px`;
         | 
| 67 | 
            +
                  preview.current.style.width = `${initialPreviewWidth}px`;
         | 
| 68 | 
            +
                } else {
         | 
| 69 | 
            +
                  // Remove inline styles for smaller screens, let CSS flex-col handle it
         | 
| 70 | 
            +
                  editor.current.style.width = "";
         | 
| 71 | 
            +
                  preview.current.style.width = "";
         | 
| 72 | 
            +
                }
         | 
| 73 | 
            +
              };
         | 
| 74 | 
            +
             | 
| 75 | 
            +
              /**
         | 
| 76 | 
            +
               * Handles resizing when the user drags the resizer
         | 
| 77 | 
            +
               * Ensures minimum widths are maintained for both panels
         | 
| 78 | 
            +
               */
         | 
| 79 | 
            +
              const handleResize = (e: MouseEvent) => {
         | 
| 80 | 
            +
                if (!editor.current || !preview.current || !resizer.current) return;
         | 
| 81 | 
            +
             | 
| 82 | 
            +
                const resizerWidth = resizer.current.offsetWidth;
         | 
| 83 | 
            +
                const minWidth = 100; // Minimum width for editor/preview
         | 
| 84 | 
            +
                const maxWidth = window.innerWidth - resizerWidth - minWidth;
         | 
| 85 | 
            +
             | 
| 86 | 
            +
                const editorWidth = e.clientX;
         | 
| 87 | 
            +
                const clampedEditorWidth = Math.max(
         | 
| 88 | 
            +
                  minWidth,
         | 
| 89 | 
            +
                  Math.min(editorWidth, maxWidth)
         | 
| 90 | 
            +
                );
         | 
| 91 | 
            +
                const calculatedPreviewWidth =
         | 
| 92 | 
            +
                  window.innerWidth - clampedEditorWidth - resizerWidth;
         | 
| 93 | 
            +
             | 
| 94 | 
            +
                editor.current.style.width = `${clampedEditorWidth}px`;
         | 
| 95 | 
            +
                preview.current.style.width = `${calculatedPreviewWidth}px`;
         | 
| 96 | 
            +
              };
         | 
| 97 | 
            +
             | 
| 98 | 
            +
              const handleMouseDown = () => {
         | 
| 99 | 
            +
                setIsResizing(true);
         | 
| 100 | 
            +
                document.addEventListener("mousemove", handleResize);
         | 
| 101 | 
            +
                document.addEventListener("mouseup", handleMouseUp);
         | 
| 102 | 
            +
              };
         | 
| 103 | 
            +
             | 
| 104 | 
            +
              const handleMouseUp = () => {
         | 
| 105 | 
            +
                setIsResizing(false);
         | 
| 106 | 
            +
                document.removeEventListener("mousemove", handleResize);
         | 
| 107 | 
            +
                document.removeEventListener("mouseup", handleMouseUp);
         | 
| 108 | 
            +
              };
         | 
| 109 | 
            +
             | 
| 110 | 
            +
              useMount(() => {
         | 
| 111 | 
            +
                if (deploy && project?._id) {
         | 
| 112 | 
            +
                  toast.success("Your project is deployed! 🎉", {
         | 
| 113 | 
            +
                    action: {
         | 
| 114 | 
            +
                      label: "See Project",
         | 
| 115 | 
            +
                      onClick: () => {
         | 
| 116 | 
            +
                        window.open(
         | 
| 117 | 
            +
                          `https://huggingface.co/spaces/${project?.space_id}`,
         | 
| 118 | 
            +
                          "_blank"
         | 
| 119 | 
            +
                        );
         | 
| 120 | 
            +
                      },
         | 
| 121 | 
            +
                    },
         | 
| 122 | 
            +
                  });
         | 
| 123 | 
            +
                  router.replace(`/projects/${project?.space_id}`);
         | 
| 124 | 
            +
                }
         | 
| 125 | 
            +
                if (htmlStorage) {
         | 
| 126 | 
            +
                  removeHtmlStorage();
         | 
| 127 | 
            +
                  toast.warning("Previous HTML content restored from local storage.");
         | 
| 128 | 
            +
                }
         | 
| 129 | 
            +
             | 
| 130 | 
            +
                resetLayout();
         | 
| 131 | 
            +
                if (!resizer.current) return;
         | 
| 132 | 
            +
                resizer.current.addEventListener("mousedown", handleMouseDown);
         | 
| 133 | 
            +
                window.addEventListener("resize", resetLayout);
         | 
| 134 | 
            +
              });
         | 
| 135 | 
            +
              useUnmount(() => {
         | 
| 136 | 
            +
                document.removeEventListener("mousemove", handleResize);
         | 
| 137 | 
            +
                document.removeEventListener("mouseup", handleMouseUp);
         | 
| 138 | 
            +
                if (resizer.current) {
         | 
| 139 | 
            +
                  resizer.current.removeEventListener("mousedown", handleMouseDown);
         | 
| 140 | 
            +
                }
         | 
| 141 | 
            +
                window.removeEventListener("resize", resetLayout);
         | 
| 142 | 
            +
              });
         | 
| 143 | 
            +
             | 
| 144 | 
            +
              // Prevent accidental navigation away when AI is working or content has changed
         | 
| 145 | 
            +
              useEvent("beforeunload", (e) => {
         | 
| 146 | 
            +
                if (isAiWorking || html !== defaultHTML) {
         | 
| 147 | 
            +
                  e.preventDefault();
         | 
| 148 | 
            +
                  return "";
         | 
| 149 | 
            +
                }
         | 
| 150 | 
            +
              });
         | 
| 151 | 
            +
             | 
| 152 | 
            +
              useUpdateEffect(() => {
         | 
| 153 | 
            +
                if (currentTab === "chat") {
         | 
| 154 | 
            +
                  // Reset editor width when switching to reasoning tab
         | 
| 155 | 
            +
                  resetLayout();
         | 
| 156 | 
            +
                  // re-add the event listener for resizing
         | 
| 157 | 
            +
                  if (resizer.current) {
         | 
| 158 | 
            +
                    resizer.current.addEventListener("mousedown", handleMouseDown);
         | 
| 159 | 
            +
                  }
         | 
| 160 | 
            +
                } else {
         | 
| 161 | 
            +
                  if (preview.current) {
         | 
| 162 | 
            +
                    // Reset preview width when switching to preview tab
         | 
| 163 | 
            +
                    preview.current.style.width = "100%";
         | 
| 164 | 
            +
                  }
         | 
| 165 | 
            +
                }
         | 
| 166 | 
            +
              }, [currentTab]);
         | 
| 167 | 
            +
             | 
| 168 | 
            +
              return (
         | 
| 169 | 
            +
                <section className="h-screen bg-slate-100 dark:bg-neutral-950 flex flex-col">
         | 
| 170 | 
            +
                  <Header tab={currentTab} onNewTab={setCurrentTab}>
         | 
| 171 | 
            +
                    {project?._id ? (
         | 
| 172 | 
            +
                      <SaveButton html={html} prompts={prompts} />
         | 
| 173 | 
            +
                    ) : (
         | 
| 174 | 
            +
                      <DeployButton html={html} prompts={prompts} />
         | 
| 175 | 
            +
                    )}
         | 
| 176 | 
            +
                  </Header>
         | 
| 177 | 
            +
                  <main className="bg-neutral-950 flex-1 max-lg:flex-col flex w-full">
         | 
| 178 | 
            +
                    {currentTab === "chat" && (
         | 
| 179 | 
            +
                      <>
         | 
| 180 | 
            +
                        <div ref={editor} className="relative">
         | 
| 181 | 
            +
                          <CopyIcon
         | 
| 182 | 
            +
                            className="size-4 absolute top-2 right-5 text-neutral-500 hover:text-neutral-300 z-2 cursor-pointer"
         | 
| 183 | 
            +
                            onClick={() => {
         | 
| 184 | 
            +
                              copyToClipboard(html);
         | 
| 185 | 
            +
                              toast.success("HTML copied to clipboard!");
         | 
| 186 | 
            +
                            }}
         | 
| 187 | 
            +
                          />
         | 
| 188 | 
            +
                          <Editor
         | 
| 189 | 
            +
                            language="html"
         | 
| 190 | 
            +
                            theme="vs-dark"
         | 
| 191 | 
            +
                            className={classNames(
         | 
| 192 | 
            +
                              "h-[calc(100dvh-98px)] lg:h-full bg-neutral-900 transition-all duration-200 ",
         | 
| 193 | 
            +
                              {
         | 
| 194 | 
            +
                                "pointer-events-none": isAiWorking,
         | 
| 195 | 
            +
                              }
         | 
| 196 | 
            +
                            )}
         | 
| 197 | 
            +
                            options={{
         | 
| 198 | 
            +
                              colorDecorators: true,
         | 
| 199 | 
            +
                              fontLigatures: true,
         | 
| 200 | 
            +
                              theme: "vs-dark",
         | 
| 201 | 
            +
                              minimap: { enabled: false },
         | 
| 202 | 
            +
                              scrollbar: {
         | 
| 203 | 
            +
                                horizontal: "hidden",
         | 
| 204 | 
            +
                              },
         | 
| 205 | 
            +
                            }}
         | 
| 206 | 
            +
                            value={html}
         | 
| 207 | 
            +
                            onChange={(value) => {
         | 
| 208 | 
            +
                              const newValue = value ?? "";
         | 
| 209 | 
            +
                              setHtml(newValue);
         | 
| 210 | 
            +
                            }}
         | 
| 211 | 
            +
                            onMount={(editor, monaco) => {
         | 
| 212 | 
            +
                              editorRef.current = editor;
         | 
| 213 | 
            +
                              monacoRef.current = monaco;
         | 
| 214 | 
            +
                            }}
         | 
| 215 | 
            +
                          />
         | 
| 216 | 
            +
                          <AskAI
         | 
| 217 | 
            +
                            html={html}
         | 
| 218 | 
            +
                            setHtml={(newHtml: string) => {
         | 
| 219 | 
            +
                              setHtml(newHtml);
         | 
| 220 | 
            +
                            }}
         | 
| 221 | 
            +
                            htmlHistory={htmlHistory}
         | 
| 222 | 
            +
                            onSuccess={(
         | 
| 223 | 
            +
                              finalHtml: string,
         | 
| 224 | 
            +
                              p: string,
         | 
| 225 | 
            +
                              updatedLines?: number[][]
         | 
| 226 | 
            +
                            ) => {
         | 
| 227 | 
            +
                              const currentHistory = [...htmlHistory];
         | 
| 228 | 
            +
                              currentHistory.unshift({
         | 
| 229 | 
            +
                                html: finalHtml,
         | 
| 230 | 
            +
                                createdAt: new Date(),
         | 
| 231 | 
            +
                                prompt: p,
         | 
| 232 | 
            +
                              });
         | 
| 233 | 
            +
                              setHtmlHistory(currentHistory);
         | 
| 234 | 
            +
                              // if xs or sm
         | 
| 235 | 
            +
                              if (window.innerWidth <= 1024) {
         | 
| 236 | 
            +
                                setCurrentTab("preview");
         | 
| 237 | 
            +
                              }
         | 
| 238 | 
            +
                              if (updatedLines && updatedLines?.length > 0) {
         | 
| 239 | 
            +
                                const decorations = updatedLines.map((line) => ({
         | 
| 240 | 
            +
                                  range: new monacoRef.current.Range(
         | 
| 241 | 
            +
                                    line[0],
         | 
| 242 | 
            +
                                    1,
         | 
| 243 | 
            +
                                    line[1],
         | 
| 244 | 
            +
                                    1
         | 
| 245 | 
            +
                                  ),
         | 
| 246 | 
            +
                                  options: {
         | 
| 247 | 
            +
                                    inlineClassName: "matched-line",
         | 
| 248 | 
            +
                                  },
         | 
| 249 | 
            +
                                }));
         | 
| 250 | 
            +
                                setTimeout(() => {
         | 
| 251 | 
            +
                                  editorRef?.current
         | 
| 252 | 
            +
                                    ?.getModel()
         | 
| 253 | 
            +
                                    ?.deltaDecorations([], decorations);
         | 
| 254 | 
            +
             | 
| 255 | 
            +
                                  editorRef.current?.revealLine(updatedLines[0][0]);
         | 
| 256 | 
            +
                                }, 100);
         | 
| 257 | 
            +
                              }
         | 
| 258 | 
            +
                            }}
         | 
| 259 | 
            +
                            isAiWorking={isAiWorking}
         | 
| 260 | 
            +
                            setisAiWorking={setIsAiWorking}
         | 
| 261 | 
            +
                            onNewPrompt={(prompt: string) => {
         | 
| 262 | 
            +
                              setPrompts((prev) => [...prev, prompt]);
         | 
| 263 | 
            +
                            }}
         | 
| 264 | 
            +
                            onScrollToBottom={() => {
         | 
| 265 | 
            +
                              editorRef.current?.revealLine(
         | 
| 266 | 
            +
                                editorRef.current?.getModel()?.getLineCount() ?? 0
         | 
| 267 | 
            +
                              );
         | 
| 268 | 
            +
                            }}
         | 
| 269 | 
            +
                          />
         | 
| 270 | 
            +
                        </div>
         | 
| 271 | 
            +
                        <div
         | 
| 272 | 
            +
                          ref={resizer}
         | 
| 273 | 
            +
                          className="bg-neutral-800 hover:bg-sky-500 active:bg-sky-500 w-1.5 cursor-col-resize h-full max-lg:hidden"
         | 
| 274 | 
            +
                        />
         | 
| 275 | 
            +
                      </>
         | 
| 276 | 
            +
                    )}
         | 
| 277 | 
            +
                    <Preview
         | 
| 278 | 
            +
                      html={html}
         | 
| 279 | 
            +
                      isResizing={isResizing}
         | 
| 280 | 
            +
                      isAiWorking={isAiWorking}
         | 
| 281 | 
            +
                      ref={preview}
         | 
| 282 | 
            +
                      device={device}
         | 
| 283 | 
            +
                      currentTab={currentTab}
         | 
| 284 | 
            +
                      iframeRef={iframeRef}
         | 
| 285 | 
            +
                    />
         | 
| 286 | 
            +
                  </main>
         | 
| 287 | 
            +
                  <Footer
         | 
| 288 | 
            +
                    onReset={() => {
         | 
| 289 | 
            +
                      if (isAiWorking) {
         | 
| 290 | 
            +
                        toast.warning("Please wait for the AI to finish working.");
         | 
| 291 | 
            +
                        return;
         | 
| 292 | 
            +
                      }
         | 
| 293 | 
            +
                      if (
         | 
| 294 | 
            +
                        window.confirm("You're about to reset the editor. Are you sure?")
         | 
| 295 | 
            +
                      ) {
         | 
| 296 | 
            +
                        setHtml(defaultHTML);
         | 
| 297 | 
            +
                        removeHtmlStorage();
         | 
| 298 | 
            +
                        editorRef.current?.revealLine(
         | 
| 299 | 
            +
                          editorRef.current?.getModel()?.getLineCount() ?? 0
         | 
| 300 | 
            +
                        );
         | 
| 301 | 
            +
                      }
         | 
| 302 | 
            +
                    }}
         | 
| 303 | 
            +
                    htmlHistory={htmlHistory}
         | 
| 304 | 
            +
                    setHtml={setHtml}
         | 
| 305 | 
            +
                    iframeRef={iframeRef}
         | 
| 306 | 
            +
                    device={device}
         | 
| 307 | 
            +
                    setDevice={setDevice}
         | 
| 308 | 
            +
                  />
         | 
| 309 | 
            +
                </section>
         | 
| 310 | 
            +
              );
         | 
| 311 | 
            +
            };
         | 
    	
        components/editor/preview/index.tsx
    ADDED
    
    | @@ -0,0 +1,78 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
            import classNames from "classnames";
         | 
| 3 | 
            +
            import { toast } from "sonner";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            import { cn } from "@/lib/utils";
         | 
| 6 | 
            +
            import { GridPattern } from "@/components/magic-ui/grid-pattern";
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            export const Preview = ({
         | 
| 9 | 
            +
              html,
         | 
| 10 | 
            +
              isResizing,
         | 
| 11 | 
            +
              isAiWorking,
         | 
| 12 | 
            +
              ref,
         | 
| 13 | 
            +
              device,
         | 
| 14 | 
            +
              currentTab,
         | 
| 15 | 
            +
              iframeRef,
         | 
| 16 | 
            +
            }: {
         | 
| 17 | 
            +
              html: string;
         | 
| 18 | 
            +
              isResizing: boolean;
         | 
| 19 | 
            +
              isAiWorking: boolean;
         | 
| 20 | 
            +
              ref: React.RefObject<HTMLDivElement | null>;
         | 
| 21 | 
            +
              iframeRef?: React.RefObject<HTMLIFrameElement | null>;
         | 
| 22 | 
            +
              device: "desktop" | "mobile";
         | 
| 23 | 
            +
              currentTab: string;
         | 
| 24 | 
            +
            }) => {
         | 
| 25 | 
            +
              return (
         | 
| 26 | 
            +
                <div
         | 
| 27 | 
            +
                  ref={ref}
         | 
| 28 | 
            +
                  className={classNames(
         | 
| 29 | 
            +
                    "w-full border-l border-gray-900 h-full relative z-0 flex items-center justify-center",
         | 
| 30 | 
            +
                    {
         | 
| 31 | 
            +
                      "lg:p-4": currentTab !== "preview",
         | 
| 32 | 
            +
                    }
         | 
| 33 | 
            +
                  )}
         | 
| 34 | 
            +
                  onClick={(e) => {
         | 
| 35 | 
            +
                    if (isAiWorking) {
         | 
| 36 | 
            +
                      e.preventDefault();
         | 
| 37 | 
            +
                      e.stopPropagation();
         | 
| 38 | 
            +
                      toast.warning("Please wait for the AI to finish working.");
         | 
| 39 | 
            +
                    }
         | 
| 40 | 
            +
                  }}
         | 
| 41 | 
            +
                >
         | 
| 42 | 
            +
                  <GridPattern
         | 
| 43 | 
            +
                    x={-1}
         | 
| 44 | 
            +
                    y={-1}
         | 
| 45 | 
            +
                    strokeDasharray={"4 2"}
         | 
| 46 | 
            +
                    className={cn(
         | 
| 47 | 
            +
                      "[mask-image:radial-gradient(900px_circle_at_center,white,transparent)]"
         | 
| 48 | 
            +
                    )}
         | 
| 49 | 
            +
                  />
         | 
| 50 | 
            +
                  <iframe
         | 
| 51 | 
            +
                    id="preview-iframe"
         | 
| 52 | 
            +
                    ref={iframeRef}
         | 
| 53 | 
            +
                    title="output"
         | 
| 54 | 
            +
                    className={classNames(
         | 
| 55 | 
            +
                      "w-full select-none transition-all duration-200 bg-black max-lg:h-full",
         | 
| 56 | 
            +
                      {
         | 
| 57 | 
            +
                        "pointer-events-none": isResizing || isAiWorking,
         | 
| 58 | 
            +
                        "lg:max-w-md lg:mx-auto lg:h-[80dvh] lg:!rounded-[42px] lg:border-[8px] lg:border-neutral-700 lg:shadow-2xl":
         | 
| 59 | 
            +
                          device === "mobile",
         | 
| 60 | 
            +
                        "h-full": device === "desktop",
         | 
| 61 | 
            +
                        "lg:border-[8px] lg:border-neutral-700 lg:shadow-2xl lg:rounded-[24px]":
         | 
| 62 | 
            +
                          currentTab !== "preview" && device === "desktop",
         | 
| 63 | 
            +
                      }
         | 
| 64 | 
            +
                    )}
         | 
| 65 | 
            +
                    srcDoc={html}
         | 
| 66 | 
            +
                    onLoad={() => {
         | 
| 67 | 
            +
                      if (iframeRef?.current?.contentWindow?.document?.body) {
         | 
| 68 | 
            +
                        iframeRef.current.contentWindow.document.body.scrollIntoView({
         | 
| 69 | 
            +
                          block: isAiWorking ? "end" : "start",
         | 
| 70 | 
            +
                          inline: "nearest",
         | 
| 71 | 
            +
                          behavior: isAiWorking ? "instant" : "smooth",
         | 
| 72 | 
            +
                        });
         | 
| 73 | 
            +
                      }
         | 
| 74 | 
            +
                    }}
         | 
| 75 | 
            +
                  />
         | 
| 76 | 
            +
                </div>
         | 
| 77 | 
            +
              );
         | 
| 78 | 
            +
            };
         | 
    	
        components/editor/save-button/index.tsx
    ADDED
    
    | @@ -0,0 +1,76 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            /* eslint-disable @typescript-eslint/no-explicit-any */
         | 
| 2 | 
            +
            import { useState } from "react";
         | 
| 3 | 
            +
            import { toast } from "sonner";
         | 
| 4 | 
            +
            import { MdSave } from "react-icons/md";
         | 
| 5 | 
            +
            import { useParams } from "next/navigation";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import Loading from "@/components/loading";
         | 
| 8 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 9 | 
            +
            import { api } from "@/lib/api";
         | 
| 10 | 
            +
             | 
| 11 | 
            +
            export function SaveButton({
         | 
| 12 | 
            +
              html,
         | 
| 13 | 
            +
              prompts,
         | 
| 14 | 
            +
            }: {
         | 
| 15 | 
            +
              html: string;
         | 
| 16 | 
            +
              prompts: string[];
         | 
| 17 | 
            +
            }) {
         | 
| 18 | 
            +
              // get params from URL
         | 
| 19 | 
            +
              const { namespace, repoId } = useParams<{
         | 
| 20 | 
            +
                namespace: string;
         | 
| 21 | 
            +
                repoId: string;
         | 
| 22 | 
            +
              }>();
         | 
| 23 | 
            +
              const [loading, setLoading] = useState(false);
         | 
| 24 | 
            +
             | 
| 25 | 
            +
              const updateSpace = async () => {
         | 
| 26 | 
            +
                setLoading(true);
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                try {
         | 
| 29 | 
            +
                  const res = await api.put(`/me/projects/${namespace}/${repoId}`, {
         | 
| 30 | 
            +
                    html,
         | 
| 31 | 
            +
                    prompts,
         | 
| 32 | 
            +
                  });
         | 
| 33 | 
            +
                  if (res.data.ok) {
         | 
| 34 | 
            +
                    toast.success("Your space is updated! 🎉", {
         | 
| 35 | 
            +
                      action: {
         | 
| 36 | 
            +
                        label: "See Space",
         | 
| 37 | 
            +
                        onClick: () => {
         | 
| 38 | 
            +
                          window.open(
         | 
| 39 | 
            +
                            `https://huggingface.co/spaces/${namespace}/${repoId}`,
         | 
| 40 | 
            +
                            "_blank"
         | 
| 41 | 
            +
                          );
         | 
| 42 | 
            +
                        },
         | 
| 43 | 
            +
                      },
         | 
| 44 | 
            +
                    });
         | 
| 45 | 
            +
                  } else {
         | 
| 46 | 
            +
                    toast.error(res?.data?.error || "Failed to update space");
         | 
| 47 | 
            +
                  }
         | 
| 48 | 
            +
                } catch (err: any) {
         | 
| 49 | 
            +
                  toast.error(err.response?.data?.error || err.message);
         | 
| 50 | 
            +
                } finally {
         | 
| 51 | 
            +
                  setLoading(false);
         | 
| 52 | 
            +
                }
         | 
| 53 | 
            +
              };
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              return (
         | 
| 56 | 
            +
                <>
         | 
| 57 | 
            +
                  <Button
         | 
| 58 | 
            +
                    variant="sky"
         | 
| 59 | 
            +
                    className="max-lg:hidden !px-4"
         | 
| 60 | 
            +
                    onClick={updateSpace}
         | 
| 61 | 
            +
                  >
         | 
| 62 | 
            +
                    <MdSave className="size-4" />
         | 
| 63 | 
            +
                    Save your Project{" "}
         | 
| 64 | 
            +
                    {loading && <Loading className="ml-2 size-4 animate-spin" />}
         | 
| 65 | 
            +
                  </Button>
         | 
| 66 | 
            +
                  <Button
         | 
| 67 | 
            +
                    variant="sky"
         | 
| 68 | 
            +
                    size="sm"
         | 
| 69 | 
            +
                    className="lg:hidden"
         | 
| 70 | 
            +
                    onClick={updateSpace}
         | 
| 71 | 
            +
                  >
         | 
| 72 | 
            +
                    Save {loading && <Loading className="ml-2 size-4 animate-spin" />}
         | 
| 73 | 
            +
                  </Button>
         | 
| 74 | 
            +
                </>
         | 
| 75 | 
            +
              );
         | 
| 76 | 
            +
            }
         | 
    	
        components/invite-friends/index.tsx
    ADDED
    
    | @@ -0,0 +1,85 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { TiUserAdd } from "react-icons/ti";
         | 
| 2 | 
            +
            import { Link } from "lucide-react";
         | 
| 3 | 
            +
            import { FaXTwitter } from "react-icons/fa6";
         | 
| 4 | 
            +
            import { useCopyToClipboard } from "react-use";
         | 
| 5 | 
            +
            import { toast } from "sonner";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 8 | 
            +
            import {
         | 
| 9 | 
            +
              Dialog,
         | 
| 10 | 
            +
              DialogContent,
         | 
| 11 | 
            +
              DialogTitle,
         | 
| 12 | 
            +
              DialogTrigger,
         | 
| 13 | 
            +
            } from "@/components/ui/dialog";
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            export function InviteFriends() {
         | 
| 16 | 
            +
              // eslint-disable-next-line @typescript-eslint/no-unused-vars
         | 
| 17 | 
            +
              const [_, copyToClipboard] = useCopyToClipboard();
         | 
| 18 | 
            +
             | 
| 19 | 
            +
              return (
         | 
| 20 | 
            +
                <Dialog>
         | 
| 21 | 
            +
                  <form>
         | 
| 22 | 
            +
                    <DialogTrigger asChild>
         | 
| 23 | 
            +
                      <Button
         | 
| 24 | 
            +
                        size="iconXs"
         | 
| 25 | 
            +
                        variant="outline"
         | 
| 26 | 
            +
                        className="!border-neutral-600 !text-neutral-400 !hover:!border-neutral-500 hover:!text-neutral-300"
         | 
| 27 | 
            +
                      >
         | 
| 28 | 
            +
                        <TiUserAdd className="size-4" />
         | 
| 29 | 
            +
                      </Button>
         | 
| 30 | 
            +
                    </DialogTrigger>
         | 
| 31 | 
            +
                    <DialogContent className="sm:max-w-lg lg:!p-8 !rounded-3xl !bg-white !border-neutral-100">
         | 
| 32 | 
            +
                      <DialogTitle className="hidden" />
         | 
| 33 | 
            +
                      <main>
         | 
| 34 | 
            +
                        <div className="flex items-center justify-start -space-x-4 mb-5">
         | 
| 35 | 
            +
                          <div className="size-11 rounded-full bg-pink-300 shadow-2xs flex items-center justify-center text-2xl">
         | 
| 36 | 
            +
                            😎
         | 
| 37 | 
            +
                          </div>
         | 
| 38 | 
            +
                          <div className="size-11 rounded-full bg-amber-300 shadow-2xs flex items-center justify-center text-2xl z-2">
         | 
| 39 | 
            +
                            😇
         | 
| 40 | 
            +
                          </div>
         | 
| 41 | 
            +
                          <div className="size-11 rounded-full bg-sky-300 shadow-2xs flex items-center justify-center text-2xl">
         | 
| 42 | 
            +
                            😜
         | 
| 43 | 
            +
                          </div>
         | 
| 44 | 
            +
                        </div>
         | 
| 45 | 
            +
                        <p className="text-xl font-semibold text-neutral-950 max-w-[200px]">
         | 
| 46 | 
            +
                          Invite your friends to join us!
         | 
| 47 | 
            +
                        </p>
         | 
| 48 | 
            +
                        <p className="text-sm text-neutral-500 mt-2 max-w-sm">
         | 
| 49 | 
            +
                          Support us and share the love and let them know about our awesome
         | 
| 50 | 
            +
                          platform.
         | 
| 51 | 
            +
                        </p>
         | 
| 52 | 
            +
                        <div className="mt-4 space-x-3.5">
         | 
| 53 | 
            +
                          <a
         | 
| 54 | 
            +
                            href="https://x.com/intent/post?url=https://enzostvs-deepsite.hf.space/&text=Checkout%20this%20awesome%20Ai%20Tool!%20Vibe%20coding%20has%20never%20been%20so%20easy✨"
         | 
| 55 | 
            +
                            target="_blank"
         | 
| 56 | 
            +
                            rel="noopener noreferrer"
         | 
| 57 | 
            +
                          >
         | 
| 58 | 
            +
                            <Button
         | 
| 59 | 
            +
                              variant="lightGray"
         | 
| 60 | 
            +
                              size="sm"
         | 
| 61 | 
            +
                              className="!text-neutral-700"
         | 
| 62 | 
            +
                            >
         | 
| 63 | 
            +
                              <FaXTwitter className="size-4" />
         | 
| 64 | 
            +
                              Share on
         | 
| 65 | 
            +
                            </Button>
         | 
| 66 | 
            +
                          </a>
         | 
| 67 | 
            +
                          <Button
         | 
| 68 | 
            +
                            variant="lightGray"
         | 
| 69 | 
            +
                            size="sm"
         | 
| 70 | 
            +
                            className="!text-neutral-700"
         | 
| 71 | 
            +
                            onClick={() => {
         | 
| 72 | 
            +
                              copyToClipboard("https://enzostvs-deepsite.hf.space/");
         | 
| 73 | 
            +
                              toast.success("Invite link copied to clipboard!");
         | 
| 74 | 
            +
                            }}
         | 
| 75 | 
            +
                          >
         | 
| 76 | 
            +
                            <Link className="size-4" />
         | 
| 77 | 
            +
                            Copy Invite Link
         | 
| 78 | 
            +
                          </Button>
         | 
| 79 | 
            +
                        </div>
         | 
| 80 | 
            +
                      </main>
         | 
| 81 | 
            +
                    </DialogContent>
         | 
| 82 | 
            +
                  </form>
         | 
| 83 | 
            +
                </Dialog>
         | 
| 84 | 
            +
              );
         | 
| 85 | 
            +
            }
         | 
    	
        components/loading/index.tsx
    ADDED
    
    | @@ -0,0 +1,41 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import classNames from "classnames";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            function Loading({
         | 
| 4 | 
            +
              overlay = true,
         | 
| 5 | 
            +
              className,
         | 
| 6 | 
            +
            }: {
         | 
| 7 | 
            +
              overlay?: boolean;
         | 
| 8 | 
            +
              className?: string;
         | 
| 9 | 
            +
            }) {
         | 
| 10 | 
            +
              return (
         | 
| 11 | 
            +
                <div
         | 
| 12 | 
            +
                  className={classNames("", {
         | 
| 13 | 
            +
                    "absolute left-0 top-0 h-full w-full flex items-center justify-center z-20 bg-black/50 rounded-full":
         | 
| 14 | 
            +
                      overlay,
         | 
| 15 | 
            +
                  })}
         | 
| 16 | 
            +
                >
         | 
| 17 | 
            +
                  <svg
         | 
| 18 | 
            +
                    className={`size-5 animate-spin text-white ${className}`}
         | 
| 19 | 
            +
                    xmlns="http://www.w3.org/2000/svg"
         | 
| 20 | 
            +
                    fill="none"
         | 
| 21 | 
            +
                    viewBox="0 0 24 24"
         | 
| 22 | 
            +
                  >
         | 
| 23 | 
            +
                    <circle
         | 
| 24 | 
            +
                      className="opacity-25"
         | 
| 25 | 
            +
                      cx="12"
         | 
| 26 | 
            +
                      cy="12"
         | 
| 27 | 
            +
                      r="10"
         | 
| 28 | 
            +
                      stroke="currentColor"
         | 
| 29 | 
            +
                      strokeWidth="4"
         | 
| 30 | 
            +
                    ></circle>
         | 
| 31 | 
            +
                    <path
         | 
| 32 | 
            +
                      className="opacity-75"
         | 
| 33 | 
            +
                      fill="currentColor"
         | 
| 34 | 
            +
                      d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
         | 
| 35 | 
            +
                    ></path>
         | 
| 36 | 
            +
                  </svg>
         | 
| 37 | 
            +
                </div>
         | 
| 38 | 
            +
              );
         | 
| 39 | 
            +
            }
         | 
| 40 | 
            +
             | 
| 41 | 
            +
            export default Loading;
         | 
    	
        components/login-modal/index.tsx
    ADDED
    
    | @@ -0,0 +1,61 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { useLocalStorage } from "react-use";
         | 
| 2 | 
            +
            import { defaultHTML } from "@/lib/consts";
         | 
| 3 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 4 | 
            +
            import { Dialog, DialogContent, DialogTitle } from "@/components/ui/dialog";
         | 
| 5 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            export const LoginModal = ({
         | 
| 8 | 
            +
              open,
         | 
| 9 | 
            +
              html,
         | 
| 10 | 
            +
              onClose,
         | 
| 11 | 
            +
              title = "Log In to use DeepSite for free",
         | 
| 12 | 
            +
              description = "Log In through your Hugging Face account to continue using DeepSite and increase your monthly free limit.",
         | 
| 13 | 
            +
            }: {
         | 
| 14 | 
            +
              open: boolean;
         | 
| 15 | 
            +
              html: string;
         | 
| 16 | 
            +
              onClose: React.Dispatch<React.SetStateAction<boolean>>;
         | 
| 17 | 
            +
              title?: string;
         | 
| 18 | 
            +
              description?: string;
         | 
| 19 | 
            +
            }) => {
         | 
| 20 | 
            +
              const { openLoginWindow } = useUser();
         | 
| 21 | 
            +
              const [, setStorage] = useLocalStorage("html_content");
         | 
| 22 | 
            +
              const handleClick = async () => {
         | 
| 23 | 
            +
                if (html !== defaultHTML) {
         | 
| 24 | 
            +
                  setStorage(html);
         | 
| 25 | 
            +
                }
         | 
| 26 | 
            +
                openLoginWindow();
         | 
| 27 | 
            +
                onClose(false);
         | 
| 28 | 
            +
              };
         | 
| 29 | 
            +
              return (
         | 
| 30 | 
            +
                <Dialog open={open} onOpenChange={onClose}>
         | 
| 31 | 
            +
                  <DialogContent className="sm:max-w-lg lg:!p-8 !rounded-3xl !bg-white !border-neutral-100">
         | 
| 32 | 
            +
                    <DialogTitle className="hidden" />
         | 
| 33 | 
            +
                    <main className="flex flex-col items-start text-left relative pt-2">
         | 
| 34 | 
            +
                      <div className="flex items-center justify-start -space-x-4 mb-5">
         | 
| 35 | 
            +
                        <div className="size-14 rounded-full bg-pink-200 shadow-2xs flex items-center justify-center text-3xl opacity-50">
         | 
| 36 | 
            +
                          💪
         | 
| 37 | 
            +
                        </div>
         | 
| 38 | 
            +
                        <div className="size-16 rounded-full bg-amber-200 shadow-2xl flex items-center justify-center text-4xl z-2">
         | 
| 39 | 
            +
                          😎
         | 
| 40 | 
            +
                        </div>
         | 
| 41 | 
            +
                        <div className="size-14 rounded-full bg-sky-200 shadow-2xs flex items-center justify-center text-3xl opacity-50">
         | 
| 42 | 
            +
                          🙌
         | 
| 43 | 
            +
                        </div>
         | 
| 44 | 
            +
                      </div>
         | 
| 45 | 
            +
                      <p className="text-2xl font-bold text-neutral-950">{title}</p>
         | 
| 46 | 
            +
                      <p className="text-neutral-500 text-base mt-2 max-w-sm">
         | 
| 47 | 
            +
                        {description}
         | 
| 48 | 
            +
                      </p>
         | 
| 49 | 
            +
                      <Button
         | 
| 50 | 
            +
                        variant="black"
         | 
| 51 | 
            +
                        size="lg"
         | 
| 52 | 
            +
                        className="w-full !text-base !h-11 mt-8"
         | 
| 53 | 
            +
                        onClick={handleClick}
         | 
| 54 | 
            +
                      >
         | 
| 55 | 
            +
                        Log In to Continue
         | 
| 56 | 
            +
                      </Button>
         | 
| 57 | 
            +
                    </main>
         | 
| 58 | 
            +
                  </DialogContent>
         | 
| 59 | 
            +
                </Dialog>
         | 
| 60 | 
            +
              );
         | 
| 61 | 
            +
            };
         | 
    	
        components/magic-ui/grid-pattern.tsx
    ADDED
    
    | @@ -0,0 +1,69 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { useId } from "react";
         | 
| 2 | 
            +
            import { cn } from "@/lib/utils";
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            interface GridPatternProps extends React.SVGProps<SVGSVGElement> {
         | 
| 5 | 
            +
              width?: number;
         | 
| 6 | 
            +
              height?: number;
         | 
| 7 | 
            +
              x?: number;
         | 
| 8 | 
            +
              y?: number;
         | 
| 9 | 
            +
              squares?: Array<[x: number, y: number]>;
         | 
| 10 | 
            +
              strokeDasharray?: string;
         | 
| 11 | 
            +
              className?: string;
         | 
| 12 | 
            +
              [key: string]: unknown;
         | 
| 13 | 
            +
            }
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            export function GridPattern({
         | 
| 16 | 
            +
              width = 40,
         | 
| 17 | 
            +
              height = 40,
         | 
| 18 | 
            +
              x = -1,
         | 
| 19 | 
            +
              y = -1,
         | 
| 20 | 
            +
              strokeDasharray = "0",
         | 
| 21 | 
            +
              squares,
         | 
| 22 | 
            +
              className,
         | 
| 23 | 
            +
              ...props
         | 
| 24 | 
            +
            }: GridPatternProps) {
         | 
| 25 | 
            +
              const id = useId();
         | 
| 26 | 
            +
             | 
| 27 | 
            +
              return (
         | 
| 28 | 
            +
                <svg
         | 
| 29 | 
            +
                  aria-hidden="true"
         | 
| 30 | 
            +
                  className={cn(
         | 
| 31 | 
            +
                    "pointer-events-none absolute inset-0 h-full w-full fill-gray-400/30 stroke-neutral-700 -z-[1]",
         | 
| 32 | 
            +
                    className
         | 
| 33 | 
            +
                  )}
         | 
| 34 | 
            +
                  {...props}
         | 
| 35 | 
            +
                >
         | 
| 36 | 
            +
                  <defs>
         | 
| 37 | 
            +
                    <pattern
         | 
| 38 | 
            +
                      id={id}
         | 
| 39 | 
            +
                      width={width}
         | 
| 40 | 
            +
                      height={height}
         | 
| 41 | 
            +
                      patternUnits="userSpaceOnUse"
         | 
| 42 | 
            +
                      x={x}
         | 
| 43 | 
            +
                      y={y}
         | 
| 44 | 
            +
                    >
         | 
| 45 | 
            +
                      <path
         | 
| 46 | 
            +
                        d={`M.5 ${height}V.5H${width}`}
         | 
| 47 | 
            +
                        fill="none"
         | 
| 48 | 
            +
                        strokeDasharray={strokeDasharray}
         | 
| 49 | 
            +
                      />
         | 
| 50 | 
            +
                    </pattern>
         | 
| 51 | 
            +
                  </defs>
         | 
| 52 | 
            +
                  <rect width="100%" height="100%" strokeWidth={0} fill={`url(#${id})`} />
         | 
| 53 | 
            +
                  {squares && (
         | 
| 54 | 
            +
                    <svg x={x} y={y} className="overflow-visible">
         | 
| 55 | 
            +
                      {squares.map(([x, y]) => (
         | 
| 56 | 
            +
                        <rect
         | 
| 57 | 
            +
                          strokeWidth="0"
         | 
| 58 | 
            +
                          key={`${x}-${y}`}
         | 
| 59 | 
            +
                          width={width - 1}
         | 
| 60 | 
            +
                          height={height - 1}
         | 
| 61 | 
            +
                          x={x * width + 1}
         | 
| 62 | 
            +
                          y={y * height + 1}
         | 
| 63 | 
            +
                        />
         | 
| 64 | 
            +
                      ))}
         | 
| 65 | 
            +
                    </svg>
         | 
| 66 | 
            +
                  )}
         | 
| 67 | 
            +
                </svg>
         | 
| 68 | 
            +
              );
         | 
| 69 | 
            +
            }
         | 
    	
        components/my-projects/index.tsx
    ADDED
    
    | @@ -0,0 +1,41 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 3 | 
            +
            import { Project } from "@/types";
         | 
| 4 | 
            +
            import { redirect } from "next/navigation";
         | 
| 5 | 
            +
            import { ProjectCard } from "./project-card";
         | 
| 6 | 
            +
            import { Plus } from "lucide-react";
         | 
| 7 | 
            +
            import Link from "next/link";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            export function MyProjects({ projects }: { projects: Project[] }) {
         | 
| 10 | 
            +
              const { user } = useUser();
         | 
| 11 | 
            +
              if (!user) {
         | 
| 12 | 
            +
                redirect("/");
         | 
| 13 | 
            +
              }
         | 
| 14 | 
            +
              return (
         | 
| 15 | 
            +
                <>
         | 
| 16 | 
            +
                  <section className="max-w-[86rem] py-12 px-4 mx-auto">
         | 
| 17 | 
            +
                    <div className="text-left">
         | 
| 18 | 
            +
                      <h1 className="text-3xl font-bold text-white">
         | 
| 19 | 
            +
                        <span className="capitalize">{user.fullname}</span>'s DeepSite
         | 
| 20 | 
            +
                        Projects
         | 
| 21 | 
            +
                      </h1>
         | 
| 22 | 
            +
                      <p className="text-muted-foreground text-base mt-1 max-w-xl">
         | 
| 23 | 
            +
                        Create, manage, and explore your DeepSite projects.
         | 
| 24 | 
            +
                      </p>
         | 
| 25 | 
            +
                    </div>
         | 
| 26 | 
            +
                    <div className="mt-8 grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-8">
         | 
| 27 | 
            +
                      <Link
         | 
| 28 | 
            +
                        href="/projects/new"
         | 
| 29 | 
            +
                        className="bg-neutral-900 rounded-xl h-44 flex items-center justify-center text-neutral-300 border border-neutral-800 hover:brightness-110 transition-all duration-200"
         | 
| 30 | 
            +
                      >
         | 
| 31 | 
            +
                        <Plus className="size-5 mr-1.5" />
         | 
| 32 | 
            +
                        Create Project
         | 
| 33 | 
            +
                      </Link>
         | 
| 34 | 
            +
                      {projects.map((project: Project) => (
         | 
| 35 | 
            +
                        <ProjectCard key={project._id} project={project} />
         | 
| 36 | 
            +
                      ))}
         | 
| 37 | 
            +
                    </div>
         | 
| 38 | 
            +
                  </section>
         | 
| 39 | 
            +
                </>
         | 
| 40 | 
            +
              );
         | 
| 41 | 
            +
            }
         | 
    	
        components/my-projects/project-card.tsx
    ADDED
    
    | @@ -0,0 +1,74 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import Link from "next/link";
         | 
| 2 | 
            +
            import { formatDistance } from "date-fns";
         | 
| 3 | 
            +
            import { EllipsisVertical, Settings } from "lucide-react";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            import { Project } from "@/types";
         | 
| 6 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 7 | 
            +
            import {
         | 
| 8 | 
            +
              DropdownMenu,
         | 
| 9 | 
            +
              DropdownMenuContent,
         | 
| 10 | 
            +
              DropdownMenuGroup,
         | 
| 11 | 
            +
              DropdownMenuItem,
         | 
| 12 | 
            +
              DropdownMenuTrigger,
         | 
| 13 | 
            +
            } from "@/components/ui/dropdown-menu";
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            export function ProjectCard({ project }: { project: Project }) {
         | 
| 16 | 
            +
              return (
         | 
| 17 | 
            +
                <div className="text-neutral-200 space-y-4 group cursor-pointer">
         | 
| 18 | 
            +
                  <Link
         | 
| 19 | 
            +
                    href={`/projects/${project.space_id}`}
         | 
| 20 | 
            +
                    className="relative bg-neutral-900 rounded-2xl overflow-hidden h-44 w-full flex items-center justify-end flex-col px-3 border border-neutral-800"
         | 
| 21 | 
            +
                  >
         | 
| 22 | 
            +
                    <iframe
         | 
| 23 | 
            +
                      src={`https://${project.space_id.replace("/", "-")}.static.hf.space/`}
         | 
| 24 | 
            +
                      frameBorder="0"
         | 
| 25 | 
            +
                      className="absolute inset-0 w-full h-full top-0 left-0 group-hover:brightness-75 transition-all duration-200 pointer-events-none"
         | 
| 26 | 
            +
                    ></iframe>
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                    <Button
         | 
| 29 | 
            +
                      variant="default"
         | 
| 30 | 
            +
                      className="w-full transition-all duration-200 translate-y-full group-hover:-translate-y-3"
         | 
| 31 | 
            +
                    >
         | 
| 32 | 
            +
                      Open project
         | 
| 33 | 
            +
                    </Button>
         | 
| 34 | 
            +
                  </Link>
         | 
| 35 | 
            +
                  <div className="flex items-start justify-between gap-3">
         | 
| 36 | 
            +
                    <div>
         | 
| 37 | 
            +
                      <p className="text-neutral-200 text-base font-semibold line-clamp-1">
         | 
| 38 | 
            +
                        {project.space_id}
         | 
| 39 | 
            +
                      </p>
         | 
| 40 | 
            +
                      <p className="text-sm text-neutral-500">
         | 
| 41 | 
            +
                        Updated{" "}
         | 
| 42 | 
            +
                        {formatDistance(
         | 
| 43 | 
            +
                          new Date(project._updatedAt || Date.now()),
         | 
| 44 | 
            +
                          new Date(),
         | 
| 45 | 
            +
                          {
         | 
| 46 | 
            +
                            addSuffix: true,
         | 
| 47 | 
            +
                          }
         | 
| 48 | 
            +
                        )}
         | 
| 49 | 
            +
                      </p>
         | 
| 50 | 
            +
                    </div>
         | 
| 51 | 
            +
                    <DropdownMenu>
         | 
| 52 | 
            +
                      <DropdownMenuTrigger asChild>
         | 
| 53 | 
            +
                        <Button variant="ghost" size="icon">
         | 
| 54 | 
            +
                          <EllipsisVertical className="text-neutral-400 size-5 hover:text-neutral-300 transition-colors duration-200 cursor-pointer" />
         | 
| 55 | 
            +
                        </Button>
         | 
| 56 | 
            +
                      </DropdownMenuTrigger>
         | 
| 57 | 
            +
                      <DropdownMenuContent className="w-56" align="start">
         | 
| 58 | 
            +
                        <DropdownMenuGroup>
         | 
| 59 | 
            +
                          <a
         | 
| 60 | 
            +
                            href={`https://huggingface.co/spaces/${project.space_id}/settings`}
         | 
| 61 | 
            +
                            target="_blank"
         | 
| 62 | 
            +
                          >
         | 
| 63 | 
            +
                            <DropdownMenuItem>
         | 
| 64 | 
            +
                              <Settings className="size-4 text-neutral-100" />
         | 
| 65 | 
            +
                              Project Settings
         | 
| 66 | 
            +
                            </DropdownMenuItem>
         | 
| 67 | 
            +
                          </a>
         | 
| 68 | 
            +
                        </DropdownMenuGroup>
         | 
| 69 | 
            +
                      </DropdownMenuContent>
         | 
| 70 | 
            +
                    </DropdownMenu>
         | 
| 71 | 
            +
                  </div>
         | 
| 72 | 
            +
                </div>
         | 
| 73 | 
            +
              );
         | 
| 74 | 
            +
            }
         | 
    	
        components/pro-modal/index.tsx
    ADDED
    
    | @@ -0,0 +1,93 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import { useLocalStorage } from "react-use";
         | 
| 2 | 
            +
            import { defaultHTML } from "@/lib/consts";
         | 
| 3 | 
            +
            import { Button } from "../ui/button";
         | 
| 4 | 
            +
            import { Dialog, DialogContent } from "../ui/dialog";
         | 
| 5 | 
            +
            import { CheckCheck } from "lucide-react";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            export const ProModal = ({
         | 
| 8 | 
            +
              open,
         | 
| 9 | 
            +
              html,
         | 
| 10 | 
            +
              onClose,
         | 
| 11 | 
            +
            }: {
         | 
| 12 | 
            +
              open: boolean;
         | 
| 13 | 
            +
              html: string;
         | 
| 14 | 
            +
              onClose: React.Dispatch<React.SetStateAction<boolean>>;
         | 
| 15 | 
            +
            }) => {
         | 
| 16 | 
            +
              const [, setStorage] = useLocalStorage("html_content");
         | 
| 17 | 
            +
              const handleProClick = () => {
         | 
| 18 | 
            +
                if (html !== defaultHTML) {
         | 
| 19 | 
            +
                  setStorage(html);
         | 
| 20 | 
            +
                }
         | 
| 21 | 
            +
              };
         | 
| 22 | 
            +
              return (
         | 
| 23 | 
            +
                <Dialog open={open} onOpenChange={onClose}>
         | 
| 24 | 
            +
                  <DialogContent className="sm:max-w-lg lg:!p-8 !rounded-3xl !bg-white !border-neutral-100">
         | 
| 25 | 
            +
                    <main className="flex flex-col items-start text-left relative pt-2">
         | 
| 26 | 
            +
                      <div className="flex items-center justify-start -space-x-4 mb-5">
         | 
| 27 | 
            +
                        <div className="size-14 rounded-full bg-pink-200 shadow-2xs flex items-center justify-center text-3xl opacity-50">
         | 
| 28 | 
            +
                          🚀
         | 
| 29 | 
            +
                        </div>
         | 
| 30 | 
            +
                        <div className="size-16 rounded-full bg-amber-200 shadow-2xl flex items-center justify-center text-4xl z-2">
         | 
| 31 | 
            +
                          🤩
         | 
| 32 | 
            +
                        </div>
         | 
| 33 | 
            +
                        <div className="size-14 rounded-full bg-sky-200 shadow-2xs flex items-center justify-center text-3xl opacity-50">
         | 
| 34 | 
            +
                          🥳
         | 
| 35 | 
            +
                        </div>
         | 
| 36 | 
            +
                      </div>
         | 
| 37 | 
            +
                      <p className="text-2xl font-bold text-neutral-950">
         | 
| 38 | 
            +
                        Only 9$ for unlimited access!
         | 
| 39 | 
            +
                      </p>
         | 
| 40 | 
            +
                      <p className="text-neutral-500 text-base mt-2 max-w-sm">
         | 
| 41 | 
            +
                        It seems like you have reached the monthly free limit of DeepSite.
         | 
| 42 | 
            +
                      </p>
         | 
| 43 | 
            +
                      <hr className="bg-neutral-200 w-full max-w-[150px] my-6" />
         | 
| 44 | 
            +
                      <p className="text-lg mt-3 text-neutral-900 font-semibold">
         | 
| 45 | 
            +
                        Upgrade to a <ProTag className="mx-1" /> Account, and unlock:
         | 
| 46 | 
            +
                      </p>
         | 
| 47 | 
            +
                      <ul className="mt-3 space-y-1 text-neutral-500">
         | 
| 48 | 
            +
                        <li className="text-sm space-x-2 flex items-center justify-start gap-2">
         | 
| 49 | 
            +
                          <CheckCheck className="text-emerald-500 size-4" />
         | 
| 50 | 
            +
                          DeepSite unlimited access to all Inference Providers
         | 
| 51 | 
            +
                        </li>
         | 
| 52 | 
            +
                        <li className="text-sm space-x-2 flex items-center justify-start gap-2">
         | 
| 53 | 
            +
                          <CheckCheck className="text-emerald-500 size-4" />
         | 
| 54 | 
            +
                          Get highest priority and 8x more quota on Spaces ZeroGPU
         | 
| 55 | 
            +
                        </li>
         | 
| 56 | 
            +
                        <li className="text-sm space-x-2 flex items-center justify-start gap-2">
         | 
| 57 | 
            +
                          <CheckCheck className="text-emerald-500 size-4" />
         | 
| 58 | 
            +
                          Activate Dataset Viewer on private datasets
         | 
| 59 | 
            +
                        </li>
         | 
| 60 | 
            +
                        <li className="text-sm space-x-2 flex items-center justify-start gap-2">
         | 
| 61 | 
            +
                          <CheckCheck className="text-emerald-500 size-4" />
         | 
| 62 | 
            +
                          Get exclusive early access to new features and updates
         | 
| 63 | 
            +
                        </li>
         | 
| 64 | 
            +
                        <li className="text-sm space-x-2 flex items-center justify-start gap-2">
         | 
| 65 | 
            +
                          <CheckCheck className="text-emerald-500 size-4" />
         | 
| 66 | 
            +
                          Get free credits across all Inference Providers
         | 
| 67 | 
            +
                        </li>
         | 
| 68 | 
            +
                        <li className="text-sm text-neutral-500 space-x-2 flex items-center justify-start gap-2 mt-3">
         | 
| 69 | 
            +
                          ... and much more!
         | 
| 70 | 
            +
                        </li>
         | 
| 71 | 
            +
                      </ul>
         | 
| 72 | 
            +
                      <Button
         | 
| 73 | 
            +
                        variant="black"
         | 
| 74 | 
            +
                        size="lg"
         | 
| 75 | 
            +
                        className="w-full !text-base !h-11 mt-8"
         | 
| 76 | 
            +
                        onClick={handleProClick}
         | 
| 77 | 
            +
                      >
         | 
| 78 | 
            +
                        Subscribe to PRO ($9/month)
         | 
| 79 | 
            +
                      </Button>
         | 
| 80 | 
            +
                    </main>
         | 
| 81 | 
            +
                  </DialogContent>
         | 
| 82 | 
            +
                </Dialog>
         | 
| 83 | 
            +
              );
         | 
| 84 | 
            +
            };
         | 
| 85 | 
            +
             | 
| 86 | 
            +
            const ProTag = ({ className }: { className?: string }) => (
         | 
| 87 | 
            +
              <span
         | 
| 88 | 
            +
                className={`${className} bg-linear-to-br shadow-green-500/10 dark:shadow-green-500/20 inline-block -skew-x-12 border border-gray-200 from-pink-300 via-green-200 to-yellow-200 text-xs font-bold text-black shadow-lg rounded-md px-2.5 py-0.5`}
         | 
| 89 | 
            +
              >
         | 
| 90 | 
            +
                PRO
         | 
| 91 | 
            +
              </span>
         | 
| 92 | 
            +
            );
         | 
| 93 | 
            +
            export default ProModal;
         | 
    	
        components/providers/tanstack-query-provider.tsx
    ADDED
    
    | @@ -0,0 +1,19 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
         | 
| 4 | 
            +
            import { ReactQueryDevtools } from "@tanstack/react-query-devtools";
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            export default function TanstackProvider({
         | 
| 7 | 
            +
              children,
         | 
| 8 | 
            +
            }: {
         | 
| 9 | 
            +
              children: React.ReactNode;
         | 
| 10 | 
            +
            }) {
         | 
| 11 | 
            +
              const queryClient = new QueryClient();
         | 
| 12 | 
            +
             | 
| 13 | 
            +
              return (
         | 
| 14 | 
            +
                <QueryClientProvider client={queryClient}>
         | 
| 15 | 
            +
                  {children}
         | 
| 16 | 
            +
                  <ReactQueryDevtools initialIsOpen={false} />
         | 
| 17 | 
            +
                </QueryClientProvider>
         | 
| 18 | 
            +
              );
         | 
| 19 | 
            +
            }
         | 
    	
        components/public/navigation/index.tsx
    ADDED
    
    | @@ -0,0 +1,156 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { useRef, useState } from "react";
         | 
| 4 | 
            +
            import Image from "next/image";
         | 
| 5 | 
            +
            import Link from "next/link";
         | 
| 6 | 
            +
            import { useMount, useUnmount } from "react-use";
         | 
| 7 | 
            +
            import classNames from "classnames";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 10 | 
            +
            import Logo from "@/assets/logo.svg";
         | 
| 11 | 
            +
            import { useUser } from "@/hooks/useUser";
         | 
| 12 | 
            +
            import { UserMenu } from "@/components/user-menu";
         | 
| 13 | 
            +
             | 
| 14 | 
            +
            const navigationLinks = [
         | 
| 15 | 
            +
              {
         | 
| 16 | 
            +
                name: "Create Website",
         | 
| 17 | 
            +
                href: "/projects/new",
         | 
| 18 | 
            +
              },
         | 
| 19 | 
            +
              {
         | 
| 20 | 
            +
                name: "Features",
         | 
| 21 | 
            +
                href: "#features",
         | 
| 22 | 
            +
              },
         | 
| 23 | 
            +
              {
         | 
| 24 | 
            +
                name: "Community",
         | 
| 25 | 
            +
                href: "#community",
         | 
| 26 | 
            +
              },
         | 
| 27 | 
            +
              {
         | 
| 28 | 
            +
                name: "Deploy",
         | 
| 29 | 
            +
                href: "#deploy",
         | 
| 30 | 
            +
              },
         | 
| 31 | 
            +
            ];
         | 
| 32 | 
            +
             | 
| 33 | 
            +
            export default function Navigation() {
         | 
| 34 | 
            +
              const { openLoginWindow, user } = useUser();
         | 
| 35 | 
            +
              const [hash, setHash] = useState("");
         | 
| 36 | 
            +
             | 
| 37 | 
            +
              const selectorRef = useRef<HTMLDivElement>(null);
         | 
| 38 | 
            +
              const linksRef = useRef<HTMLLIElement[]>(
         | 
| 39 | 
            +
                new Array(navigationLinks.length).fill(null)
         | 
| 40 | 
            +
              );
         | 
| 41 | 
            +
              const [isScrolled, setIsScrolled] = useState(false);
         | 
| 42 | 
            +
             | 
| 43 | 
            +
              useMount(() => {
         | 
| 44 | 
            +
                const handleScroll = () => {
         | 
| 45 | 
            +
                  const scrollTop = window.scrollY;
         | 
| 46 | 
            +
                  setIsScrolled(scrollTop > 100);
         | 
| 47 | 
            +
                };
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                const initialHash = window.location.hash;
         | 
| 50 | 
            +
                if (initialHash) {
         | 
| 51 | 
            +
                  setHash(initialHash);
         | 
| 52 | 
            +
                  calculateSelectorPosition(initialHash);
         | 
| 53 | 
            +
                }
         | 
| 54 | 
            +
             | 
| 55 | 
            +
                window.addEventListener("scroll", handleScroll);
         | 
| 56 | 
            +
              });
         | 
| 57 | 
            +
             | 
| 58 | 
            +
              useUnmount(() => {
         | 
| 59 | 
            +
                window.removeEventListener("scroll", () => {});
         | 
| 60 | 
            +
              });
         | 
| 61 | 
            +
             | 
| 62 | 
            +
              const handleClick = (href: string) => {
         | 
| 63 | 
            +
                setHash(href);
         | 
| 64 | 
            +
                calculateSelectorPosition(href);
         | 
| 65 | 
            +
              };
         | 
| 66 | 
            +
             | 
| 67 | 
            +
              const calculateSelectorPosition = (href: string) => {
         | 
| 68 | 
            +
                if (selectorRef.current && linksRef.current) {
         | 
| 69 | 
            +
                  const index = navigationLinks.findIndex((l) => l.href === href);
         | 
| 70 | 
            +
                  const targetLink = linksRef.current[index];
         | 
| 71 | 
            +
                  if (targetLink) {
         | 
| 72 | 
            +
                    const targetRect = targetLink.getBoundingClientRect();
         | 
| 73 | 
            +
                    selectorRef.current.style.left = targetRect.left + "px";
         | 
| 74 | 
            +
                    selectorRef.current.style.width = targetRect.width + "px";
         | 
| 75 | 
            +
                  }
         | 
| 76 | 
            +
                }
         | 
| 77 | 
            +
              };
         | 
| 78 | 
            +
             | 
| 79 | 
            +
              return (
         | 
| 80 | 
            +
                <div
         | 
| 81 | 
            +
                  className={classNames(
         | 
| 82 | 
            +
                    "sticky top-0 z-10 transition-all duration-200 backdrop-blur-md",
         | 
| 83 | 
            +
                    {
         | 
| 84 | 
            +
                      "bg-black/30": isScrolled,
         | 
| 85 | 
            +
                    }
         | 
| 86 | 
            +
                  )}
         | 
| 87 | 
            +
                >
         | 
| 88 | 
            +
                  <nav className="grid grid-cols-3 p-4 container mx-auto">
         | 
| 89 | 
            +
                    <Link href="/" className="flex items-center gap-1">
         | 
| 90 | 
            +
                      <Image
         | 
| 91 | 
            +
                        src={Logo}
         | 
| 92 | 
            +
                        className="w-9 mr-1"
         | 
| 93 | 
            +
                        alt="DeepSite Logo"
         | 
| 94 | 
            +
                        width={64}
         | 
| 95 | 
            +
                        height={64}
         | 
| 96 | 
            +
                      />
         | 
| 97 | 
            +
                      <p className="font-sans text-white text-xl font-bold">DeepSite</p>
         | 
| 98 | 
            +
                    </Link>
         | 
| 99 | 
            +
                    <ul className="flex items-center justify-center gap-6">
         | 
| 100 | 
            +
                      {navigationLinks.map((link) => (
         | 
| 101 | 
            +
                        <li
         | 
| 102 | 
            +
                          key={link.name}
         | 
| 103 | 
            +
                          ref={(el) => {
         | 
| 104 | 
            +
                            const index = navigationLinks.findIndex(
         | 
| 105 | 
            +
                              (l) => l.href === link.href
         | 
| 106 | 
            +
                            );
         | 
| 107 | 
            +
                            if (el && linksRef.current[index] !== el) {
         | 
| 108 | 
            +
                              linksRef.current[index] = el;
         | 
| 109 | 
            +
                            }
         | 
| 110 | 
            +
                          }}
         | 
| 111 | 
            +
                          className="inline-block font-sans text-sm"
         | 
| 112 | 
            +
                        >
         | 
| 113 | 
            +
                          <Link
         | 
| 114 | 
            +
                            href={link.href}
         | 
| 115 | 
            +
                            className={classNames(
         | 
| 116 | 
            +
                              "text-neutral-500 hover:text-primary transition-colors",
         | 
| 117 | 
            +
                              {
         | 
| 118 | 
            +
                                "text-primary": hash === link.href,
         | 
| 119 | 
            +
                              }
         | 
| 120 | 
            +
                            )}
         | 
| 121 | 
            +
                            onClick={() => {
         | 
| 122 | 
            +
                              handleClick(link.href);
         | 
| 123 | 
            +
                            }}
         | 
| 124 | 
            +
                          >
         | 
| 125 | 
            +
                            {link.name}
         | 
| 126 | 
            +
                          </Link>
         | 
| 127 | 
            +
                        </li>
         | 
| 128 | 
            +
                      ))}
         | 
| 129 | 
            +
                      <div
         | 
| 130 | 
            +
                        ref={selectorRef}
         | 
| 131 | 
            +
                        className={classNames(
         | 
| 132 | 
            +
                          "h-1 absolute bottom-4 transition-all duration-200 flex items-center justify-center",
         | 
| 133 | 
            +
                          {
         | 
| 134 | 
            +
                            "opacity-0": !hash,
         | 
| 135 | 
            +
                          }
         | 
| 136 | 
            +
                        )}
         | 
| 137 | 
            +
                      >
         | 
| 138 | 
            +
                        <div className="size-1 bg-white rounded-full" />
         | 
| 139 | 
            +
                      </div>
         | 
| 140 | 
            +
                    </ul>
         | 
| 141 | 
            +
                    <div className="flex items-center justify-end gap-2">
         | 
| 142 | 
            +
                      {user ? (
         | 
| 143 | 
            +
                        <UserMenu className="!pl-3 !pr-4 !py-2 !h-auto !rounded-lg" />
         | 
| 144 | 
            +
                      ) : (
         | 
| 145 | 
            +
                        <>
         | 
| 146 | 
            +
                          <Button variant="link" size={"sm"} onClick={openLoginWindow}>
         | 
| 147 | 
            +
                            Log In
         | 
| 148 | 
            +
                          </Button>
         | 
| 149 | 
            +
                          <Button size={"sm"}>Sign Up</Button>
         | 
| 150 | 
            +
                        </>
         | 
| 151 | 
            +
                      )}
         | 
| 152 | 
            +
                    </div>
         | 
| 153 | 
            +
                  </nav>
         | 
| 154 | 
            +
                </div>
         | 
| 155 | 
            +
              );
         | 
| 156 | 
            +
            }
         | 
    	
        components/space/ask-ai/index.tsx
    ADDED
    
    | @@ -0,0 +1,41 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { ArrowUp } from "lucide-react";
         | 
| 4 | 
            +
            import { PiGearSixFill } from "react-icons/pi";
         | 
| 5 | 
            +
            import { TiUserAdd } from "react-icons/ti";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { Button } from "@/components/ui/button";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            export const AskAi = () => {
         | 
| 10 | 
            +
              return (
         | 
| 11 | 
            +
                <div className="bg-neutral-800 border border-neutral-700 rounded-2xl ring-[4px] focus-within:ring-neutral-500/30 focus-within:border-neutral-600 ring-transparent group">
         | 
| 12 | 
            +
                  <textarea
         | 
| 13 | 
            +
                    rows={3}
         | 
| 14 | 
            +
                    className="w-full bg-transparent text-sm outline-none text-white placeholder:text-neutral-400 p-4 resize-none mb-1"
         | 
| 15 | 
            +
                    placeholder="Ask DeepSite anything..."
         | 
| 16 | 
            +
                    onChange={() => {}}
         | 
| 17 | 
            +
                    onKeyDown={() => {}}
         | 
| 18 | 
            +
                  />
         | 
| 19 | 
            +
                  <div className="flex items-center justify-between gap-2 px-4 pb-3">
         | 
| 20 | 
            +
                    <div className="flex-1 flex justify-start">
         | 
| 21 | 
            +
                      <Button
         | 
| 22 | 
            +
                        size="iconXs"
         | 
| 23 | 
            +
                        variant="outline"
         | 
| 24 | 
            +
                        className="!border-neutral-600 !text-neutral-400 !hover:!border-neutral-500 hover:!text-neutral-300"
         | 
| 25 | 
            +
                      >
         | 
| 26 | 
            +
                        <TiUserAdd className="size-4" />
         | 
| 27 | 
            +
                      </Button>
         | 
| 28 | 
            +
                    </div>
         | 
| 29 | 
            +
                    <div className="flex items-center justify-end gap-2">
         | 
| 30 | 
            +
                      <Button variant="black" size="sm">
         | 
| 31 | 
            +
                        <PiGearSixFill className="size-4" />
         | 
| 32 | 
            +
                        Settings
         | 
| 33 | 
            +
                      </Button>
         | 
| 34 | 
            +
                      <Button size="iconXs">
         | 
| 35 | 
            +
                        <ArrowUp className="size-4" />
         | 
| 36 | 
            +
                      </Button>
         | 
| 37 | 
            +
                    </div>
         | 
| 38 | 
            +
                  </div>
         | 
| 39 | 
            +
                </div>
         | 
| 40 | 
            +
              );
         | 
| 41 | 
            +
            };
         | 
    	
        components/ui/avatar.tsx
    ADDED
    
    | @@ -0,0 +1,53 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import * as React from "react"
         | 
| 4 | 
            +
            import * as AvatarPrimitive from "@radix-ui/react-avatar"
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            import { cn } from "@/lib/utils"
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            function Avatar({
         | 
| 9 | 
            +
              className,
         | 
| 10 | 
            +
              ...props
         | 
| 11 | 
            +
            }: React.ComponentProps<typeof AvatarPrimitive.Root>) {
         | 
| 12 | 
            +
              return (
         | 
| 13 | 
            +
                <AvatarPrimitive.Root
         | 
| 14 | 
            +
                  data-slot="avatar"
         | 
| 15 | 
            +
                  className={cn(
         | 
| 16 | 
            +
                    "relative flex size-8 shrink-0 overflow-hidden rounded-full",
         | 
| 17 | 
            +
                    className
         | 
| 18 | 
            +
                  )}
         | 
| 19 | 
            +
                  {...props}
         | 
| 20 | 
            +
                />
         | 
| 21 | 
            +
              )
         | 
| 22 | 
            +
            }
         | 
| 23 | 
            +
             | 
| 24 | 
            +
            function AvatarImage({
         | 
| 25 | 
            +
              className,
         | 
| 26 | 
            +
              ...props
         | 
| 27 | 
            +
            }: React.ComponentProps<typeof AvatarPrimitive.Image>) {
         | 
| 28 | 
            +
              return (
         | 
| 29 | 
            +
                <AvatarPrimitive.Image
         | 
| 30 | 
            +
                  data-slot="avatar-image"
         | 
| 31 | 
            +
                  className={cn("aspect-square size-full", className)}
         | 
| 32 | 
            +
                  {...props}
         | 
| 33 | 
            +
                />
         | 
| 34 | 
            +
              )
         | 
| 35 | 
            +
            }
         | 
| 36 | 
            +
             | 
| 37 | 
            +
            function AvatarFallback({
         | 
| 38 | 
            +
              className,
         | 
| 39 | 
            +
              ...props
         | 
| 40 | 
            +
            }: React.ComponentProps<typeof AvatarPrimitive.Fallback>) {
         | 
| 41 | 
            +
              return (
         | 
| 42 | 
            +
                <AvatarPrimitive.Fallback
         | 
| 43 | 
            +
                  data-slot="avatar-fallback"
         | 
| 44 | 
            +
                  className={cn(
         | 
| 45 | 
            +
                    "bg-muted flex size-full items-center justify-center rounded-full",
         | 
| 46 | 
            +
                    className
         | 
| 47 | 
            +
                  )}
         | 
| 48 | 
            +
                  {...props}
         | 
| 49 | 
            +
                />
         | 
| 50 | 
            +
              )
         | 
| 51 | 
            +
            }
         | 
| 52 | 
            +
             | 
| 53 | 
            +
            export { Avatar, AvatarImage, AvatarFallback }
         | 
    	
        components/ui/button.tsx
    ADDED
    
    | @@ -0,0 +1,67 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import * as React from "react";
         | 
| 2 | 
            +
            import { Slot } from "@radix-ui/react-slot";
         | 
| 3 | 
            +
            import { cva, type VariantProps } from "class-variance-authority";
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            import { cn } from "@/lib/utils";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            const buttonVariants = cva(
         | 
| 8 | 
            +
              "inline-flex items-center cursor-pointer justify-center gap-2 whitespace-nowrap rounded-full text-sm font-sans font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
         | 
| 9 | 
            +
              {
         | 
| 10 | 
            +
                variants: {
         | 
| 11 | 
            +
                  variant: {
         | 
| 12 | 
            +
                    default:
         | 
| 13 | 
            +
                      "bg-primary text-primary-foreground shadow-xs hover:bg-primary/90",
         | 
| 14 | 
            +
                    destructive:
         | 
| 15 | 
            +
                      "bg-red-500 text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 [&_svg]:!text-white",
         | 
| 16 | 
            +
                    outline:
         | 
| 17 | 
            +
                      "border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50",
         | 
| 18 | 
            +
                    secondary:
         | 
| 19 | 
            +
                      "bg-secondary text-secondary-foreground shadow-xs hover:bg-secondary/80",
         | 
| 20 | 
            +
                    ghost:
         | 
| 21 | 
            +
                      "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
         | 
| 22 | 
            +
                    lightGray: "bg-neutral-200/60 hover:bg-neutral-200",
         | 
| 23 | 
            +
                    link: "text-primary underline-offset-4 hover:underline",
         | 
| 24 | 
            +
                    ghostDarker:
         | 
| 25 | 
            +
                      "text-white shadow-xs focus-visible:ring-black/40 bg-black/40 hover:bg-black/70",
         | 
| 26 | 
            +
                    black: "bg-neutral-950 text-neutral-300 hover:brightness-110",
         | 
| 27 | 
            +
                    sky: "bg-sky-500 text-white hover:brightness-110",
         | 
| 28 | 
            +
                  },
         | 
| 29 | 
            +
                  size: {
         | 
| 30 | 
            +
                    default: "h-9 px-4 py-2 has-[>svg]:px-3",
         | 
| 31 | 
            +
                    sm: "h-8 rounded-full text-[13px] gap-1.5 px-3",
         | 
| 32 | 
            +
                    lg: "h-10 rounded-full px-6 has-[>svg]:px-4",
         | 
| 33 | 
            +
                    icon: "size-9",
         | 
| 34 | 
            +
                    iconXs: "size-7",
         | 
| 35 | 
            +
                    iconXss: "size-6",
         | 
| 36 | 
            +
                    xs: "h-6 text-xs rounded-full pl-2 pr-2 gap-1",
         | 
| 37 | 
            +
                  },
         | 
| 38 | 
            +
                },
         | 
| 39 | 
            +
                defaultVariants: {
         | 
| 40 | 
            +
                  variant: "default",
         | 
| 41 | 
            +
                  size: "default",
         | 
| 42 | 
            +
                },
         | 
| 43 | 
            +
              }
         | 
| 44 | 
            +
            );
         | 
| 45 | 
            +
             | 
| 46 | 
            +
            function Button({
         | 
| 47 | 
            +
              className,
         | 
| 48 | 
            +
              variant,
         | 
| 49 | 
            +
              size,
         | 
| 50 | 
            +
              asChild = false,
         | 
| 51 | 
            +
              ...props
         | 
| 52 | 
            +
            }: React.ComponentProps<"button"> &
         | 
| 53 | 
            +
              VariantProps<typeof buttonVariants> & {
         | 
| 54 | 
            +
                asChild?: boolean;
         | 
| 55 | 
            +
              }) {
         | 
| 56 | 
            +
              const Comp = asChild ? Slot : "button";
         | 
| 57 | 
            +
             | 
| 58 | 
            +
              return (
         | 
| 59 | 
            +
                <Comp
         | 
| 60 | 
            +
                  data-slot="button"
         | 
| 61 | 
            +
                  className={cn(buttonVariants({ variant, size, className }))}
         | 
| 62 | 
            +
                  {...props}
         | 
| 63 | 
            +
                />
         | 
| 64 | 
            +
              );
         | 
| 65 | 
            +
            }
         | 
| 66 | 
            +
             | 
| 67 | 
            +
            export { Button, buttonVariants };
         | 
    	
        components/ui/dialog.tsx
    ADDED
    
    | @@ -0,0 +1,143 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import * as React from "react"
         | 
| 4 | 
            +
            import * as DialogPrimitive from "@radix-ui/react-dialog"
         | 
| 5 | 
            +
            import { XIcon } from "lucide-react"
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { cn } from "@/lib/utils"
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            function Dialog({
         | 
| 10 | 
            +
              ...props
         | 
| 11 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Root>) {
         | 
| 12 | 
            +
              return <DialogPrimitive.Root data-slot="dialog" {...props} />
         | 
| 13 | 
            +
            }
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            function DialogTrigger({
         | 
| 16 | 
            +
              ...props
         | 
| 17 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Trigger>) {
         | 
| 18 | 
            +
              return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} />
         | 
| 19 | 
            +
            }
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            function DialogPortal({
         | 
| 22 | 
            +
              ...props
         | 
| 23 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Portal>) {
         | 
| 24 | 
            +
              return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} />
         | 
| 25 | 
            +
            }
         | 
| 26 | 
            +
             | 
| 27 | 
            +
            function DialogClose({
         | 
| 28 | 
            +
              ...props
         | 
| 29 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Close>) {
         | 
| 30 | 
            +
              return <DialogPrimitive.Close data-slot="dialog-close" {...props} />
         | 
| 31 | 
            +
            }
         | 
| 32 | 
            +
             | 
| 33 | 
            +
            function DialogOverlay({
         | 
| 34 | 
            +
              className,
         | 
| 35 | 
            +
              ...props
         | 
| 36 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Overlay>) {
         | 
| 37 | 
            +
              return (
         | 
| 38 | 
            +
                <DialogPrimitive.Overlay
         | 
| 39 | 
            +
                  data-slot="dialog-overlay"
         | 
| 40 | 
            +
                  className={cn(
         | 
| 41 | 
            +
                    "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
         | 
| 42 | 
            +
                    className
         | 
| 43 | 
            +
                  )}
         | 
| 44 | 
            +
                  {...props}
         | 
| 45 | 
            +
                />
         | 
| 46 | 
            +
              )
         | 
| 47 | 
            +
            }
         | 
| 48 | 
            +
             | 
| 49 | 
            +
            function DialogContent({
         | 
| 50 | 
            +
              className,
         | 
| 51 | 
            +
              children,
         | 
| 52 | 
            +
              showCloseButton = true,
         | 
| 53 | 
            +
              ...props
         | 
| 54 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Content> & {
         | 
| 55 | 
            +
              showCloseButton?: boolean
         | 
| 56 | 
            +
            }) {
         | 
| 57 | 
            +
              return (
         | 
| 58 | 
            +
                <DialogPortal data-slot="dialog-portal">
         | 
| 59 | 
            +
                  <DialogOverlay />
         | 
| 60 | 
            +
                  <DialogPrimitive.Content
         | 
| 61 | 
            +
                    data-slot="dialog-content"
         | 
| 62 | 
            +
                    className={cn(
         | 
| 63 | 
            +
                      "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 sm:max-w-lg",
         | 
| 64 | 
            +
                      className
         | 
| 65 | 
            +
                    )}
         | 
| 66 | 
            +
                    {...props}
         | 
| 67 | 
            +
                  >
         | 
| 68 | 
            +
                    {children}
         | 
| 69 | 
            +
                    {showCloseButton && (
         | 
| 70 | 
            +
                      <DialogPrimitive.Close
         | 
| 71 | 
            +
                        data-slot="dialog-close"
         | 
| 72 | 
            +
                        className="ring-offset-background focus:ring-ring data-[state=open]:bg-accent data-[state=open]:text-muted-foreground absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4"
         | 
| 73 | 
            +
                      >
         | 
| 74 | 
            +
                        <XIcon />
         | 
| 75 | 
            +
                        <span className="sr-only">Close</span>
         | 
| 76 | 
            +
                      </DialogPrimitive.Close>
         | 
| 77 | 
            +
                    )}
         | 
| 78 | 
            +
                  </DialogPrimitive.Content>
         | 
| 79 | 
            +
                </DialogPortal>
         | 
| 80 | 
            +
              )
         | 
| 81 | 
            +
            }
         | 
| 82 | 
            +
             | 
| 83 | 
            +
            function DialogHeader({ className, ...props }: React.ComponentProps<"div">) {
         | 
| 84 | 
            +
              return (
         | 
| 85 | 
            +
                <div
         | 
| 86 | 
            +
                  data-slot="dialog-header"
         | 
| 87 | 
            +
                  className={cn("flex flex-col gap-2 text-center sm:text-left", className)}
         | 
| 88 | 
            +
                  {...props}
         | 
| 89 | 
            +
                />
         | 
| 90 | 
            +
              )
         | 
| 91 | 
            +
            }
         | 
| 92 | 
            +
             | 
| 93 | 
            +
            function DialogFooter({ className, ...props }: React.ComponentProps<"div">) {
         | 
| 94 | 
            +
              return (
         | 
| 95 | 
            +
                <div
         | 
| 96 | 
            +
                  data-slot="dialog-footer"
         | 
| 97 | 
            +
                  className={cn(
         | 
| 98 | 
            +
                    "flex flex-col-reverse gap-2 sm:flex-row sm:justify-end",
         | 
| 99 | 
            +
                    className
         | 
| 100 | 
            +
                  )}
         | 
| 101 | 
            +
                  {...props}
         | 
| 102 | 
            +
                />
         | 
| 103 | 
            +
              )
         | 
| 104 | 
            +
            }
         | 
| 105 | 
            +
             | 
| 106 | 
            +
            function DialogTitle({
         | 
| 107 | 
            +
              className,
         | 
| 108 | 
            +
              ...props
         | 
| 109 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Title>) {
         | 
| 110 | 
            +
              return (
         | 
| 111 | 
            +
                <DialogPrimitive.Title
         | 
| 112 | 
            +
                  data-slot="dialog-title"
         | 
| 113 | 
            +
                  className={cn("text-lg leading-none font-semibold", className)}
         | 
| 114 | 
            +
                  {...props}
         | 
| 115 | 
            +
                />
         | 
| 116 | 
            +
              )
         | 
| 117 | 
            +
            }
         | 
| 118 | 
            +
             | 
| 119 | 
            +
            function DialogDescription({
         | 
| 120 | 
            +
              className,
         | 
| 121 | 
            +
              ...props
         | 
| 122 | 
            +
            }: React.ComponentProps<typeof DialogPrimitive.Description>) {
         | 
| 123 | 
            +
              return (
         | 
| 124 | 
            +
                <DialogPrimitive.Description
         | 
| 125 | 
            +
                  data-slot="dialog-description"
         | 
| 126 | 
            +
                  className={cn("text-muted-foreground text-sm", className)}
         | 
| 127 | 
            +
                  {...props}
         | 
| 128 | 
            +
                />
         | 
| 129 | 
            +
              )
         | 
| 130 | 
            +
            }
         | 
| 131 | 
            +
             | 
| 132 | 
            +
            export {
         | 
| 133 | 
            +
              Dialog,
         | 
| 134 | 
            +
              DialogClose,
         | 
| 135 | 
            +
              DialogContent,
         | 
| 136 | 
            +
              DialogDescription,
         | 
| 137 | 
            +
              DialogFooter,
         | 
| 138 | 
            +
              DialogHeader,
         | 
| 139 | 
            +
              DialogOverlay,
         | 
| 140 | 
            +
              DialogPortal,
         | 
| 141 | 
            +
              DialogTitle,
         | 
| 142 | 
            +
              DialogTrigger,
         | 
| 143 | 
            +
            }
         | 
    	
        components/ui/dropdown-menu.tsx
    ADDED
    
    | @@ -0,0 +1,257 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import * as React from "react";
         | 
| 4 | 
            +
            import * as DropdownMenuPrimitive from "@radix-ui/react-dropdown-menu";
         | 
| 5 | 
            +
            import { CheckIcon, ChevronRightIcon, CircleIcon } from "lucide-react";
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { cn } from "@/lib/utils";
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            function DropdownMenu({
         | 
| 10 | 
            +
              ...props
         | 
| 11 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Root>) {
         | 
| 12 | 
            +
              return <DropdownMenuPrimitive.Root data-slot="dropdown-menu" {...props} />;
         | 
| 13 | 
            +
            }
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            function DropdownMenuPortal({
         | 
| 16 | 
            +
              ...props
         | 
| 17 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Portal>) {
         | 
| 18 | 
            +
              return (
         | 
| 19 | 
            +
                <DropdownMenuPrimitive.Portal data-slot="dropdown-menu-portal" {...props} />
         | 
| 20 | 
            +
              );
         | 
| 21 | 
            +
            }
         | 
| 22 | 
            +
             | 
| 23 | 
            +
            function DropdownMenuTrigger({
         | 
| 24 | 
            +
              ...props
         | 
| 25 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Trigger>) {
         | 
| 26 | 
            +
              return (
         | 
| 27 | 
            +
                <DropdownMenuPrimitive.Trigger
         | 
| 28 | 
            +
                  data-slot="dropdown-menu-trigger"
         | 
| 29 | 
            +
                  {...props}
         | 
| 30 | 
            +
                />
         | 
| 31 | 
            +
              );
         | 
| 32 | 
            +
            }
         | 
| 33 | 
            +
             | 
| 34 | 
            +
            function DropdownMenuContent({
         | 
| 35 | 
            +
              className,
         | 
| 36 | 
            +
              sideOffset = 4,
         | 
| 37 | 
            +
              ...props
         | 
| 38 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Content>) {
         | 
| 39 | 
            +
              return (
         | 
| 40 | 
            +
                <DropdownMenuPrimitive.Portal>
         | 
| 41 | 
            +
                  <DropdownMenuPrimitive.Content
         | 
| 42 | 
            +
                    data-slot="dropdown-menu-content"
         | 
| 43 | 
            +
                    sideOffset={sideOffset}
         | 
| 44 | 
            +
                    className={cn(
         | 
| 45 | 
            +
                      "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-dropdown-menu-content-available-height) min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
         | 
| 46 | 
            +
                      className
         | 
| 47 | 
            +
                    )}
         | 
| 48 | 
            +
                    {...props}
         | 
| 49 | 
            +
                  />
         | 
| 50 | 
            +
                </DropdownMenuPrimitive.Portal>
         | 
| 51 | 
            +
              );
         | 
| 52 | 
            +
            }
         | 
| 53 | 
            +
             | 
| 54 | 
            +
            function DropdownMenuGroup({
         | 
| 55 | 
            +
              ...props
         | 
| 56 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Group>) {
         | 
| 57 | 
            +
              return (
         | 
| 58 | 
            +
                <DropdownMenuPrimitive.Group data-slot="dropdown-menu-group" {...props} />
         | 
| 59 | 
            +
              );
         | 
| 60 | 
            +
            }
         | 
| 61 | 
            +
             | 
| 62 | 
            +
            function DropdownMenuItem({
         | 
| 63 | 
            +
              className,
         | 
| 64 | 
            +
              inset,
         | 
| 65 | 
            +
              variant = "default",
         | 
| 66 | 
            +
              ...props
         | 
| 67 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Item> & {
         | 
| 68 | 
            +
              inset?: boolean;
         | 
| 69 | 
            +
              variant?: "default" | "destructive";
         | 
| 70 | 
            +
            }) {
         | 
| 71 | 
            +
              return (
         | 
| 72 | 
            +
                <DropdownMenuPrimitive.Item
         | 
| 73 | 
            +
                  data-slot="dropdown-menu-item"
         | 
| 74 | 
            +
                  data-inset={inset}
         | 
| 75 | 
            +
                  data-variant={variant}
         | 
| 76 | 
            +
                  className={cn(
         | 
| 77 | 
            +
                    "focus:bg-accent focus:text-accent-foreground data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 dark:data-[variant=destructive]:focus:bg-destructive/20 data-[variant=destructive]:focus:text-destructive data-[variant=destructive]:*:[svg]:!text-destructive [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-pointer items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
         | 
| 78 | 
            +
                    className
         | 
| 79 | 
            +
                  )}
         | 
| 80 | 
            +
                  {...props}
         | 
| 81 | 
            +
                />
         | 
| 82 | 
            +
              );
         | 
| 83 | 
            +
            }
         | 
| 84 | 
            +
             | 
| 85 | 
            +
            function DropdownMenuCheckboxItem({
         | 
| 86 | 
            +
              className,
         | 
| 87 | 
            +
              children,
         | 
| 88 | 
            +
              checked,
         | 
| 89 | 
            +
              ...props
         | 
| 90 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.CheckboxItem>) {
         | 
| 91 | 
            +
              return (
         | 
| 92 | 
            +
                <DropdownMenuPrimitive.CheckboxItem
         | 
| 93 | 
            +
                  data-slot="dropdown-menu-checkbox-item"
         | 
| 94 | 
            +
                  className={cn(
         | 
| 95 | 
            +
                    "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
         | 
| 96 | 
            +
                    className
         | 
| 97 | 
            +
                  )}
         | 
| 98 | 
            +
                  checked={checked}
         | 
| 99 | 
            +
                  {...props}
         | 
| 100 | 
            +
                >
         | 
| 101 | 
            +
                  <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
         | 
| 102 | 
            +
                    <DropdownMenuPrimitive.ItemIndicator>
         | 
| 103 | 
            +
                      <CheckIcon className="size-4" />
         | 
| 104 | 
            +
                    </DropdownMenuPrimitive.ItemIndicator>
         | 
| 105 | 
            +
                  </span>
         | 
| 106 | 
            +
                  {children}
         | 
| 107 | 
            +
                </DropdownMenuPrimitive.CheckboxItem>
         | 
| 108 | 
            +
              );
         | 
| 109 | 
            +
            }
         | 
| 110 | 
            +
             | 
| 111 | 
            +
            function DropdownMenuRadioGroup({
         | 
| 112 | 
            +
              ...props
         | 
| 113 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioGroup>) {
         | 
| 114 | 
            +
              return (
         | 
| 115 | 
            +
                <DropdownMenuPrimitive.RadioGroup
         | 
| 116 | 
            +
                  data-slot="dropdown-menu-radio-group"
         | 
| 117 | 
            +
                  {...props}
         | 
| 118 | 
            +
                />
         | 
| 119 | 
            +
              );
         | 
| 120 | 
            +
            }
         | 
| 121 | 
            +
             | 
| 122 | 
            +
            function DropdownMenuRadioItem({
         | 
| 123 | 
            +
              className,
         | 
| 124 | 
            +
              children,
         | 
| 125 | 
            +
              ...props
         | 
| 126 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioItem>) {
         | 
| 127 | 
            +
              return (
         | 
| 128 | 
            +
                <DropdownMenuPrimitive.RadioItem
         | 
| 129 | 
            +
                  data-slot="dropdown-menu-radio-item"
         | 
| 130 | 
            +
                  className={cn(
         | 
| 131 | 
            +
                    "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
         | 
| 132 | 
            +
                    className
         | 
| 133 | 
            +
                  )}
         | 
| 134 | 
            +
                  {...props}
         | 
| 135 | 
            +
                >
         | 
| 136 | 
            +
                  <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
         | 
| 137 | 
            +
                    <DropdownMenuPrimitive.ItemIndicator>
         | 
| 138 | 
            +
                      <CircleIcon className="size-2 fill-current" />
         | 
| 139 | 
            +
                    </DropdownMenuPrimitive.ItemIndicator>
         | 
| 140 | 
            +
                  </span>
         | 
| 141 | 
            +
                  {children}
         | 
| 142 | 
            +
                </DropdownMenuPrimitive.RadioItem>
         | 
| 143 | 
            +
              );
         | 
| 144 | 
            +
            }
         | 
| 145 | 
            +
             | 
| 146 | 
            +
            function DropdownMenuLabel({
         | 
| 147 | 
            +
              className,
         | 
| 148 | 
            +
              inset,
         | 
| 149 | 
            +
              ...props
         | 
| 150 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Label> & {
         | 
| 151 | 
            +
              inset?: boolean;
         | 
| 152 | 
            +
            }) {
         | 
| 153 | 
            +
              return (
         | 
| 154 | 
            +
                <DropdownMenuPrimitive.Label
         | 
| 155 | 
            +
                  data-slot="dropdown-menu-label"
         | 
| 156 | 
            +
                  data-inset={inset}
         | 
| 157 | 
            +
                  className={cn(
         | 
| 158 | 
            +
                    "px-2 py-1.5 text-sm font-medium data-[inset]:pl-8",
         | 
| 159 | 
            +
                    className
         | 
| 160 | 
            +
                  )}
         | 
| 161 | 
            +
                  {...props}
         | 
| 162 | 
            +
                />
         | 
| 163 | 
            +
              );
         | 
| 164 | 
            +
            }
         | 
| 165 | 
            +
             | 
| 166 | 
            +
            function DropdownMenuSeparator({
         | 
| 167 | 
            +
              className,
         | 
| 168 | 
            +
              ...props
         | 
| 169 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Separator>) {
         | 
| 170 | 
            +
              return (
         | 
| 171 | 
            +
                <DropdownMenuPrimitive.Separator
         | 
| 172 | 
            +
                  data-slot="dropdown-menu-separator"
         | 
| 173 | 
            +
                  className={cn("bg-border -mx-1 my-1 h-px", className)}
         | 
| 174 | 
            +
                  {...props}
         | 
| 175 | 
            +
                />
         | 
| 176 | 
            +
              );
         | 
| 177 | 
            +
            }
         | 
| 178 | 
            +
             | 
| 179 | 
            +
            function DropdownMenuShortcut({
         | 
| 180 | 
            +
              className,
         | 
| 181 | 
            +
              ...props
         | 
| 182 | 
            +
            }: React.ComponentProps<"span">) {
         | 
| 183 | 
            +
              return (
         | 
| 184 | 
            +
                <span
         | 
| 185 | 
            +
                  data-slot="dropdown-menu-shortcut"
         | 
| 186 | 
            +
                  className={cn(
         | 
| 187 | 
            +
                    "text-muted-foreground ml-auto text-xs tracking-widest",
         | 
| 188 | 
            +
                    className
         | 
| 189 | 
            +
                  )}
         | 
| 190 | 
            +
                  {...props}
         | 
| 191 | 
            +
                />
         | 
| 192 | 
            +
              );
         | 
| 193 | 
            +
            }
         | 
| 194 | 
            +
             | 
| 195 | 
            +
            function DropdownMenuSub({
         | 
| 196 | 
            +
              ...props
         | 
| 197 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.Sub>) {
         | 
| 198 | 
            +
              return <DropdownMenuPrimitive.Sub data-slot="dropdown-menu-sub" {...props} />;
         | 
| 199 | 
            +
            }
         | 
| 200 | 
            +
             | 
| 201 | 
            +
            function DropdownMenuSubTrigger({
         | 
| 202 | 
            +
              className,
         | 
| 203 | 
            +
              inset,
         | 
| 204 | 
            +
              children,
         | 
| 205 | 
            +
              ...props
         | 
| 206 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.SubTrigger> & {
         | 
| 207 | 
            +
              inset?: boolean;
         | 
| 208 | 
            +
            }) {
         | 
| 209 | 
            +
              return (
         | 
| 210 | 
            +
                <DropdownMenuPrimitive.SubTrigger
         | 
| 211 | 
            +
                  data-slot="dropdown-menu-sub-trigger"
         | 
| 212 | 
            +
                  data-inset={inset}
         | 
| 213 | 
            +
                  className={cn(
         | 
| 214 | 
            +
                    "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[inset]:pl-8",
         | 
| 215 | 
            +
                    className
         | 
| 216 | 
            +
                  )}
         | 
| 217 | 
            +
                  {...props}
         | 
| 218 | 
            +
                >
         | 
| 219 | 
            +
                  {children}
         | 
| 220 | 
            +
                  <ChevronRightIcon className="ml-auto size-4" />
         | 
| 221 | 
            +
                </DropdownMenuPrimitive.SubTrigger>
         | 
| 222 | 
            +
              );
         | 
| 223 | 
            +
            }
         | 
| 224 | 
            +
             | 
| 225 | 
            +
            function DropdownMenuSubContent({
         | 
| 226 | 
            +
              className,
         | 
| 227 | 
            +
              ...props
         | 
| 228 | 
            +
            }: React.ComponentProps<typeof DropdownMenuPrimitive.SubContent>) {
         | 
| 229 | 
            +
              return (
         | 
| 230 | 
            +
                <DropdownMenuPrimitive.SubContent
         | 
| 231 | 
            +
                  data-slot="dropdown-menu-sub-content"
         | 
| 232 | 
            +
                  className={cn(
         | 
| 233 | 
            +
                    "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-lg",
         | 
| 234 | 
            +
                    className
         | 
| 235 | 
            +
                  )}
         | 
| 236 | 
            +
                  {...props}
         | 
| 237 | 
            +
                />
         | 
| 238 | 
            +
              );
         | 
| 239 | 
            +
            }
         | 
| 240 | 
            +
             | 
| 241 | 
            +
            export {
         | 
| 242 | 
            +
              DropdownMenu,
         | 
| 243 | 
            +
              DropdownMenuPortal,
         | 
| 244 | 
            +
              DropdownMenuTrigger,
         | 
| 245 | 
            +
              DropdownMenuContent,
         | 
| 246 | 
            +
              DropdownMenuGroup,
         | 
| 247 | 
            +
              DropdownMenuLabel,
         | 
| 248 | 
            +
              DropdownMenuItem,
         | 
| 249 | 
            +
              DropdownMenuCheckboxItem,
         | 
| 250 | 
            +
              DropdownMenuRadioGroup,
         | 
| 251 | 
            +
              DropdownMenuRadioItem,
         | 
| 252 | 
            +
              DropdownMenuSeparator,
         | 
| 253 | 
            +
              DropdownMenuShortcut,
         | 
| 254 | 
            +
              DropdownMenuSub,
         | 
| 255 | 
            +
              DropdownMenuSubTrigger,
         | 
| 256 | 
            +
              DropdownMenuSubContent,
         | 
| 257 | 
            +
            };
         | 
    	
        components/ui/input.tsx
    ADDED
    
    | @@ -0,0 +1,21 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            import * as React from "react"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import { cn } from "@/lib/utils"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            function Input({ className, type, ...props }: React.ComponentProps<"input">) {
         | 
| 6 | 
            +
              return (
         | 
| 7 | 
            +
                <input
         | 
| 8 | 
            +
                  type={type}
         | 
| 9 | 
            +
                  data-slot="input"
         | 
| 10 | 
            +
                  className={cn(
         | 
| 11 | 
            +
                    "file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 border-input flex h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
         | 
| 12 | 
            +
                    "focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]",
         | 
| 13 | 
            +
                    "aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
         | 
| 14 | 
            +
                    className
         | 
| 15 | 
            +
                  )}
         | 
| 16 | 
            +
                  {...props}
         | 
| 17 | 
            +
                />
         | 
| 18 | 
            +
              )
         | 
| 19 | 
            +
            }
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            export { Input }
         | 
    	
        components/ui/popover.tsx
    ADDED
    
    | @@ -0,0 +1,48 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client";
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import * as React from "react";
         | 
| 4 | 
            +
            import * as PopoverPrimitive from "@radix-ui/react-popover";
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            import { cn } from "@/lib/utils";
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            function Popover({
         | 
| 9 | 
            +
              ...props
         | 
| 10 | 
            +
            }: React.ComponentProps<typeof PopoverPrimitive.Root>) {
         | 
| 11 | 
            +
              return <PopoverPrimitive.Root data-slot="popover" {...props} />;
         | 
| 12 | 
            +
            }
         | 
| 13 | 
            +
             | 
| 14 | 
            +
            function PopoverTrigger({
         | 
| 15 | 
            +
              ...props
         | 
| 16 | 
            +
            }: React.ComponentProps<typeof PopoverPrimitive.Trigger>) {
         | 
| 17 | 
            +
              return <PopoverPrimitive.Trigger data-slot="popover-trigger" {...props} />;
         | 
| 18 | 
            +
            }
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            function PopoverContent({
         | 
| 21 | 
            +
              className,
         | 
| 22 | 
            +
              align = "center",
         | 
| 23 | 
            +
              sideOffset = 4,
         | 
| 24 | 
            +
              ...props
         | 
| 25 | 
            +
            }: React.ComponentProps<typeof PopoverPrimitive.Content>) {
         | 
| 26 | 
            +
              return (
         | 
| 27 | 
            +
                <PopoverPrimitive.Portal>
         | 
| 28 | 
            +
                  <PopoverPrimitive.Content
         | 
| 29 | 
            +
                    data-slot="popover-content"
         | 
| 30 | 
            +
                    align={align}
         | 
| 31 | 
            +
                    sideOffset={sideOffset}
         | 
| 32 | 
            +
                    className={cn(
         | 
| 33 | 
            +
                      "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-72 origin-(--radix-popover-content-transform-origin) rounded-md border p-4 shadow-md outline-hidden",
         | 
| 34 | 
            +
                      className
         | 
| 35 | 
            +
                    )}
         | 
| 36 | 
            +
                    {...props}
         | 
| 37 | 
            +
                  />
         | 
| 38 | 
            +
                </PopoverPrimitive.Portal>
         | 
| 39 | 
            +
              );
         | 
| 40 | 
            +
            }
         | 
| 41 | 
            +
             | 
| 42 | 
            +
            function PopoverAnchor({
         | 
| 43 | 
            +
              ...props
         | 
| 44 | 
            +
            }: React.ComponentProps<typeof PopoverPrimitive.Anchor>) {
         | 
| 45 | 
            +
              return <PopoverPrimitive.Anchor data-slot="popover-anchor" {...props} />;
         | 
| 46 | 
            +
            }
         | 
| 47 | 
            +
             | 
| 48 | 
            +
            export { Popover, PopoverTrigger, PopoverContent, PopoverAnchor };
         | 
    	
        components/ui/select.tsx
    ADDED
    
    | @@ -0,0 +1,185 @@ | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | |
|  | 
|  | |
| 1 | 
            +
            "use client"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            import * as React from "react"
         | 
| 4 | 
            +
            import * as SelectPrimitive from "@radix-ui/react-select"
         | 
| 5 | 
            +
            import { CheckIcon, ChevronDownIcon, ChevronUpIcon } from "lucide-react"
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            import { cn } from "@/lib/utils"
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            function Select({
         | 
| 10 | 
            +
              ...props
         | 
| 11 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Root>) {
         | 
| 12 | 
            +
              return <SelectPrimitive.Root data-slot="select" {...props} />
         | 
| 13 | 
            +
            }
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            function SelectGroup({
         | 
| 16 | 
            +
              ...props
         | 
| 17 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Group>) {
         | 
| 18 | 
            +
              return <SelectPrimitive.Group data-slot="select-group" {...props} />
         | 
| 19 | 
            +
            }
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            function SelectValue({
         | 
| 22 | 
            +
              ...props
         | 
| 23 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Value>) {
         | 
| 24 | 
            +
              return <SelectPrimitive.Value data-slot="select-value" {...props} />
         | 
| 25 | 
            +
            }
         | 
| 26 | 
            +
             | 
| 27 | 
            +
            function SelectTrigger({
         | 
| 28 | 
            +
              className,
         | 
| 29 | 
            +
              size = "default",
         | 
| 30 | 
            +
              children,
         | 
| 31 | 
            +
              ...props
         | 
| 32 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Trigger> & {
         | 
| 33 | 
            +
              size?: "sm" | "default"
         | 
| 34 | 
            +
            }) {
         | 
| 35 | 
            +
              return (
         | 
| 36 | 
            +
                <SelectPrimitive.Trigger
         | 
| 37 | 
            +
                  data-slot="select-trigger"
         | 
| 38 | 
            +
                  data-size={size}
         | 
| 39 | 
            +
                  className={cn(
         | 
| 40 | 
            +
                    "border-input data-[placeholder]:text-muted-foreground [&_svg:not([class*='text-'])]:text-muted-foreground focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive dark:bg-input/30 dark:hover:bg-input/50 flex w-fit items-center justify-between gap-2 rounded-md border bg-transparent px-3 py-2 text-sm whitespace-nowrap shadow-xs transition-[color,box-shadow] outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50 data-[size=default]:h-9 data-[size=sm]:h-8 *:data-[slot=select-value]:line-clamp-1 *:data-[slot=select-value]:flex *:data-[slot=select-value]:items-center *:data-[slot=select-value]:gap-2 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
         | 
| 41 | 
            +
                    className
         | 
| 42 | 
            +
                  )}
         | 
| 43 | 
            +
                  {...props}
         | 
| 44 | 
            +
                >
         | 
| 45 | 
            +
                  {children}
         | 
| 46 | 
            +
                  <SelectPrimitive.Icon asChild>
         | 
| 47 | 
            +
                    <ChevronDownIcon className="size-4 opacity-50" />
         | 
| 48 | 
            +
                  </SelectPrimitive.Icon>
         | 
| 49 | 
            +
                </SelectPrimitive.Trigger>
         | 
| 50 | 
            +
              )
         | 
| 51 | 
            +
            }
         | 
| 52 | 
            +
             | 
| 53 | 
            +
            function SelectContent({
         | 
| 54 | 
            +
              className,
         | 
| 55 | 
            +
              children,
         | 
| 56 | 
            +
              position = "popper",
         | 
| 57 | 
            +
              ...props
         | 
| 58 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Content>) {
         | 
| 59 | 
            +
              return (
         | 
| 60 | 
            +
                <SelectPrimitive.Portal>
         | 
| 61 | 
            +
                  <SelectPrimitive.Content
         | 
| 62 | 
            +
                    data-slot="select-content"
         | 
| 63 | 
            +
                    className={cn(
         | 
| 64 | 
            +
                      "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 relative z-50 max-h-(--radix-select-content-available-height) min-w-[8rem] origin-(--radix-select-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border shadow-md",
         | 
| 65 | 
            +
                      position === "popper" &&
         | 
| 66 | 
            +
                        "data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1",
         | 
| 67 | 
            +
                      className
         | 
| 68 | 
            +
                    )}
         | 
| 69 | 
            +
                    position={position}
         | 
| 70 | 
            +
                    {...props}
         | 
| 71 | 
            +
                  >
         | 
| 72 | 
            +
                    <SelectScrollUpButton />
         | 
| 73 | 
            +
                    <SelectPrimitive.Viewport
         | 
| 74 | 
            +
                      className={cn(
         | 
| 75 | 
            +
                        "p-1",
         | 
| 76 | 
            +
                        position === "popper" &&
         | 
| 77 | 
            +
                          "h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)] scroll-my-1"
         | 
| 78 | 
            +
                      )}
         | 
| 79 | 
            +
                    >
         | 
| 80 | 
            +
                      {children}
         | 
| 81 | 
            +
                    </SelectPrimitive.Viewport>
         | 
| 82 | 
            +
                    <SelectScrollDownButton />
         | 
| 83 | 
            +
                  </SelectPrimitive.Content>
         | 
| 84 | 
            +
                </SelectPrimitive.Portal>
         | 
| 85 | 
            +
              )
         | 
| 86 | 
            +
            }
         | 
| 87 | 
            +
             | 
| 88 | 
            +
            function SelectLabel({
         | 
| 89 | 
            +
              className,
         | 
| 90 | 
            +
              ...props
         | 
| 91 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Label>) {
         | 
| 92 | 
            +
              return (
         | 
| 93 | 
            +
                <SelectPrimitive.Label
         | 
| 94 | 
            +
                  data-slot="select-label"
         | 
| 95 | 
            +
                  className={cn("text-muted-foreground px-2 py-1.5 text-xs", className)}
         | 
| 96 | 
            +
                  {...props}
         | 
| 97 | 
            +
                />
         | 
| 98 | 
            +
              )
         | 
| 99 | 
            +
            }
         | 
| 100 | 
            +
             | 
| 101 | 
            +
            function SelectItem({
         | 
| 102 | 
            +
              className,
         | 
| 103 | 
            +
              children,
         | 
| 104 | 
            +
              ...props
         | 
| 105 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Item>) {
         | 
| 106 | 
            +
              return (
         | 
| 107 | 
            +
                <SelectPrimitive.Item
         | 
| 108 | 
            +
                  data-slot="select-item"
         | 
| 109 | 
            +
                  className={cn(
         | 
| 110 | 
            +
                    "focus:bg-accent focus:text-accent-foreground [&_svg:not([class*='text-'])]:text-muted-foreground relative flex w-full cursor-default items-center gap-2 rounded-sm py-1.5 pr-8 pl-2 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 *:[span]:last:flex *:[span]:last:items-center *:[span]:last:gap-2",
         | 
| 111 | 
            +
                    className
         | 
| 112 | 
            +
                  )}
         | 
| 113 | 
            +
                  {...props}
         | 
| 114 | 
            +
                >
         | 
| 115 | 
            +
                  <span className="absolute right-2 flex size-3.5 items-center justify-center">
         | 
| 116 | 
            +
                    <SelectPrimitive.ItemIndicator>
         | 
| 117 | 
            +
                      <CheckIcon className="size-4" />
         | 
| 118 | 
            +
                    </SelectPrimitive.ItemIndicator>
         | 
| 119 | 
            +
                  </span>
         | 
| 120 | 
            +
                  <SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
         | 
| 121 | 
            +
                </SelectPrimitive.Item>
         | 
| 122 | 
            +
              )
         | 
| 123 | 
            +
            }
         | 
| 124 | 
            +
             | 
| 125 | 
            +
            function SelectSeparator({
         | 
| 126 | 
            +
              className,
         | 
| 127 | 
            +
              ...props
         | 
| 128 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.Separator>) {
         | 
| 129 | 
            +
              return (
         | 
| 130 | 
            +
                <SelectPrimitive.Separator
         | 
| 131 | 
            +
                  data-slot="select-separator"
         | 
| 132 | 
            +
                  className={cn("bg-border pointer-events-none -mx-1 my-1 h-px", className)}
         | 
| 133 | 
            +
                  {...props}
         | 
| 134 | 
            +
                />
         | 
| 135 | 
            +
              )
         | 
| 136 | 
            +
            }
         | 
| 137 | 
            +
             | 
| 138 | 
            +
            function SelectScrollUpButton({
         | 
| 139 | 
            +
              className,
         | 
| 140 | 
            +
              ...props
         | 
| 141 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.ScrollUpButton>) {
         | 
| 142 | 
            +
              return (
         | 
| 143 | 
            +
                <SelectPrimitive.ScrollUpButton
         | 
| 144 | 
            +
                  data-slot="select-scroll-up-button"
         | 
| 145 | 
            +
                  className={cn(
         | 
| 146 | 
            +
                    "flex cursor-default items-center justify-center py-1",
         | 
| 147 | 
            +
                    className
         | 
| 148 | 
            +
                  )}
         | 
| 149 | 
            +
                  {...props}
         | 
| 150 | 
            +
                >
         | 
| 151 | 
            +
                  <ChevronUpIcon className="size-4" />
         | 
| 152 | 
            +
                </SelectPrimitive.ScrollUpButton>
         | 
| 153 | 
            +
              )
         | 
| 154 | 
            +
            }
         | 
| 155 | 
            +
             | 
| 156 | 
            +
            function SelectScrollDownButton({
         | 
| 157 | 
            +
              className,
         | 
| 158 | 
            +
              ...props
         | 
| 159 | 
            +
            }: React.ComponentProps<typeof SelectPrimitive.ScrollDownButton>) {
         | 
| 160 | 
            +
              return (
         | 
| 161 | 
            +
                <SelectPrimitive.ScrollDownButton
         | 
| 162 | 
            +
                  data-slot="select-scroll-down-button"
         | 
| 163 | 
            +
                  className={cn(
         | 
| 164 | 
            +
                    "flex cursor-default items-center justify-center py-1",
         | 
| 165 | 
            +
                    className
         | 
| 166 | 
            +
                  )}
         | 
| 167 | 
            +
                  {...props}
         | 
| 168 | 
            +
                >
         | 
| 169 | 
            +
                  <ChevronDownIcon className="size-4" />
         | 
| 170 | 
            +
                </SelectPrimitive.ScrollDownButton>
         | 
| 171 | 
            +
              )
         | 
| 172 | 
            +
            }
         | 
| 173 | 
            +
             | 
| 174 | 
            +
            export {
         | 
| 175 | 
            +
              Select,
         | 
| 176 | 
            +
              SelectContent,
         | 
| 177 | 
            +
              SelectGroup,
         | 
| 178 | 
            +
              SelectItem,
         | 
| 179 | 
            +
              SelectLabel,
         | 
| 180 | 
            +
              SelectScrollDownButton,
         | 
| 181 | 
            +
              SelectScrollUpButton,
         | 
| 182 | 
            +
              SelectSeparator,
         | 
| 183 | 
            +
              SelectTrigger,
         | 
| 184 | 
            +
              SelectValue,
         | 
| 185 | 
            +
            }
         | 
