ultisuite-client/components/admin/settings/sections/drive-mount-oauth-section.tsx
R3D347HR4Y ad1370ea7e
Some checks are pending
E2E / Playwright e2e (push) Waiting to run
feat: enhance configuration and add new demo layouts
- Introduced turbopack alias for canvas in next.config.mjs.
- Updated package.json scripts for development and branding tasks.
- Added new dependencies for Tiptap extensions.
- Implemented new demo layouts for agenda, contacts, drive, and mail applications.
- Enhanced globals.css for improved theming and splash screen animations.
- Added OAuth callback handling for drive mounts.
- Updated layout components to integrate new demo shells and improve structure.
2026-06-12 19:10:24 +02:00

155 lines
5.5 KiB
TypeScript

"use client"
import { useEffect, useState } from "react"
import { Check, Copy } from "lucide-react"
import { toast } from "sonner"
import { useOrgSettingsStore } from "@/lib/admin-settings/org-settings-store"
import type { DriveMountOAuthProvider, DriveMountOAuthSettings } from "@/lib/admin-settings/org-settings-types"
import { Button } from "@/components/ui/button"
import { Input } from "@/components/ui/input"
import { Label } from "@/components/ui/label"
import { Switch } from "@/components/ui/switch"
import { buildDriveMountOAuthRedirectURI } from "@/lib/drive/drive-mount-oauth"
const PROVIDERS: { id: DriveMountOAuthProvider; label: string; hint: string }[] = [
{
id: "google",
label: "Google Drive",
hint: "Console Google Cloud — API Drive, redirect URI ci-dessous",
},
{
id: "dropbox",
label: "Dropbox",
hint: "App Dropbox — permissions files.metadata.read, files.content.read/write",
},
{
id: "microsoft",
label: "Microsoft OneDrive",
hint: "Azure AD — Microsoft Graph Files.ReadWrite",
},
]
const SECRET_KEYS: Record<DriveMountOAuthProvider, "mount_oauth_google" | "mount_oauth_dropbox" | "mount_oauth_microsoft"> = {
google: "mount_oauth_google",
dropbox: "mount_oauth_dropbox",
microsoft: "mount_oauth_microsoft",
}
export function DriveMountOAuthSection({
draft,
onChange,
}: {
draft: DriveMountOAuthSettings
onChange: (next: DriveMountOAuthSettings) => void
}) {
const secrets = useOrgSettingsStore((s) => s.meta?.secrets)
const [redirectUri, setRedirectUri] = useState("")
const [copied, setCopied] = useState(false)
useEffect(() => {
setRedirectUri(buildDriveMountOAuthRedirectURI())
}, [])
const updateProvider = (provider: DriveMountOAuthProvider, patch: Partial<DriveMountOAuthSettings[typeof provider]>) => {
onChange({
...draft,
[provider]: { ...draft[provider], ...patch },
})
}
const copyRedirectUri = async () => {
const uri = redirectUri || buildDriveMountOAuthRedirectURI()
try {
await navigator.clipboard.writeText(uri)
setCopied(true)
toast.success("URI de redirection copiée")
window.setTimeout(() => setCopied(false), 2000)
} catch {
toast.error("Impossible de copier l'URI")
}
}
return (
<div className="space-y-4 rounded-lg border p-4">
<div>
<h3 className="text-sm font-medium">Connexion cloud (OAuth)</h3>
<p className="mt-1 text-xs text-muted-foreground">
Permet aux utilisateurs de monter Google Drive, Dropbox ou OneDrive depuis UltiDrive.
</p>
</div>
<div>
<Label>URI de redirection OAuth</Label>
<div className="mt-1 flex gap-2">
<Input
className="h-9 flex-1 font-mono text-xs"
readOnly
value={redirectUri}
placeholder="Chargement…"
/>
<Button
type="button"
variant="outline"
size="sm"
className="h-9 shrink-0 gap-1.5 px-3"
onClick={() => void copyRedirectUri()}
disabled={!redirectUri}
>
{copied ? <Check className="h-4 w-4" /> : <Copy className="h-4 w-4" />}
Copier
</Button>
</div>
<p className="mt-1 text-xs text-muted-foreground">
Basée sur l&apos;URL actuelle du navigateur. Enregistrez-la chez chaque fournisseur OAuth (Google, Dropbox, Microsoft).
</p>
</div>
<div className="space-y-4">
{PROVIDERS.map(({ id, label, hint }) => {
const provider = draft[id]
const configured = Boolean(secrets?.[SECRET_KEYS[id]]?.configured)
return (
<div key={id} className="space-y-3 rounded-md border p-3">
<label className="flex items-center justify-between gap-4">
<div>
<p className="text-sm font-medium">{label}</p>
<p className="text-xs text-muted-foreground">{hint}</p>
</div>
<Switch
checked={provider.enabled}
onCheckedChange={(enabled) => updateProvider(id, { enabled })}
/>
</label>
{provider.enabled ? (
<div className="grid gap-3 sm:grid-cols-2">
<div className="sm:col-span-2">
<Label>Client ID</Label>
<Input
className="mt-1 h-9 font-mono text-xs"
value={provider.client_id}
onChange={(e) => updateProvider(id, { client_id: e.target.value })}
autoComplete="off"
/>
</div>
<div className="sm:col-span-2">
<Label>Client secret</Label>
<Input
className="mt-1 h-9 font-mono text-xs"
type="password"
value={provider.client_secret}
onChange={(e) => updateProvider(id, { client_secret: e.target.value })}
placeholder={configured ? "•••••••• (laisser vide pour conserver)" : "Coller le secret"}
autoComplete="off"
/>
{configured && !provider.client_secret.trim() ? (
<p className="mt-1 text-xs text-muted-foreground">Secret configuré</p>
) : null}
</div>
</div>
) : null}
</div>
)
})}
</div>
</div>
)
}