'use client'
import { useState, useCallback, useEffect, useRef } from 'react'
import dynamic from 'next/dynamic'
import { AnimatePresence, motion } from 'framer-motion'
import { X, GearSix, Buildings, User } from '@phosphor-icons/react'
import { Button, Spinner } from '@ciphera-net/ui'
import { useUnifiedSettings } from '@/lib/unified-settings-context'
import { useAuth } from '@/lib/auth/context'
import { useSite } from '@/lib/swr/dashboard'
import { listSites, type Site } from '@/lib/api/sites'
// Lazy-load tab components — only loaded when the tab is first rendered
const tabLoader = () =>
const SiteGeneralTab = dynamic(() => import('./tabs/SiteGeneralTab'), { loading: tabLoader })
const SiteGoalsTab = dynamic(() => import('./tabs/SiteGoalsTab'), { loading: tabLoader })
const SiteVisibilityTab = dynamic(() => import('./tabs/SiteVisibilityTab'), { loading: tabLoader })
const SitePrivacyTab = dynamic(() => import('./tabs/SitePrivacyTab'), { loading: tabLoader })
const SiteBotSpamTab = dynamic(() => import('./tabs/SiteBotSpamTab'), { loading: tabLoader })
const SiteReportsTab = dynamic(() => import('./tabs/SiteReportsTab'), { loading: tabLoader })
const SiteIntegrationsTab = dynamic(() => import('./tabs/SiteIntegrationsTab'), { loading: tabLoader })
const WorkspaceGeneralTab = dynamic(() => import('./tabs/WorkspaceGeneralTab'), { loading: tabLoader })
const WorkspaceBillingTab = dynamic(() => import('./tabs/WorkspaceBillingTab'), { loading: tabLoader })
const WorkspaceMembersTab = dynamic(() => import('./tabs/WorkspaceMembersTab'), { loading: tabLoader })
const WorkspaceNotificationsTab = dynamic(() => import('./tabs/WorkspaceNotificationsTab'), { loading: tabLoader })
const WorkspaceAuditTab = dynamic(() => import('./tabs/WorkspaceAuditTab'), { loading: tabLoader })
const AccountProfileTab = dynamic(() => import('./tabs/AccountProfileTab'), { loading: tabLoader })
const AccountSecurityTab = dynamic(() => import('./tabs/AccountSecurityTab'), { loading: tabLoader })
const AccountDevicesTab = dynamic(() => import('./tabs/AccountDevicesTab'), { loading: tabLoader })
// ─── Types ──────────────────────────────────────────────────────
type SettingsContext = 'site' | 'workspace' | 'account'
interface TabDef {
id: string
label: string
}
const SITE_TABS: TabDef[] = [
{ id: 'general', label: 'General' },
{ id: 'goals', label: 'Goals' },
{ id: 'visibility', label: 'Visibility' },
{ id: 'privacy', label: 'Privacy' },
{ id: 'bot-spam', label: 'Bot & Spam' },
{ id: 'reports', label: 'Reports' },
{ id: 'integrations', label: 'Integrations' },
]
const WORKSPACE_TABS: TabDef[] = [
{ id: 'general', label: 'General' },
{ id: 'members', label: 'Members' },
{ id: 'billing', label: 'Billing' },
{ id: 'notifications', label: 'Notifications' },
{ id: 'audit', label: 'Audit Log' },
]
const ACCOUNT_TABS: TabDef[] = [
{ id: 'profile', label: 'Profile' },
{ id: 'security', label: 'Security' },
{ id: 'devices', label: 'Devices' },
]
// ─── Context Switcher ───────────────────────────────────────────
function ContextSwitcher({
active,
onChange,
activeSiteDomain,
}: {
active: SettingsContext
onChange: (ctx: SettingsContext) => void
activeSiteDomain: string | null
}) {
return (
{/* Site button — locked to current site, no dropdown */}
{activeSiteDomain && (
onChange('site')}
className={`flex items-center gap-2 px-3 py-2 text-sm font-medium rounded-lg transition-all duration-200 ${
active === 'site'
? 'bg-neutral-700 text-white shadow-sm'
: 'text-neutral-400 hover:text-white'
}`}
>
{activeSiteDomain}
)}
onChange('workspace')}
className={`flex items-center gap-2 px-3 py-2 text-sm font-medium rounded-lg transition-all duration-200 ${
active === 'workspace'
? 'bg-neutral-700 text-white shadow-sm'
: 'text-neutral-400 hover:text-white'
}`}
>
Organization
onChange('account')}
className={`flex items-center gap-2 px-3 py-2 text-sm font-medium rounded-lg transition-all duration-200 ${
active === 'account'
? 'bg-neutral-700 text-white shadow-sm'
: 'text-neutral-400 hover:text-white'
}`}
>
Account
)
}
// ─── Tab Bar ────────────────────────────────────────────────────
function TabBar({
tabs,
activeTab,
onChange,
}: {
tabs: TabDef[]
activeTab: string
onChange: (id: string) => void
}) {
return (
{tabs.map(tab => (
onChange(tab.id)}
className={`relative px-3 py-2 text-sm font-medium whitespace-nowrap rounded-lg transition-all duration-200 ${
activeTab === tab.id
? 'text-brand-orange'
: 'text-neutral-500 hover:text-neutral-300'
}`}
>
{tab.label}
{activeTab === tab.id && (
)}
))}
)
}
// ─── Tab Content ────────────────────────────────────────────────
function ComingSoon({ label }: { label: string }) {
return (
{label}
This section is being migrated. For now, use the existing settings page.
)
}
function TabContent({
context,
activeTab,
siteId,
onDirtyChange,
onRegisterSave,
}: {
context: SettingsContext
activeTab: string
siteId: string | null
onDirtyChange: (dirty: boolean) => void
onRegisterSave: (fn: () => Promise) => void
}) {
const dirtyProps = { onDirtyChange, onRegisterSave }
// Site tabs
if (context === 'site' && siteId) {
switch (activeTab) {
case 'general': return
case 'goals': return
case 'visibility': return
case 'privacy': return
case 'bot-spam': return
case 'reports': return
case 'integrations': return
}
}
// Workspace tabs
if (context === 'workspace') {
switch (activeTab) {
case 'general': return
case 'billing': return
case 'members': return
case 'notifications': return
case 'audit': return
}
}
// Account tabs
if (context === 'account') {
switch (activeTab) {
case 'profile': return
case 'security': return
case 'devices': return
}
}
return null
}
// ─── Main Modal ─────────────────────────────────────────────────
export default function UnifiedSettingsModal() {
const { isOpen, openUnifiedSettings, closeUnifiedSettings: closeSettings, initialTab: initTab } = useUnifiedSettings()
const { user } = useAuth()
const [context, setContext] = useState('site')
const [siteTabs, setSiteTabs] = useState('general')
const [workspaceTabs, setWorkspaceTabs] = useState('general')
const [accountTabs, setAccountTabs] = useState('profile')
const [sites, setSites] = useState([])
const [activeSiteId, setActiveSiteId] = useState(null)
// ─── Dirty state & pending navigation ────────────────────────
const isDirtyRef = useRef(false)
const [isDirtyVisible, setIsDirtyVisible] = useState(false)
const pendingActionRef = useRef<(() => void) | null>(null)
const [hasPendingAction, setHasPendingAction] = useState(false)
const saveHandlerRef = useRef<(() => Promise) | null>(null)
const [saving, setSaving] = useState(false)
const [showGlass, setShowGlass] = useState(false)
const handleDirtyChange = useCallback((dirty: boolean) => {
isDirtyRef.current = dirty
setIsDirtyVisible(dirty)
// If user saved and there was a pending action, execute it
if (!dirty && pendingActionRef.current) {
const action = pendingActionRef.current
pendingActionRef.current = null
setHasPendingAction(false)
action()
}
}, [])
const handleRegisterSave = useCallback((fn: () => Promise) => {
saveHandlerRef.current = fn
}, [])
const handleSaveFromBar = useCallback(async () => {
if (!saveHandlerRef.current) return
setSaving(true)
try {
await saveHandlerRef.current()
} finally {
setSaving(false)
}
}, [])
/** Run action if clean, or store as pending if dirty */
const guardedAction = useCallback((action: () => void) => {
if (isDirtyRef.current) {
pendingActionRef.current = action
setHasPendingAction(true)
} else {
action()
}
}, [])
const handleDiscard = useCallback(() => {
isDirtyRef.current = false
setIsDirtyVisible(false)
setHasPendingAction(false)
saveHandlerRef.current = null
const action = pendingActionRef.current
pendingActionRef.current = null
action?.()
}, [])
// Apply initial tab when modal opens
useEffect(() => {
if (isOpen && initTab) {
if (initTab.context) setContext(initTab.context)
if (initTab.tab) {
if (initTab.context === 'site') setSiteTabs(initTab.tab)
else if (initTab.context === 'workspace') setWorkspaceTabs(initTab.tab)
else if (initTab.context === 'account') setAccountTabs(initTab.tab)
}
}
}, [isOpen, initTab])
// Reset dirty state when modal opens
useEffect(() => {
if (isOpen) {
isDirtyRef.current = false
pendingActionRef.current = null
setHasPendingAction(false)
setShowGlass(true)
}
}, [isOpen])
// Detect site from URL and load sites list when modal opens
useEffect(() => {
if (!isOpen || !user?.org_id) return
if (typeof window !== 'undefined') {
const match = window.location.pathname.match(/\/sites\/([a-f0-9-]+)/)
if (match) {
setActiveSiteId(match[1])
// Only default to site context if no specific context was requested
if (!initTab?.context) setContext('site')
} else {
setActiveSiteId(null)
if (!initTab?.context) setContext('workspace')
}
}
// Only fetch sites if we don't have them yet
if (sites.length === 0) {
listSites().then(data => {
setSites(Array.isArray(data) ? data : [])
}).catch(() => {})
}
}, [isOpen, user?.org_id])
// Global keyboard shortcuts: `,` toggles settings, Escape closes
useEffect(() => {
const handler = (e: KeyboardEvent) => {
const tag = (e.target as HTMLElement)?.tagName
if (tag === 'INPUT' || tag === 'TEXTAREA' || tag === 'SELECT') return
if (e.key === ',' && !e.metaKey && !e.ctrlKey && !e.altKey) {
e.preventDefault()
if (isOpen) guardedAction(closeSettings)
else openUnifiedSettings()
}
if (e.key === 'Escape' && isOpen) {
guardedAction(closeSettings)
}
}
window.addEventListener('keydown', handler)
return () => window.removeEventListener('keydown', handler)
}, [isOpen, openUnifiedSettings, closeSettings, guardedAction])
const tabs = context === 'site' ? SITE_TABS : context === 'workspace' ? WORKSPACE_TABS : ACCOUNT_TABS
const activeTab = context === 'site' ? siteTabs : context === 'workspace' ? workspaceTabs : accountTabs
const setActiveTab = context === 'site' ? setSiteTabs : context === 'workspace' ? setWorkspaceTabs : setAccountTabs
const handleContextChange = useCallback((ctx: SettingsContext) => {
guardedAction(() => {
setContext(ctx)
if (ctx === 'site') setSiteTabs('general')
else if (ctx === 'workspace') setWorkspaceTabs('general')
else if (ctx === 'account') setAccountTabs('profile')
})
}, [guardedAction])
const handleTabChange = useCallback((tabId: string) => {
guardedAction(() => setActiveTab(tabId))
}, [guardedAction, setActiveTab])
const handleClose = useCallback(() => {
guardedAction(closeSettings)
}, [guardedAction, closeSettings])
const handleBackdropClick = useCallback(() => {
guardedAction(closeSettings)
}, [guardedAction, closeSettings])
return (
<>
{/* Backdrop — fades in/out */}
{/* Glass panel — always mounted, fades out on close */}
e.stopPropagation()}
>
{/* Content animates in/out */}
setShowGlass(false)}>
{isOpen && (
{/* Header */}
Settings
{/* Context Switcher */}
s.id === activeSiteId)?.domain ?? null}
/>
{/* Tabs */}
{/* Content */}
{/* Save bar */}
{isDirtyVisible && (
{hasPendingAction ? 'Save or discard to continue' : 'Unsaved changes'}
{hasPendingAction && (
Discard
)}
{saving ? 'Saving...' : 'Save Changes'}
)}
)}
>
)
}