Files
breakpilot-compliance/admin-compliance/app/sdk/sso/_components/SSOConfigCard.tsx
Sharang Parnerkar 2fb6b98bc5 refactor(admin): split sso page.tsx into colocated components
Extract types, constants, helpers, and UI pieces (shared LoadingSkeleton/
EmptyState/StatusBadge/CopyButton, SSOConfigFormModal, DeleteConfirmModal,
ConnectionTestPanel, SSOConfigCard, SSOUsersTable, SSOInfoSection) into
_components/ and _types.ts to bring page.tsx from 1482 LOC to 339 LOC
(under the 500 hard cap). Behavior preserved.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-11 22:53:08 +02:00

159 lines
6.1 KiB
TypeScript

'use client'
import { useState } from 'react'
import {
ChevronDown,
ChevronUp,
Key,
Pencil,
Trash2,
} from 'lucide-react'
import type { SSOConfig } from '../_types'
import { AVAILABLE_ROLES } from './constants'
import { formatDate } from './helpers'
import { CopyButton, StatusBadge } from './shared'
import { ConnectionTestPanel } from './ConnectionTestPanel'
export function SSOConfigCard({
config,
onEdit,
onDelete,
onToggle,
}: {
config: SSOConfig
onEdit: () => void
onDelete: () => void
onToggle: () => void
}) {
const [expanded, setExpanded] = useState(false)
return (
<div className="bg-white rounded-xl border border-slate-200 hover:border-purple-300 transition-colors">
{/* Main Row */}
<div className="p-5">
<div className="flex items-start justify-between">
<div className="flex items-start gap-4 flex-1 min-w-0">
<div className="w-10 h-10 bg-purple-100 text-purple-700 rounded-lg flex items-center justify-center flex-shrink-0">
<Key className="w-5 h-5" />
</div>
<div className="flex-1 min-w-0">
<div className="flex items-center gap-2 mb-1 flex-wrap">
<h3 className="font-semibold text-slate-900">{config.name}</h3>
<span className="px-2 py-0.5 bg-blue-100 text-blue-700 rounded text-xs font-medium uppercase">
{config.provider_type}
</span>
<StatusBadge enabled={config.enabled} />
</div>
<p className="text-sm text-slate-500 font-mono truncate">{config.oidc_issuer_url}</p>
<div className="flex items-center gap-4 mt-2 text-xs text-slate-400">
<span>Client ID: {(config.oidc_client_id || '').substring(0, 12)}...</span>
<span>Scopes: {(config.oidc_scopes || []).join(', ')}</span>
<span>Erstellt: {formatDate(config.created_at)}</span>
</div>
</div>
</div>
{/* Actions */}
<div className="flex items-center gap-2 ml-4 flex-shrink-0">
<button
onClick={onToggle}
className={`relative inline-flex h-6 w-11 items-center rounded-full transition-colors ${
config.enabled ? 'bg-purple-600' : 'bg-slate-300'
}`}
title={config.enabled ? 'Deaktivieren' : 'Aktivieren'}
>
<span
className={`inline-block h-4 w-4 transform rounded-full bg-white transition-transform ${
config.enabled ? 'translate-x-6' : 'translate-x-1'
}`}
/>
</button>
<button
onClick={onEdit}
className="p-2 text-slate-400 hover:text-purple-600 transition-colors"
title="Bearbeiten"
>
<Pencil className="w-4 h-4" />
</button>
<button
onClick={onDelete}
className="p-2 text-slate-400 hover:text-red-500 transition-colors"
title="Loeschen"
>
<Trash2 className="w-4 h-4" />
</button>
<button
onClick={() => setExpanded(!expanded)}
className="p-2 text-slate-400 hover:text-slate-600 transition-colors"
title={expanded ? 'Zuklappen' : 'Details anzeigen'}
>
{expanded ? <ChevronUp className="w-4 h-4" /> : <ChevronDown className="w-4 h-4" />}
</button>
</div>
</div>
</div>
{/* Expanded Details */}
{expanded && (
<div className="px-5 pb-5 space-y-4 border-t border-slate-100 pt-4">
{/* Config Details Grid */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 text-sm">
<div>
<span className="text-slate-500">Redirect URI:</span>
<div className="flex items-center gap-1 mt-0.5">
<span className="font-mono text-slate-700 truncate text-xs">{config.oidc_redirect_uri}</span>
<CopyButton value={config.oidc_redirect_uri} />
</div>
</div>
<div>
<span className="text-slate-500">Auto-Provisioning:</span>
<div className="mt-0.5 text-slate-700">
{config.auto_provision ? 'Aktiviert' : 'Deaktiviert'}
</div>
</div>
<div>
<span className="text-slate-500">Standard-Rolle:</span>
<div className="mt-0.5 text-slate-700">
{config.default_role_id || 'Standard'}
</div>
</div>
<div>
<span className="text-slate-500">Scopes:</span>
<div className="flex gap-1 mt-0.5 flex-wrap">
{(config.oidc_scopes || []).map(scope => (
<span key={scope} className="px-2 py-0.5 bg-slate-100 text-slate-600 rounded text-xs">
{scope}
</span>
))}
</div>
</div>
</div>
{/* Role Mappings */}
{config.role_mapping && Object.keys(config.role_mapping).length > 0 && (
<div>
<h4 className="text-sm font-medium text-slate-700 mb-2">Rollen-Mapping</h4>
<div className="space-y-1">
{Object.entries(config.role_mapping).map(([group, role]) => (
<div key={group} className="flex items-center gap-2 text-sm">
<span className="px-2 py-0.5 bg-blue-100 text-blue-700 rounded text-xs font-mono">
{group}
</span>
<span className="text-slate-400">&rarr;</span>
<span className="px-2 py-0.5 bg-purple-100 text-purple-700 rounded text-xs">
{AVAILABLE_ROLES.find(r => r.value === role)?.label || role}
</span>
</div>
))}
</div>
</div>
)}
{/* Connection Test */}
<ConnectionTestPanel config={config} />
</div>
)}
</div>
)
}