import { useState, useEffect, useCallback, useRef } from "react";
import { Button } from "@/components/ui/button";
import { Globe } from "lucide-react";
import { IpcClient } from "@/ipc/ipc_client";
import { useSettings } from "@/hooks/useSettings";
import { useLoadApp } from "@/hooks/useLoadApp";
import { useVercelDeployments } from "@/hooks/useVercelDeployments";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import {} from "@/components/ui/dialog";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import { App } from "@/ipc/ipc_types";
interface VercelConnectorProps {
appId: number | null;
folderName: string;
}
interface VercelProject {
id: string;
name: string;
framework: string | null;
}
interface ConnectedVercelConnectorProps {
appId: number;
app: App;
refreshApp: () => void;
}
interface UnconnectedVercelConnectorProps {
appId: number | null;
folderName: string;
settings: any;
refreshSettings: () => void;
refreshApp: () => void;
}
function ConnectedVercelConnector({
appId,
app,
refreshApp,
}: ConnectedVercelConnectorProps) {
const {
deployments,
isLoading: isLoadingDeployments,
error: deploymentsError,
getDeployments: handleGetDeployments,
disconnectProject,
isDisconnecting,
disconnectError,
} = useVercelDeployments(appId);
const handleDisconnectProject = async () => {
await disconnectProject();
refreshApp();
};
return (
);
}
function UnconnectedVercelConnector({
appId,
folderName,
settings,
refreshSettings,
refreshApp,
}: UnconnectedVercelConnectorProps) {
// --- Manual Token Entry State ---
const [accessToken, setAccessToken] = useState("");
const [isSavingToken, setIsSavingToken] = useState(false);
const [tokenError, setTokenError] = useState(null);
const [tokenSuccess, setTokenSuccess] = useState(false);
// --- Project Setup State ---
const [projectSetupMode, setProjectSetupMode] = useState<
"create" | "existing"
>("create");
const [availableProjects, setAvailableProjects] = useState(
[],
);
const [isLoadingProjects, setIsLoadingProjects] = useState(false);
const [selectedProject, setSelectedProject] = useState("");
// Create new project state
const [projectName, setProjectName] = useState(folderName);
const [projectAvailable, setProjectAvailable] = useState(
null,
);
const [projectCheckError, setProjectCheckError] = useState(
null,
);
const [isCheckingProject, setIsCheckingProject] = useState(false);
const [isCreatingProject, setIsCreatingProject] = useState(false);
const [createProjectError, setCreateProjectError] = useState(
null,
);
const [createProjectSuccess, setCreateProjectSuccess] =
useState(false);
const debounceTimeoutRef = useRef(null);
// Load available projects when Vercel is connected
useEffect(() => {
if (settings?.vercelAccessToken && projectSetupMode === "existing") {
loadAvailableProjects();
}
}, [settings?.vercelAccessToken, projectSetupMode]);
// Cleanup debounce timer on unmount
useEffect(() => {
return () => {
if (debounceTimeoutRef.current) {
clearTimeout(debounceTimeoutRef.current);
}
};
}, []);
const loadAvailableProjects = async () => {
setIsLoadingProjects(true);
try {
const projects = await IpcClient.getInstance().listVercelProjects();
setAvailableProjects(projects);
} catch (error) {
console.error("Failed to load Vercel projects:", error);
} finally {
setIsLoadingProjects(false);
}
};
const handleSaveAccessToken = async (e: React.FormEvent) => {
e.preventDefault();
if (!accessToken.trim()) return;
setIsSavingToken(true);
setTokenError(null);
setTokenSuccess(false);
try {
await IpcClient.getInstance().saveVercelAccessToken({
token: accessToken.trim(),
});
setTokenSuccess(true);
setAccessToken("");
refreshSettings();
} catch (err: any) {
setTokenError(err.message || "Failed to save access token.");
} finally {
setIsSavingToken(false);
}
};
const checkProjectAvailability = useCallback(async (name: string) => {
setProjectCheckError(null);
setProjectAvailable(null);
if (!name) return;
setIsCheckingProject(true);
try {
const result = await IpcClient.getInstance().isVercelProjectAvailable({
name,
});
setProjectAvailable(result.available);
if (!result.available) {
setProjectCheckError(result.error || "Project name is not available.");
}
} catch (err: any) {
setProjectCheckError(
err.message || "Failed to check project availability.",
);
} finally {
setIsCheckingProject(false);
}
}, []);
const debouncedCheckProjectAvailability = useCallback(
(name: string) => {
if (debounceTimeoutRef.current) {
clearTimeout(debounceTimeoutRef.current);
}
debounceTimeoutRef.current = setTimeout(() => {
checkProjectAvailability(name);
}, 500);
},
[checkProjectAvailability],
);
const handleSetupProject = async (e: React.FormEvent) => {
e.preventDefault();
if (!appId) return;
setCreateProjectError(null);
setIsCreatingProject(true);
setCreateProjectSuccess(false);
try {
if (projectSetupMode === "create") {
await IpcClient.getInstance().createVercelProject({
name: projectName,
appId,
});
} else {
await IpcClient.getInstance().connectToExistingVercelProject({
projectId: selectedProject,
appId,
});
}
setCreateProjectSuccess(true);
setProjectCheckError(null);
refreshApp();
} catch (err: any) {
setCreateProjectError(
err.message ||
`Failed to ${projectSetupMode === "create" ? "create" : "connect to"} project.`,
);
} finally {
setIsCreatingProject(false);
}
};
if (!settings?.vercelAccessToken) {
return (
Connect to Vercel
To connect your app to Vercel, you'll need to create an access
token:
- If you don't have a Vercel account, sign up first
- Go to Vercel settings to create a token
- Copy the token and paste it below
{tokenError && (
)}
{tokenSuccess && (
Successfully connected to Vercel! You can now set up your
project below.
)}
);
}
return (
{/* Collapsible Header */}
Set up your Vercel project
{/* Collapsible Content */}
{/* Mode Selection */}
{createProjectError && (
{createProjectError}
)}
{createProjectSuccess && (
{projectSetupMode === "create"
? "Project created and linked!"
: "Connected to project!"}
)}
);
}
export function VercelConnector({ appId, folderName }: VercelConnectorProps) {
const { app, refreshApp } = useLoadApp(appId);
const { settings, refreshSettings } = useSettings();
if (app?.vercelProjectId && appId) {
return (
);
} else {
return (
);
}
}