@duffcloudservices/site-forms 0.1.0 → 0.1.1
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 +51 -4
- package/dist/index.js +13 -13
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
- package/src/DcsForm.vue +5 -1
package/README.md
CHANGED
|
@@ -41,9 +41,9 @@ release-workflow details.
|
|
|
41
41
|
|
|
42
42
|
## Vite setup
|
|
43
43
|
|
|
44
|
-
|
|
44
|
+
Three pieces are required in the consuming site:
|
|
45
45
|
|
|
46
|
-
1. **`vite-plugin-yaml`** so
|
|
46
|
+
1. **`vite-plugin-yaml`** so YAML modules return parsed objects:
|
|
47
47
|
|
|
48
48
|
```ts
|
|
49
49
|
// vite.config.ts
|
|
@@ -57,13 +57,58 @@ Two pieces are required in the consuming site:
|
|
|
57
57
|
Without it, the loader falls back to parsing raw strings via
|
|
58
58
|
`js-yaml`, which works but pays the parse cost at boot.
|
|
59
59
|
|
|
60
|
-
2. **
|
|
60
|
+
2. **A `formsModules` loader** in your site that does the
|
|
61
|
+
`import.meta.glob` from a path Vite can resolve (see below).
|
|
62
|
+
|
|
63
|
+
3. **Env vars** the runtime reads:
|
|
61
64
|
|
|
62
65
|
| Variable | Purpose |
|
|
63
66
|
| ----------------------- | ---------------------------------------------------- |
|
|
64
67
|
| `VITE_DCS_PUBLIC_API` | Base URL of the DCS public API (no trailing slash). |
|
|
65
68
|
| `VITE_DCS_SITE_SLUG` | Default site slug used when the prop is omitted. |
|
|
66
69
|
|
|
70
|
+
### Why the `formsModules` prop is required in real sites
|
|
71
|
+
|
|
72
|
+
`<DcsForm/>` ships with an internal `import.meta.glob('/.dcs/forms/*.yaml')`
|
|
73
|
+
fallback, but Vite resolves the leading `/` against the **consumer's
|
|
74
|
+
Vite project root** (the directory containing `vite.config.ts`).
|
|
75
|
+
On every customer-site repo today, the `.dcs/forms/` directory lives
|
|
76
|
+
at the **repo root**, one or more levels above the Vite root
|
|
77
|
+
(typically `site/` or `docs/`). The internal glob therefore matches
|
|
78
|
+
nothing and you get:
|
|
79
|
+
|
|
80
|
+
```
|
|
81
|
+
[@duffcloudservices/site-forms] No form definition found for "contact".
|
|
82
|
+
Expected a YAML at /.dcs/forms/contact.yaml.
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
The fix is a one-file loader the rest of your site imports from.
|
|
86
|
+
|
|
87
|
+
#### Vue SPA (`vite.config.ts` in `site/`)
|
|
88
|
+
|
|
89
|
+
```ts
|
|
90
|
+
// site/src/dcs-forms.ts
|
|
91
|
+
const modules = import.meta.glob('../../.dcs/forms/*.yaml', {
|
|
92
|
+
eager: true,
|
|
93
|
+
import: 'default',
|
|
94
|
+
})
|
|
95
|
+
export const dcsFormsModules: Record<string, unknown> = modules
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
#### VitePress (`vite` block in `docs/.vitepress/config.ts`)
|
|
99
|
+
|
|
100
|
+
```ts
|
|
101
|
+
// docs/.vitepress/dcs-forms-loader.ts
|
|
102
|
+
const modules = import.meta.glob('../../.dcs/forms/*.yaml', {
|
|
103
|
+
eager: true,
|
|
104
|
+
import: 'default',
|
|
105
|
+
})
|
|
106
|
+
export const dcsFormsModules: Record<string, unknown> = modules
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
The relative depth (`../../`) depends on where the loader file lives
|
|
110
|
+
relative to the repo root. Adjust as needed.
|
|
111
|
+
|
|
67
112
|
## Usage
|
|
68
113
|
|
|
69
114
|
Place a YAML file at `<site>/.dcs/forms/contact.yaml`:
|
|
@@ -93,11 +138,13 @@ Then in any page component:
|
|
|
93
138
|
```vue
|
|
94
139
|
<script setup lang="ts">
|
|
95
140
|
import { DcsForm } from '@duffcloudservices/site-forms'
|
|
141
|
+
import { dcsFormsModules } from '@/dcs-forms'
|
|
96
142
|
</script>
|
|
97
143
|
|
|
98
144
|
<template>
|
|
99
145
|
<DcsForm
|
|
100
146
|
form-id="contact"
|
|
147
|
+
:forms-modules="dcsFormsModules"
|
|
101
148
|
@submit-success="onSuccess"
|
|
102
149
|
@submit-error="onError"
|
|
103
150
|
/>
|
|
@@ -113,7 +160,7 @@ import { DcsForm } from '@duffcloudservices/site-forms'
|
|
|
113
160
|
| `definitionOverride` | `PortalFormDefinition` | — | Used by the portal preview iframe to show in-flight edits. |
|
|
114
161
|
| `apiBase` | `string` | `import.meta.env.VITE_DCS_PUBLIC_API` | Override for tests / non-prod environments. |
|
|
115
162
|
| `captchaToken` | `string` | — | Attached to the submission payload when set. |
|
|
116
|
-
| `formsModules` | `Record<string, unknown>` | `import.meta.glob(
|
|
163
|
+
| `formsModules` | `Record<string, unknown>` | internal fallback glob (rarely matches) | **Required in real sites.** Pass a `Record<string, unknown>` from your own `import.meta.glob('../../.dcs/forms/*.yaml', { eager: true, import: 'default' })` — see Vite setup section. |
|
|
117
164
|
|
|
118
165
|
## Emits
|
|
119
166
|
|
package/dist/index.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { reactive as X, ref as
|
|
1
|
+
import { reactive as X, ref as C, computed as p, defineComponent as I, createElementBlock as f, openBlock as l, normalizeClass as Z, renderSlot as S, createCommentVNode as k, createTextVNode as A, toDisplayString as b, createBlock as w, withCtx as V, createElementVNode as v, Fragment as T, renderList as B, watch as H, onMounted as ee, unref as u, resolveDynamicComponent as te } from "vue";
|
|
2
2
|
import ie from "ajv";
|
|
3
3
|
import re from "ajv-formats";
|
|
4
4
|
import K from "js-yaml";
|
|
@@ -60,7 +60,7 @@ function _(e) {
|
|
|
60
60
|
return t;
|
|
61
61
|
}
|
|
62
62
|
function ae(e) {
|
|
63
|
-
const { definition: t } = e, r = X(_(t)), i =
|
|
63
|
+
const { definition: t } = e, r = X(_(t)), i = C({}), o = C({}), d = C(!1), s = C(!1), c = C(null), a = C(0), n = p(() => t.fields), m = p(
|
|
64
64
|
() => t.steps && t.steps.length > 0 ? t.steps : null
|
|
65
65
|
), g = p(() => {
|
|
66
66
|
const h = m.value;
|
|
@@ -79,7 +79,7 @@ function ae(e) {
|
|
|
79
79
|
function y(h, F) {
|
|
80
80
|
r[h] = F, i.value[h] && (i.value = { ...i.value, [h]: void 0 });
|
|
81
81
|
}
|
|
82
|
-
function
|
|
82
|
+
function D(h) {
|
|
83
83
|
o.value = { ...o.value, [h]: !0 };
|
|
84
84
|
}
|
|
85
85
|
function $(h) {
|
|
@@ -134,7 +134,7 @@ function ae(e) {
|
|
|
134
134
|
isLastStep: O,
|
|
135
135
|
visibleFields: z,
|
|
136
136
|
setValue: y,
|
|
137
|
-
touch:
|
|
137
|
+
touch: D,
|
|
138
138
|
validateCurrentScope: L,
|
|
139
139
|
validateAll: j,
|
|
140
140
|
next: J,
|
|
@@ -270,7 +270,7 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
270
270
|
}, Pe = {
|
|
271
271
|
key: 0,
|
|
272
272
|
class: "dcs-form-field__help"
|
|
273
|
-
},
|
|
273
|
+
}, De = {
|
|
274
274
|
key: 0,
|
|
275
275
|
class: "dcs-form-field__error",
|
|
276
276
|
role: "alert"
|
|
@@ -301,11 +301,11 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
301
301
|
e.field.helpText ? (l(), f("p", Pe, b(e.field.helpText), 1)) : k("", !0)
|
|
302
302
|
]),
|
|
303
303
|
S(t.$slots, "error", {}, () => [
|
|
304
|
-
e.error ? (l(), f("p",
|
|
304
|
+
e.error ? (l(), f("p", De, b(e.error), 1)) : k("", !0)
|
|
305
305
|
])
|
|
306
306
|
], 10, Ve));
|
|
307
307
|
}
|
|
308
|
-
}),
|
|
308
|
+
}), Ce = ["id", "type", "name", "value", "placeholder", "required", "aria-invalid", "aria-describedby"], N = /* @__PURE__ */ I({
|
|
309
309
|
__name: "DcsFormText",
|
|
310
310
|
props: {
|
|
311
311
|
field: {},
|
|
@@ -348,7 +348,7 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
348
348
|
"aria-describedby": e.error ? `${o.value}-error` : void 0,
|
|
349
349
|
onInput: c[0] || (c[0] = (a) => i("update:modelValue", a.target.value)),
|
|
350
350
|
onBlur: c[1] || (c[1] = (a) => i("blur"))
|
|
351
|
-
}, null, 40,
|
|
351
|
+
}, null, 40, Ce)
|
|
352
352
|
])
|
|
353
353
|
]),
|
|
354
354
|
_: 3
|
|
@@ -772,7 +772,7 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
772
772
|
return N;
|
|
773
773
|
}
|
|
774
774
|
}
|
|
775
|
-
const O =
|
|
775
|
+
const O = C(null);
|
|
776
776
|
async function z(y) {
|
|
777
777
|
if (y.preventDefault(), !s.value) return;
|
|
778
778
|
if (!n.validateAll()) {
|
|
@@ -801,9 +801,9 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
801
801
|
}
|
|
802
802
|
return ee(() => {
|
|
803
803
|
s.value || console.warn(
|
|
804
|
-
`[@duffcloudservices/site-forms] No form definition found for "${r.formId}". Expected a YAML at
|
|
804
|
+
`[@duffcloudservices/site-forms] No form definition found for "${r.formId}". Expected a YAML at .dcs/forms/${r.formId}.yaml. In customer-site repos the YAML lives above the Vite root, so the internal auto-glob will not find it — pass formsModules explicitly: <DcsForm form-id="${r.formId}" :forms-modules="dcsFormsModules" />. See @duffcloudservices/site-forms README "Vite setup" section.`
|
|
805
805
|
);
|
|
806
|
-
}), (y,
|
|
806
|
+
}), (y, D) => s.value ? u(n).submitted.value ? (l(), f("div", {
|
|
807
807
|
key: 1,
|
|
808
808
|
class: "dcs-form dcs-form--success",
|
|
809
809
|
"data-form-key": e.formId
|
|
@@ -863,13 +863,13 @@ const Ve = ["data-form-field-key"], we = ["for"], qe = {
|
|
|
863
863
|
key: 0,
|
|
864
864
|
type: "button",
|
|
865
865
|
class: "dcs-form__btn dcs-form__btn--prev",
|
|
866
|
-
onClick:
|
|
866
|
+
onClick: D[0] || (D[0] = ($) => u(n).prev())
|
|
867
867
|
}, " Previous ")) : k("", !0),
|
|
868
868
|
u(n).steps.value && !u(n).isLastStep.value ? (l(), f("button", {
|
|
869
869
|
key: 1,
|
|
870
870
|
type: "button",
|
|
871
871
|
class: "dcs-form__btn dcs-form__btn--next",
|
|
872
|
-
onClick:
|
|
872
|
+
onClick: D[1] || (D[1] = ($) => u(n).next())
|
|
873
873
|
}, " Next ")) : k("", !0),
|
|
874
874
|
!u(n).steps.value || u(n).isLastStep.value ? (l(), f("button", {
|
|
875
875
|
key: 2,
|
package/dist/index.js.map
CHANGED
|
@@ -1 +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":"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"}
|
|
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 `In customer-site repos the YAML lives above the Vite root, so the ` +\r\n `internal auto-glob will not find it — pass formsModules explicitly: ` +\r\n `<DcsForm form-id=\"${props.formId}\" :forms-modules=\"dcsFormsModules\" />. ` +\r\n `See @duffcloudservices/site-forms README \"Vite setup\" section.`,\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":"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"}
|
package/package.json
CHANGED
package/src/DcsForm.vue
CHANGED
|
@@ -190,7 +190,11 @@ onMounted(() => {
|
|
|
190
190
|
// eslint-disable-next-line no-console
|
|
191
191
|
console.warn(
|
|
192
192
|
`[@duffcloudservices/site-forms] No form definition found for "${props.formId}". ` +
|
|
193
|
-
`Expected a YAML at
|
|
193
|
+
`Expected a YAML at .dcs/forms/${props.formId}.yaml. ` +
|
|
194
|
+
`In customer-site repos the YAML lives above the Vite root, so the ` +
|
|
195
|
+
`internal auto-glob will not find it — pass formsModules explicitly: ` +
|
|
196
|
+
`<DcsForm form-id="${props.formId}" :forms-modules="dcsFormsModules" />. ` +
|
|
197
|
+
`See @duffcloudservices/site-forms README "Vite setup" section.`,
|
|
194
198
|
)
|
|
195
199
|
}
|
|
196
200
|
})
|