import type { z } from "zod"; function infoMessageId(fieldId: string) { return `${fieldId}-info`; } export function getNestedValue( obj: Record, path: string, ): unknown { const parts = parsePath(path); let current: unknown = obj; for (const part of parts) { if (current === null || current === undefined) return undefined; if (typeof part === "number") { current = (current as unknown[])[part]; } else { current = (current as Record)[part]; } } return current; } function parsePath(path: string): (string | number)[] { const parts: (string | number)[] = []; const regex = /([^.[[\]]+)|\[(\d+)\]/g; const matches = path.matchAll(regex); for (const match of matches) { if (match[1] !== undefined) { parts.push(match[1]); } else if (match[2] !== undefined) { parts.push(Number(match[2])); } } return parts; } export function setNestedValue( obj: Record, path: string, value: unknown, ): Record { const parts = parsePath(path); if (parts.length === 0) return obj; if (parts.length === 1) { const key = parts[0]!; if (typeof key === "number") { const arr = Array.isArray(obj) ? [...obj] : []; arr[key] = value; return arr as unknown as Record; } return { ...obj, [key]: value }; } const [head, ...tail] = parts; const tailPath = tail .map((p) => (typeof p === "number" ? `[${p}]` : p)) .join(".") .replace(/\.\[/g, "["); if (typeof head === "number") { const arr = Array.isArray(obj) ? [...obj] : []; arr[head] = setNestedValue( (arr[head] as Record) ?? {}, tailPath, value, ); return arr as unknown as Record; } const nested = (obj[head] as Record) ?? {}; return { ...obj, [head]: setNestedValue(nested, tailPath, value), }; } export function getNestedSchema( schema: z.ZodObject, path: string, ): z.ZodType | undefined { const parts = parsePath(path); let current: z.ZodType = schema; for (const part of parts) { const unwrapped = unwrapSchema(current); if (typeof part === "number") { const def = unwrapped._def as { type?: string; element?: z.ZodType; }; if (def.type === "array" && def.element) { current = def.element; } else { return undefined; } } else if ("shape" in unwrapped && unwrapped.shape) { const nextSchema = (unwrapped.shape as Record)[part]; if (!nextSchema) return undefined; current = nextSchema; } else { return undefined; } } return current; } function unwrapSchema(schema: z.ZodType): z.ZodType { const def = schema._def ?? (schema as unknown as { def: unknown }).def; const typeName = (def as { typeName?: string }).typeName ?? (def as { type?: string }).type; if ( typeName === "ZodNullable" || typeName === "ZodOptional" || typeName === "ZodDefault" || typeName === "nullable" || typeName === "optional" || typeName === "default" ) { const inner = (def as unknown as { innerType: z.ZodType }).innerType; return unwrapSchema(inner); } if (typeName === "ZodEffects" || typeName === "effects") { return unwrapSchema((def as unknown as { schema: z.ZodType }).schema); } return schema; } export function errorMessageId(fieldId: string) { return `${fieldId}-error`; } export function validateField( schema: z.ZodObject, name: string, value: unknown, ): string | undefined { const fieldSchema = name.includes(".") ? getNestedSchema(schema, name) : (schema.shape[name] as z.ZodType | undefined); if (!fieldSchema) return undefined; const result = fieldSchema.safeParse(value); if (result.success) return undefined; const issue = result.error.issues[0]; if (!issue) return undefined; if ( issue.code === "invalid_type" && (value === null || value === undefined || value === "") ) { return "forms:errors.required"; } if (issue.code === "too_small" && issue.minimum === 1) { return "forms:errors.required"; } return issue.message; } export function ariaAttributes({ id, error, bottomText, required, }: { id: string; error?: string; bottomText?: string; required?: boolean; }) { return { "aria-invalid": error ? ("true" as const) : undefined, "aria-describedby": bottomText ? infoMessageId(id) : undefined, "aria-errormessage": error ? errorMessageId(id) : undefined, "aria-required": required ? ("true" as const) : undefined, }; }