@duffcloudservices/site-forms 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +213 -0
- package/dist/DcsForm.vue.d.ts +67 -0
- package/dist/composables/useDcsForm.d.ts +36 -0
- package/dist/composables/useFormSubmission.d.ts +18 -0
- package/dist/composables/useFormValidation.d.ts +19 -0
- package/dist/fields/DcsFormCheckbox.vue.d.ts +12 -0
- package/dist/fields/DcsFormCheckboxGroup.vue.d.ts +12 -0
- package/dist/fields/DcsFormDate.vue.d.ts +14 -0
- package/dist/fields/DcsFormFieldWrapper.vue.d.ts +27 -0
- package/dist/fields/DcsFormFile.vue.d.ts +12 -0
- package/dist/fields/DcsFormHidden.vue.d.ts +7 -0
- package/dist/fields/DcsFormHtmlBlock.vue.d.ts +6 -0
- package/dist/fields/DcsFormRadio.vue.d.ts +12 -0
- package/dist/fields/DcsFormSection.vue.d.ts +21 -0
- package/dist/fields/DcsFormSelect.vue.d.ts +35 -0
- package/dist/fields/DcsFormText.vue.d.ts +34 -0
- package/dist/fields/DcsFormTextarea.vue.d.ts +34 -0
- package/dist/index.d.ts +22 -0
- package/dist/index.js +918 -0
- package/dist/index.js.map +1 -0
- package/dist/loaders/yaml.d.ts +12 -0
- package/dist/schema/validate.d.ts +9 -0
- package/dist/types.d.ts +106 -0
- package/package.json +73 -0
- package/src/DcsForm.vue +299 -0
- package/src/__tests__/fields.test.ts +82 -0
- package/src/__tests__/multi-step.test.ts +46 -0
- package/src/__tests__/schema.test.ts +42 -0
- package/src/__tests__/submission.test.ts +77 -0
- package/src/__tests__/visible-if.test.ts +111 -0
- package/src/composables/useDcsForm.ts +201 -0
- package/src/composables/useFormSubmission.ts +113 -0
- package/src/composables/useFormValidation.ts +127 -0
- package/src/fields/DcsFormCheckbox.vue +35 -0
- package/src/fields/DcsFormCheckboxGroup.vue +52 -0
- package/src/fields/DcsFormDate.vue +34 -0
- package/src/fields/DcsFormFieldWrapper.vue +39 -0
- package/src/fields/DcsFormFile.vue +38 -0
- package/src/fields/DcsFormHidden.vue +17 -0
- package/src/fields/DcsFormHtmlBlock.vue +19 -0
- package/src/fields/DcsFormRadio.vue +45 -0
- package/src/fields/DcsFormSection.vue +19 -0
- package/src/fields/DcsFormSelect.vue +62 -0
- package/src/fields/DcsFormText.vue +54 -0
- package/src/fields/DcsFormTextarea.vue +43 -0
- package/src/index.ts +51 -0
- package/src/loaders/yaml.ts +51 -0
- package/src/schema/form-definition.schema.json +633 -0
- package/src/schema/validate.ts +58 -0
- package/src/shims.d.ts +10 -0
- package/src/types.ts +140 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../src/composables/useFormValidation.ts","../src/composables/useDcsForm.ts","../src/composables/useFormSubmission.ts","../src/schema/validate.ts","../src/loaders/yaml.ts","../src/fields/DcsFormFieldWrapper.vue","../src/fields/DcsFormText.vue","../src/fields/DcsFormTextarea.vue","../src/fields/DcsFormSelect.vue","../src/fields/DcsFormRadio.vue","../src/fields/DcsFormCheckboxGroup.vue","../src/fields/DcsFormCheckbox.vue","../src/fields/DcsFormDate.vue","../src/fields/DcsFormFile.vue","../src/fields/DcsFormHidden.vue","../src/fields/DcsFormSection.vue","../src/fields/DcsFormHtmlBlock.vue","../src/DcsForm.vue"],"sourcesContent":["import type {\r\n PortalFormDefinition,\r\n PortalFormField,\r\n FormErrors,\r\n FormValues,\r\n} from '../types'\r\n\r\n/**\r\n * Returns true iff `field` is currently visible given the form's\r\n * other values. Hidden fields are never validated and never sent.\r\n */\r\nexport function isFieldVisible(\r\n field: PortalFormField,\r\n values: FormValues,\r\n): boolean {\r\n if (!field.visibleIf) return true\r\n const sibling = values[field.visibleIf.fieldId]\r\n return sibling === field.visibleIf.equals\r\n}\r\n\r\n/**\r\n * Validates a single field's value. Returns an error string or\r\n * `undefined` if valid. Layout-only field types (section-heading,\r\n * html-block) and hidden fields never produce errors.\r\n */\r\nexport function validateField(\r\n field: PortalFormField,\r\n value: unknown,\r\n): string | undefined {\r\n if (\r\n field.type === 'section-heading' ||\r\n field.type === 'html-block' ||\r\n field.type === 'hidden'\r\n ) {\r\n return undefined\r\n }\r\n\r\n const isEmpty =\r\n value === undefined ||\r\n value === null ||\r\n value === '' ||\r\n (Array.isArray(value) && value.length === 0)\r\n\r\n if (field.required && isEmpty) {\r\n return `${field.label} is required`\r\n }\r\n if (isEmpty) return undefined\r\n\r\n // Type-specific built-ins\r\n if (field.type === 'email' && typeof value === 'string') {\r\n if (!/^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(value)) {\r\n return `${field.label} must be a valid email address`\r\n }\r\n }\r\n\r\n const v = field.validation\r\n if (!v) return undefined\r\n\r\n if (typeof value === 'string') {\r\n if (v.minLength != null && value.length < v.minLength) {\r\n return `${field.label} must be at least ${v.minLength} characters`\r\n }\r\n if (v.maxLength != null && value.length > v.maxLength) {\r\n return `${field.label} must be at most ${v.maxLength} characters`\r\n }\r\n if (v.regex) {\r\n try {\r\n if (!new RegExp(v.regex).test(value)) {\r\n return `${field.label} is invalid`\r\n }\r\n } catch {\r\n // bad regex in the definition — treat as no rule\r\n }\r\n }\r\n }\r\n\r\n if (typeof value === 'number') {\r\n if (v.min != null && value < v.min) {\r\n return `${field.label} must be at least ${v.min}`\r\n }\r\n if (v.max != null && value > v.max) {\r\n return `${field.label} must be at most ${v.max}`\r\n }\r\n }\r\n\r\n if (field.type === 'file' && v.accept && v.accept.length > 0) {\r\n const file = value as File | null\r\n if (file && typeof File !== 'undefined' && file instanceof File) {\r\n const accepted = v.accept.some((a) => {\r\n if (a.startsWith('.')) {\r\n return file.name.toLowerCase().endsWith(a.toLowerCase())\r\n }\r\n return file.type === a\r\n })\r\n if (!accepted) {\r\n return `${field.label} must be one of: ${v.accept.join(', ')}`\r\n }\r\n }\r\n }\r\n\r\n return undefined\r\n}\r\n\r\n/**\r\n * Validates an entire form. Skips fields that are not visible.\r\n * If `fieldIds` is supplied, only those fields are validated\r\n * (used for per-step validation in multi-step forms).\r\n */\r\nexport function validateForm(\r\n def: PortalFormDefinition,\r\n values: FormValues,\r\n fieldIds?: string[],\r\n): FormErrors {\r\n const errors: FormErrors = {}\r\n const ids = fieldIds ? new Set(fieldIds) : null\r\n for (const field of def.fields) {\r\n if (ids && !ids.has(field.id)) continue\r\n if (!isFieldVisible(field, values)) continue\r\n const err = validateField(field, values[field.id])\r\n if (err) errors[field.id] = err\r\n }\r\n return errors\r\n}\r\n\r\nexport function hasErrors(errors: FormErrors): boolean {\r\n return Object.values(errors).some((e) => !!e)\r\n}\r\n","import { computed, reactive, ref, type ComputedRef, type Ref } from 'vue'\r\nimport type {\r\n PortalFormDefinition,\r\n PortalFormField,\r\n PortalFormStep,\r\n FormErrors,\r\n FormValues,\r\n} from '../types'\r\nimport {\r\n hasErrors,\r\n isFieldVisible,\r\n validateForm,\r\n} from './useFormValidation'\r\n\r\nexport interface UseDcsFormOptions {\r\n definition: PortalFormDefinition\r\n}\r\n\r\nexport interface UseDcsFormReturn {\r\n values: FormValues\r\n errors: Ref<FormErrors>\r\n touched: Ref<Record<string, boolean>>\r\n submitting: Ref<boolean>\r\n submitted: Ref<boolean>\r\n submitError: Ref<string | null>\r\n /** All fields, with layout-only types preserved. */\r\n fields: ComputedRef<PortalFormField[]>\r\n /** Steps if multi-step, else null. */\r\n steps: ComputedRef<PortalFormStep[] | null>\r\n currentStepIndex: Ref<number>\r\n currentStep: ComputedRef<PortalFormStep | null>\r\n currentStepFields: ComputedRef<PortalFormField[]>\r\n isFirstStep: ComputedRef<boolean>\r\n isLastStep: ComputedRef<boolean>\r\n visibleFields: ComputedRef<PortalFormField[]>\r\n /** Sets a field value and clears that field's error. */\r\n setValue: (id: string, value: unknown) => void\r\n /** Marks a field as touched + revalidates only that field's row. */\r\n touch: (id: string) => void\r\n /** Validates the current step (multi-step) or the entire form. */\r\n validateCurrentScope: () => boolean\r\n validateAll: () => boolean\r\n next: () => boolean\r\n prev: () => void\r\n reset: () => void\r\n /** Returns the values that should actually be submitted (visible only). */\r\n collectSubmissionValues: () => FormValues\r\n}\r\n\r\nfunction defaultsFor(def: PortalFormDefinition): FormValues {\r\n const out: FormValues = {}\r\n for (const f of def.fields) {\r\n if (f.defaultValue !== undefined) {\r\n out[f.id] = f.defaultValue as unknown\r\n } else if (f.type === 'checkbox') {\r\n out[f.id] = false\r\n } else if (f.type === 'multiselect' || f.type === 'checkbox-group') {\r\n out[f.id] = [] as string[]\r\n } else {\r\n out[f.id] = ''\r\n }\r\n }\r\n return out\r\n}\r\n\r\nexport function useDcsForm(opts: UseDcsFormOptions): UseDcsFormReturn {\r\n const { definition } = opts\r\n const values = reactive<FormValues>(defaultsFor(definition))\r\n const errors = ref<FormErrors>({})\r\n const touched = ref<Record<string, boolean>>({})\r\n const submitting = ref(false)\r\n const submitted = ref(false)\r\n const submitError = ref<string | null>(null)\r\n const currentStepIndex = ref(0)\r\n\r\n const fields = computed(() => definition.fields)\r\n const steps = computed<PortalFormStep[] | null>(() =>\r\n definition.steps && definition.steps.length > 0 ? definition.steps : null,\r\n )\r\n const currentStep = computed<PortalFormStep | null>(() => {\r\n const s = steps.value\r\n return s ? (s[currentStepIndex.value] ?? null) : null\r\n })\r\n const currentStepFields = computed<PortalFormField[]>(() => {\r\n const step = currentStep.value\r\n if (!step) return definition.fields\r\n const ids = new Set(step.fieldIds)\r\n return definition.fields.filter((f) => ids.has(f.id))\r\n })\r\n const isFirstStep = computed(() => currentStepIndex.value === 0)\r\n const isLastStep = computed(() => {\r\n const s = steps.value\r\n return !s || currentStepIndex.value >= s.length - 1\r\n })\r\n const visibleFields = computed(() =>\r\n currentStepFields.value.filter((f) => isFieldVisible(f, values as FormValues)),\r\n )\r\n\r\n function setValue(id: string, value: unknown): void {\r\n ;(values as FormValues)[id] = value\r\n if (errors.value[id]) {\r\n errors.value = { ...errors.value, [id]: undefined }\r\n }\r\n }\r\n\r\n function touch(id: string): void {\r\n touched.value = { ...touched.value, [id]: true }\r\n }\r\n\r\n function validateScope(scopeFieldIds?: string[]): boolean {\r\n const next = validateForm(definition, values as FormValues, scopeFieldIds)\r\n // Merge with existing errors but only for the validated scope so we\r\n // don't wipe out errors from other steps.\r\n if (scopeFieldIds) {\r\n const merged = { ...errors.value }\r\n for (const id of scopeFieldIds) {\r\n merged[id] = next[id]\r\n }\r\n errors.value = merged\r\n } else {\r\n errors.value = next\r\n }\r\n return !hasErrors(errors.value)\r\n }\r\n\r\n function validateCurrentScope(): boolean {\r\n const scope = currentStep.value?.fieldIds\r\n return validateScope(scope)\r\n }\r\n\r\n function validateAll(): boolean {\r\n return validateScope()\r\n }\r\n\r\n function next(): boolean {\r\n if (!steps.value) return false\r\n if (!validateCurrentScope()) return false\r\n if (currentStepIndex.value < steps.value.length - 1) {\r\n currentStepIndex.value++\r\n return true\r\n }\r\n return false\r\n }\r\n\r\n function prev(): void {\r\n if (currentStepIndex.value > 0) currentStepIndex.value--\r\n }\r\n\r\n function reset(): void {\r\n const fresh = defaultsFor(definition)\r\n for (const k of Object.keys(values as object)) {\r\n delete (values as FormValues)[k]\r\n }\r\n Object.assign(values as FormValues, fresh)\r\n errors.value = {}\r\n touched.value = {}\r\n submitting.value = false\r\n submitted.value = false\r\n submitError.value = null\r\n currentStepIndex.value = 0\r\n }\r\n\r\n function collectSubmissionValues(): FormValues {\r\n const out: FormValues = {}\r\n for (const f of definition.fields) {\r\n if (\r\n f.type === 'section-heading' ||\r\n f.type === 'html-block'\r\n )\r\n continue\r\n if (!isFieldVisible(f, values as FormValues)) continue\r\n out[f.id] = (values as FormValues)[f.id]\r\n }\r\n return out\r\n }\r\n\r\n return {\r\n values: values as FormValues,\r\n errors,\r\n touched,\r\n submitting,\r\n submitted,\r\n submitError,\r\n fields,\r\n steps,\r\n currentStepIndex,\r\n currentStep,\r\n currentStepFields,\r\n isFirstStep,\r\n isLastStep,\r\n visibleFields,\r\n setValue,\r\n touch,\r\n validateCurrentScope,\r\n validateAll,\r\n next,\r\n prev,\r\n reset,\r\n collectSubmissionValues,\r\n }\r\n}\r\n","import type {\r\n DcsFormSubmitPayload,\r\n DcsFormSubmitSuccess,\r\n DcsFormSubmitError,\r\n} from '../types'\r\n\r\nexport interface SubmitOptions {\r\n apiBase: string\r\n siteSlug: string\r\n payload: DcsFormSubmitPayload\r\n /** Number of retries on 5xx / network errors. Default 1. */\r\n retries?: number\r\n /** Optional fetch implementation override (tests). */\r\n fetchImpl?: typeof fetch\r\n}\r\n\r\n/**\r\n * POSTs a form submission to\r\n * `${apiBase}/public/sites/{siteSlug}/form-submissions`.\r\n *\r\n * Uses JSON for plain values and `multipart/form-data` when any\r\n * value is a `File` (file-upload fields).\r\n */\r\nexport async function submitFormValues(\r\n opts: SubmitOptions,\r\n): Promise<DcsFormSubmitSuccess> {\r\n const { apiBase, siteSlug, payload } = opts\r\n const retries = opts.retries ?? 1\r\n const fetchImpl = opts.fetchImpl ?? fetch\r\n const url = `${apiBase.replace(/\\/$/, '')}/public/sites/${encodeURIComponent(\r\n siteSlug,\r\n )}/form-submissions`\r\n\r\n const hasFile = Object.values(payload.values).some(\r\n (v) => typeof File !== 'undefined' && v instanceof File,\r\n )\r\n\r\n let lastError: Error | undefined\r\n let lastStatus: number | undefined\r\n for (let attempt = 0; attempt <= retries; attempt++) {\r\n try {\r\n const init: RequestInit = hasFile\r\n ? { method: 'POST', body: buildFormData(payload) }\r\n : {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify(payload),\r\n }\r\n const res = await fetchImpl(url, init)\r\n lastStatus = res.status\r\n if (res.ok) {\r\n const body = await safeJson(res)\r\n return { payload, response: body }\r\n }\r\n // Retry only on 5xx\r\n if (res.status < 500) {\r\n const body = await safeText(res)\r\n const err: DcsFormSubmitError = {\r\n payload,\r\n status: res.status,\r\n error: new Error(`Submission failed (${res.status}): ${body}`),\r\n }\r\n throw err\r\n }\r\n lastError = new Error(`Server error ${res.status}`)\r\n } catch (e) {\r\n // If `e` is already a DcsFormSubmitError-shaped object, rethrow.\r\n if (e && typeof e === 'object' && 'payload' in e && 'error' in e) {\r\n throw e\r\n }\r\n lastError = e as Error\r\n }\r\n }\r\n\r\n throw {\r\n payload,\r\n status: lastStatus,\r\n error: lastError ?? new Error('Submission failed'),\r\n } satisfies DcsFormSubmitError\r\n}\r\n\r\nfunction buildFormData(payload: DcsFormSubmitPayload): FormData {\r\n const fd = new FormData()\r\n fd.append('formId', payload.formId)\r\n if (payload.captchaToken) fd.append('captchaToken', payload.captchaToken)\r\n for (const [key, value] of Object.entries(payload.values)) {\r\n if (value === undefined || value === null) continue\r\n if (value instanceof File) {\r\n fd.append(`values[${key}]`, value)\r\n } else if (Array.isArray(value)) {\r\n for (const item of value) fd.append(`values[${key}][]`, String(item))\r\n } else {\r\n fd.append(`values[${key}]`, String(value))\r\n }\r\n }\r\n return fd\r\n}\r\n\r\nasync function safeJson(res: Response): Promise<unknown> {\r\n try {\r\n return await res.json()\r\n } catch {\r\n return null\r\n }\r\n}\r\n\r\nasync function safeText(res: Response): Promise<string> {\r\n try {\r\n return await res.text()\r\n } catch {\r\n return ''\r\n }\r\n}\r\n","/**\r\n * Validates a loaded form definition against the form-definition JSON\r\n * Schema emitted by `@dcs/contracts`. Dev-only — failures are logged\r\n * via `console.warn` rather than thrown so a malformed YAML never\r\n * takes down a customer site in production.\r\n *\r\n * Schema is bundled as a snapshot of\r\n * `contracts/dist/form-definition.schema.json` to avoid a runtime\r\n * dependency on the contracts build output.\r\n */\r\nimport Ajv, { type ErrorObject, type ValidateFunction } from 'ajv'\r\nimport addFormats from 'ajv-formats'\r\nimport schema from './form-definition.schema.json'\r\nimport type { PortalFormDefinition } from '../types'\r\n\r\nlet validator: ValidateFunction | null = null\r\n\r\nfunction getValidator(): ValidateFunction {\r\n if (validator) return validator\r\n const ajv = new Ajv({\r\n allErrors: true,\r\n strict: false,\r\n discriminator: false,\r\n })\r\n addFormats(ajv)\r\n validator = ajv.compile(schema as object)\r\n return validator\r\n}\r\n\r\nexport interface SchemaValidationResult {\r\n valid: boolean\r\n errors: ErrorObject[]\r\n}\r\n\r\nexport function validateFormDefinition(\r\n def: PortalFormDefinition | unknown,\r\n): SchemaValidationResult {\r\n const v = getValidator()\r\n const valid = v(def) as boolean\r\n return { valid, errors: valid ? [] : (v.errors ?? []) }\r\n}\r\n\r\n/** Logs a `console.warn` when invalid, but only when `import.meta.env.DEV`. */\r\nexport function warnIfInvalid(\r\n formId: string,\r\n def: PortalFormDefinition | unknown,\r\n isDev: boolean,\r\n): SchemaValidationResult {\r\n const result = validateFormDefinition(def)\r\n if (!result.valid && isDev) {\r\n // eslint-disable-next-line no-console\r\n console.warn(\r\n `[@duffcloudservices/site-forms] Form \"${formId}\" failed schema validation:`,\r\n result.errors,\r\n )\r\n }\r\n return result\r\n}\r\n","import yaml from 'js-yaml'\r\nimport type { PortalFormDefinition } from '../types'\r\n\r\n/**\r\n * Eagerly load every form YAML under `/.dcs/forms/*.yaml` from the\r\n * consuming Vite app and parse them into PortalFormDefinition objects.\r\n *\r\n * Vite returns the modules as raw strings (when using `?raw` or the\r\n * default text loader for unknown extensions) or as parsed objects\r\n * (when `vite-plugin-yaml` is installed). This helper handles both.\r\n */\r\nexport function loadFormDefinitions(\r\n modules: Record<string, unknown>,\r\n): Record<string, PortalFormDefinition> {\r\n const out: Record<string, PortalFormDefinition> = {}\r\n for (const [path, mod] of Object.entries(modules)) {\r\n const formId = extractFormId(path)\r\n const def = parseModule(mod)\r\n if (def) {\r\n out[def.formId ?? formId] = def\r\n }\r\n }\r\n return out\r\n}\r\n\r\nfunction extractFormId(path: string): string {\r\n const file = path.split('/').pop() ?? path\r\n return file.replace(/\\.ya?ml$/i, '')\r\n}\r\n\r\nfunction parseModule(mod: unknown): PortalFormDefinition | null {\r\n if (!mod) return null\r\n if (typeof mod === 'string') {\r\n return yaml.load(mod) as PortalFormDefinition\r\n }\r\n if (typeof mod === 'object') {\r\n // vite-plugin-yaml returns parsed objects, possibly wrapped in\r\n // `{ default: ... }` when imported via `import: 'default'`.\r\n const maybeDefault = (mod as { default?: unknown }).default\r\n if (maybeDefault && typeof maybeDefault === 'object') {\r\n return maybeDefault as PortalFormDefinition\r\n }\r\n return mod as PortalFormDefinition\r\n }\r\n return null\r\n}\r\n\r\n/** Parse a single raw YAML string into a PortalFormDefinition. */\r\nexport function parseFormYaml(raw: string): PortalFormDefinition {\r\n return yaml.load(raw) as PortalFormDefinition\r\n}\r\n","<script setup lang=\"ts\">\r\nimport type { PortalFormField } from '../types'\r\n\r\ndefineProps<{\r\n field: PortalFormField\r\n error?: string\r\n /** Optional id to use for the input — wrappers use it for label `for=`. */\r\n inputId?: string\r\n}>()\r\n</script>\r\n\r\n<template>\r\n <div\r\n class=\"dcs-form-field\"\r\n :class=\"[`dcs-form-field--${field.type}`, `dcs-form-field--width-${field.width ?? 'full'}`]\"\r\n :data-form-field-key=\"field.id\"\r\n >\r\n <slot name=\"label\">\r\n <label\r\n v-if=\"field.type !== 'checkbox' && field.type !== 'hidden' && field.type !== 'section-heading' && field.type !== 'html-block'\"\r\n :for=\"inputId ?? `field-${field.id}`\"\r\n class=\"dcs-form-field__label\"\r\n >\r\n {{ field.label }}\r\n <span v-if=\"field.required\" aria-hidden=\"true\" class=\"dcs-form-field__required\">*</span>\r\n </label>\r\n </slot>\r\n\r\n <slot />\r\n\r\n <slot name=\"help\">\r\n <p v-if=\"field.helpText\" class=\"dcs-form-field__help\">{{ field.helpText }}</p>\r\n </slot>\r\n\r\n <slot name=\"error\">\r\n <p v-if=\"error\" class=\"dcs-form-field__error\" role=\"alert\">{{ error }}</p>\r\n </slot>\r\n </div>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string | number | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string]\r\n blur: []\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\nconst inputType = computed(() => {\r\n switch (props.field.type) {\r\n case 'email':\r\n return 'email'\r\n case 'tel':\r\n return 'tel'\r\n default:\r\n return 'text'\r\n }\r\n})\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <slot\r\n name=\"input\"\r\n :id=\"inputId\"\r\n :value=\"modelValue\"\r\n :on-input=\"(e: Event) => emit('update:modelValue', (e.target as HTMLInputElement).value)\"\r\n :on-blur=\"() => emit('blur')\"\r\n >\r\n <input\r\n :id=\"inputId\"\r\n class=\"dcs-form-input\"\r\n :type=\"inputType\"\r\n :name=\"field.id\"\r\n :value=\"modelValue ?? ''\"\r\n :placeholder=\"field.placeholder\"\r\n :required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n :aria-describedby=\"error ? `${inputId}-error` : undefined\"\r\n @input=\"emit('update:modelValue', ($event.target as HTMLInputElement).value)\"\r\n @blur=\"emit('blur')\"\r\n />\r\n </slot>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string]\r\n blur: []\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <slot\r\n name=\"input\"\r\n :id=\"inputId\"\r\n :value=\"modelValue\"\r\n :on-input=\"(e: Event) => emit('update:modelValue', (e.target as HTMLTextAreaElement).value)\"\r\n :on-blur=\"() => emit('blur')\"\r\n >\r\n <textarea\r\n :id=\"inputId\"\r\n class=\"dcs-form-input dcs-form-textarea\"\r\n :name=\"field.id\"\r\n :placeholder=\"field.placeholder\"\r\n :required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n rows=\"5\"\r\n :value=\"modelValue ?? ''\"\r\n @input=\"emit('update:modelValue', ($event.target as HTMLTextAreaElement).value)\"\r\n @blur=\"emit('blur')\"\r\n />\r\n </slot>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string | string[] | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string | string[]]\r\n blur: []\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\nconst isMulti = computed(() => props.field.type === 'multiselect')\r\nconst options = computed(() => props.field.options ?? [])\r\n\r\nfunction onChange(e: Event): void {\r\n const sel = e.target as HTMLSelectElement\r\n if (isMulti.value) {\r\n const values: string[] = []\r\n for (const opt of Array.from(sel.selectedOptions)) values.push(opt.value)\r\n emit('update:modelValue', values)\r\n } else {\r\n emit('update:modelValue', sel.value)\r\n }\r\n}\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <slot\r\n name=\"input\"\r\n :id=\"inputId\"\r\n :value=\"modelValue\"\r\n :options=\"options\"\r\n :on-change=\"onChange\"\r\n >\r\n <select\r\n :id=\"inputId\"\r\n class=\"dcs-form-input dcs-form-select\"\r\n :name=\"field.id\"\r\n :required=\"field.required\"\r\n :multiple=\"isMulti\"\r\n :aria-invalid=\"!!error\"\r\n :value=\"modelValue ?? (isMulti ? [] : '')\"\r\n @change=\"onChange\"\r\n @blur=\"emit('blur')\"\r\n >\r\n <option v-if=\"!isMulti\" value=\"\" disabled>\r\n {{ field.placeholder ?? 'Select…' }}\r\n </option>\r\n <option v-for=\"opt in options\" :key=\"opt.value\" :value=\"opt.value\">\r\n {{ opt.label }}\r\n </option>\r\n </select>\r\n </slot>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string]\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\nconst options = computed(() => props.field.options ?? [])\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <fieldset\r\n class=\"dcs-form-radio-group\"\r\n role=\"radiogroup\"\r\n :aria-required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n >\r\n <legend class=\"sr-only\">{{ field.label }}</legend>\r\n <label\r\n v-for=\"opt in options\"\r\n :key=\"opt.value\"\r\n class=\"dcs-form-radio\"\r\n >\r\n <input\r\n type=\"radio\"\r\n :name=\"field.id\"\r\n :value=\"opt.value\"\r\n :checked=\"modelValue === opt.value\"\r\n @change=\"emit('update:modelValue', opt.value)\"\r\n />\r\n <span>{{ opt.label }}</span>\r\n </label>\r\n </fieldset>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string[] | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string[]]\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\nconst options = computed(() => props.field.options ?? [])\r\nconst current = computed(() => props.modelValue ?? [])\r\n\r\nfunction toggle(value: string, checked: boolean): void {\r\n const set = new Set(current.value)\r\n if (checked) set.add(value)\r\n else set.delete(value)\r\n emit('update:modelValue', Array.from(set))\r\n}\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <fieldset\r\n class=\"dcs-form-checkbox-group\"\r\n :aria-required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n >\r\n <legend class=\"sr-only\">{{ field.label }}</legend>\r\n <label\r\n v-for=\"opt in options\"\r\n :key=\"opt.value\"\r\n class=\"dcs-form-checkbox\"\r\n >\r\n <input\r\n type=\"checkbox\"\r\n :name=\"`${field.id}[]`\"\r\n :value=\"opt.value\"\r\n :checked=\"current.includes(opt.value)\"\r\n @change=\"toggle(opt.value, ($event.target as HTMLInputElement).checked)\"\r\n />\r\n <span>{{ opt.label }}</span>\r\n </label>\r\n </fieldset>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: boolean | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: boolean]\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <template #label><span class=\"sr-only\">{{ field.label }}</span></template>\r\n <label class=\"dcs-form-checkbox-single\" :for=\"inputId\">\r\n <input\r\n :id=\"inputId\"\r\n type=\"checkbox\"\r\n :name=\"field.id\"\r\n :checked=\"!!modelValue\"\r\n :required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n @change=\"emit('update:modelValue', ($event.target as HTMLInputElement).checked)\"\r\n />\r\n <span>{{ field.label }}<span v-if=\"field.required\" aria-hidden=\"true\"> *</span></span>\r\n </label>\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: string | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: string]\r\n blur: []\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <input\r\n :id=\"inputId\"\r\n class=\"dcs-form-input dcs-form-date\"\r\n type=\"date\"\r\n :name=\"field.id\"\r\n :required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n :value=\"modelValue ?? ''\"\r\n @input=\"emit('update:modelValue', ($event.target as HTMLInputElement).value)\"\r\n @blur=\"emit('blur')\"\r\n />\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport type { PortalFormField } from '../types'\r\nimport DcsFormFieldWrapper from './DcsFormFieldWrapper.vue'\r\n\r\nconst props = defineProps<{\r\n field: PortalFormField\r\n modelValue: File | undefined\r\n error?: string\r\n}>()\r\n\r\nconst emit = defineEmits<{\r\n 'update:modelValue': [value: File | undefined]\r\n}>()\r\n\r\nconst inputId = computed(() => `field-${props.field.id}`)\r\nconst accept = computed(() => (props.field.validation?.accept ?? []).join(','))\r\n\r\nfunction onChange(e: Event): void {\r\n const target = e.target as HTMLInputElement\r\n emit('update:modelValue', target.files?.[0])\r\n}\r\n</script>\r\n\r\n<template>\r\n <DcsFormFieldWrapper :field=\"field\" :error=\"error\" :input-id=\"inputId\">\r\n <input\r\n :id=\"inputId\"\r\n class=\"dcs-form-input dcs-form-file\"\r\n type=\"file\"\r\n :name=\"field.id\"\r\n :required=\"field.required\"\r\n :aria-invalid=\"!!error\"\r\n :accept=\"accept || undefined\"\r\n @change=\"onChange\"\r\n />\r\n </DcsFormFieldWrapper>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport type { PortalFormField } from '../types'\r\n\r\ndefineProps<{\r\n field: PortalFormField\r\n modelValue: string | number | undefined\r\n}>()\r\n</script>\r\n\r\n<template>\r\n <input\r\n type=\"hidden\"\r\n :name=\"field.id\"\r\n :value=\"modelValue ?? (field.defaultValue as string | number | undefined) ?? ''\"\r\n :data-form-field-key=\"field.id\"\r\n />\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport type { PortalFormField } from '../types'\r\n\r\ndefineProps<{\r\n field: PortalFormField\r\n}>()\r\n</script>\r\n\r\n<template>\r\n <div\r\n class=\"dcs-form-section\"\r\n :data-form-field-key=\"field.id\"\r\n >\r\n <slot>\r\n <h3 class=\"dcs-form-section__heading\">{{ field.label }}</h3>\r\n <p v-if=\"field.helpText\" class=\"dcs-form-section__help\">{{ field.helpText }}</p>\r\n </slot>\r\n </div>\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport type { PortalFormField } from '../types'\r\n\r\ndefineProps<{\r\n field: PortalFormField\r\n}>()\r\n</script>\r\n\r\n<template>\r\n <!--\r\n `html` is sanitized server-side per the schema; rendering with v-html\r\n is intentional. Site authors must keep portal-side sanitization on.\r\n -->\r\n <div\r\n class=\"dcs-form-html-block\"\r\n :data-form-field-key=\"field.id\"\r\n v-html=\"field.html ?? ''\"\r\n />\r\n</template>\r\n","<script setup lang=\"ts\">\r\nimport { computed, onMounted, ref, watch } from 'vue'\r\nimport type {\r\n DcsFormSubmitError,\r\n DcsFormSubmitSuccess,\r\n PortalFormDefinition,\r\n PortalFormField,\r\n} from './types'\r\nimport { useDcsForm } from './composables/useDcsForm'\r\nimport { submitFormValues } from './composables/useFormSubmission'\r\nimport { warnIfInvalid } from './schema/validate'\r\nimport { loadFormDefinitions } from './loaders/yaml'\r\n\r\nimport DcsFormText from './fields/DcsFormText.vue'\r\nimport DcsFormTextarea from './fields/DcsFormTextarea.vue'\r\nimport DcsFormSelect from './fields/DcsFormSelect.vue'\r\nimport DcsFormRadio from './fields/DcsFormRadio.vue'\r\nimport DcsFormCheckboxGroup from './fields/DcsFormCheckboxGroup.vue'\r\nimport DcsFormCheckbox from './fields/DcsFormCheckbox.vue'\r\nimport DcsFormDate from './fields/DcsFormDate.vue'\r\nimport DcsFormFile from './fields/DcsFormFile.vue'\r\nimport DcsFormHidden from './fields/DcsFormHidden.vue'\r\nimport DcsFormSection from './fields/DcsFormSection.vue'\r\nimport DcsFormHtmlBlock from './fields/DcsFormHtmlBlock.vue'\r\n\r\ninterface Props {\r\n /** Kebab-case form id; matches `.dcs/forms/<formId>.yaml`. */\r\n formId: string\r\n /** Site slug used for the submission endpoint path. */\r\n siteSlug?: string\r\n /** Override the loaded definition (used by the portal preview iframe). */\r\n definitionOverride?: PortalFormDefinition\r\n /** Override the API base URL; defaults to `VITE_DCS_PUBLIC_API`. */\r\n apiBase?: string\r\n /** Optional captcha token, attached to the submission payload. */\r\n captchaToken?: string\r\n /**\r\n * Optional override for the YAML modules map. Mostly for tests; in\r\n * real apps the build-time `import.meta.glob` call below is used.\r\n */\r\n formsModules?: Record<string, unknown>\r\n}\r\n\r\nconst props = defineProps<Props>()\r\n\r\nconst emit = defineEmits<{\r\n 'submit-success': [event: DcsFormSubmitSuccess]\r\n 'submit-error': [event: DcsFormSubmitError]\r\n 'validation-error': [errors: Record<string, string | undefined>]\r\n}>()\r\n\r\n// Eager glob — Vite inlines every site form at build time. Customer\r\n// sites should configure `vite-plugin-yaml` so YAML is parsed to an\r\n// object; without it the loader falls back to parsing raw strings.\r\nconst eagerFormsModules = (import.meta as unknown as {\r\n glob: (\r\n pattern: string,\r\n opts: { eager: true; import: string; query?: string },\r\n ) => Record<string, unknown>\r\n}).glob('/.dcs/forms/*.yaml', { eager: true, import: 'default' })\r\n\r\nconst definitions = computed(() =>\r\n loadFormDefinitions(props.formsModules ?? eagerFormsModules),\r\n)\r\n\r\nconst definition = computed<PortalFormDefinition | null>(() => {\r\n if (props.definitionOverride) return props.definitionOverride\r\n return definitions.value[props.formId] ?? null\r\n})\r\n\r\nconst isDev = !!(import.meta as unknown as { env?: { DEV?: boolean } }).env?.DEV\r\n\r\nwatch(\r\n definition,\r\n (def) => {\r\n if (def) warnIfInvalid(props.formId, def, isDev)\r\n },\r\n { immediate: true },\r\n)\r\n\r\n// We always call useDcsForm with *some* definition so hooks render\r\n// stably; if the form is missing we'll show an error state instead.\r\nconst safeDefinition = computed<PortalFormDefinition>(\r\n () =>\r\n definition.value ?? {\r\n formId: props.formId,\r\n title: '',\r\n submission: { kind: 'lead' },\r\n fields: [],\r\n },\r\n)\r\n\r\nconst form = useDcsForm({ definition: safeDefinition.value })\r\n\r\n// If the resolved definition changes (e.g. preview iframe updates),\r\n// re-create derived state by resetting.\r\nwatch(\r\n () => safeDefinition.value,\r\n () => form.reset(),\r\n)\r\n\r\nconst apiBase = computed(\r\n () =>\r\n props.apiBase ??\r\n (import.meta as unknown as { env?: { VITE_DCS_PUBLIC_API?: string } }).env\r\n ?.VITE_DCS_PUBLIC_API ??\r\n '',\r\n)\r\n\r\nconst resolvedSiteSlug = computed(\r\n () =>\r\n props.siteSlug ??\r\n (import.meta as unknown as { env?: { VITE_DCS_SITE_SLUG?: string } }).env\r\n ?.VITE_DCS_SITE_SLUG ??\r\n '',\r\n)\r\n\r\nconst submitLabel = computed(\r\n () => safeDefinition.value.submitLabel ?? 'Send',\r\n)\r\n\r\nfunction fieldComponent(field: PortalFormField) {\r\n switch (field.type) {\r\n case 'text':\r\n case 'email':\r\n case 'tel':\r\n return DcsFormText\r\n case 'textarea':\r\n return DcsFormTextarea\r\n case 'select':\r\n case 'multiselect':\r\n return DcsFormSelect\r\n case 'radio':\r\n return DcsFormRadio\r\n case 'checkbox-group':\r\n return DcsFormCheckboxGroup\r\n case 'checkbox':\r\n return DcsFormCheckbox\r\n case 'date':\r\n return DcsFormDate\r\n case 'file':\r\n return DcsFormFile\r\n case 'hidden':\r\n return DcsFormHidden\r\n case 'section-heading':\r\n return DcsFormSection\r\n case 'html-block':\r\n return DcsFormHtmlBlock\r\n default:\r\n return DcsFormText\r\n }\r\n}\r\n\r\nconst formEl = ref<HTMLFormElement | null>(null)\r\n\r\nasync function onSubmit(e: Event): Promise<void> {\r\n e.preventDefault()\r\n if (!definition.value) return\r\n const ok = form.validateAll()\r\n if (!ok) {\r\n emit('validation-error', form.errors.value)\r\n return\r\n }\r\n form.submitting.value = true\r\n form.submitError.value = null\r\n const payload = {\r\n formId: props.formId,\r\n values: form.collectSubmissionValues(),\r\n captchaToken: props.captchaToken,\r\n }\r\n try {\r\n const result = await submitFormValues({\r\n apiBase: apiBase.value,\r\n siteSlug: resolvedSiteSlug.value,\r\n payload,\r\n })\r\n form.submitted.value = true\r\n emit('submit-success', result)\r\n } catch (err) {\r\n const e2 = err as DcsFormSubmitError\r\n form.submitError.value = e2.error?.message ?? 'Submission failed'\r\n emit('submit-error', e2)\r\n } finally {\r\n form.submitting.value = false\r\n }\r\n}\r\n\r\nonMounted(() => {\r\n if (!definition.value) {\r\n // eslint-disable-next-line no-console\r\n console.warn(\r\n `[@duffcloudservices/site-forms] No form definition found for \"${props.formId}\". ` +\r\n `Expected a YAML at /.dcs/forms/${props.formId}.yaml.`,\r\n )\r\n }\r\n})\r\n</script>\r\n\r\n<template>\r\n <div v-if=\"!definition\" class=\"dcs-form dcs-form--missing\" :data-form-key=\"formId\">\r\n <slot name=\"missing\" :form-id=\"formId\">\r\n <p>Form “{{ formId }}” is not configured.</p>\r\n </slot>\r\n </div>\r\n\r\n <div v-else-if=\"form.submitted.value\" class=\"dcs-form dcs-form--success\" :data-form-key=\"formId\">\r\n <slot name=\"success\" :definition=\"definition\">\r\n <p>{{ definition.successMessage ?? 'Thanks — we received your message.' }}</p>\r\n </slot>\r\n </div>\r\n\r\n <form\r\n v-else\r\n ref=\"formEl\"\r\n class=\"dcs-form\"\r\n :data-form-key=\"formId\"\r\n novalidate\r\n @submit=\"onSubmit\"\r\n >\r\n <slot name=\"header\" :definition=\"definition\">\r\n <header class=\"dcs-form__header\">\r\n <h2 class=\"dcs-form__title\">{{ definition.title }}</h2>\r\n <p v-if=\"definition.description\" class=\"dcs-form__description\">\r\n {{ definition.description }}\r\n </p>\r\n </header>\r\n </slot>\r\n\r\n <div v-if=\"form.steps.value\" class=\"dcs-form__progress\" aria-live=\"polite\">\r\n <slot\r\n name=\"progress\"\r\n :current=\"form.currentStepIndex.value\"\r\n :total=\"form.steps.value.length\"\r\n :step=\"form.currentStep.value\"\r\n >\r\n <p>\r\n Step {{ form.currentStepIndex.value + 1 }} of\r\n {{ form.steps.value.length }}\r\n <template v-if=\"form.currentStep.value\">\r\n — {{ form.currentStep.value.title }}\r\n </template>\r\n </p>\r\n </slot>\r\n </div>\r\n\r\n <div class=\"dcs-form__fields\">\r\n <component\r\n :is=\"fieldComponent(field)\"\r\n v-for=\"field in form.visibleFields.value\"\r\n :key=\"field.id\"\r\n :field=\"field\"\r\n :model-value=\"form.values[field.id]\"\r\n :error=\"form.errors.value[field.id]\"\r\n @update:model-value=\"(v: unknown) => form.setValue(field.id, v)\"\r\n @blur=\"form.touch(field.id)\"\r\n />\r\n </div>\r\n\r\n <div v-if=\"form.submitError.value\" class=\"dcs-form__submit-error\" role=\"alert\">\r\n {{ form.submitError.value }}\r\n </div>\r\n\r\n <div class=\"dcs-form__actions\">\r\n <slot\r\n name=\"actions\"\r\n :is-first-step=\"form.isFirstStep.value\"\r\n :is-last-step=\"form.isLastStep.value\"\r\n :submitting=\"form.submitting.value\"\r\n :prev=\"form.prev\"\r\n :next=\"form.next\"\r\n >\r\n <button\r\n v-if=\"form.steps.value && !form.isFirstStep.value\"\r\n type=\"button\"\r\n class=\"dcs-form__btn dcs-form__btn--prev\"\r\n @click=\"form.prev()\"\r\n >\r\n Previous\r\n </button>\r\n <button\r\n v-if=\"form.steps.value && !form.isLastStep.value\"\r\n type=\"button\"\r\n class=\"dcs-form__btn dcs-form__btn--next\"\r\n @click=\"form.next()\"\r\n >\r\n Next\r\n </button>\r\n <button\r\n v-if=\"!form.steps.value || form.isLastStep.value\"\r\n type=\"submit\"\r\n class=\"dcs-form__btn dcs-form__btn--submit\"\r\n :disabled=\"form.submitting.value\"\r\n >\r\n {{ form.submitting.value ? 'Sending…' : submitLabel }}\r\n </button>\r\n </slot>\r\n </div>\r\n </form>\r\n</template>\r\n"],"names":["isFieldVisible","field","values","validateField","value","isEmpty","v","file","a","validateForm","def","fieldIds","errors","ids","err","hasErrors","e","defaultsFor","out","f","useDcsForm","opts","definition","reactive","ref","touched","submitting","submitted","submitError","currentStepIndex","fields","computed","steps","currentStep","s","currentStepFields","step","isFirstStep","isLastStep","visibleFields","setValue","id","touch","validateScope","scopeFieldIds","next","merged","validateCurrentScope","scope","validateAll","prev","reset","fresh","k","collectSubmissionValues","submitFormValues","apiBase","siteSlug","payload","retries","fetchImpl","url","hasFile","lastError","lastStatus","attempt","init","buildFormData","res","body","safeJson","safeText","fd","key","item","validator","getValidator","ajv","Ajv","addFormats","schema","validateFormDefinition","valid","warnIfInvalid","formId","isDev","result","loadFormDefinitions","modules","path","mod","extractFormId","parseModule","yaml","maybeDefault","parseFormYaml","raw","_createElementBlock","_normalizeClass","__props","_renderSlot","_ctx","_hoisted_3","_openBlock","_hoisted_4","_toDisplayString","_hoisted_5","props","emit","__emit","inputId","inputType","_createBlock","DcsFormFieldWrapper","_createElementVNode","$event","isMulti","options","onChange","sel","opt","_hoisted_2","_Fragment","_renderList","current","toggle","checked","set","_hoisted_1","_createTextVNode","accept","target","eagerFormsModules","definitions","watch","safeDefinition","form","__vite_import_meta_env__","resolvedSiteSlug","submitLabel","fieldComponent","DcsFormText","DcsFormTextarea","DcsFormSelect","DcsFormRadio","DcsFormCheckboxGroup","DcsFormCheckbox","DcsFormDate","DcsFormFile","DcsFormHidden","DcsFormSection","DcsFormHtmlBlock","formEl","onSubmit","e2","onMounted","_unref","_hoisted_6","_hoisted_7","_hoisted_8","_resolveDynamicComponent","_hoisted_9","_hoisted_10","_cache","_hoisted_11"],"mappings":";;;;AAWO,SAASA,EACdC,GACAC,GACS;AACT,SAAKD,EAAM,YACKC,EAAOD,EAAM,UAAU,OAAO,MAC3BA,EAAM,UAAU,SAFN;AAG/B;AAOO,SAASE,GACdF,GACAG,GACoB;AACpB,MACEH,EAAM,SAAS,qBACfA,EAAM,SAAS,gBACfA,EAAM,SAAS;AAEf;AAGF,QAAMI,IAEJD,KAAU,QACVA,MAAU,MACT,MAAM,QAAQA,CAAK,KAAKA,EAAM,WAAW;AAE5C,MAAIH,EAAM,YAAYI;AACpB,WAAO,GAAGJ,EAAM,KAAK;AAEvB,MAAII,EAAS;AAGb,MAAIJ,EAAM,SAAS,WAAW,OAAOG,KAAU,YACzC,CAAC,6BAA6B,KAAKA,CAAK;AAC1C,WAAO,GAAGH,EAAM,KAAK;AAIzB,QAAMK,IAAIL,EAAM;AAChB,MAAKK,GAEL;AAAA,QAAI,OAAOF,KAAU,UAAU;AAC7B,UAAIE,EAAE,aAAa,QAAQF,EAAM,SAASE,EAAE;AAC1C,eAAO,GAAGL,EAAM,KAAK,qBAAqBK,EAAE,SAAS;AAEvD,UAAIA,EAAE,aAAa,QAAQF,EAAM,SAASE,EAAE;AAC1C,eAAO,GAAGL,EAAM,KAAK,oBAAoBK,EAAE,SAAS;AAEtD,UAAIA,EAAE;AACJ,YAAI;AACF,cAAI,CAAC,IAAI,OAAOA,EAAE,KAAK,EAAE,KAAKF,CAAK;AACjC,mBAAO,GAAGH,EAAM,KAAK;AAAA,QAEzB,QAAQ;AAAA,QAER;AAAA,IAEJ;AAEA,QAAI,OAAOG,KAAU,UAAU;AAC7B,UAAIE,EAAE,OAAO,QAAQF,IAAQE,EAAE;AAC7B,eAAO,GAAGL,EAAM,KAAK,qBAAqBK,EAAE,GAAG;AAEjD,UAAIA,EAAE,OAAO,QAAQF,IAAQE,EAAE;AAC7B,eAAO,GAAGL,EAAM,KAAK,oBAAoBK,EAAE,GAAG;AAAA,IAElD;AAEA,QAAIL,EAAM,SAAS,UAAUK,EAAE,UAAUA,EAAE,OAAO,SAAS,GAAG;AAC5D,YAAMC,IAAOH;AACb,UAAIG,KAAQ,OAAO,OAAS,OAAeA,aAAgB,QAOrD,CANaD,EAAE,OAAO,KAAK,CAACE,MAC1BA,EAAE,WAAW,GAAG,IACXD,EAAK,KAAK,YAAA,EAAc,SAASC,EAAE,aAAa,IAElDD,EAAK,SAASC,CACtB;AAEC,eAAO,GAAGP,EAAM,KAAK,oBAAoBK,EAAE,OAAO,KAAK,IAAI,CAAC;AAAA,IAGlE;AAAA;AAGF;AAOO,SAASG,GACdC,GACAR,GACAS,GACY;AACZ,QAAMC,IAAqB,CAAA,GACrBC,IAAMF,IAAW,IAAI,IAAIA,CAAQ,IAAI;AAC3C,aAAWV,KAASS,EAAI,QAAQ;AAE9B,QADIG,KAAO,CAACA,EAAI,IAAIZ,EAAM,EAAE,KACxB,CAACD,EAAeC,GAAOC,CAAM,EAAG;AACpC,UAAMY,IAAMX,GAAcF,GAAOC,EAAOD,EAAM,EAAE,CAAC;AACjD,IAAIa,MAAKF,EAAOX,EAAM,EAAE,IAAIa;AAAA,EAC9B;AACA,SAAOF;AACT;AAEO,SAASG,GAAUH,GAA6B;AACrD,SAAO,OAAO,OAAOA,CAAM,EAAE,KAAK,CAACI,MAAM,CAAC,CAACA,CAAC;AAC9C;AC7EA,SAASC,EAAYP,GAAuC;AAC1D,QAAMQ,IAAkB,CAAA;AACxB,aAAWC,KAAKT,EAAI;AAClB,IAAIS,EAAE,iBAAiB,SACrBD,EAAIC,EAAE,EAAE,IAAIA,EAAE,eACLA,EAAE,SAAS,aACpBD,EAAIC,EAAE,EAAE,IAAI,KACHA,EAAE,SAAS,iBAAiBA,EAAE,SAAS,mBAChDD,EAAIC,EAAE,EAAE,IAAI,CAAA,IAEZD,EAAIC,EAAE,EAAE,IAAI;AAGhB,SAAOD;AACT;AAEO,SAASE,GAAWC,GAA2C;AACpE,QAAM,EAAE,YAAAC,MAAeD,GACjBnB,IAASqB,EAAqBN,EAAYK,CAAU,CAAC,GACrDV,IAASY,EAAgB,EAAE,GAC3BC,IAAUD,EAA6B,EAAE,GACzCE,IAAaF,EAAI,EAAK,GACtBG,IAAYH,EAAI,EAAK,GACrBI,IAAcJ,EAAmB,IAAI,GACrCK,IAAmBL,EAAI,CAAC,GAExBM,IAASC,EAAS,MAAMT,EAAW,MAAM,GACzCU,IAAQD;AAAA,IAAkC,MAC9CT,EAAW,SAASA,EAAW,MAAM,SAAS,IAAIA,EAAW,QAAQ;AAAA,EAAA,GAEjEW,IAAcF,EAAgC,MAAM;AACxD,UAAMG,IAAIF,EAAM;AAChB,WAAOE,IAAKA,EAAEL,EAAiB,KAAK,KAAK,OAAQ;AAAA,EACnD,CAAC,GACKM,IAAoBJ,EAA4B,MAAM;AAC1D,UAAMK,IAAOH,EAAY;AACzB,QAAI,CAACG,EAAM,QAAOd,EAAW;AAC7B,UAAMT,IAAM,IAAI,IAAIuB,EAAK,QAAQ;AACjC,WAAOd,EAAW,OAAO,OAAO,CAACH,MAAMN,EAAI,IAAIM,EAAE,EAAE,CAAC;AAAA,EACtD,CAAC,GACKkB,IAAcN,EAAS,MAAMF,EAAiB,UAAU,CAAC,GACzDS,IAAaP,EAAS,MAAM;AAChC,UAAMG,IAAIF,EAAM;AAChB,WAAO,CAACE,KAAKL,EAAiB,SAASK,EAAE,SAAS;AAAA,EACpD,CAAC,GACKK,IAAgBR;AAAA,IAAS,MAC7BI,EAAkB,MAAM,OAAO,CAAChB,MAAMnB,EAAemB,GAAGjB,CAAoB,CAAC;AAAA,EAAA;AAG/E,WAASsC,EAASC,GAAYrC,GAAsB;AAChD,IAAAF,EAAsBuC,CAAE,IAAIrC,GAC1BQ,EAAO,MAAM6B,CAAE,MACjB7B,EAAO,QAAQ,EAAE,GAAGA,EAAO,OAAO,CAAC6B,CAAE,GAAG,OAAA;AAAA,EAE5C;AAEA,WAASC,EAAMD,GAAkB;AAC/B,IAAAhB,EAAQ,QAAQ,EAAE,GAAGA,EAAQ,OAAO,CAACgB,CAAE,GAAG,GAAA;AAAA,EAC5C;AAEA,WAASE,EAAcC,GAAmC;AACxD,UAAMC,IAAOpC,GAAaa,GAAYpB,GAAsB0C,CAAa;AAGzE,QAAIA,GAAe;AACjB,YAAME,IAAS,EAAE,GAAGlC,EAAO,MAAA;AAC3B,iBAAW6B,KAAMG;AACf,QAAAE,EAAOL,CAAE,IAAII,EAAKJ,CAAE;AAEtB,MAAA7B,EAAO,QAAQkC;AAAA,IACjB;AACE,MAAAlC,EAAO,QAAQiC;AAEjB,WAAO,CAAC9B,GAAUH,EAAO,KAAK;AAAA,EAChC;AAEA,WAASmC,IAAgC;AACvC,UAAMC,IAAQf,EAAY,OAAO;AACjC,WAAOU,EAAcK,CAAK;AAAA,EAC5B;AAEA,WAASC,IAAuB;AAC9B,WAAON,EAAA;AAAA,EACT;AAEA,WAASE,IAAgB;AAEvB,WADI,CAACb,EAAM,SACP,CAACe,EAAA,IAA+B,KAChClB,EAAiB,QAAQG,EAAM,MAAM,SAAS,KAChDH,EAAiB,SACV,MAEF;AAAA,EACT;AAEA,WAASqB,IAAa;AACpB,IAAIrB,EAAiB,QAAQ,KAAGA,EAAiB;AAAA,EACnD;AAEA,WAASsB,IAAc;AACrB,UAAMC,IAAQnC,EAAYK,CAAU;AACpC,eAAW+B,KAAK,OAAO,KAAKnD,CAAgB;AAC1C,aAAQA,EAAsBmD,CAAC;AAEjC,WAAO,OAAOnD,GAAsBkD,CAAK,GACzCxC,EAAO,QAAQ,CAAA,GACfa,EAAQ,QAAQ,CAAA,GAChBC,EAAW,QAAQ,IACnBC,EAAU,QAAQ,IAClBC,EAAY,QAAQ,MACpBC,EAAiB,QAAQ;AAAA,EAC3B;AAEA,WAASyB,IAAsC;AAC7C,UAAMpC,IAAkB,CAAA;AACxB,eAAWC,KAAKG,EAAW;AACzB,MACEH,EAAE,SAAS,qBACXA,EAAE,SAAS,gBAGRnB,EAAemB,GAAGjB,CAAoB,MAC3CgB,EAAIC,EAAE,EAAE,IAAKjB,EAAsBiB,EAAE,EAAE;AAEzC,WAAOD;AAAA,EACT;AAEA,SAAO;AAAA,IACL,QAAAhB;AAAA,IACA,QAAAU;AAAA,IACA,SAAAa;AAAA,IACA,YAAAC;AAAA,IACA,WAAAC;AAAA,IACA,aAAAC;AAAA,IACA,QAAAE;AAAA,IACA,OAAAE;AAAA,IACA,kBAAAH;AAAA,IACA,aAAAI;AAAA,IACA,mBAAAE;AAAA,IACA,aAAAE;AAAA,IACA,YAAAC;AAAA,IACA,eAAAC;AAAA,IACA,UAAAC;AAAA,IACA,OAAAE;AAAA,IACA,sBAAAK;AAAA,IACA,aAAAE;AAAA,IACA,MAAAJ;AAAA,IACA,MAAAK;AAAA,IACA,OAAAC;AAAA,IACA,yBAAAG;AAAA,EAAA;AAEJ;ACjLA,eAAsBC,GACpBlC,GAC+B;AAC/B,QAAM,EAAE,SAAAmC,GAAS,UAAAC,GAAU,SAAAC,EAAA,IAAYrC,GACjCsC,IAAUtC,EAAK,WAAW,GAC1BuC,IAAYvC,EAAK,aAAa,OAC9BwC,IAAM,GAAGL,EAAQ,QAAQ,OAAO,EAAE,CAAC,iBAAiB;AAAA,IACxDC;AAAA,EAAA,CACD,qBAEKK,IAAU,OAAO,OAAOJ,EAAQ,MAAM,EAAE;AAAA,IAC5C,CAACpD,MAAM,OAAO,OAAS,OAAeA,aAAa;AAAA,EAAA;AAGrD,MAAIyD,GACAC;AACJ,WAASC,IAAU,GAAGA,KAAWN,GAASM;AACxC,QAAI;AACF,YAAMC,IAAoBJ,IACtB,EAAE,QAAQ,QAAQ,MAAMK,GAAcT,CAAO,MAC7C;AAAA,QACE,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAA;AAAA,QAC3B,MAAM,KAAK,UAAUA,CAAO;AAAA,MAAA,GAE5BU,IAAM,MAAMR,EAAUC,GAAKK,CAAI;AAErC,UADAF,IAAaI,EAAI,QACbA,EAAI,IAAI;AACV,cAAMC,IAAO,MAAMC,GAASF,CAAG;AAC/B,eAAO,EAAE,SAAAV,GAAS,UAAUW,EAAA;AAAA,MAC9B;AAEA,UAAID,EAAI,SAAS,KAAK;AACpB,cAAMC,IAAO,MAAME,GAASH,CAAG;AAM/B,cALgC;AAAA,UAC9B,SAAAV;AAAA,UACA,QAAQU,EAAI;AAAA,UACZ,OAAO,IAAI,MAAM,sBAAsBA,EAAI,MAAM,MAAMC,CAAI,EAAE;AAAA,QAAA;AAAA,MAGjE;AACA,MAAAN,IAAY,IAAI,MAAM,gBAAgBK,EAAI,MAAM,EAAE;AAAA,IACpD,SAASpD,GAAG;AAEV,UAAIA,KAAK,OAAOA,KAAM,YAAY,aAAaA,KAAK,WAAWA;AAC7D,cAAMA;AAER,MAAA+C,IAAY/C;AAAA,IACd;AAGF,QAAM;AAAA,IACJ,SAAA0C;AAAA,IACA,QAAQM;AAAA,IACR,OAAOD,KAAa,IAAI,MAAM,mBAAmB;AAAA,EAAA;AAErD;AAEA,SAASI,GAAcT,GAAyC;AAC9D,QAAMc,IAAK,IAAI,SAAA;AACf,EAAAA,EAAG,OAAO,UAAUd,EAAQ,MAAM,GAC9BA,EAAQ,gBAAcc,EAAG,OAAO,gBAAgBd,EAAQ,YAAY;AACxE,aAAW,CAACe,GAAKrE,CAAK,KAAK,OAAO,QAAQsD,EAAQ,MAAM;AACtD,QAA2BtD,KAAU;AACrC,UAAIA,aAAiB;AACnB,QAAAoE,EAAG,OAAO,UAAUC,CAAG,KAAKrE,CAAK;AAAA,eACxB,MAAM,QAAQA,CAAK;AAC5B,mBAAWsE,KAAQtE,EAAO,CAAAoE,EAAG,OAAO,UAAUC,CAAG,OAAO,OAAOC,CAAI,CAAC;AAAA;AAEpE,QAAAF,EAAG,OAAO,UAAUC,CAAG,KAAK,OAAOrE,CAAK,CAAC;AAG7C,SAAOoE;AACT;AAEA,eAAeF,GAASF,GAAiC;AACvD,MAAI;AACF,WAAO,MAAMA,EAAI,KAAA;AAAA,EACnB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,eAAeG,GAASH,GAAgC;AACtD,MAAI;AACF,WAAO,MAAMA,EAAI,KAAA;AAAA,EACnB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;;;;;;;;;;;ACjGA,IAAIO,IAAqC;AAEzC,SAASC,KAAiC;AACxC,MAAID,EAAW,QAAOA;AACtB,QAAME,IAAM,IAAIC,GAAI;AAAA,IAClB,WAAW;AAAA,IACX,QAAQ;AAAA,IACR,eAAe;AAAA,EAAA,CAChB;AACD,SAAAC,GAAWF,CAAG,GACdF,IAAYE,EAAI,QAAQG,EAAgB,GACjCL;AACT;AAOO,SAASM,GACdvE,GACwB;AACxB,QAAMJ,IAAIsE,GAAA,GACJM,IAAQ5E,EAAEI,CAAG;AACnB,SAAO,EAAE,OAAAwE,GAAO,QAAQA,IAAQ,CAAA,IAAM5E,EAAE,UAAU,GAAC;AACrD;AAGO,SAAS6E,GACdC,GACA1E,GACA2E,GACwB;AACxB,QAAMC,IAASL,GAAuBvE,CAAG;AACzC,SAAI,CAAC4E,EAAO,SAASD,KAEnB,QAAQ;AAAA,IACN,yCAAyCD,CAAM;AAAA,IAC/CE,EAAO;AAAA,EAAA,GAGJA;AACT;AC9CO,SAASC,GACdC,GACsC;AACtC,QAAMtE,IAA4C,CAAA;AAClD,aAAW,CAACuE,GAAMC,CAAG,KAAK,OAAO,QAAQF,CAAO,GAAG;AACjD,UAAMJ,IAASO,GAAcF,CAAI,GAC3B/E,IAAMkF,GAAYF,CAAG;AAC3B,IAAIhF,MACFQ,EAAIR,EAAI,UAAU0E,CAAM,IAAI1E;AAAA,EAEhC;AACA,SAAOQ;AACT;AAEA,SAASyE,GAAcF,GAAsB;AAE3C,UADaA,EAAK,MAAM,GAAG,EAAE,SAASA,GAC1B,QAAQ,aAAa,EAAE;AACrC;AAEA,SAASG,GAAYF,GAA2C;AAC9D,MAAI,CAACA,EAAK,QAAO;AACjB,MAAI,OAAOA,KAAQ;AACjB,WAAOG,EAAK,KAAKH,CAAG;AAEtB,MAAI,OAAOA,KAAQ,UAAU;AAG3B,UAAMI,IAAgBJ,EAA8B;AACpD,WAAII,KAAgB,OAAOA,KAAiB,WACnCA,IAEFJ;AAAA,EACT;AACA,SAAO;AACT;AAGO,SAASK,GAAcC,GAAmC;AAC/D,SAAOH,EAAK,KAAKG,CAAG;AACtB;;;;;;;;;;;;;;;;;;;;2BCtCEC,EAyBM,OAAA;AAAA,MAxBJ,OAAKC,EAAA,CAAC,kBAAgB,CAAA,mBACMC,EAAA,MAAM,IAAI,IAAA,yBAA6BA,EAAA,MAAM,SAAK,MAAA,EAAA,CAAA,CAAA;AAAA,MAC7E,uBAAqBA,EAAA,MAAM;AAAA,IAAA;MAE5BC,EASOC,uBATP,MASO;AAAA,QAPGF,EAAA,MAAM,SAAI,cAAmBA,EAAA,MAAM,SAAI,YAAiBA,EAAA,MAAM,SAAI,qBAA0BA,EAAA,MAAM,SAAI,qBAD9GF,EAOQ,SAAA;AAAA;UALL,KAAKE,EAAA,WAAO,SAAaA,EAAA,MAAM,EAAE;AAAA,UAClC,OAAM;AAAA,QAAA;cAEHA,EAAA,MAAM,KAAK,IAAG,KACjB,CAAA;AAAA,UAAYA,EAAA,MAAM,iBAAlBF,EAAwF,QAAxFK,IAAgF,GAAC;;;MAIrFF,EAAQC,EAAA,QAAA,SAAA;AAAA,MAERD,EAEOC,sBAFP,MAEO;AAAA,QADIF,EAAA,MAAM,YAAfI,EAAA,GAAAN,EAA8E,KAA9EO,IAA8EC,EAArBN,EAAA,MAAM,QAAQ,GAAA,CAAA;;MAGzEC,EAEOC,uBAFP,MAEO;AAAA,QADIF,EAAA,cAATF,EAA0E,KAA1ES,IAA0ED,EAAZN,EAAA,KAAK,GAAA,CAAA;;;;;;;;;;;;;AC9BzE,UAAMQ,IAAQR,GAMRS,IAAOC,GAKPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE,GAClDI,IAAYhF,EAAS,MAAM;AAC/B,cAAQ4E,EAAM,MAAM,MAAA;AAAA,QAClB,KAAK;AACH,iBAAO;AAAA,QACT,KAAK;AACH,iBAAO;AAAA,QACT;AACE,iBAAO;AAAA,MAAA;AAAA,IAEb,CAAC;2BAICK,EAsBsBC,GAAA;AAAA,MAtBA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAoBO;AAAA,QApBPV,EAoBOC,EAAA,QAAA,SAAA;AAAA,UAlBJ,IAAIS,EAAA;AAAA,UACJ,OAAOX,EAAA;AAAA,UACP,SAAQ,CAAGnF,MAAa4F,uBAA2B5F,EAAE,OAA4B,KAAK;AAAA,UACtF,cAAe4F,EAAI,MAAA;AAAA,QAAA,GALtB,MAoBO;AAAA,UAbLM,EAYE,SAAA;AAAA,YAXC,IAAIJ,EAAA;AAAA,YACL,OAAM;AAAA,YACL,MAAMC,EAAA;AAAA,YACN,MAAMZ,EAAA,MAAM;AAAA,YACZ,OAAOA,EAAA,cAAU;AAAA,YACjB,aAAaA,EAAA,MAAM;AAAA,YACnB,UAAUA,EAAA,MAAM;AAAA,YAChB,kBAAgBA,EAAA;AAAA,YAChB,oBAAkBA,EAAA,QAAK,GAAMW,EAAA,KAAO,WAAW;AAAA,YAC/C,gCAAOF,EAAI,qBAAuBO,EAAO,OAA4B,KAAK;AAAA,YAC1E,+BAAMP,EAAI,MAAA;AAAA,UAAA;;;;;;;;;;;;;;;AC5CnB,UAAMD,IAAQR,GAMRS,IAAOC,GAKPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE;2BAItDK,EAqBsBC,GAAA;AAAA,MArBA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAmBO;AAAA,QAnBPV,EAmBOC,EAAA,QAAA,SAAA;AAAA,UAjBJ,IAAIS,EAAA;AAAA,UACJ,OAAOX,EAAA;AAAA,UACP,SAAQ,CAAGnF,MAAa4F,uBAA2B5F,EAAE,OAA+B,KAAK;AAAA,UACzF,cAAe4F,EAAI,MAAA;AAAA,QAAA,GALtB,MAmBO;AAAA,UAZLM,EAWE,YAAA;AAAA,YAVC,IAAIJ,EAAA;AAAA,YACL,OAAM;AAAA,YACL,MAAMX,EAAA,MAAM;AAAA,YACZ,aAAaA,EAAA,MAAM;AAAA,YACnB,UAAUA,EAAA,MAAM;AAAA,YAChB,kBAAgBA,EAAA;AAAA,YACjB,MAAK;AAAA,YACJ,OAAOA,EAAA,cAAU;AAAA,YACjB,gCAAOS,EAAI,qBAAuBO,EAAO,OAA+B,KAAK;AAAA,YAC7E,+BAAMP,EAAI,MAAA;AAAA,UAAA;;;;;;;;;;;;;;;;;;;ACjCnB,UAAMD,IAAQR,GAMRS,IAAOC,GAKPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE,GAClDS,IAAUrF,EAAS,MAAM4E,EAAM,MAAM,SAAS,aAAa,GAC3DU,IAAUtF,EAAS,MAAM4E,EAAM,MAAM,WAAW,EAAE;AAExD,aAASW,EAAStG,GAAgB;AAChC,YAAMuG,IAAMvG,EAAE;AACd,UAAIoG,EAAQ,OAAO;AACjB,cAAMlH,IAAmB,CAAA;AACzB,mBAAWsH,KAAO,MAAM,KAAKD,EAAI,eAAe,EAAG,CAAArH,EAAO,KAAKsH,EAAI,KAAK;AACxE,QAAAZ,EAAK,qBAAqB1G,CAAM;AAAA,MAClC;AACE,QAAA0G,EAAK,qBAAqBW,EAAI,KAAK;AAAA,IAEvC;2BAIEP,EA2BsBC,GAAA;AAAA,MA3BA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAyBO;AAAA,QAzBPV,EAyBOC,EAAA,QAAA,SAAA;AAAA,UAvBJ,IAAIS,EAAA;AAAA,UACJ,OAAOX,EAAA;AAAA,UACP,SAASkB,EAAA;AAAA,UACT,UAAAC;AAAA,QAAA,GALH,MAyBO;AAAA,UAlBLJ,EAiBS,UAAA;AAAA,YAhBN,IAAIJ,EAAA;AAAA,YACL,OAAM;AAAA,YACL,MAAMX,EAAA,MAAM;AAAA,YACZ,UAAUA,EAAA,MAAM;AAAA,YAChB,UAAUiB,EAAA;AAAA,YACV,kBAAgBjB,EAAA;AAAA,YAChB,OAAOA,EAAA,eAAeiB,EAAA,QAAO,CAAA,IAAA;AAAA,YAC7B,UAAAE;AAAA,YACA,+BAAMV,EAAI,MAAA;AAAA,UAAA;YAEIQ,EAAA,qBAAfb,KAAAN,EAES,UAFTwB,IAEShB,EADJN,EAAA,MAAM,eAAW,SAAA,GAAA,CAAA;AAAA,oBAEtBF,EAESyB,GAAA,MAAAC,EAFaN,EAAA,OAAO,CAAdG,YAAfvB,EAES,UAAA;AAAA,cAFuB,KAAKuB,EAAI;AAAA,cAAQ,OAAOA,EAAI;AAAA,YAAA,GACvDf,EAAAe,EAAI,KAAK,GAAA,GAAAlB,EAAA;;;;;;;;;;;;;;;;ACnDtB,UAAMK,IAAQR,GAMRS,IAAOC,GAIPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE,GAClDU,IAAUtF,EAAS,MAAM4E,EAAM,MAAM,WAAW,EAAE;2BAItDK,EAuBsBC,GAAA;AAAA,MAvBA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAqBW;AAAA,QArBXI,EAqBW,YAAA;AAAA,UApBT,OAAM;AAAA,UACN,MAAK;AAAA,UACJ,iBAAef,EAAA,MAAM;AAAA,UACrB,kBAAgBA,EAAA;AAAA,QAAA;UAEjBe,EAAkD,UAAlDO,IAAkDhB,EAAvBN,EAAA,MAAM,KAAK,GAAA,CAAA;AAAA,kBACtCF,EAaQyB,GAAA,MAAAC,EAZQN,EAAA,OAAO,CAAdG,YADTvB,EAaQ,SAAA;AAAA,YAXL,KAAKuB,EAAI;AAAA,YACV,OAAM;AAAA,UAAA;YAENN,EAME,SAAA;AAAA,cALA,MAAK;AAAA,cACJ,MAAMf,EAAA,MAAM;AAAA,cACZ,OAAOqB,EAAI;AAAA,cACX,SAASrB,EAAA,eAAeqB,EAAI;AAAA,cAC5B,UAAM,CAAAL,MAAEP,EAAI,qBAAsBY,EAAI,KAAK;AAAA,YAAA;YAE9CN,EAA4B,QAAA,MAAAT,EAAnBe,EAAI,KAAK,GAAA,CAAA;AAAA,UAAA;;;;;;;;;;;;;;;ACnC1B,UAAMb,IAAQR,GAMRS,IAAOC,GAIPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE,GAClDU,IAAUtF,EAAS,MAAM4E,EAAM,MAAM,WAAW,EAAE,GAClDiB,IAAU7F,EAAS,MAAM4E,EAAM,cAAc,CAAA,CAAE;AAErD,aAASkB,EAAOzH,GAAe0H,GAAwB;AACrD,YAAMC,IAAM,IAAI,IAAIH,EAAQ,KAAK;AACjC,MAAIE,IAASC,EAAI,IAAI3H,CAAK,IACrB2H,EAAI,OAAO3H,CAAK,GACrBwG,EAAK,qBAAqB,MAAM,KAAKmB,CAAG,CAAC;AAAA,IAC3C;2BAIEf,EAsBsBC,GAAA;AAAA,MAtBA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAoBW;AAAA,QApBXI,EAoBW,YAAA;AAAA,UAnBT,OAAM;AAAA,UACL,iBAAef,EAAA,MAAM;AAAA,UACrB,kBAAgBA,EAAA;AAAA,QAAA;UAEjBe,EAAkD,UAAlDO,IAAkDhB,EAAvBN,EAAA,MAAM,KAAK,GAAA,CAAA;AAAA,kBACtCF,EAaQyB,GAAA,MAAAC,EAZQN,EAAA,OAAO,CAAdG,YADTvB,EAaQ,SAAA;AAAA,YAXL,KAAKuB,EAAI;AAAA,YACV,OAAM;AAAA,UAAA;YAENN,EAME,SAAA;AAAA,cALA,MAAK;AAAA,cACJ,MAAI,GAAKf,EAAA,MAAM,EAAE;AAAA,cACjB,OAAOqB,EAAI;AAAA,cACX,SAASI,EAAA,MAAQ,SAASJ,EAAI,KAAK;AAAA,cACnC,UAAM,CAAAL,MAAEU,EAAOL,EAAI,OAAQL,EAAO,OAA4B,OAAO;AAAA,YAAA;YAExED,EAA4B,QAAA,MAAAT,EAAnBe,EAAI,KAAK,GAAA,CAAA;AAAA,UAAA;;;;;;;;;;;;;;;;;;AC1C1B,UAAMb,IAAQR,GAMRS,IAAOC,GAIPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE;2BAItDK,EAcsBC,GAAA;AAAA,MAdA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;MACjD,SAAM,MAA8C;AAAA,QAA9CI,EAA8C,QAA9Cc,IAA8CvB,EAArBN,EAAA,MAAM,KAAK,GAAA,CAAA;AAAA,MAAA;iBACrD,MAWQ;AAAA,QAXRe,EAWQ,SAAA;AAAA,UAXD,OAAM;AAAA,UAA4B,KAAKJ,EAAA;AAAA,QAAA;UAC5CI,EAQE,SAAA;AAAA,YAPC,IAAIJ,EAAA;AAAA,YACL,MAAK;AAAA,YACJ,MAAMX,EAAA,MAAM;AAAA,YACZ,WAAWA,EAAA;AAAA,YACX,UAAUA,EAAA,MAAM;AAAA,YAChB,kBAAgBA,EAAA;AAAA,YAChB,iCAAQS,EAAI,qBAAuBO,EAAO,OAA4B,OAAO;AAAA,UAAA;UAEhFD,EAAsF,QAAA,MAAA;AAAA,YAA7Ee,EAAAxB,EAAAN,EAAA,MAAM,KAAK,GAAA,CAAA;AAAA,YAAeA,EAAA,MAAM,iBAAlBF,EAAwD,QAAxDO,IAA+C,IAAE;;;;;;;;;;;;;;;;AC1B9E,UAAMG,IAAQR,GAMRS,IAAOC,GAKPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE;2BAItDK,EAYsBC,GAAA;AAAA,MAZA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MAUE;AAAA,QAVFI,EAUE,SAAA;AAAA,UATC,IAAIJ,EAAA;AAAA,UACL,OAAM;AAAA,UACN,MAAK;AAAA,UACJ,MAAMX,EAAA,MAAM;AAAA,UACZ,UAAUA,EAAA,MAAM;AAAA,UAChB,kBAAgBA,EAAA;AAAA,UAChB,OAAOA,EAAA,cAAU;AAAA,UACjB,gCAAOS,EAAI,qBAAuBO,EAAO,OAA4B,KAAK;AAAA,UAC1E,+BAAMP,EAAI,MAAA;AAAA,QAAA;;;;;;;;;;;;;;ACzBjB,UAAMD,IAAQR,GAMRS,IAAOC,GAIPC,IAAU/E,EAAS,MAAM,SAAS4E,EAAM,MAAM,EAAE,EAAE,GAClDuB,IAASnG,EAAS,OAAO4E,EAAM,MAAM,YAAY,UAAU,CAAA,GAAI,KAAK,GAAG,CAAC;AAE9E,aAASW,EAAStG,GAAgB;AAChC,YAAMmH,IAASnH,EAAE;AACjB,MAAA4F,EAAK,qBAAqBuB,EAAO,QAAQ,CAAC,CAAC;AAAA,IAC7C;2BAIEnB,EAWsBC,GAAA;AAAA,MAXA,OAAOd,EAAA;AAAA,MAAQ,OAAOA,EAAA;AAAA,MAAQ,YAAUW,EAAA;AAAA,IAAA;iBAC5D,MASE;AAAA,QATFI,EASE,SAAA;AAAA,UARC,IAAIJ,EAAA;AAAA,UACL,OAAM;AAAA,UACN,MAAK;AAAA,UACJ,MAAMX,EAAA,MAAM;AAAA,UACZ,UAAUA,EAAA,MAAM;AAAA,UAChB,kBAAgBA,EAAA;AAAA,UAChB,QAAQ+B,EAAA,SAAU;AAAA,UAClB,UAAAZ;AAAA,QAAA;;;;;;;;;;;;2BCxBLrB,EAKE,SAAA;AAAA,MAJA,MAAK;AAAA,MACJ,MAAME,EAAA,MAAM;AAAA,MACZ,OAAOA,EAAA,cAAeA,EAAA,MAAM,gBAAY;AAAA,MACxC,uBAAqBA,EAAA,MAAM;AAAA,IAAA;;;;;;;;;;;2BCL9BF,EAQM,OAAA;AAAA,MAPJ,OAAM;AAAA,MACL,uBAAqBE,EAAA,MAAM;AAAA,IAAA;MAE5BC,EAGOC,yBAHP,MAGO;AAAA,QAFLa,EAA4D,MAA5DO,IAA4DhB,EAAnBN,EAAA,MAAM,KAAK,GAAA,CAAA;AAAA,QAC3CA,EAAA,MAAM,YAAfI,EAAA,GAAAN,EAAgF,KAAhFK,IAAgFG,EAArBN,EAAA,MAAM,QAAQ,GAAA,CAAA;;;;;;;;;;2BCF7EF,EAIE,OAAA;AAAA,MAHA,OAAM;AAAA,MACL,uBAAqBE,EAAA,MAAM;AAAA,MAC5B,WAAQA,EAAA,MAAM,QAAI;AAAA,IAAA;;;;;;;;;;;;;;;;;;;;;;;;;AC2BtB,UAAMQ,IAAQR,GAERS,IAAOC,GASPuB,IAAqB,uBAAA,OAAA,EAAA,GAOrBC,IAActG;AAAA,MAAS,MAC3BwD,GAAoBoB,EAAM,gBAAgByB,CAAiB;AAAA,IAAA,GAGvD9G,IAAaS,EAAsC,MACnD4E,EAAM,qBAA2BA,EAAM,qBACpC0B,EAAY,MAAM1B,EAAM,MAAM,KAAK,IAC3C,GAEKtB,IAAQ;AAEd,IAAAiD;AAAA,MACEhH;AAAA,MACA,CAACZ,MAAQ;AACP,QAAIA,KAAKyE,GAAcwB,EAAM,QAAQjG,GAAK2E,CAAK;AAAA,MACjD;AAAA,MACA,EAAE,WAAW,GAAA;AAAA,IAAK;AAKpB,UAAMkD,IAAiBxG;AAAA,MACrB,MACET,EAAW,SAAS;AAAA,QAClB,QAAQqF,EAAM;AAAA,QACd,OAAO;AAAA,QACP,YAAY,EAAE,MAAM,OAAA;AAAA,QACpB,QAAQ,CAAA;AAAA,MAAC;AAAA,IACX,GAGE6B,IAAOpH,GAAW,EAAE,YAAYmH,EAAe,OAAO;AAI5D,IAAAD;AAAA,MACE,MAAMC,EAAe;AAAA,MACrB,MAAMC,EAAK,MAAA;AAAA,IAAM;AAGnB,UAAMhF,IAAUzB;AAAA,MACd,MACE4E,EAAM,WACL8B,GACG,uBACJ;AAAA,IAAA,GAGEC,IAAmB3G;AAAA,MACvB,MACE4E,EAAM,YACL8B,GACG,sBACJ;AAAA,IAAA,GAGEE,IAAc5G;AAAA,MAClB,MAAMwG,EAAe,MAAM,eAAe;AAAA,IAAA;AAG5C,aAASK,EAAe3I,GAAwB;AAC9C,cAAQA,EAAM,MAAA;AAAA,QACZ,KAAK;AAAA,QACL,KAAK;AAAA,QACL,KAAK;AACH,iBAAO4I;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AAAA,QACL,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT,KAAK;AACH,iBAAOC;AAAAA,QACT;AACE,iBAAOV;AAAAA,MAAA;AAAA,IAEb;AAEA,UAAMW,IAAShI,EAA4B,IAAI;AAE/C,mBAAeiI,EAASzI,GAAyB;AAE/C,UADAA,EAAE,eAAA,GACE,CAACM,EAAW,MAAO;AAEvB,UAAI,CADOkH,EAAK,YAAA,GACP;AACP,QAAA5B,EAAK,oBAAoB4B,EAAK,OAAO,KAAK;AAC1C;AAAA,MACF;AACA,MAAAA,EAAK,WAAW,QAAQ,IACxBA,EAAK,YAAY,QAAQ;AACzB,YAAM9E,IAAU;AAAA,QACd,QAAQiD,EAAM;AAAA,QACd,QAAQ6B,EAAK,wBAAA;AAAA,QACb,cAAc7B,EAAM;AAAA,MAAA;AAEtB,UAAI;AACF,cAAMrB,IAAS,MAAM/B,GAAiB;AAAA,UACpC,SAASC,EAAQ;AAAA,UACjB,UAAUkF,EAAiB;AAAA,UAC3B,SAAAhF;AAAA,QAAA,CACD;AACD,QAAA8E,EAAK,UAAU,QAAQ,IACvB5B,EAAK,kBAAkBtB,CAAM;AAAA,MAC/B,SAASxE,GAAK;AACZ,cAAM4I,IAAK5I;AACX,QAAA0H,EAAK,YAAY,QAAQkB,EAAG,OAAO,WAAW,qBAC9C9C,EAAK,gBAAgB8C,CAAE;AAAA,MACzB,UAAA;AACE,QAAAlB,EAAK,WAAW,QAAQ;AAAA,MAC1B;AAAA,IACF;AAEA,WAAAmB,GAAU,MAAM;AACd,MAAKrI,EAAW,SAEd,QAAQ;AAAA,QACN,iEAAiEqF,EAAM,MAAM,qCACzCA,EAAM,MAAM;AAAA,MAAA;AAAA,IAGtD,CAAC,aAIarF,EAAA,QAMIsI,EAAApB,CAAA,EAAK,UAAU,cAA/BvC,EAIM,OAAA;AAAA;MAJgC,OAAM;AAAA,MAA8B,iBAAeE,EAAA;AAAA,IAAA;MACvFC,EAEOC,EAAA,QAAA,WAAA,EAFe,YAAY/E,EAAA,MAAA,GAAlC,MAEO;AAAA,QADL4F,EAA8E,KAAA,MAAAT,EAAxEnF,EAAA,MAAW,kBAAc,oCAAA,GAAA,CAAA;AAAA,MAAA;uBAInC2E,EAsFO,QAAA;AAAA;eApFD;AAAA,MAAJ,KAAIuD;AAAA,MACJ,OAAM;AAAA,MACL,iBAAerD,EAAA;AAAA,MAChB,YAAA;AAAA,MACC,UAAAsD;AAAA,IAAA;MAEDrD,EAOOC,EAAA,QAAA,UAAA,EAPc,YAAY/E,EAAA,MAAA,GAAjC,MAOO;AAAA,QANL4F,EAKS,UALTV,IAKS;AAAA,UAJPU,EAAuD,MAAvDR,IAAuDD,EAAxBnF,EAAA,MAAW,KAAK,GAAA,CAAA;AAAA,UACtCA,EAAA,MAAW,eAApBiF,EAAA,GAAAN,EAEI,KAFJ4D,IAEIpD,EADCnF,EAAA,MAAW,WAAW,GAAA,CAAA;;;MAKpBsI,EAAApB,CAAA,EAAK,MAAM,SAAtBjC,KAAAN,EAeM,OAfN6D,IAeM;AAAA,QAdJ1D,EAaOC,EAAA,QAAA,YAAA;AAAA,UAXJ,SAASuD,EAAApB,CAAA,EAAK,iBAAiB;AAAA,UAC/B,OAAOoB,EAAApB,CAAA,EAAK,MAAM,MAAM;AAAA,UACxB,MAAMoB,EAAApB,CAAA,EAAK,YAAY;AAAA,QAAA,GAJ1B,MAaO;AAAA,UAPLtB,EAMI,KAAA,MAAA;AAAA,YANDe,EAAA,aACO2B,EAAApB,CAAA,EAAK,iBAAiB,QAAK,CAAA,IAAO,SAC1C/B,EAAGmD,EAAApB,CAAA,EAAK,MAAM,MAAM,MAAM,IAAG,KAC7B,CAAA;AAAA,YAAgBoB,EAAApB,CAAA,EAAK,YAAY,cAAjCvC,EAEWyB,GAAA,EAAA,KAAA,EAAA,GAAA;AAAA,cAF6BO,EAAA,UACjC2B,EAAApB,CAAA,EAAK,YAAY,MAAM,KAAK,GAAA,CAAA;AAAA,YAAA;;;;MAMzCtB,EAWM,OAXN6C,IAWM;AAAA,SAVJxD,EAAA,EAAA,GAAAN,EASEyB,WAPgBkC,EAAApB,CAAA,EAAK,cAAc,QAA5BvI,YAFT+G,EASEgD,GARKpB,EAAe3I,CAAK,CAAA,GAAA;AAAA,UAExB,KAAKA,EAAM;AAAA,UACX,OAAAA;AAAA,UACA,eAAa2J,EAAApB,CAAA,EAAK,OAAOvI,EAAM,EAAE;AAAA,UACjC,OAAO2J,KAAK,OAAO,MAAM3J,EAAM,EAAE;AAAA,UACjC,uBAAkB,CAAGK,MAAesJ,EAAApB,CAAA,EAAK,SAASvI,EAAM,IAAIK,CAAC;AAAA,UAC7D,eAAMsJ,EAAApB,CAAA,EAAK,MAAMvI,EAAM,EAAE;AAAA,QAAA;;MAInB2J,EAAApB,CAAA,EAAK,YAAY,SAA5BjC,KAAAN,EAEM,OAFNgE,IAEMxD,EADDmD,KAAK,YAAY,KAAK,GAAA,CAAA;MAG3B1C,EAkCM,OAlCNgD,IAkCM;AAAA,QAjCJ9D,EAgCOC,EAAA,QAAA,WAAA;AAAA,UA9BJ,aAAeuD,EAAApB,CAAA,EAAK,YAAY;AAAA,UAChC,YAAcoB,EAAApB,CAAA,EAAK,WAAW;AAAA,UAC9B,YAAYoB,EAAApB,CAAA,EAAK,WAAW;AAAA,UAC5B,MAAMoB,EAAApB,CAAA,EAAK;AAAA,UACX,MAAMoB,EAAApB,CAAA,EAAK;AAAA,QAAA,GANd,MAgCO;AAAA,UAvBGoB,EAAApB,CAAA,EAAK,MAAM,UAAUoB,EAAApB,CAAA,EAAK,YAAY,cAD9CvC,EAOS,UAAA;AAAA;YALP,MAAK;AAAA,YACL,OAAM;AAAA,YACL,SAAKkE,EAAA,CAAA,MAAAA,EAAA,CAAA,IAAA,CAAAhD,MAAEyC,EAAApB,CAAA,EAAK,KAAA;AAAA,UAAI,GAClB,YAED;UAEQoB,EAAApB,CAAA,EAAK,MAAM,UAAUoB,EAAApB,CAAA,EAAK,WAAW,cAD7CvC,EAOS,UAAA;AAAA;YALP,MAAK;AAAA,YACL,OAAM;AAAA,YACL,SAAKkE,EAAA,CAAA,MAAAA,EAAA,CAAA,IAAA,CAAAhD,MAAEyC,EAAApB,CAAA,EAAK,KAAA;AAAA,UAAI,GAClB,QAED;WAESoB,EAAApB,CAAA,EAAK,MAAM,SAASoB,EAAApB,CAAA,EAAK,WAAW,cAD7CvC,EAOS,UAAA;AAAA;YALP,MAAK;AAAA,YACL,OAAM;AAAA,YACL,UAAU2D,EAAApB,CAAA,EAAK,WAAW;AAAA,UAAA,GAExB/B,EAAAmD,EAAApB,CAAA,EAAK,WAAW,qBAAqBG,EAAA,KAAW,GAAA,GAAAyB,EAAA;;;wBA9F3DnE,EAIM,OAAA;AAAA;MAJkB,OAAM;AAAA,MAA8B,iBAAeE,EAAA;AAAA,IAAA;MACzEC,EAEOC,EAAA,QAAA,WAAA,EAFe,QAASF,EAAA,OAAA,GAA/B,MAEO;AAAA,QADLe,EAAyD,KAAA,MAAtD,WAAYT,EAAGN,EAAA,MAAM,IAAG,wBAA0B,CAAA;AAAA,MAAA;;;;"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { PortalFormDefinition } from '../types';
|
|
2
|
+
/**
|
|
3
|
+
* Eagerly load every form YAML under `/.dcs/forms/*.yaml` from the
|
|
4
|
+
* consuming Vite app and parse them into PortalFormDefinition objects.
|
|
5
|
+
*
|
|
6
|
+
* Vite returns the modules as raw strings (when using `?raw` or the
|
|
7
|
+
* default text loader for unknown extensions) or as parsed objects
|
|
8
|
+
* (when `vite-plugin-yaml` is installed). This helper handles both.
|
|
9
|
+
*/
|
|
10
|
+
export declare function loadFormDefinitions(modules: Record<string, unknown>): Record<string, PortalFormDefinition>;
|
|
11
|
+
/** Parse a single raw YAML string into a PortalFormDefinition. */
|
|
12
|
+
export declare function parseFormYaml(raw: string): PortalFormDefinition;
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { ErrorObject } from 'ajv';
|
|
2
|
+
import { PortalFormDefinition } from '../types';
|
|
3
|
+
export interface SchemaValidationResult {
|
|
4
|
+
valid: boolean;
|
|
5
|
+
errors: ErrorObject[];
|
|
6
|
+
}
|
|
7
|
+
export declare function validateFormDefinition(def: PortalFormDefinition | unknown): SchemaValidationResult;
|
|
8
|
+
/** Logs a `console.warn` when invalid, but only when `import.meta.env.DEV`. */
|
|
9
|
+
export declare function warnIfInvalid(formId: string, def: PortalFormDefinition | unknown, isDev: boolean): SchemaValidationResult;
|
package/dist/types.d.ts
ADDED
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Form type definitions — vendored snapshot of the canonical form types
|
|
3
|
+
* generated from `contracts/spec/domains/portal-site-forms.yaml` (see
|
|
4
|
+
* `contracts/generated/typescript/types.gen.ts`).
|
|
5
|
+
*
|
|
6
|
+
* This is an intentional fork of the generated types so the published
|
|
7
|
+
* package has zero runtime/peer dependency on the (workspace-internal,
|
|
8
|
+
* never published) `@dcs/contracts` package. When the form schema in
|
|
9
|
+
* `contracts/spec/domains/portal-site-forms.yaml` changes, refresh both
|
|
10
|
+
* this file and `src/schema/form-definition.schema.json`.
|
|
11
|
+
*
|
|
12
|
+
* Consumers should import these types from `@duffcloudservices/site-forms`
|
|
13
|
+
* so the package remains the single import surface for managed-form
|
|
14
|
+
* runtime code on customer sites.
|
|
15
|
+
*/
|
|
16
|
+
export type PortalFormFieldType = 'text' | 'email' | 'tel' | 'textarea' | 'select' | 'multiselect' | 'radio' | 'checkbox' | 'checkbox-group' | 'date' | 'file' | 'hidden' | 'section-heading' | 'html-block';
|
|
17
|
+
export type PortalFormFieldWidth = 'full' | 'half' | 'third';
|
|
18
|
+
export interface PortalFormFieldOption {
|
|
19
|
+
value: string;
|
|
20
|
+
label: string;
|
|
21
|
+
}
|
|
22
|
+
export interface PortalFormFieldValidation {
|
|
23
|
+
regex?: string;
|
|
24
|
+
minLength?: number;
|
|
25
|
+
maxLength?: number;
|
|
26
|
+
min?: number;
|
|
27
|
+
max?: number;
|
|
28
|
+
accept?: string[];
|
|
29
|
+
}
|
|
30
|
+
export interface PortalFormFieldVisibleIf {
|
|
31
|
+
fieldId: string;
|
|
32
|
+
equals: string | number | boolean;
|
|
33
|
+
}
|
|
34
|
+
export interface PortalFormField {
|
|
35
|
+
id: string;
|
|
36
|
+
type: PortalFormFieldType;
|
|
37
|
+
label: string;
|
|
38
|
+
helpText?: string;
|
|
39
|
+
placeholder?: string;
|
|
40
|
+
defaultValue?: string | number | boolean | string[];
|
|
41
|
+
required?: boolean;
|
|
42
|
+
width?: PortalFormFieldWidth;
|
|
43
|
+
options?: PortalFormFieldOption[];
|
|
44
|
+
validation?: PortalFormFieldValidation;
|
|
45
|
+
visibleIf?: PortalFormFieldVisibleIf;
|
|
46
|
+
/**
|
|
47
|
+
* Marks the field as collecting Protected Health Information.
|
|
48
|
+
* When true the value must never appear in notification emails
|
|
49
|
+
* or logs and the owning site must be in the Medical category
|
|
50
|
+
* (see `compliance.instructions.md`).
|
|
51
|
+
*/
|
|
52
|
+
phi?: boolean;
|
|
53
|
+
html?: string;
|
|
54
|
+
}
|
|
55
|
+
export interface PortalFormStep {
|
|
56
|
+
id: string;
|
|
57
|
+
title: string;
|
|
58
|
+
description?: string;
|
|
59
|
+
fieldIds: string[];
|
|
60
|
+
}
|
|
61
|
+
export interface PortalFormSubmissionLeadConfig {
|
|
62
|
+
kind: 'lead';
|
|
63
|
+
category?: string;
|
|
64
|
+
}
|
|
65
|
+
export interface PortalFormSubmissionEmailConfig {
|
|
66
|
+
kind: 'email';
|
|
67
|
+
to: string[];
|
|
68
|
+
subjectTemplate?: string;
|
|
69
|
+
}
|
|
70
|
+
export interface PortalFormSubmissionWebhookConfig {
|
|
71
|
+
kind: 'webhook';
|
|
72
|
+
url: string;
|
|
73
|
+
signingSecretRef?: string;
|
|
74
|
+
}
|
|
75
|
+
export type PortalFormSubmissionConfig = PortalFormSubmissionLeadConfig | PortalFormSubmissionEmailConfig | PortalFormSubmissionWebhookConfig;
|
|
76
|
+
export interface PortalFormDefinition {
|
|
77
|
+
formId: string;
|
|
78
|
+
title: string;
|
|
79
|
+
description?: string;
|
|
80
|
+
submitLabel?: string;
|
|
81
|
+
successMessage?: string;
|
|
82
|
+
submission: PortalFormSubmissionConfig;
|
|
83
|
+
steps?: PortalFormStep[];
|
|
84
|
+
fields: PortalFormField[];
|
|
85
|
+
version?: number;
|
|
86
|
+
createdAt?: string;
|
|
87
|
+
updatedAt?: string;
|
|
88
|
+
}
|
|
89
|
+
/** Map of field id -> current value held by the form. */
|
|
90
|
+
export type FormValues = Record<string, unknown>;
|
|
91
|
+
/** Per-field validation errors keyed by field id. */
|
|
92
|
+
export type FormErrors = Record<string, string | undefined>;
|
|
93
|
+
export interface DcsFormSubmitPayload {
|
|
94
|
+
formId: string;
|
|
95
|
+
values: FormValues;
|
|
96
|
+
captchaToken?: string;
|
|
97
|
+
}
|
|
98
|
+
export interface DcsFormSubmitSuccess {
|
|
99
|
+
payload: DcsFormSubmitPayload;
|
|
100
|
+
response: unknown;
|
|
101
|
+
}
|
|
102
|
+
export interface DcsFormSubmitError {
|
|
103
|
+
payload: DcsFormSubmitPayload;
|
|
104
|
+
error: Error;
|
|
105
|
+
status?: number;
|
|
106
|
+
}
|
package/package.json
ADDED
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@duffcloudservices/site-forms",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Shared <DcsForm/> runtime for DCS customer sites — renders managed form definitions from .dcs/forms/<formId>.yaml",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"files": [
|
|
7
|
+
"dist",
|
|
8
|
+
"src"
|
|
9
|
+
],
|
|
10
|
+
"main": "./dist/index.js",
|
|
11
|
+
"module": "./dist/index.js",
|
|
12
|
+
"types": "./dist/index.d.ts",
|
|
13
|
+
"exports": {
|
|
14
|
+
".": {
|
|
15
|
+
"types": "./dist/index.d.ts",
|
|
16
|
+
"import": "./dist/index.js"
|
|
17
|
+
},
|
|
18
|
+
"./style.css": "./dist/style.css"
|
|
19
|
+
},
|
|
20
|
+
"scripts": {
|
|
21
|
+
"build": "vite build",
|
|
22
|
+
"dev": "vite build --watch",
|
|
23
|
+
"test": "vitest run",
|
|
24
|
+
"test:watch": "vitest",
|
|
25
|
+
"type-check": "vue-tsc --noEmit -p tsconfig.json",
|
|
26
|
+
"prepublishOnly": "pnpm run build"
|
|
27
|
+
},
|
|
28
|
+
"peerDependencies": {
|
|
29
|
+
"vue": "^3.5.0"
|
|
30
|
+
},
|
|
31
|
+
"dependencies": {
|
|
32
|
+
"ajv": "^8.17.1",
|
|
33
|
+
"ajv-formats": "^3.0.1",
|
|
34
|
+
"js-yaml": "^4.1.0"
|
|
35
|
+
},
|
|
36
|
+
"devDependencies": {
|
|
37
|
+
"@types/js-yaml": "^4.0.9",
|
|
38
|
+
"@types/node": "^22.0.0",
|
|
39
|
+
"@vitejs/plugin-vue": "^6.0.0",
|
|
40
|
+
"@vue/test-utils": "^2.4.6",
|
|
41
|
+
"@vue/tsconfig": "^0.7.0",
|
|
42
|
+
"jsdom": "^26.0.0",
|
|
43
|
+
"typescript": "~5.8.0",
|
|
44
|
+
"vite": "^7.0.0",
|
|
45
|
+
"vite-plugin-dts": "^4.5.0",
|
|
46
|
+
"vitest": "^3.0.0",
|
|
47
|
+
"vue": "^3.5.18",
|
|
48
|
+
"vue-tsc": "^3.0.0"
|
|
49
|
+
},
|
|
50
|
+
"keywords": [
|
|
51
|
+
"dcs",
|
|
52
|
+
"forms",
|
|
53
|
+
"vue",
|
|
54
|
+
"site-forms",
|
|
55
|
+
"duff-cloud-services"
|
|
56
|
+
],
|
|
57
|
+
"author": "Duff Cloud Services",
|
|
58
|
+
"license": "MIT",
|
|
59
|
+
"repository": {
|
|
60
|
+
"type": "git",
|
|
61
|
+
"url": "https://github.com/duffn/dcs"
|
|
62
|
+
},
|
|
63
|
+
"homepage": "https://portal.duffcloudservices.com",
|
|
64
|
+
"bugs": {
|
|
65
|
+
"url": "https://github.com/duffn/dcs/issues"
|
|
66
|
+
},
|
|
67
|
+
"engines": {
|
|
68
|
+
"node": ">=18.0.0"
|
|
69
|
+
},
|
|
70
|
+
"publishConfig": {
|
|
71
|
+
"access": "public"
|
|
72
|
+
}
|
|
73
|
+
}
|