263 lines
7.9 KiB
TypeScript
Vendored
263 lines
7.9 KiB
TypeScript
Vendored
import { useCallback, useEffect, useState } from 'react'
|
|
import { LayoutDashboard } from 'lucide-react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { toast } from 'sonner'
|
|
import { useAuthStore } from '@/stores/auth-store'
|
|
import { api } from '@/lib/api'
|
|
import { Button } from '@/components/ui/button'
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardDescription,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from '@/components/ui/card'
|
|
import { Switch } from '@/components/ui/switch'
|
|
|
|
type SidebarModuleConfig = {
|
|
enabled: boolean
|
|
[key: string]: boolean
|
|
}
|
|
|
|
type SidebarModulesConfig = Record<string, SidebarModuleConfig>
|
|
|
|
type SectionDef = {
|
|
key: string
|
|
title: string
|
|
description: string
|
|
modules: { key: string; title: string; description: string }[]
|
|
}
|
|
|
|
export function SidebarModulesCard() {
|
|
const { t } = useTranslation()
|
|
const [loading, setLoading] = useState(false)
|
|
const [config, setConfig] = useState<SidebarModulesConfig>({})
|
|
const currentUser = useAuthStore((s) => s.auth.user)
|
|
const setUser = useAuthStore((s) => s.auth.setUser)
|
|
|
|
const sectionDefs: SectionDef[] = [
|
|
{
|
|
key: 'chat',
|
|
title: t('Chat Area'),
|
|
description: t('Playground and chat functions'),
|
|
modules: [
|
|
{
|
|
key: 'playground',
|
|
title: t('Playground'),
|
|
description: t('AI model testing environment'),
|
|
},
|
|
{
|
|
key: 'chat',
|
|
title: t('Chat'),
|
|
description: t('Chat session management'),
|
|
},
|
|
],
|
|
},
|
|
{
|
|
key: 'console',
|
|
title: t('Console Area'),
|
|
description: t('Data management and log viewing'),
|
|
modules: [
|
|
{
|
|
key: 'detail',
|
|
title: t('Dashboard'),
|
|
description: t('System data statistics'),
|
|
},
|
|
{
|
|
key: 'token',
|
|
title: t('Token Management'),
|
|
description: t('API token management'),
|
|
},
|
|
{
|
|
key: 'log',
|
|
title: t('Usage Logs'),
|
|
description: t('API usage records'),
|
|
},
|
|
{
|
|
key: 'midjourney',
|
|
title: t('Drawing Logs'),
|
|
description: t('Drawing task records'),
|
|
},
|
|
{
|
|
key: 'task',
|
|
title: t('Task Logs'),
|
|
description: t('System task records'),
|
|
},
|
|
],
|
|
},
|
|
{
|
|
key: 'personal',
|
|
title: t('Personal Center Area'),
|
|
description: t('User personal functions'),
|
|
modules: [
|
|
{
|
|
key: 'topup',
|
|
title: t('Wallet Management'),
|
|
description: t('Balance and top-up management'),
|
|
},
|
|
{
|
|
key: 'personal',
|
|
title: t('Personal Settings'),
|
|
description: t('Personal info settings'),
|
|
},
|
|
],
|
|
},
|
|
]
|
|
|
|
const loadConfig = useCallback(async () => {
|
|
try {
|
|
const res = await api.get('/api/user/self')
|
|
if (res.data.success && res.data.data?.sidebar_modules) {
|
|
const raw = res.data.data.sidebar_modules
|
|
const parsed = typeof raw === 'string' ? JSON.parse(raw) : raw
|
|
setConfig(parsed)
|
|
} else {
|
|
const defaults: SidebarModulesConfig = {}
|
|
for (const sec of sectionDefs) {
|
|
defaults[sec.key] = { enabled: true }
|
|
for (const mod of sec.modules) defaults[sec.key][mod.key] = true
|
|
}
|
|
setConfig(defaults)
|
|
}
|
|
} catch {
|
|
/* ignore */
|
|
}
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
loadConfig()
|
|
}, [loadConfig])
|
|
|
|
const toggleSection = (sectionKey: string, val: boolean) => {
|
|
setConfig((prev) => ({
|
|
...prev,
|
|
[sectionKey]: { ...prev[sectionKey], enabled: val },
|
|
}))
|
|
}
|
|
|
|
const toggleModule = (
|
|
sectionKey: string,
|
|
moduleKey: string,
|
|
val: boolean
|
|
) => {
|
|
setConfig((prev) => ({
|
|
...prev,
|
|
[sectionKey]: { ...prev[sectionKey], [moduleKey]: val },
|
|
}))
|
|
}
|
|
|
|
const handleSave = async () => {
|
|
setLoading(true)
|
|
try {
|
|
const serialized = JSON.stringify(config)
|
|
const res = await api.put('/api/user/self', {
|
|
sidebar_modules: serialized,
|
|
})
|
|
if (res.data.success) {
|
|
// Sync to auth-store so useSidebarConfig re-runs and the sidebar
|
|
// updates immediately without needing a page refresh.
|
|
if (currentUser) {
|
|
setUser({ ...currentUser, sidebar_modules: serialized })
|
|
}
|
|
toast.success(t('Saved successfully'))
|
|
} else {
|
|
toast.error(res.data.message || t('Save failed'))
|
|
}
|
|
} catch {
|
|
toast.error(t('Save failed, please retry'))
|
|
} finally {
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
const handleReset = () => {
|
|
const defaults: SidebarModulesConfig = {}
|
|
for (const sec of sectionDefs) {
|
|
defaults[sec.key] = { enabled: true }
|
|
for (const mod of sec.modules) defaults[sec.key][mod.key] = true
|
|
}
|
|
setConfig(defaults)
|
|
toast.success(t('Reset to default configuration'))
|
|
}
|
|
|
|
return (
|
|
<Card className='gap-0 overflow-hidden py-0'>
|
|
<CardHeader className='border-b p-3 !pb-3 sm:p-5 sm:!pb-5'>
|
|
<div className='flex items-center gap-3'>
|
|
<div className='bg-muted flex h-8 w-8 shrink-0 items-center justify-center rounded-lg sm:h-9 sm:w-9'>
|
|
<LayoutDashboard className='h-4 w-4' />
|
|
</div>
|
|
<div className='min-w-0'>
|
|
<CardTitle className='text-lg tracking-tight sm:text-xl'>
|
|
{t('Sidebar Personal Settings')}
|
|
</CardTitle>
|
|
<CardDescription className='text-xs sm:text-sm'>
|
|
{t('Customize sidebar display content')}
|
|
</CardDescription>
|
|
</div>
|
|
</div>
|
|
</CardHeader>
|
|
<CardContent className='space-y-4 p-3 sm:space-y-5 sm:p-5'>
|
|
{sectionDefs.map((section) => {
|
|
const sectionEnabled = config[section.key]?.enabled !== false
|
|
return (
|
|
<div
|
|
key={section.key}
|
|
className='bg-background/60 rounded-xl border p-3'
|
|
>
|
|
<div className='flex items-start justify-between gap-3'>
|
|
<div className='min-w-0'>
|
|
<p className='text-sm font-medium'>{section.title}</p>
|
|
<p className='text-muted-foreground text-xs'>
|
|
{section.description}
|
|
</p>
|
|
</div>
|
|
<Switch
|
|
checked={sectionEnabled}
|
|
onCheckedChange={(v) => toggleSection(section.key, v)}
|
|
/>
|
|
</div>
|
|
<div className='mt-3 grid grid-cols-1 gap-2 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-1'>
|
|
{section.modules.map((mod) => (
|
|
<div
|
|
key={mod.key}
|
|
className={`flex min-h-16 items-center justify-between rounded-lg border p-3 transition-opacity ${
|
|
sectionEnabled ? '' : 'opacity-50'
|
|
}`}
|
|
>
|
|
<div className='mr-2 min-w-0'>
|
|
<p className='truncate text-sm font-medium'>
|
|
{mod.title}
|
|
</p>
|
|
<p className='text-muted-foreground truncate text-xs'>
|
|
{mod.description}
|
|
</p>
|
|
</div>
|
|
<Switch
|
|
checked={config[section.key]?.[mod.key] !== false}
|
|
onCheckedChange={(v) =>
|
|
toggleModule(section.key, mod.key, v)
|
|
}
|
|
disabled={!sectionEnabled}
|
|
/>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)
|
|
})}
|
|
|
|
<div className='flex flex-col-reverse gap-2 border-t pt-4 sm:flex-row sm:justify-end'>
|
|
<Button variant='outline' onClick={handleReset}>
|
|
{t('Reset to Default')}
|
|
</Button>
|
|
<Button onClick={handleSave} disabled={loading}>
|
|
{loading ? t('Saving...') : t('Save Changes')}
|
|
</Button>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
)
|
|
}
|