feat: support primary color customization

This commit is contained in:
codytseng
2025-10-18 23:18:44 +08:00
parent b17846f264
commit 28dad7373f
21 changed files with 644 additions and 43 deletions

View File

@@ -1,4 +1,5 @@
import { Label } from '@/components/ui/label'
import { PRIMARY_COLORS, TPrimaryColor } from '@/constants'
import SecondaryPageLayout from '@/layouts/SecondaryPageLayout'
import { cn } from '@/lib/utils'
import { useTheme } from '@/providers/ThemeProvider'
@@ -15,11 +16,11 @@ const THEMES = [
const AppearanceSettingsPage = forwardRef(({ index }: { index?: number }, ref) => {
const { t } = useTranslation()
const { themeSetting, setThemeSetting } = useTheme()
const { themeSetting, setThemeSetting, primaryColor, setPrimaryColor } = useTheme()
return (
<SecondaryPageLayout ref={ref} index={index} title={t('Appearance')}>
<div className="space-y-4 mt-3">
<div className="space-y-4 my-3">
<div className="flex flex-col gap-2 px-4">
<Label className="text-base">{t('Theme')}</Label>
<div className="grid grid-cols-2 md:grid-cols-4 gap-4 w-full">
@@ -31,7 +32,9 @@ const AppearanceSettingsPage = forwardRef(({ index }: { index?: number }, ref) =
}}
className={cn(
'flex flex-col items-center gap-2 py-4 rounded-lg border-2 transition-all',
themeSetting === key ? 'border-primary' : 'border-border hover:border-primary/60'
themeSetting === key
? 'border-primary'
: 'border-border hover:border-muted-foreground/40'
)}
>
<div className="flex items-center justify-center w-8 h-8">{icon}</div>
@@ -40,6 +43,31 @@ const AppearanceSettingsPage = forwardRef(({ index }: { index?: number }, ref) =
))}
</div>
</div>
<div className="flex flex-col gap-2 px-4">
<Label className="text-base">{t('Primary color')}</Label>
<div className="grid grid-cols-4 gap-4 w-full">
{Object.entries(PRIMARY_COLORS).map(([key, config]) => (
<button
key={key}
onClick={() => setPrimaryColor(key as TPrimaryColor)}
className={cn(
'flex flex-col items-center gap-2 py-4 rounded-lg border-2 transition-all',
primaryColor === key
? 'border-primary'
: 'border-border hover:border-muted-foreground/40'
)}
>
<div
className="size-8 rounded-full shadow-md"
style={{
backgroundColor: `hsl(${config.light.primary})`
}}
/>
<span className="text-xs font-medium">{t(config.name)}</span>
</button>
))}
</div>
</div>
</div>
</SecondaryPageLayout>
)