@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 CHANGED
@@ -41,9 +41,9 @@ release-workflow details.
41
41
 
42
42
  ## Vite setup
43
43
 
44
- Two pieces are required in the consuming site:
44
+ Three pieces are required in the consuming site:
45
45
 
46
- 1. **`vite-plugin-yaml`** so `import.meta.glob('/.dcs/forms/*.yaml', { eager: true, import: 'default' })` returns parsed objects:
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. **Env vars** the runtime reads:
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(...)` | Test-only override for the YAML modules map. |
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 D, 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";
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 = D({}), o = D({}), d = D(!1), s = D(!1), c = D(null), a = D(0), n = p(() => t.fields), m = p(
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 C(h) {
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: C,
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
- }, Ce = {
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", Ce, b(e.error), 1)) : k("", !0)
304
+ e.error ? (l(), f("p", De, b(e.error), 1)) : k("", !0)
305
305
  ])
306
306
  ], 10, Ve));
307
307
  }
308
- }), De = ["id", "type", "name", "value", "placeholder", "required", "aria-invalid", "aria-describedby"], N = /* @__PURE__ */ I({
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, De)
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 = D(null);
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 /.dcs/forms/${r.formId}.yaml.`
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, C) => s.value ? u(n).submitted.value ? (l(), f("div", {
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: C[0] || (C[0] = ($) => u(n).prev())
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: C[1] || (C[1] = ($) => u(n).next())
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 &ldquo;{{ formId }}&rdquo; 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 &ldquo;{{ formId }}&rdquo; 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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@duffcloudservices/site-forms",
3
- "version": "0.1.0",
3
+ "version": "0.1.1",
4
4
  "description": "Shared <DcsForm/> runtime for DCS customer sites — renders managed form definitions from .dcs/forms/<formId>.yaml",
5
5
  "type": "module",
6
6
  "files": [
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 /.dcs/forms/${props.formId}.yaml.`,
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
  })