@defra/forms-engine-plugin 0.0.4 → 0.0.6

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.
Files changed (260) hide show
  1. package/.server/server/index.js +0 -4
  2. package/.server/server/index.js.map +1 -1
  3. package/.server/server/plugins/engine/helpers.js +3 -0
  4. package/.server/server/plugins/engine/helpers.js.map +1 -1
  5. package/.server/server/plugins/engine/index.js +27 -1
  6. package/.server/server/plugins/engine/index.js.map +1 -1
  7. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js +2 -4
  8. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js.map +1 -1
  9. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js +4 -10
  10. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js.map +1 -1
  11. package/.server/server/plugins/engine/pageControllers/StatusPageController.js +2 -3
  12. package/.server/server/plugins/engine/pageControllers/StatusPageController.js.map +1 -1
  13. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js +2 -4
  14. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js.map +1 -1
  15. package/.server/server/plugins/engine/plugin.js +65 -6
  16. package/.server/server/plugins/engine/plugin.js.map +1 -1
  17. package/.server/server/plugins/engine/types.js.map +1 -1
  18. package/.server/server/{views → plugins/engine/views}/components/service-banner/template.test.js +1 -1
  19. package/.server/server/plugins/engine/views/components/service-banner/template.test.js.map +1 -0
  20. package/.server/server/{views → plugins/engine/views}/components/tag-env/template.test.js +1 -1
  21. package/.server/server/plugins/engine/views/components/tag-env/template.test.js.map +1 -0
  22. package/.server/server/services/cacheService.js +5 -2
  23. package/.server/server/services/cacheService.js.map +1 -1
  24. package/.server/typings/hapi/index.d.js.map +1 -1
  25. package/README.md +215 -4
  26. package/package.json +3 -3
  27. package/src/client/javascripts/application.js +87 -0
  28. package/src/client/javascripts/file-upload.js +386 -0
  29. package/src/client/stylesheets/_code.scss +33 -0
  30. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  31. package/src/client/stylesheets/_prose.scss +56 -0
  32. package/src/client/stylesheets/_service-banner.scss +24 -0
  33. package/src/client/stylesheets/_summary-list.scss +28 -0
  34. package/src/client/stylesheets/_tag-env.scss +24 -0
  35. package/src/client/stylesheets/application.scss +14 -0
  36. package/src/common/cookies.js +58 -0
  37. package/src/common/cookies.test.js +23 -0
  38. package/src/common/types.js +5 -0
  39. package/src/config/index.ts +271 -0
  40. package/src/index.ts +31 -0
  41. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  42. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  43. package/src/server/common/helpers/logging/logger.ts +7 -0
  44. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  45. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  46. package/src/server/common/helpers/redis-client.js +70 -0
  47. package/src/server/constants.js +1 -0
  48. package/src/server/forms/README.md +10 -0
  49. package/src/server/forms/components.json +1015 -0
  50. package/src/server/forms/report-a-terrorist.json +270 -0
  51. package/src/server/forms/runner-components-test.json +365 -0
  52. package/src/server/forms/test.json +581 -0
  53. package/src/server/index.test.ts +582 -0
  54. package/src/server/index.ts +135 -0
  55. package/src/server/plugins/blankie.test.ts +73 -0
  56. package/src/server/plugins/blankie.ts +48 -0
  57. package/src/server/plugins/crumb.ts +20 -0
  58. package/src/server/plugins/engine/README.md +87 -0
  59. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  60. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  61. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  62. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  63. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  64. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  65. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  66. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  67. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  68. package/src/server/plugins/engine/components/Details.ts +30 -0
  69. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  70. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  71. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  72. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  73. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  74. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  75. package/src/server/plugins/engine/components/Html.ts +29 -0
  76. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  77. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  78. package/src/server/plugins/engine/components/List.test.ts +76 -0
  79. package/src/server/plugins/engine/components/List.ts +72 -0
  80. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  81. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  82. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  83. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  84. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  85. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  86. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  87. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  88. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  89. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  90. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  91. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  92. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  93. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  94. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  95. package/src/server/plugins/engine/components/TextField.ts +96 -0
  96. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  97. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  98. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  99. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  100. package/src/server/plugins/engine/components/constants.ts +1 -0
  101. package/src/server/plugins/engine/components/helpers.ts +330 -0
  102. package/src/server/plugins/engine/components/index.ts +24 -0
  103. package/src/server/plugins/engine/components/types.ts +117 -0
  104. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  105. package/src/server/plugins/engine/helpers.test.ts +791 -0
  106. package/src/server/plugins/engine/helpers.ts +384 -0
  107. package/src/server/plugins/engine/index.ts +47 -0
  108. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  109. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  110. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  111. package/src/server/plugins/engine/models/Section.ts +0 -0
  112. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  113. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  114. package/src/server/plugins/engine/models/index.ts +2 -0
  115. package/src/server/plugins/engine/models/types.ts +114 -0
  116. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  117. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  118. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  119. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  120. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  121. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  122. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  123. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  124. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1116 -0
  125. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +447 -0
  126. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  127. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  128. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  129. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +565 -0
  130. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  131. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  132. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  133. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  134. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +51 -0
  135. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +262 -0
  136. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  137. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  138. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  139. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  140. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  141. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  142. package/src/server/plugins/engine/plugin.ts +753 -0
  143. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  144. package/src/server/plugins/engine/services/formsService.js +46 -0
  145. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  146. package/src/server/plugins/engine/services/index.js +3 -0
  147. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  148. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  149. package/src/server/plugins/engine/services/uploadService.js +60 -0
  150. package/src/server/plugins/engine/types.ts +317 -0
  151. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  152. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  153. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  154. package/src/server/plugins/engine/views/components/debug/macro.njk +3 -0
  155. package/src/server/plugins/engine/views/components/debug/template.njk +13 -0
  156. package/src/server/plugins/engine/views/components/details.html +6 -0
  157. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  158. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  159. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  160. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  161. package/src/server/plugins/engine/views/components/html.html +3 -0
  162. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  163. package/src/server/plugins/engine/views/components/list.html +36 -0
  164. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  165. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  166. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  167. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  168. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  169. package/src/server/plugins/engine/views/components/service-banner/macro.njk +3 -0
  170. package/src/server/plugins/engine/views/components/service-banner/template.njk +20 -0
  171. package/src/server/plugins/engine/views/components/service-banner/template.test.js +43 -0
  172. package/src/server/plugins/engine/views/components/tag-env/macro.njk +3 -0
  173. package/src/server/plugins/engine/views/components/tag-env/template.njk +30 -0
  174. package/src/server/plugins/engine/views/components/tag-env/template.test.js +66 -0
  175. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  176. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  177. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  178. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  179. package/src/server/plugins/engine/views/confirmation.html +19 -0
  180. package/src/server/plugins/engine/views/file-upload.html +45 -0
  181. package/src/server/plugins/engine/views/index.html +39 -0
  182. package/src/server/plugins/engine/views/item-delete.html +56 -0
  183. package/src/server/plugins/engine/views/layout.html +199 -0
  184. package/src/server/plugins/engine/views/partials/components.html +6 -0
  185. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  186. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  187. package/src/server/plugins/engine/views/partials/form.html +15 -0
  188. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  189. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  190. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  191. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  192. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  193. package/src/server/plugins/engine/views/summary.html +50 -0
  194. package/src/server/plugins/errorPages.ts +58 -0
  195. package/src/server/plugins/nunjucks/context.js +88 -0
  196. package/src/server/plugins/nunjucks/context.test.js +142 -0
  197. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  198. package/src/server/plugins/nunjucks/environment.js +116 -0
  199. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  200. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  201. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  202. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  203. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  204. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  205. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  206. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  207. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  208. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  209. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  210. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  211. package/src/server/plugins/nunjucks/index.js +3 -0
  212. package/src/server/plugins/nunjucks/plugin.js +40 -0
  213. package/src/server/plugins/nunjucks/render.js +42 -0
  214. package/src/server/plugins/nunjucks/types.js +40 -0
  215. package/src/server/plugins/pulse.ts +11 -0
  216. package/src/server/plugins/router.ts +201 -0
  217. package/src/server/plugins/session.ts +28 -0
  218. package/src/server/routes/health.js +13 -0
  219. package/src/server/routes/health.test.js +35 -0
  220. package/src/server/routes/index.test.ts +125 -0
  221. package/src/server/routes/index.ts +2 -0
  222. package/src/server/routes/public.ts +47 -0
  223. package/src/server/routes/types.ts +48 -0
  224. package/src/server/schemas/index.ts +34 -0
  225. package/src/server/secure-context.js +43 -0
  226. package/src/server/services/cacheService.test.ts +277 -0
  227. package/src/server/services/cacheService.ts +138 -0
  228. package/src/server/services/httpService.test.js +491 -0
  229. package/src/server/services/httpService.ts +50 -0
  230. package/src/server/services/index.ts +1 -0
  231. package/src/server/types.ts +54 -0
  232. package/src/server/utils/notify.test.ts +37 -0
  233. package/src/server/utils/notify.ts +50 -0
  234. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  235. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  236. package/src/server/utils/utils.js +24 -0
  237. package/src/server/utils/utils.test.js +54 -0
  238. package/src/server/views/404.html +16 -0
  239. package/src/server/views/500.html +19 -0
  240. package/src/server/views/help/accessibility-statement.html +58 -0
  241. package/src/server/views/help/cookie-preferences.html +57 -0
  242. package/src/server/views/help/cookies.html +71 -0
  243. package/src/server/views/help/get-support.html +37 -0
  244. package/src/server/views/help/privacy-notice.html +68 -0
  245. package/src/server/views/help/terms-and-conditions.html +83 -0
  246. package/src/typings/hapi/index.d.ts +87 -0
  247. package/src/typings/hapi-tracing/index.d.ts +6 -0
  248. package/src/typings/index.d.ts +3 -0
  249. package/src/typings/joi/index.d.ts +22 -0
  250. package/.server/server/views/components/service-banner/template.test.js.map +0 -1
  251. package/.server/server/views/components/tag-env/template.test.js.map +0 -1
  252. /package/.server/server/{views → plugins/engine/views}/components/debug/macro.njk +0 -0
  253. /package/.server/server/{views → plugins/engine/views}/components/debug/template.njk +0 -0
  254. /package/.server/server/{views → plugins/engine/views}/components/service-banner/macro.njk +0 -0
  255. /package/.server/server/{views → plugins/engine/views}/components/service-banner/template.njk +0 -0
  256. /package/.server/server/{views → plugins/engine/views}/components/tag-env/macro.njk +0 -0
  257. /package/.server/server/{views → plugins/engine/views}/components/tag-env/template.njk +0 -0
  258. /package/.server/server/{views → plugins/engine/views}/confirmation.html +0 -0
  259. /package/.server/server/{views → plugins/engine/views}/layout.html +0 -0
  260. /package/.server/server/{views → plugins/engine/views}/summary.html +0 -0
