All checks were successful
Build + deploy releases / build-and-deploy (push) Successful in 8s
Schema-driven form renderer plus zddc-server endpoints that turn any
<name>.form.yaml into a working data-collection form at <path>/<name>.form.html.
Submissions land in <path>/<name>/<YYYY-MM-DD>-<email-sanitized>.yaml,
ACL-gated by the existing .zddc cascade. The form posts back to its own URL;
the server strips ".html" and routes by what's underneath, so create and
update use the same client-side code path.
Form spec dialect: JSON Schema 2020-12 + RJSF-style ui:* hints, written in
YAML. Chosen for LLM authorability — it's the canonical structured-output
target for OpenAI/Anthropic, and the ui:* convention is the most-trained UI
hint vocabulary. Supported subset for v0: type (string/number/integer/boolean/
array/object), enum, min/max, minLength/maxLength, required, additionalProperties:
false, properties, items, format (date, email). Round-trip mode is form-as-truth:
submission YAML is regenerated each save, comments are not preserved (the v1
file-as-truth mode for hand-edited files like .zddc itself is deferred).
New components:
* form/ — sixth single-file HTML tool, vanilla JS renderer (~760 LoC)
* zddc/internal/jsonschema/ — focused JSON Schema validator covering only
the v0 keyword subset. Match-implementation-cost-to-surface-used: a full
library brings 70%+ surface we don't use; revisit when v1 adds $ref +
oneOf + if/then/else.
* zddc/internal/handler/formhandler.go — RecognizeFormRequest / ServeForm,
capability-URL re-edit, atomic submission writes via the new
zddc.WriteAtomic helper extracted from writer.go.
* dispatch() in zddc-server/main.go now intercepts *.form.html and
*.yaml.html before the static-file path; spec existence is the trigger.
Build pipeline: form joins ZDDC_RELEASE_TOOLS in lockstep, gets its own
embedded copy in handler/form.html (separate from the apps cascade —
the form renderer is fixed, not subject to per-folder version overrides).
Tests: 5 new Playwright specs (form-safety) + 14 new Go tests across the
validator and handler. All 172 Playwright tests + 10 Go packages green.
End-to-end manual verification: GET empty → POST 201 + capability URL →
GET re-edit (pre-filled) → POST update → 200, raw YAML browsable, ACL
deny → 403.
Docs: form/ section added to AGENTS.md and ARCHITECTURE.md. AGENTS.md
also documents the implementation-vs-dependency policy. CLAUDE.md repo-shape
list extended.
Deferred (v1+): .zddc editor migration onto this system, file-as-truth
lossless YAML round-trip, ui:show-when conditional visibility, oneOf/anyOf,
apps-cascade preview hook, cascade-fetched form definitions.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
226 lines
8.5 KiB
JavaScript
226 lines
8.5 KiB
JavaScript
(function (app) {
|
|
'use strict';
|
|
|
|
const u = app.modules.util;
|
|
|
|
// Build the standard label / description / input / help / error scaffold
|
|
// shared by all primitive widgets. Returns { wrap, errEl }.
|
|
function fieldContainer(opts) {
|
|
const wrap = u.h('div', { className: 'form-field' });
|
|
if (opts.label) {
|
|
const lbl = u.h('label', { className: 'form-field__label', for: opts.id });
|
|
lbl.appendChild(document.createTextNode(opts.label));
|
|
if (opts.required) {
|
|
lbl.appendChild(u.h('span', { className: 'required-mark' }, '*'));
|
|
}
|
|
wrap.appendChild(lbl);
|
|
}
|
|
if (opts.description) {
|
|
wrap.appendChild(u.h('div', { className: 'form-field__description' }, opts.description));
|
|
}
|
|
wrap.appendChild(opts.input);
|
|
if (opts.help) {
|
|
wrap.appendChild(u.h('div', { className: 'form-field__help' }, opts.help));
|
|
}
|
|
const errEl = u.h('div', { className: 'form-field__error', hidden: true });
|
|
wrap.appendChild(errEl);
|
|
return { wrap: wrap, errEl: errEl };
|
|
}
|
|
|
|
function coerceEnum(rawValue, options) {
|
|
for (let i = 0; i < options.length; i++) {
|
|
if (String(options[i]) === rawValue) {
|
|
return options[i];
|
|
}
|
|
}
|
|
return rawValue;
|
|
}
|
|
|
|
function makePrimitive(schema, ui, path, value, options) {
|
|
const id = u.uid('w');
|
|
const required = !!options.required;
|
|
const label = (ui && ui['ui:title']) || schema.title || options.fieldName || '';
|
|
const description = (ui && ui['ui:description']) || schema.description || '';
|
|
const help = (ui && ui['ui:help']) || '';
|
|
const placeholder = (ui && ui['ui:placeholder']) || '';
|
|
const widget = (ui && ui['ui:widget']) || '';
|
|
const readonly = !!(ui && ui['ui:readonly']);
|
|
const autofocus = !!(ui && ui['ui:autofocus']);
|
|
|
|
let input;
|
|
let read;
|
|
|
|
const t = schema.type;
|
|
|
|
if (t === 'boolean') {
|
|
// Render boolean as a single checkbox with an inline label, suppressing
|
|
// the standard label-above layout for cleaner UX.
|
|
const cb = u.h('input', { type: 'checkbox', id: id });
|
|
if (value === true) {
|
|
cb.checked = true;
|
|
}
|
|
if (readonly) {
|
|
cb.disabled = true;
|
|
}
|
|
const wrap = u.h('div', { className: 'form-field form-field--boolean' });
|
|
const inlineLabel = u.h('label', { for: id, className: 'form-field__checkbox-inline' });
|
|
inlineLabel.appendChild(cb);
|
|
inlineLabel.appendChild(document.createTextNode(' '));
|
|
inlineLabel.appendChild(document.createTextNode(label || ''));
|
|
if (required) {
|
|
inlineLabel.appendChild(u.h('span', { className: 'required-mark' }, '*'));
|
|
}
|
|
wrap.appendChild(inlineLabel);
|
|
if (description) {
|
|
wrap.appendChild(u.h('div', { className: 'form-field__description' }, description));
|
|
}
|
|
if (help) {
|
|
wrap.appendChild(u.h('div', { className: 'form-field__help' }, help));
|
|
}
|
|
const errEl = u.h('div', { className: 'form-field__error', hidden: true });
|
|
wrap.appendChild(errEl);
|
|
|
|
return widgetObject(wrap, errEl, path, function () {
|
|
return cb.checked;
|
|
});
|
|
}
|
|
|
|
if (Array.isArray(schema.enum)) {
|
|
const opts = schema.enum;
|
|
if (widget === 'radio') {
|
|
input = u.h('div', { className: 'form-field__radio-group' });
|
|
opts.forEach(function (opt, idx) {
|
|
const radioId = id + '-' + idx;
|
|
const radio = u.h('input', { type: 'radio', name: id, id: radioId, value: String(opt) });
|
|
if (value === opt) {
|
|
radio.checked = true;
|
|
}
|
|
if (readonly) {
|
|
radio.disabled = true;
|
|
}
|
|
const lbl = u.h('label', { for: radioId });
|
|
lbl.appendChild(radio);
|
|
lbl.appendChild(document.createTextNode(' ' + String(opt)));
|
|
input.appendChild(lbl);
|
|
});
|
|
read = function () {
|
|
const checked = input.querySelector('input[type="radio"]:checked');
|
|
return checked ? coerceEnum(checked.value, opts) : undefined;
|
|
};
|
|
} else {
|
|
input = u.h('select', { id: id, className: 'form-field__select' });
|
|
if (!required) {
|
|
input.appendChild(u.h('option', { value: '' }, '— select —'));
|
|
}
|
|
opts.forEach(function (opt) {
|
|
const o = u.h('option', { value: String(opt) }, String(opt));
|
|
if (value === opt) {
|
|
o.selected = true;
|
|
}
|
|
input.appendChild(o);
|
|
});
|
|
if (readonly) {
|
|
input.disabled = true;
|
|
}
|
|
read = function () {
|
|
if (input.value === '') {
|
|
return undefined;
|
|
}
|
|
return coerceEnum(input.value, opts);
|
|
};
|
|
}
|
|
} else if (t === 'number' || t === 'integer') {
|
|
input = u.h('input', {
|
|
type: 'number',
|
|
id: id,
|
|
className: 'form-field__input',
|
|
step: t === 'integer' ? '1' : 'any'
|
|
});
|
|
if (placeholder) {
|
|
input.placeholder = placeholder;
|
|
}
|
|
if (value != null) {
|
|
input.value = String(value);
|
|
}
|
|
if (readonly) {
|
|
input.readOnly = true;
|
|
}
|
|
if (autofocus) {
|
|
input.autofocus = true;
|
|
}
|
|
read = function () {
|
|
const v = input.value.trim();
|
|
if (v === '') {
|
|
return undefined;
|
|
}
|
|
const n = Number(v);
|
|
// If the user typed something non-numeric, return the raw string and
|
|
// let server validation produce a friendly error.
|
|
return Number.isFinite(n) ? n : v;
|
|
};
|
|
} else {
|
|
// Default: string-shaped input.
|
|
const fmt = schema.format;
|
|
if (widget === 'textarea') {
|
|
input = u.h('textarea', { id: id, className: 'form-field__textarea' });
|
|
} else {
|
|
let inputType = 'text';
|
|
if (fmt === 'date') {
|
|
inputType = 'date';
|
|
} else if (fmt === 'email') {
|
|
inputType = 'email';
|
|
}
|
|
input = u.h('input', { type: inputType, id: id, className: 'form-field__input' });
|
|
}
|
|
if (placeholder) {
|
|
input.placeholder = placeholder;
|
|
}
|
|
if (value != null) {
|
|
input.value = String(value);
|
|
}
|
|
if (readonly) {
|
|
input.readOnly = true;
|
|
}
|
|
if (autofocus) {
|
|
input.autofocus = true;
|
|
}
|
|
read = function () {
|
|
return input.value === '' ? undefined : input.value;
|
|
};
|
|
}
|
|
|
|
const built = fieldContainer({
|
|
id: id,
|
|
label: label,
|
|
description: description,
|
|
help: help,
|
|
required: required,
|
|
input: input
|
|
});
|
|
|
|
return widgetObject(built.wrap, built.errEl, path, read);
|
|
}
|
|
|
|
// Common widget shape used by both primitive and the wrapper above.
|
|
function widgetObject(wrapEl, errEl, path, read) {
|
|
return {
|
|
el: wrapEl,
|
|
path: path,
|
|
type: 'primitive',
|
|
read: read,
|
|
setError: function (msg) {
|
|
errEl.textContent = msg;
|
|
errEl.hidden = false;
|
|
wrapEl.classList.add('form-field--invalid');
|
|
},
|
|
clearErrors: function () {
|
|
errEl.textContent = '';
|
|
errEl.hidden = true;
|
|
wrapEl.classList.remove('form-field--invalid');
|
|
},
|
|
child: function () { return null; }
|
|
};
|
|
}
|
|
|
|
app.modules.widgets = { makePrimitive: makePrimitive };
|
|
})(window.formApp);
|