219 lines
7.2 KiB
TypeScript
219 lines
7.2 KiB
TypeScript
import { useState, useEffect } from "react";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Label } from "@/components/ui/label";
|
|
import { SupabaseSchema } from "@/lib/schemas";
|
|
import { IpcClient } from "@/ipc/ipc_client";
|
|
import { toast } from "sonner";
|
|
import { useSettings } from "@/hooks/useSettings";
|
|
import { useSupabase } from "@/hooks/useSupabase";
|
|
import {
|
|
Select,
|
|
SelectContent,
|
|
SelectItem,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
} from "@/components/ui/select";
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardDescription,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from "@/components/ui/card";
|
|
import { Skeleton } from "@/components/ui/skeleton";
|
|
import { useLoadApp } from "@/hooks/useLoadApp";
|
|
import { useDeepLink } from "@/contexts/DeepLinkContext";
|
|
const OAUTH_CLIENT_ID = "bf747de7-60bb-48a2-9015-6494e0b04983";
|
|
// @ts-ignore
|
|
import supabaseLogoLight from "../../assets/supabase/supabase-logo-wordmark--light.svg";
|
|
// @ts-ignore
|
|
import supabaseLogoDark from "../../assets/supabase/supabase-logo-wordmark--dark.svg";
|
|
// @ts-ignore
|
|
import connectSupabaseDark from "../../assets/supabase/connect-supabase-dark.svg";
|
|
// @ts-ignore
|
|
import connectSupabaseLight from "../../assets/supabase/connect-supabase-light.svg";
|
|
|
|
import { ExternalLink } from "lucide-react";
|
|
import { useTheme } from "@/contexts/ThemeContext";
|
|
|
|
export function SupabaseConnector({ appId }: { appId: number }) {
|
|
const { settings, refreshSettings } = useSettings();
|
|
const { app, refreshApp } = useLoadApp(appId);
|
|
const { lastDeepLink } = useDeepLink();
|
|
const { isDarkMode } = useTheme();
|
|
useEffect(() => {
|
|
const handleDeepLink = async () => {
|
|
if (lastDeepLink?.type === "supabase-oauth-return") {
|
|
await refreshSettings();
|
|
await refreshApp();
|
|
}
|
|
};
|
|
handleDeepLink();
|
|
}, [lastDeepLink]);
|
|
const {
|
|
projects,
|
|
loading,
|
|
error,
|
|
loadProjects,
|
|
setAppProject,
|
|
unsetAppProject,
|
|
} = useSupabase();
|
|
const currentProjectId = app?.supabaseProjectId;
|
|
|
|
useEffect(() => {
|
|
// Load projects when the component mounts and user is connected
|
|
if (settings?.supabase?.accessToken) {
|
|
loadProjects();
|
|
}
|
|
}, [settings?.supabase?.accessToken, loadProjects]);
|
|
|
|
const handleProjectSelect = async (projectId: string) => {
|
|
try {
|
|
await setAppProject(projectId, appId);
|
|
toast.success("Project connected to app successfully");
|
|
await refreshApp();
|
|
} catch (error) {
|
|
toast.error("Failed to connect project to app");
|
|
}
|
|
};
|
|
|
|
const handleUnsetProject = async () => {
|
|
try {
|
|
await unsetAppProject(appId);
|
|
toast.success("Project disconnected from app successfully");
|
|
await refreshApp();
|
|
} catch (error) {
|
|
console.error("Failed to disconnect project:", error);
|
|
toast.error("Failed to disconnect project from app");
|
|
}
|
|
};
|
|
|
|
if (settings?.supabase?.accessToken) {
|
|
if (app?.supabaseProjectName) {
|
|
return (
|
|
<Card className="mt-1">
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center justify-between">
|
|
Supabase Project{" "}
|
|
<Button
|
|
variant="outline"
|
|
onClick={() => {
|
|
IpcClient.getInstance().openExternalUrl(
|
|
`https://supabase.com/dashboard/project/${app.supabaseProjectId}`
|
|
);
|
|
}}
|
|
className="ml-2 px-2 py-1"
|
|
style={{ display: "inline-flex", alignItems: "center" }}
|
|
asChild
|
|
>
|
|
<div className="flex items-center gap-2">
|
|
<img
|
|
src={isDarkMode ? supabaseLogoDark : supabaseLogoLight}
|
|
alt="Supabase Logo"
|
|
style={{ height: 20, width: "auto", marginRight: 4 }}
|
|
/>
|
|
<ExternalLink className="h-4 w-4" />
|
|
</div>
|
|
</Button>
|
|
</CardTitle>
|
|
<CardDescription>
|
|
This app is connected to project: {app.supabaseProjectName}
|
|
</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<Button variant="destructive" onClick={handleUnsetProject}>
|
|
Disconnect Project
|
|
</Button>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
return (
|
|
<Card className="mt-1">
|
|
<CardHeader>
|
|
<CardTitle>Supabase Projects</CardTitle>
|
|
<CardDescription>
|
|
Select a Supabase project to connect to this app
|
|
</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
{loading ? (
|
|
<div className="space-y-2">
|
|
<Skeleton className="h-4 w-full" />
|
|
<Skeleton className="h-10 w-full" />
|
|
</div>
|
|
) : error ? (
|
|
<div className="text-red-500">
|
|
Error loading projects: {error.message}
|
|
<Button
|
|
variant="outline"
|
|
className="mt-2"
|
|
onClick={() => loadProjects()}
|
|
>
|
|
Retry
|
|
</Button>
|
|
</div>
|
|
) : (
|
|
<div className="space-y-4">
|
|
{projects.length === 0 ? (
|
|
<p className="text-sm text-gray-500">
|
|
No projects found in your Supabase account.
|
|
</p>
|
|
) : (
|
|
<>
|
|
<div className="space-y-2">
|
|
<Label htmlFor="project-select">Project</Label>
|
|
<Select
|
|
value={currentProjectId || ""}
|
|
onValueChange={handleProjectSelect}
|
|
>
|
|
<SelectTrigger id="project-select">
|
|
<SelectValue placeholder="Select a project" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
{projects.map((project) => (
|
|
<SelectItem key={project.id} value={project.id}>
|
|
{project.name || project.id}
|
|
</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
|
|
{currentProjectId && (
|
|
<div className="text-sm text-gray-500">
|
|
This app is connected to project:{" "}
|
|
{projects.find((p) => p.id === currentProjectId)?.name ||
|
|
currentProjectId}
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
</div>
|
|
)}
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="flex flex-col space-y-4 p-4 border rounded-md">
|
|
<div className="flex flex-col md:flex-row items-center justify-between">
|
|
<h2 className="text-lg font-medium">Integrations</h2>
|
|
<img
|
|
onClick={() => {
|
|
IpcClient.getInstance().openExternalUrl(
|
|
"https://supabase-oauth.dyad.sh/api/connect-supabase/login"
|
|
);
|
|
}}
|
|
src={isDarkMode ? connectSupabaseDark : connectSupabaseLight}
|
|
alt="Connect to Supabase"
|
|
className="w-full h-10 min-h-8 min-w-20 cursor-pointer"
|
|
// className="h-10"
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|