@@ -0,0 +1,701 @@
1
+ import { ComponentType, type NumberFieldComponent } from '@defra/forms-model'
2
+
3
+ import { ComponentCollection } from '~/src/server/plugins/engine/components/ComponentCollection.js'
4
+ import { NumberField } from '~/src/server/plugins/engine/components/NumberField.js'
5
+ import {
6
+ getAnswer,
7
+ type Field
8
+ } from '~/src/server/plugins/engine/components/helpers.js'
9
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
10
+ import definition from '~/test/form/definitions/blank.js'
11
+ import { getFormData, getFormState } from '~/test/helpers/component-helpers.js'
12
+
13
+ describe('NumberField', () => {
14
+ let model: FormModel
15
+
16
+ beforeEach(() => {
17
+ model = new FormModel(definition, {
18
+ basePath: 'test'
19
+ })
20
+ })
21
+
22
+ describe('Defaults', () => {
23
+ let def: NumberFieldComponent
24
+ let collection: ComponentCollection
25
+ let field: Field
26
+
27
+ beforeEach(() => {
28
+ def = {
29
+ title: 'Example number field',
30
+ name: 'myComponent',
31
+ type: ComponentType.NumberField,
32
+ options: {},
33
+ schema: {}
34
+ } satisfies NumberFieldComponent
35
+
36
+ collection = new ComponentCollection([def], { model })
37
+ field = collection.fields[0]
38
+ })
39
+
40
+ describe('Schema', () => {
41
+ it('uses component title as label', () => {
42
+ const { formSchema } = collection
43
+ const { keys } = formSchema.describe()
44
+
45
+ expect(keys).toHaveProperty(
46
+ 'myComponent',
47
+ expect.objectContaining({
48
+ flags: expect.objectContaining({
49
+ label: 'Example number field'
50
+ })
51
+ })
52
+ )
53
+ })
54
+
55
+ it('uses component name as keys', () => {
56
+ const { formSchema } = collection
57
+ const { keys } = formSchema.describe()
58
+
59
+ expect(field.keys).toEqual(['myComponent'])
60
+ expect(field.collection).toBeUndefined()
61
+
62
+ for (const key of field.keys) {
63
+ expect(keys).toHaveProperty(key)
64
+ }
65
+ })
66
+
67
+ it('is required by default', () => {
68
+ const { formSchema } = collection
69
+ const { keys } = formSchema.describe()
70
+
71
+ expect(keys).toHaveProperty(
72
+ 'myComponent',
73
+ expect.objectContaining({
74
+ flags: expect.objectContaining({
75
+ presence: 'required'
76
+ })
77
+ })
78
+ )
79
+ })
80
+
81
+ it('is optional when configured', () => {
82
+ const collectionOptional = new ComponentCollection(
83
+ [{ ...def, options: { required: false } }],
84
+ { model }
85
+ )
86
+
87
+ const { formSchema } = collectionOptional
88
+ const { keys } = formSchema.describe()
89
+
90
+ expect(keys).toHaveProperty(
91
+ 'myComponent',
92
+ expect.objectContaining({ allow: [''] })
93
+ )
94
+
95
+ const result = collectionOptional.validate(getFormData(''))
96
+ expect(result.errors).toBeUndefined()
97
+ })
98
+
99
+ it('accepts valid values', () => {
100
+ const result1 = collection.validate(getFormData('1'))
101
+ const result2 = collection.validate(getFormData('10'))
102
+ const result3 = collection.validate(getFormData('2024'))
103
+ const result4 = collection.validate(getFormData(' 2020'))
104
+
105
+ expect(result1.errors).toBeUndefined()
106
+ expect(result2.errors).toBeUndefined()
107
+ expect(result3.errors).toBeUndefined()
108
+ expect(result4.errors).toBeUndefined()
109
+ })
110
+
111
+ it('adds errors for empty value', () => {
112
+ const result = collection.validate(getFormData(''))
113
+
114
+ expect(result.errors).toEqual([
115
+ expect.objectContaining({
116
+ text: 'Enter example number field'
117
+ })
118
+ ])
119
+
120
+ expect(result.errors).toEqual([
121
+ expect.objectContaining({
122
+ text: 'Enter example number field'
123
+ })
124
+ ])
125
+ })
126
+
127
+ it('adds errors for invalid values', () => {
128
+ const result1 = collection.validate(getFormData(['invalid']))
129
+ const result2 = collection.validate(
130
+ // @ts-expect-error - Allow invalid param for test
131
+ getFormData({ unknown: 'invalid' })
132
+ )
133
+
134
+ expect(result1.errors).toBeTruthy()
135
+ expect(result2.errors).toBeTruthy()
136
+ })
137
+ })
138
+
139
+ describe('State', () => {
140
+ it('returns text from state', () => {
141
+ const state1 = getFormState(2024)
142
+ const state2 = getFormState(null)
143
+
144
+ const answer1 = getAnswer(field, state1)
145
+ const answer2 = getAnswer(field, state2)
146
+
147
+ expect(answer1).toBe('2024')
148
+ expect(answer2).toBe('')
149
+ })
150
+
151
+ it('returns payload from state', () => {
152
+ const state1 = getFormState(2024)
153
+ const state2 = getFormState(null)
154
+
155
+ const payload1 = field.getFormDataFromState(state1)
156
+ const payload2 = field.getFormDataFromState(state2)
157
+
158
+ expect(payload1).toEqual(getFormData(2024))
159
+ expect(payload2).toEqual(getFormData())
160
+ })
161
+
162
+ it('returns value from state', () => {
163
+ const state1 = getFormState(2024)
164
+ const state2 = getFormState(null)
165
+
166
+ const value1 = field.getFormValueFromState(state1)
167
+ const value2 = field.getFormValueFromState(state2)
168
+
169
+ expect(value1).toBe(2024)
170
+ expect(value2).toBeUndefined()
171
+ })
172
+
173
+ it('returns context for conditions and form submission', () => {
174
+ const state1 = getFormState(2024)
175
+ const state2 = getFormState(null)
176
+
177
+ const value1 = field.getContextValueFromState(state1)
178
+ const value2 = field.getContextValueFromState(state2)
179
+
180
+ expect(value1).toBe(2024)
181
+ expect(value2).toBeNull()
182
+ })
183
+
184
+ it('returns state from payload', () => {
185
+ const payload1 = getFormData(2024)
186
+ const payload2 = getFormData()
187
+
188
+ const value1 = field.getStateFromValidForm(payload1)
189
+ const value2 = field.getStateFromValidForm(payload2)
190
+
191
+ expect(value1).toEqual(getFormState(2024))
192
+ expect(value2).toEqual(getFormState(null))
193
+ })
194
+ })
195
+
196
+ describe('View model', () => {
197
+ it('sets Nunjucks component defaults', () => {
198
+ const viewModel = field.getViewModel(getFormData(2024))
199
+
200
+ expect(viewModel).toEqual(
201
+ expect.objectContaining({
202
+ label: { text: def.title },
203
+ name: 'myComponent',
204
+ id: 'myComponent',
205
+ value: 2024
206
+ })
207
+ )
208
+ })
209
+
210
+ it('sets Nunjucks component prefix and suffix', () => {
211
+ const componentCustom = new NumberField(
212
+ { ...def, options: { prefix: '£', suffix: 'per item' } },
213
+ { model }
214
+ )
215
+
216
+ const viewModel = componentCustom.getViewModel(getFormData(99.99))
217
+
218
+ expect(viewModel.prefix).toEqual({ text: '£' })
219
+ expect(viewModel.suffix).toEqual({ text: 'per item' })
220
+ })
221
+
222
+ it('sets Nunjucks component inputmode attribute when precision is not defined', () => {
223
+ const componentCustom = new NumberField(
224
+ { ...def, schema: { precision: undefined } },
225
+ { model }
226
+ )
227
+
228
+ const viewModel = componentCustom.getViewModel(getFormData(99))
229
+
230
+ expect(viewModel.attributes).toHaveProperty('inputmode', 'numeric')
231
+ })
232
+
233
+ it('sets Nunjucks component inputmode attribute when precision is 0', () => {
234
+ const componentCustom = new NumberField(
235
+ { ...def, schema: { precision: 0 } },
236
+ { model }
237
+ )
238
+
239
+ const viewModel = componentCustom.getViewModel(getFormData(99))
240
+
241
+ expect(viewModel.attributes).toHaveProperty('inputmode', 'numeric')
242
+ })
243
+
244
+ it('does not set Nunjucks component inputmode attribute when precision is positive', () => {
245
+ const componentCustom = new NumberField(
246
+ { ...def, schema: { precision: 2 } },
247
+ { model }
248
+ )
249
+
250
+ const viewModel = componentCustom.getViewModel(getFormData(99.99))
251
+
252
+ expect(viewModel.attributes).not.toHaveProperty('inputmode', 'numeric')
253
+ })
254
+ })
255
+
256
+ it('sets Nunjucks component value when invalid', () => {
257
+ const viewModel = field.getViewModel(getFormData('AA'))
258
+
259
+ expect(viewModel).toHaveProperty('value', 'AA')
260
+ })
261
+ })
262
+
263
+ describe('Validation', () => {
264
+ describe.each([
265
+ {
266
+ description: 'Trim empty spaces',
267
+ component: {
268
+ title: 'Example number field',
269
+ name: 'myComponent',
270
+ type: ComponentType.NumberField,
271
+ options: {},
272
+ schema: {}
273
+ } satisfies NumberFieldComponent,
274
+ assertions: [
275
+ {
276
+ input: getFormData(' 2024'),
277
+ output: { value: getFormData(2024) }
278
+ },
279
+ {
280
+ input: getFormData('2024 '),
281
+ output: { value: getFormData(2024) }
282
+ },
283
+ {
284
+ input: getFormData(' 2024 \n\n'),
285
+ output: { value: getFormData(2024) }
286
+ }
287
+ ]
288
+ },
289
+ {
290
+ description: 'Number validation',
291
+ component: {
292
+ title: 'Example number field',
293
+ name: 'myComponent',
294
+ type: ComponentType.NumberField,
295
+ options: {},
296
+ schema: {}
297
+ } satisfies NumberFieldComponent,
298
+ assertions: [
299
+ {
300
+ input: getFormData('Not a number'),
301
+ output: {
302
+ value: getFormData('Not a number'),
303
+ errors: [
304
+ expect.objectContaining({
305
+ text: 'Example number field must be a number'
306
+ })
307
+ ]
308
+ }
309
+ },
310
+ {
311
+ input: getFormData('£99.99'),
312
+ output: {
313
+ value: getFormData('£99.99'),
314
+ errors: [
315
+ expect.objectContaining({
316
+ text: 'Example number field must be a number'
317
+ })
318
+ ]
319
+ }
320
+ },
321
+ {
322
+ input: getFormData('100.55'),
323
+ output: { value: getFormData(100.55) }
324
+ },
325
+ {
326
+ input: getFormData('3.14159'),
327
+ output: { value: getFormData(3.14159) }
328
+ }
329
+ ]
330
+ },
331
+ {
332
+ description: 'Schema precision (integers only)',
333
+ component: {
334
+ title: 'Example number field',
335
+ name: 'myComponent',
336
+ type: ComponentType.NumberField,
337
+ options: {},
338
+ schema: {
339
+ precision: 0
340
+ }
341
+ } satisfies NumberFieldComponent,
342
+ assertions: [
343
+ {
344
+ input: getFormData('3.14159'),
345
+ output: {
346
+ value: getFormData(3.14159),
347
+ errors: [
348
+ expect.objectContaining({
349
+ text: 'Example number field must be a whole number'
350
+ })
351
+ ]
352
+ }
353
+ },
354
+ {
355
+ input: getFormData('3'),
356
+ output: { value: getFormData(3) }
357
+ }
358
+ ]
359
+ },
360
+ {
361
+ description: 'Schema precision (integers only when negative)',
362
+ component: {
363
+ title: 'Example number field',
364
+ name: 'myComponent',
365
+ type: ComponentType.NumberField,
366
+ options: {},
367
+ schema: {
368
+ precision: -1
369
+ }
370
+ } satisfies NumberFieldComponent,
371
+ assertions: [
372
+ {
373
+ input: getFormData('3.14159'),
374
+ output: {
375
+ value: getFormData(3.14159),
376
+ errors: [
377
+ expect.objectContaining({
378
+ text: 'Example number field must be a whole number'
379
+ })
380
+ ]
381
+ }
382
+ },
383
+ {
384
+ input: getFormData('3'),
385
+ output: { value: getFormData(3) }
386
+ }
387
+ ]
388
+ },
389
+ {
390
+ description: 'Schema precision (1 decimal place)',
391
+ component: {
392
+ title: 'Example number field',
393
+ name: 'myComponent',
394
+ type: ComponentType.NumberField,
395
+ options: {},
396
+ schema: {
397
+ precision: 1
398
+ }
399
+ } satisfies NumberFieldComponent,
400
+ assertions: [
401
+ {
402
+ input: getFormData('3.14159'),
403
+ output: {
404
+ value: getFormData(3.14159),
405
+ errors: [
406
+ expect.objectContaining({
407
+ text: 'Example number field must have 1 or fewer decimal places'
408
+ })
409
+ ]
410
+ }
411
+ },
412
+ {
413
+ input: getFormData('3.1'),
414
+ output: { value: getFormData(3.1) }
415
+ }
416
+ ]
417
+ },
418
+ {
419
+ description: 'Schema precision (2 decimal places)',
420
+ component: {
421
+ title: 'Example number field',
422
+ name: 'myComponent',
423
+ type: ComponentType.NumberField,
424
+ options: {},
425
+ schema: {
426
+ precision: 2
427
+ }
428
+ } satisfies NumberFieldComponent,
429
+ assertions: [
430
+ {
431
+ input: getFormData('3.14159'),
432
+ output: {
433
+ value: getFormData(3.14159),
434
+ errors: [
435
+ expect.objectContaining({
436
+ text: 'Example number field must have 2 or fewer decimal places'
437
+ })
438
+ ]
439
+ }
440
+ },
441
+ {
442
+ input: getFormData('3.1'),
443
+ output: { value: getFormData(3.1) }
444
+ },
445
+ {
446
+ input: getFormData('3.14'),
447
+ output: { value: getFormData(3.14) }
448
+ }
449
+ ]
450
+ },
451
+ {
452
+ description: 'Schema precision with unsafe numbers',
453
+ component: {
454
+ title: 'Example number field',
455
+ name: 'myComponent',
456
+ type: ComponentType.NumberField,
457
+ options: {},
458
+ schema: {
459
+ precision: 2
460
+ }
461
+ } satisfies NumberFieldComponent,
462
+ assertions: [
463
+ {
464
+ input: getFormData('64811494532973582'),
465
+ output: {
466
+ value: getFormData(64811494532973580),
467
+ errors: [
468
+ expect.objectContaining({
469
+ text: 'Enter example number field in the correct format'
470
+ })
471
+ ]
472
+ }
473
+ },
474
+ {
475
+ input: getFormData('3.1'),
476
+ output: { value: getFormData(3.1) }
477
+ },
478
+ {
479
+ input: getFormData('3.14'),
480
+ output: { value: getFormData(3.14) }
481
+ }
482
+ ]
483
+ },
484
+ {
485
+ description: 'Schema min and max',
486
+ component: {
487
+ title: 'Example number field',
488
+ name: 'myComponent',
489
+ type: ComponentType.NumberField,
490
+ options: {},
491
+ schema: {
492
+ min: 5,
493
+ max: 8
494
+ }
495
+ } satisfies NumberFieldComponent,
496
+ assertions: [
497
+ {
498
+ input: getFormData('4'),
499
+ output: {
500
+ value: getFormData(4),
501
+ errors: [
502
+ expect.objectContaining({
503
+ text: 'Example number field must be 5 or higher'
504
+ })
505
+ ]
506
+ }
507
+ },
508
+ {
509
+ input: getFormData('10'),
510
+ output: {
511
+ value: getFormData(10),
512
+ errors: [
513
+ expect.objectContaining({
514
+ text: 'Example number field must be 8 or lower'
515
+ })
516
+ ]
517
+ }
518
+ }
519
+ ]
520
+ },
521
+ {
522
+ description: 'Custom validation message',
523
+ component: {
524
+ title: 'Example number field',
525
+ name: 'myComponent',
526
+ type: ComponentType.NumberField,
527
+ options: {
528
+ customValidationMessage: 'This is a custom error',
529
+ customValidationMessages: {
530
+ 'any.required': 'This is not used',
531
+ 'number.base': 'This is not used',
532
+ 'number.min': 'This is not used',
533
+ 'number.max': 'This is not used'
534
+ }
535
+ },
536
+ schema: {}
537
+ } satisfies NumberFieldComponent,
538
+ assertions: [
539
+ {
540
+ input: getFormData(''),
541
+ output: {
542
+ value: getFormData(''),
543
+ errors: [
544
+ expect.objectContaining({
545
+ text: 'This is a custom error'
546
+ })
547
+ ]
548
+ }
549
+ },
550
+ {
551
+ input: getFormData('AA'),
552
+ output: {
553
+ value: getFormData('AA'),
554
+ errors: [
555
+ expect.objectContaining({
556
+ text: 'This is a custom error'
557
+ })
558
+ ]
559
+ }
560
+ },
561
+ {
562
+ input: getFormData('invalid'),
563
+ output: {
564
+ value: getFormData('invalid'),
565
+ errors: [
566
+ expect.objectContaining({
567
+ text: 'This is a custom error'
568
+ })
569
+ ]
570
+ }
571
+ }
572
+ ]
573
+ },
574
+ {
575
+ description: 'Custom validation messages (multiple)',
576
+ component: {
577
+ title: 'Example number field',
578
+ name: 'myComponent',
579
+ type: ComponentType.NumberField,
580
+ options: {
581
+ customValidationMessages: {
582
+ 'any.required': 'This is a custom required error',
583
+ 'number.base': 'This is a custom number error',
584
+ 'number.max': 'This is a custom max number error',
585
+ 'number.min': 'This is a custom min number error'
586
+ }
587
+ },
588
+ schema: {
589
+ min: 5,
590
+ max: 8
591
+ }
592
+ } satisfies NumberFieldComponent,
593
+ assertions: [
594
+ {
595
+ input: getFormData(''),
596
+ output: {
597
+ value: getFormData(''),
598
+ errors: [
599
+ expect.objectContaining({
600
+ text: 'This is a custom required error'
601
+ })
602
+ ]
603
+ }
604
+ },
605
+ {
606
+ input: getFormData('AA'),
607
+ output: {
608
+ value: getFormData('AA'),
609
+ errors: [
610
+ expect.objectContaining({
611
+ text: 'This is a custom number error'
612
+ })
613
+ ]
614
+ }
615
+ },
616
+ {
617
+ input: getFormData('4'),
618
+ output: {
619
+ value: getFormData(4),
620
+ errors: [
621
+ expect.objectContaining({
622
+ text: 'This is a custom min number error'
623
+ })
624
+ ]
625
+ }
626
+ },
627
+ {
628
+ input: getFormData('10'),
629
+ output: {
630
+ value: getFormData(10),
631
+ errors: [
632
+ expect.objectContaining({
633
+ text: 'This is a custom max number error'
634
+ })
635
+ ]
636
+ }
637
+ }
638
+ ]
639
+ },
640
+ {
641
+ description: 'Custom validation overrides schema precision message',
642
+ component: {
643
+ title: 'Example number field',
644
+ name: 'myComponent',
645
+ type: ComponentType.NumberField,
646
+ options: {
647
+ customValidationMessage: 'This is a custom error'
648
+ },
649
+ schema: {
650
+ precision: 2
651
+ }
652
+ } satisfies NumberFieldComponent,
653
+ assertions: [
654
+ {
655
+ input: getFormData('3.14159'),
656
+ output: {
657
+ value: getFormData(3.14159),
658
+ errors: [
659
+ expect.objectContaining({
660
+ text: 'This is a custom error'
661
+ })
662
+ ]
663
+ }
664
+ }
665
+ ]
666
+ },
667
+ {
668
+ description: 'Optional field',
669
+ component: {
670
+ title: 'Example number field',
671
+ name: 'myComponent',
672
+ type: ComponentType.NumberField,
673
+ options: {
674
+ required: false
675
+ },
676
+ schema: {}
677
+ } satisfies NumberFieldComponent,
678
+ assertions: [
679
+ {
680
+ input: getFormData(''),
681
+ output: { value: getFormData('') }
682
+ }
683
+ ]
684
+ }
685
+ ])('$description', ({ component: def, assertions }) => {
686
+ let collection: ComponentCollection
687
+
688
+ beforeEach(() => {
689
+ collection = new ComponentCollection([def], { model })
690
+ })
691
+
692
+ it.each([...assertions])(
693
+ 'validates custom example',
694
+ ({ input, output }) => {
695
+ const result = collection.validate(input)
696
+ expect(result).toEqual(output)
697
+ }
698
+ )
699
+ })
700
+ })
701
+ })