From ac1536b19f5765e082650615e0a5748f731f8c58 Mon Sep 17 00:00:00 2001 From: duyua9 Date: Mon, 18 May 2026 22:03:25 +0800 Subject: [PATCH] fix(web): render object config values structurally (#10949) --- web/src/components/AutoField.tsx | 95 +++++++++++++++++++++++++------- 1 file changed, 75 insertions(+), 20 deletions(-) diff --git a/web/src/components/AutoField.tsx b/web/src/components/AutoField.tsx index f7afd150b..0f96d4204 100644 --- a/web/src/components/AutoField.tsx +++ b/web/src/components/AutoField.tsx @@ -17,6 +17,71 @@ function FieldHint({ schema, schemaKey }: { schema: Record; sch ); } +function isRecord(value: unknown): value is Record { + return typeof value === "object" && value !== null && !Array.isArray(value); +} + +function formatScalar(value: unknown): string { + if (value === undefined || value === null) return ""; + if (typeof value === "string") return value; + if (typeof value === "number" || typeof value === "boolean") return String(value); + return JSON.stringify(value); +} + +function NestedValueEditor({ + fieldKey, + value, + onChange, +}: { + fieldKey: string; + value: unknown; + onChange: (v: unknown) => void; +}) { + if (isRecord(value)) { + return ( +
+ {Object.entries(value).map(([subKey, subVal]) => ( +
+ + onChange({ ...value, [subKey]: next })} + /> +
+ ))} +
+ ); + } + + if (Array.isArray(value)) { + return ( +
+ {value.map((item, index) => ( +
+ + + onChange(value.map((existing, i) => (i === index ? next : existing))) + } + /> +
+ ))} +
+ ); + } + + return ( + onChange(e.target.value)} + className="text-xs" + /> + ); +} + export function AutoField({ schemaKey, schema, @@ -26,6 +91,16 @@ export function AutoField({ const rawLabel = schemaKey.split(".").pop() ?? schemaKey; const label = rawLabel.replace(/_/g, " ").replace(/\b\w/g, (c) => c.toUpperCase()); + if (isRecord(value) || (Array.isArray(value) && value.some((item) => isRecord(item)))) { + return ( +
+ + + +
+ ); + } + if (schema.type === "boolean") { return (
@@ -114,26 +189,6 @@ export function AutoField({ ); } - if (typeof value === "object" && value !== null && !Array.isArray(value)) { - const obj = value as Record; - return ( -
- - - {Object.entries(obj).map(([subKey, subVal]) => ( -
- - onChange({ ...obj, [subKey]: e.target.value })} - className="text-xs" - /> -
- ))} -
- ); - } - return (