- Home sidebar icon: always cyan, bg-accent-dim only when route is "/" - Mobile TopBar: add left padding so hamburger isn't hidden behind logo - Landing page: bump card border color (#1e2130 → #2a2f3d) for better contrast - Replace all font-label references (40 occurrences, 19 files) with font-mono or font-sans - Remove deprecated --font-label CSS variable from index.css - Convert hardcoded hex in layout inline styles to CSS variables (light-mode ready) - Add @types/react-syntax-highlighter for script builder types Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
71 lines
2.3 KiB
TypeScript
71 lines
2.3 KiB
TypeScript
import { Terminal } from 'lucide-react'
|
|
import { useScriptGeneratorStore } from '@/store/scriptGeneratorStore'
|
|
import { ScriptParameterField } from './ScriptParameterField'
|
|
import type { ScriptParametersSchema, ScriptParameter } from '@/types'
|
|
|
|
interface Props {
|
|
canGenerate: boolean
|
|
}
|
|
|
|
export function ScriptParameterForm({ canGenerate }: Props) {
|
|
const selectedTemplate = useScriptGeneratorStore(s => s.selectedTemplate)
|
|
const paramValues = useScriptGeneratorStore(s => s.paramValues)
|
|
const formErrors = useScriptGeneratorStore(s => s.formErrors)
|
|
|
|
if (!selectedTemplate) return null
|
|
|
|
const schema = selectedTemplate.parameters_schema as ScriptParametersSchema
|
|
const parameters = (schema?.parameters ?? []).slice().sort((a, b) => a.order - b.order)
|
|
|
|
// Group parameters: null-group first, then named groups in order of first appearance
|
|
const ungrouped = parameters.filter(p => p.group === null)
|
|
const groupOrder: string[] = []
|
|
const grouped: Record<string, ScriptParameter[]> = {}
|
|
for (const p of parameters) {
|
|
if (p.group !== null) {
|
|
if (!grouped[p.group]) {
|
|
grouped[p.group] = []
|
|
groupOrder.push(p.group)
|
|
}
|
|
grouped[p.group].push(p)
|
|
}
|
|
}
|
|
|
|
const renderParam = (param: ScriptParameter) => (
|
|
<ScriptParameterField
|
|
key={param.key}
|
|
param={param}
|
|
value={paramValues[param.key] ?? ''}
|
|
error={formErrors[param.key] || undefined}
|
|
disabled={!canGenerate}
|
|
/>
|
|
)
|
|
|
|
if (parameters.length === 0) {
|
|
return (
|
|
<div className="flex items-center gap-2 rounded-lg border border-border bg-card px-3 py-3">
|
|
<Terminal size={14} className="text-muted-foreground shrink-0" />
|
|
<p className="text-xs text-muted-foreground">
|
|
This template has no parameters — click <span className="text-foreground font-medium">Generate</span> to produce the script.
|
|
</p>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<div className="flex flex-col gap-4">
|
|
{ungrouped.map(renderParam)}
|
|
{groupOrder.map(group => (
|
|
<div key={group}>
|
|
<p className="font-sans text-[0.625rem] uppercase tracking-[0.1em] text-muted-foreground mb-3">
|
|
{group}
|
|
</p>
|
|
<div className="flex flex-col gap-4">
|
|
{grouped[group].map(renderParam)}
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)
|
|
}
|