Files
chaos-api/web/default/src/features/system-settings/models/claude-settings-card.tsx
T
t0ng7u b08febaa3c refactor: system settings UI for consistent, compact layouts
Redesign the system settings interface to align with the rest of the console experience by using fixed header actions, removing redundant subtitles, respecting global content width, and standardizing responsive form layouts.

Introduce reusable settings layout primitives for forms, switch rows, grouped controls, nested control sections, title status indicators, and page action portals. Replace duplicated card-style switch markup with explicit compact components, improve nested switch readability, and reduce visual noise across authentication, billing, content, integrations, maintenance, models, and request-limit settings.

Also complete missing i18n translations, remove obsolete subtitle translation keys, refine i18n sync reporting, fix sidebar truncation for long labels, and verify the frontend with type checking and lint diagnostics.
2026-05-25 00:34:26 +08:00

279 lines
9.2 KiB
TypeScript
Vendored
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
/*
Copyright (C) 2023-2026 QuantumNous
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
For commercial licensing, please contact support@quantumnous.com
*/
import { useEffect, useRef } from 'react'
import * as z from 'zod'
import { useForm } from 'react-hook-form'
import { zodResolver } from '@hookform/resolvers/zod'
import { useTranslation } from 'react-i18next'
import { toast } from 'sonner'
import {
Form,
FormControl,
FormDescription,
FormField,
FormItem,
FormLabel,
FormMessage,
} from '@/components/ui/form'
import { Input } from '@/components/ui/input'
import { Switch } from '@/components/ui/switch'
import { Textarea } from '@/components/ui/textarea'
import {
SettingsForm,
SettingsSwitchContent,
SettingsControlGroup,
SettingsSwitchItem,
} from '../components/settings-form-layout'
import { SettingsPageFormActions } from '../components/settings-page-context'
import { SettingsSection } from '../components/settings-section'
import { useUpdateOption } from '../hooks/use-update-option'
import {
formatJsonForTextarea,
normalizeJsonString,
validateJsonString,
} from './utils'
const schema = z.object({
claude: z.object({
model_headers_settings: z.string().superRefine((value, ctx) => {
const result = validateJsonString(value)
if (!result.valid) {
ctx.addIssue({
code: z.ZodIssueCode.custom,
message: result.message || 'Invalid JSON',
})
}
}),
default_max_tokens: z.string().superRefine((value, ctx) => {
const result = validateJsonString(value)
if (!result.valid) {
ctx.addIssue({
code: z.ZodIssueCode.custom,
message: result.message || 'Invalid JSON',
})
}
}),
thinking_adapter_enabled: z.boolean(),
thinking_adapter_budget_tokens_percentage: z.coerce
.number()
.min(0.1, { message: 'Must be at least 0.1' })
.max(1, { message: 'Must be 1 or less' }),
}),
})
type ClaudeSettingsFormValues = z.output<typeof schema>
type ClaudeSettingsFormInput = z.input<typeof schema>
type FlatClaudeSettings = {
'claude.model_headers_settings': string
'claude.default_max_tokens': string
'claude.thinking_adapter_enabled': boolean
'claude.thinking_adapter_budget_tokens_percentage': number
}
type ClaudeSettingsCardProps = {
defaultValues: ClaudeSettingsFormInput
}
export function ClaudeSettingsCard({ defaultValues }: ClaudeSettingsCardProps) {
const { t } = useTranslation()
const updateOption = useUpdateOption()
const normalizedDefaultsRef = useRef<FlatClaudeSettings>({
'claude.model_headers_settings': normalizeJsonString(
defaultValues.claude.model_headers_settings
),
'claude.default_max_tokens': normalizeJsonString(
defaultValues.claude.default_max_tokens
),
'claude.thinking_adapter_enabled':
defaultValues.claude.thinking_adapter_enabled,
'claude.thinking_adapter_budget_tokens_percentage': Number(
defaultValues.claude.thinking_adapter_budget_tokens_percentage
),
})
const buildFormDefaults = (
values: ClaudeSettingsFormInput
): ClaudeSettingsFormInput => ({
claude: {
model_headers_settings: formatJsonForTextarea(
values.claude.model_headers_settings
),
default_max_tokens: formatJsonForTextarea(
values.claude.default_max_tokens
),
thinking_adapter_enabled: values.claude.thinking_adapter_enabled,
thinking_adapter_budget_tokens_percentage:
values.claude.thinking_adapter_budget_tokens_percentage,
},
})
const form = useForm<
ClaudeSettingsFormInput,
unknown,
ClaudeSettingsFormValues
>({
resolver: zodResolver(schema),
defaultValues: buildFormDefaults(defaultValues),
})
useEffect(() => {
normalizedDefaultsRef.current = {
'claude.model_headers_settings': normalizeJsonString(
defaultValues.claude.model_headers_settings
),
'claude.default_max_tokens': normalizeJsonString(
defaultValues.claude.default_max_tokens
),
'claude.thinking_adapter_enabled':
defaultValues.claude.thinking_adapter_enabled,
'claude.thinking_adapter_budget_tokens_percentage': Number(
defaultValues.claude.thinking_adapter_budget_tokens_percentage
),
}
form.reset(buildFormDefaults(defaultValues))
}, [defaultValues, form])
const onSubmit = async (values: ClaudeSettingsFormValues) => {
const normalized: FlatClaudeSettings = {
'claude.model_headers_settings': normalizeJsonString(
values.claude.model_headers_settings
),
'claude.default_max_tokens': normalizeJsonString(
values.claude.default_max_tokens
),
'claude.thinking_adapter_enabled': values.claude.thinking_adapter_enabled,
'claude.thinking_adapter_budget_tokens_percentage':
values.claude.thinking_adapter_budget_tokens_percentage,
}
const updates = (
Object.keys(normalized) as Array<keyof FlatClaudeSettings>
).filter((key) => normalized[key] !== normalizedDefaultsRef.current[key])
if (updates.length === 0) {
toast.info(t('No changes to save'))
return
}
for (const key of updates) {
await updateOption.mutateAsync({ key, value: normalized[key] })
}
}
return (
<SettingsSection title={t('Claude')}>
<Form {...form}>
{/* eslint-disable-next-line react-hooks/refs */}
<SettingsForm onSubmit={form.handleSubmit(onSubmit)}>
<SettingsPageFormActions
onSave={form.handleSubmit(onSubmit)}
isSaving={updateOption.isPending}
/>
<FormField
control={form.control}
name='claude.model_headers_settings'
render={({ field }) => (
<FormItem>
<FormLabel>{t('Request Header Overrides')}</FormLabel>
<FormControl>
<Textarea rows={8} {...field} />
</FormControl>
<FormDescription>
{t(
'Provide per-model header overrides as JSON. Useful for enabling beta features such as expanded context windows.'
)}
</FormDescription>
<FormMessage />
</FormItem>
)}
/>
<FormField
control={form.control}
name='claude.default_max_tokens'
render={({ field }) => (
<FormItem>
<FormLabel>{t('Default Max Tokens')}</FormLabel>
<FormControl>
<Textarea rows={8} {...field} />
</FormControl>
<FormDescription>
{t('Example')}{' '}
{`{ "default": 8192, "claude-3-haiku-20240307": 4096, "claude-3-opus-20240229": 4096, "claude-3-7-sonnet-20250219-thinking": 8192 }`}
</FormDescription>
<FormMessage />
</FormItem>
)}
/>
<SettingsControlGroup>
<FormField
control={form.control}
name='claude.thinking_adapter_enabled'
render={({ field }) => (
<SettingsSwitchItem>
<SettingsSwitchContent>
<FormLabel>{t('Thinking Adapter')}</FormLabel>
<FormDescription>
{t(
'Translate `-thinking` suffixes into Anthropic native thinking models while keeping pricing predictable.'
)}
</FormDescription>
</SettingsSwitchContent>
<FormControl>
<Switch
checked={field.value}
onCheckedChange={field.onChange}
/>
</FormControl>
</SettingsSwitchItem>
)}
/>
<FormField
control={form.control}
name='claude.thinking_adapter_budget_tokens_percentage'
render={({ field }) => (
<FormItem>
<FormLabel>{t('Budget Tokens Ratio')}</FormLabel>
<FormControl>
<Input
{...field}
value={String(field.value ?? '')}
onChange={(event) => field.onChange(event.target.value)}
/>
</FormControl>
<FormDescription>
{t(
'Budget tokens = max tokens × ratio. Accepts a decimal between 0.1 and 1.'
)}
</FormDescription>
<FormMessage />
</FormItem>
)}
/>
</SettingsControlGroup>
</SettingsForm>
</Form>
</SettingsSection>
)
}