mirror of
https://github.com/YspCoder/clawgo.git
synced 2026-04-27 14:07:29 +08:00
91 lines
3.9 KiB
TypeScript
91 lines
3.9 KiB
TypeScript
import React, { useState } from 'react';
|
||
import { RefreshCw, Save } from 'lucide-react';
|
||
import { useTranslation } from 'react-i18next';
|
||
import { useAppContext } from '../context/AppContext';
|
||
import RecursiveConfig from '../components/RecursiveConfig';
|
||
|
||
function setPath(obj: any, path: string, value: any) {
|
||
const keys = path.split('.')
|
||
const next = JSON.parse(JSON.stringify(obj || {}))
|
||
let cur = next
|
||
for (let i = 0; i < keys.length - 1; i++) {
|
||
const k = keys[i]
|
||
if (typeof cur[k] !== 'object' || cur[k] === null) cur[k] = {}
|
||
cur = cur[k]
|
||
}
|
||
cur[keys[keys.length - 1]] = value
|
||
return next
|
||
}
|
||
|
||
const Config: React.FC = () => {
|
||
const { t } = useTranslation();
|
||
const { cfg, setCfg, cfgRaw, setCfgRaw, loadConfig, hotReloadFieldDetails, q } = useAppContext();
|
||
const [showRaw, setShowRaw] = useState(false);
|
||
|
||
async function saveConfig() {
|
||
try {
|
||
const payload = showRaw ? JSON.parse(cfgRaw) : cfg;
|
||
const r = await fetch(`/webui/api/config${q}`, {
|
||
method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(payload),
|
||
});
|
||
alert(await r.text());
|
||
} catch (e) {
|
||
alert('Failed to save config: ' + e);
|
||
}
|
||
}
|
||
|
||
return (
|
||
<div className="p-8 max-w-5xl mx-auto space-y-6 flex flex-col min-h-full">
|
||
<div className="flex items-center justify-between">
|
||
<h1 className="text-2xl font-semibold tracking-tight">{t('configuration')}</h1>
|
||
<div className="flex items-center gap-1 bg-zinc-900/80 p-1 rounded-lg border border-zinc-800">
|
||
<button onClick={() => setShowRaw(false)} className={`px-4 py-1.5 text-sm font-medium rounded-md transition-all ${!showRaw ? 'bg-zinc-800 text-white shadow-sm' : 'text-zinc-400 hover:text-zinc-200'}`}>{t('form')}</button>
|
||
<button onClick={() => setShowRaw(true)} className={`px-4 py-1.5 text-sm font-medium rounded-md transition-all ${showRaw ? 'bg-zinc-800 text-white shadow-sm' : 'text-zinc-400 hover:text-zinc-200'}`}>{t('rawJson')}</button>
|
||
</div>
|
||
</div>
|
||
|
||
<div className="flex items-center gap-3">
|
||
<button onClick={loadConfig} className="flex items-center gap-2 px-4 py-2 bg-zinc-800 hover:bg-zinc-700 rounded-lg text-sm font-medium transition-colors">
|
||
<RefreshCw className="w-4 h-4" /> {t('reload')}
|
||
</button>
|
||
<button onClick={saveConfig} className="flex items-center gap-2 px-4 py-2 bg-indigo-600 hover:bg-indigo-500 text-white rounded-lg text-sm font-medium transition-colors shadow-sm">
|
||
<Save className="w-4 h-4" /> {t('saveChanges')}
|
||
</button>
|
||
</div>
|
||
|
||
<div className="bg-zinc-900/40 border border-zinc-800/80 rounded-2xl p-4">
|
||
<div className="text-sm font-semibold text-zinc-300 mb-2">热更新字段(完整)</div>
|
||
<div className="grid grid-cols-1 md:grid-cols-2 gap-2 text-xs">
|
||
{hotReloadFieldDetails.map((it) => (
|
||
<div key={it.path} className="p-2 rounded bg-zinc-950 border border-zinc-800">
|
||
<div className="font-mono text-zinc-200">{it.path}</div>
|
||
<div className="text-zinc-400">{it.name || ''}{it.description ? ` · ${it.description}` : ''}</div>
|
||
</div>
|
||
))}
|
||
</div>
|
||
</div>
|
||
|
||
<div className="flex-1 bg-zinc-900/40 border border-zinc-800/80 rounded-2xl overflow-hidden flex flex-col shadow-sm">
|
||
{!showRaw ? (
|
||
<div className="p-8 overflow-y-auto">
|
||
<RecursiveConfig
|
||
data={cfg}
|
||
labels={t('configLabels', { returnObjects: true }) as Record<string, string>}
|
||
onChange={(path, val) => setCfg(v => setPath(v, path, val))}
|
||
/>
|
||
</div>
|
||
) : (
|
||
<textarea
|
||
value={cfgRaw}
|
||
onChange={(e) => setCfgRaw(e.target.value)}
|
||
className="flex-1 w-full bg-zinc-950 p-6 font-mono text-sm text-zinc-300 focus:outline-none resize-none"
|
||
spellCheck={false}
|
||
/>
|
||
)}
|
||
</div>
|
||
</div>
|
||
);
|
||
};
|
||
|
||
export default Config;
|