@defra/forms-engine-plugin 0.0.3 → 0.0.4

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 (328) hide show
  1. package/.public/assets/fonts/bold-affa96571d-v2.woff +0 -0
  2. package/.public/assets/fonts/bold-b542beb274-v2.woff2 +0 -0
  3. package/.public/assets/fonts/light-94a07e06a1-v2.woff2 +0 -0
  4. package/.public/assets/fonts/light-f591b13f7d-v2.woff +0 -0
  5. package/.public/assets/images/favicon.ico +0 -0
  6. package/.public/assets/images/favicon.svg +1 -0
  7. package/.public/assets/images/govuk-crest.svg +1 -0
  8. package/.public/assets/images/govuk-icon-180.png +0 -0
  9. package/.public/assets/images/govuk-icon-192.png +0 -0
  10. package/.public/assets/images/govuk-icon-512.png +0 -0
  11. package/.public/assets/images/govuk-icon-mask.svg +1 -0
  12. package/.public/assets/images/govuk-opengraph-image.png +0 -0
  13. package/.public/assets/manifest.json +39 -0
  14. package/.public/assets-manifest.json +24 -0
  15. package/.public/javascripts/application.0fd8c18.min.js +3 -0
  16. package/.public/javascripts/application.0fd8c18.min.js.LICENSE.txt +58 -0
  17. package/.public/javascripts/application.0fd8c18.min.js.map +1 -0
  18. package/.public/javascripts/file-upload.b2f18f5.min.js +2 -0
  19. package/.public/javascripts/file-upload.b2f18f5.min.js.map +1 -0
  20. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js +2 -0
  21. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js.map +1 -0
  22. package/.public/stylesheets/application.e340021.min.css +14 -0
  23. package/.public/stylesheets/application.e340021.min.css.map +1 -0
  24. package/package.json +6 -2
  25. package/.browserslistrc +0 -16
  26. package/.editorconfig +0 -9
  27. package/.eslintrc.cjs +0 -266
  28. package/.github/dependabot.yml +0 -85
  29. package/.github/workflows/check-pull-request.yml +0 -209
  30. package/.github/workflows/pr-notifier.yml +0 -98
  31. package/.github/workflows/publish.yml +0 -111
  32. package/.husky/pre-commit +0 -1
  33. package/.lintstagedrc.js +0 -4
  34. package/.nvmrc +0 -1
  35. package/.prettierignore +0 -8
  36. package/.prettierrc.cjs +0 -26
  37. package/Dockerfile +0 -61
  38. package/Procfile +0 -1
  39. package/babel.config.cjs +0 -55
  40. package/compose/aws.env +0 -4
  41. package/compose/start-localstack.sh +0 -26
  42. package/docker-compose.yaml +0 -86
  43. package/globals.d.ts +0 -1
  44. package/jest.config.cjs +0 -54
  45. package/jest.environment.js +0 -4
  46. package/jest.setup.cjs +0 -14
  47. package/postcss.config.js +0 -26
  48. package/sonar-project.properties +0 -17
  49. package/src/client/javascripts/application.js +0 -87
  50. package/src/client/javascripts/file-upload.js +0 -386
  51. package/src/client/stylesheets/_code.scss +0 -33
  52. package/src/client/stylesheets/_govuk-frontend.scss +0 -4
  53. package/src/client/stylesheets/_prose.scss +0 -56
  54. package/src/client/stylesheets/_service-banner.scss +0 -24
  55. package/src/client/stylesheets/_summary-list.scss +0 -28
  56. package/src/client/stylesheets/_tag-env.scss +0 -24
  57. package/src/client/stylesheets/application.scss +0 -14
  58. package/src/common/cookies.js +0 -58
  59. package/src/common/cookies.test.js +0 -23
  60. package/src/common/types.js +0 -5
  61. package/src/config/index.ts +0 -271
  62. package/src/index.ts +0 -31
  63. package/src/server/common/helpers/logging/logger-options.test.ts +0 -50
  64. package/src/server/common/helpers/logging/logger-options.ts +0 -46
  65. package/src/server/common/helpers/logging/logger.ts +0 -7
  66. package/src/server/common/helpers/logging/request-logger.ts +0 -9
  67. package/src/server/common/helpers/logging/request-tracing.js +0 -10
  68. package/src/server/common/helpers/redis-client.js +0 -70
  69. package/src/server/constants.js +0 -1
  70. package/src/server/forms/README.md +0 -10
  71. package/src/server/forms/components.json +0 -1015
  72. package/src/server/forms/report-a-terrorist.json +0 -270
  73. package/src/server/forms/runner-components-test.json +0 -365
  74. package/src/server/forms/test.json +0 -581
  75. package/src/server/index.test.ts +0 -582
  76. package/src/server/index.ts +0 -140
  77. package/src/server/plugins/blankie.test.ts +0 -73
  78. package/src/server/plugins/blankie.ts +0 -48
  79. package/src/server/plugins/crumb.ts +0 -20
  80. package/src/server/plugins/engine/README.md +0 -87
  81. package/src/server/plugins/engine/components/AutocompleteField.test.ts +0 -294
  82. package/src/server/plugins/engine/components/AutocompleteField.ts +0 -49
  83. package/src/server/plugins/engine/components/CheckboxesField.test.ts +0 -379
  84. package/src/server/plugins/engine/components/CheckboxesField.ts +0 -106
  85. package/src/server/plugins/engine/components/ComponentBase.ts +0 -97
  86. package/src/server/plugins/engine/components/ComponentCollection.ts +0 -278
  87. package/src/server/plugins/engine/components/DatePartsField.test.ts +0 -822
  88. package/src/server/plugins/engine/components/DatePartsField.ts +0 -264
  89. package/src/server/plugins/engine/components/Details.test.ts +0 -49
  90. package/src/server/plugins/engine/components/Details.ts +0 -30
  91. package/src/server/plugins/engine/components/EmailAddressField.test.ts +0 -395
  92. package/src/server/plugins/engine/components/EmailAddressField.ts +0 -55
  93. package/src/server/plugins/engine/components/FileUploadField.test.ts +0 -778
  94. package/src/server/plugins/engine/components/FileUploadField.ts +0 -262
  95. package/src/server/plugins/engine/components/FormComponent.ts +0 -249
  96. package/src/server/plugins/engine/components/Html.test.ts +0 -48
  97. package/src/server/plugins/engine/components/Html.ts +0 -29
  98. package/src/server/plugins/engine/components/InsetText.test.ts +0 -48
  99. package/src/server/plugins/engine/components/InsetText.ts +0 -27
  100. package/src/server/plugins/engine/components/List.test.ts +0 -76
  101. package/src/server/plugins/engine/components/List.ts +0 -72
  102. package/src/server/plugins/engine/components/ListFormComponent.ts +0 -140
  103. package/src/server/plugins/engine/components/MonthYearField.test.ts +0 -567
  104. package/src/server/plugins/engine/components/MonthYearField.ts +0 -222
  105. package/src/server/plugins/engine/components/MultilineTextField.test.ts +0 -558
  106. package/src/server/plugins/engine/components/MultilineTextField.ts +0 -138
  107. package/src/server/plugins/engine/components/NumberField.test.ts +0 -701
  108. package/src/server/plugins/engine/components/NumberField.ts +0 -163
  109. package/src/server/plugins/engine/components/RadiosField.test.ts +0 -288
  110. package/src/server/plugins/engine/components/RadiosField.ts +0 -24
  111. package/src/server/plugins/engine/components/SelectField.test.ts +0 -288
  112. package/src/server/plugins/engine/components/SelectField.ts +0 -47
  113. package/src/server/plugins/engine/components/SelectionControlField.ts +0 -43
  114. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +0 -356
  115. package/src/server/plugins/engine/components/TelephoneNumberField.ts +0 -67
  116. package/src/server/plugins/engine/components/TextField.test.ts +0 -489
  117. package/src/server/plugins/engine/components/TextField.ts +0 -96
  118. package/src/server/plugins/engine/components/UkAddressField.test.ts +0 -623
  119. package/src/server/plugins/engine/components/UkAddressField.ts +0 -172
  120. package/src/server/plugins/engine/components/YesNoField.test.ts +0 -248
  121. package/src/server/plugins/engine/components/YesNoField.ts +0 -31
  122. package/src/server/plugins/engine/components/constants.ts +0 -1
  123. package/src/server/plugins/engine/components/helpers.ts +0 -330
  124. package/src/server/plugins/engine/components/index.ts +0 -24
  125. package/src/server/plugins/engine/components/types.ts +0 -117
  126. package/src/server/plugins/engine/configureEnginePlugin.ts +0 -47
  127. package/src/server/plugins/engine/helpers.test.ts +0 -791
  128. package/src/server/plugins/engine/helpers.ts +0 -379
  129. package/src/server/plugins/engine/index.ts +0 -7
  130. package/src/server/plugins/engine/models/FormModel.test.ts +0 -42
  131. package/src/server/plugins/engine/models/FormModel.ts +0 -443
  132. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  133. package/src/server/plugins/engine/models/Section.ts +0 -0
  134. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +0 -209
  135. package/src/server/plugins/engine/models/SummaryViewModel.ts +0 -220
  136. package/src/server/plugins/engine/models/index.ts +0 -2
  137. package/src/server/plugins/engine/models/types.ts +0 -114
  138. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +0 -143
  139. package/src/server/plugins/engine/outputFormatters/human/v1.ts +0 -73
  140. package/src/server/plugins/engine/outputFormatters/index.test.ts +0 -17
  141. package/src/server/plugins/engine/outputFormatters/index.ts +0 -44
  142. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +0 -229
  143. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +0 -140
  144. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +0 -229
  145. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +0 -153
  146. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +0 -1108
  147. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +0 -446
  148. package/src/server/plugins/engine/pageControllers/PageController.test.ts +0 -205
  149. package/src/server/plugins/engine/pageControllers/PageController.ts +0 -176
  150. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +0 -1264
  151. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +0 -561
  152. package/src/server/plugins/engine/pageControllers/README.md +0 -28
  153. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +0 -264
  154. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +0 -458
  155. package/src/server/plugins/engine/pageControllers/StartPageController.ts +0 -18
  156. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +0 -50
  157. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +0 -261
  158. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +0 -28
  159. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +0 -19
  160. package/src/server/plugins/engine/pageControllers/helpers.test.ts +0 -198
  161. package/src/server/plugins/engine/pageControllers/helpers.ts +0 -101
  162. package/src/server/plugins/engine/pageControllers/index.ts +0 -10
  163. package/src/server/plugins/engine/pageControllers/validationOptions.ts +0 -89
  164. package/src/server/plugins/engine/plugin.ts +0 -673
  165. package/src/server/plugins/engine/services/formSubmissionService.js +0 -46
  166. package/src/server/plugins/engine/services/formsService.js +0 -46
  167. package/src/server/plugins/engine/services/formsService.test.js +0 -90
  168. package/src/server/plugins/engine/services/index.js +0 -3
  169. package/src/server/plugins/engine/services/notifyService.test.ts +0 -132
  170. package/src/server/plugins/engine/services/notifyService.ts +0 -64
  171. package/src/server/plugins/engine/services/uploadService.js +0 -60
  172. package/src/server/plugins/engine/types.ts +0 -315
  173. package/src/server/plugins/engine/views/components/autocompletefield.html +0 -5
  174. package/src/server/plugins/engine/views/components/checkboxesfield.html +0 -5
  175. package/src/server/plugins/engine/views/components/datepartsfield.html +0 -5
  176. package/src/server/plugins/engine/views/components/details.html +0 -6
  177. package/src/server/plugins/engine/views/components/emailaddressfield.html +0 -5
  178. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +0 -8
  179. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +0 -3
  180. package/src/server/plugins/engine/views/components/fileuploadfield.html +0 -24
  181. package/src/server/plugins/engine/views/components/html.html +0 -3
  182. package/src/server/plugins/engine/views/components/insettext.html +0 -7
  183. package/src/server/plugins/engine/views/components/list.html +0 -36
  184. package/src/server/plugins/engine/views/components/monthyearfield.html +0 -5
  185. package/src/server/plugins/engine/views/components/multilinetextfield.html +0 -10
  186. package/src/server/plugins/engine/views/components/numberfield.html +0 -5
  187. package/src/server/plugins/engine/views/components/radiosfield.html +0 -5
  188. package/src/server/plugins/engine/views/components/selectfield.html +0 -5
  189. package/src/server/plugins/engine/views/components/telephonenumberfield.html +0 -5
  190. package/src/server/plugins/engine/views/components/textfield.html +0 -5
  191. package/src/server/plugins/engine/views/components/ukaddressfield.html +0 -25
  192. package/src/server/plugins/engine/views/components/yesnofield.html +0 -5
  193. package/src/server/plugins/engine/views/file-upload.html +0 -45
  194. package/src/server/plugins/engine/views/index.html +0 -39
  195. package/src/server/plugins/engine/views/item-delete.html +0 -56
  196. package/src/server/plugins/engine/views/partials/components.html +0 -6
  197. package/src/server/plugins/engine/views/partials/conditional-components.html +0 -3
  198. package/src/server/plugins/engine/views/partials/debug.html +0 -44
  199. package/src/server/plugins/engine/views/partials/form.html +0 -15
  200. package/src/server/plugins/engine/views/partials/heading.html +0 -16
  201. package/src/server/plugins/engine/views/partials/preview-banner.html +0 -32
  202. package/src/server/plugins/engine/views/partials/preview-banner.test.js +0 -122
  203. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +0 -10
  204. package/src/server/plugins/engine/views/repeat-list-summary.html +0 -53
  205. package/src/server/plugins/errorPages.ts +0 -58
  206. package/src/server/plugins/nunjucks/context.js +0 -88
  207. package/src/server/plugins/nunjucks/context.test.js +0 -142
  208. package/src/server/plugins/nunjucks/enviroment.test.js +0 -201
  209. package/src/server/plugins/nunjucks/environment.js +0 -116
  210. package/src/server/plugins/nunjucks/filters/answer.js +0 -27
  211. package/src/server/plugins/nunjucks/filters/answer.test.js +0 -89
  212. package/src/server/plugins/nunjucks/filters/evaluate.js +0 -21
  213. package/src/server/plugins/nunjucks/filters/field.js +0 -28
  214. package/src/server/plugins/nunjucks/filters/field.test.js +0 -75
  215. package/src/server/plugins/nunjucks/filters/highlight.js +0 -11
  216. package/src/server/plugins/nunjucks/filters/href.js +0 -30
  217. package/src/server/plugins/nunjucks/filters/href.test.js +0 -80
  218. package/src/server/plugins/nunjucks/filters/index.js +0 -8
  219. package/src/server/plugins/nunjucks/filters/inspect.js +0 -15
  220. package/src/server/plugins/nunjucks/filters/page.js +0 -24
  221. package/src/server/plugins/nunjucks/filters/page.test.js +0 -65
  222. package/src/server/plugins/nunjucks/index.js +0 -3
  223. package/src/server/plugins/nunjucks/plugin.js +0 -40
  224. package/src/server/plugins/nunjucks/render.js +0 -42
  225. package/src/server/plugins/nunjucks/types.js +0 -40
  226. package/src/server/plugins/pulse.ts +0 -11
  227. package/src/server/plugins/router.ts +0 -201
  228. package/src/server/plugins/session.ts +0 -28
  229. package/src/server/routes/health.js +0 -13
  230. package/src/server/routes/health.test.js +0 -35
  231. package/src/server/routes/index.test.ts +0 -125
  232. package/src/server/routes/index.ts +0 -2
  233. package/src/server/routes/public.ts +0 -47
  234. package/src/server/routes/types.ts +0 -48
  235. package/src/server/schemas/index.ts +0 -34
  236. package/src/server/secure-context.js +0 -43
  237. package/src/server/services/cacheService.test.ts +0 -276
  238. package/src/server/services/cacheService.ts +0 -131
  239. package/src/server/services/httpService.test.js +0 -491
  240. package/src/server/services/httpService.ts +0 -50
  241. package/src/server/services/index.ts +0 -1
  242. package/src/server/types.ts +0 -54
  243. package/src/server/utils/notify.test.ts +0 -37
  244. package/src/server/utils/notify.ts +0 -50
  245. package/src/server/utils/secure-context/get-trust-store-certs.js +0 -11
  246. package/src/server/utils/secure-context/get-trust-store-certs.test.js +0 -19
  247. package/src/server/utils/utils.js +0 -24
  248. package/src/server/utils/utils.test.js +0 -54
  249. package/src/server/views/404.html +0 -16
  250. package/src/server/views/500.html +0 -19
  251. package/src/server/views/components/debug/macro.njk +0 -3
  252. package/src/server/views/components/debug/template.njk +0 -13
  253. package/src/server/views/components/service-banner/macro.njk +0 -3
  254. package/src/server/views/components/service-banner/template.njk +0 -20
  255. package/src/server/views/components/service-banner/template.test.js +0 -43
  256. package/src/server/views/components/tag-env/macro.njk +0 -3
  257. package/src/server/views/components/tag-env/template.njk +0 -30
  258. package/src/server/views/components/tag-env/template.test.js +0 -66
  259. package/src/server/views/confirmation.html +0 -19
  260. package/src/server/views/help/accessibility-statement.html +0 -58
  261. package/src/server/views/help/cookie-preferences.html +0 -57
  262. package/src/server/views/help/cookies.html +0 -71
  263. package/src/server/views/help/get-support.html +0 -37
  264. package/src/server/views/help/privacy-notice.html +0 -68
  265. package/src/server/views/help/terms-and-conditions.html +0 -83
  266. package/src/server/views/layout.html +0 -199
  267. package/src/server/views/summary.html +0 -50
  268. package/src/typings/hapi/index.d.ts +0 -95
  269. package/src/typings/hapi-tracing/index.d.ts +0 -6
  270. package/src/typings/index.d.ts +0 -3
  271. package/src/typings/joi/index.d.ts +0 -22
  272. package/stylelint.config.js +0 -10
  273. package/test/client/javascripts/file-upload.test.js +0 -1197
  274. package/test/condition/checkboxes.test.js +0 -112
  275. package/test/condition/radios.test.js +0 -112
  276. package/test/condition/text.test.js +0 -103
  277. package/test/fixtures/assets-manifest.json +0 -4
  278. package/test/fixtures/form.js +0 -86
  279. package/test/fixtures/index.js +0 -2
  280. package/test/fixtures/list.js +0 -92
  281. package/test/form/cookies.test.js +0 -338
  282. package/test/form/csrf.test.js +0 -87
  283. package/test/form/definitions/basic.js +0 -101
  284. package/test/form/definitions/blank.js +0 -10
  285. package/test/form/definitions/checkboxes.json +0 -88
  286. package/test/form/definitions/components.json +0 -452
  287. package/test/form/definitions/conditional-reveal.js +0 -140
  288. package/test/form/definitions/conditions-basic.js +0 -187
  289. package/test/form/definitions/conditions-complex.js +0 -338
  290. package/test/form/definitions/conditions-dates.js +0 -78
  291. package/test/form/definitions/conditions-escaping.js +0 -143
  292. package/test/form/definitions/demo-cph-number.js +0 -3099
  293. package/test/form/definitions/feedback.json +0 -45
  294. package/test/form/definitions/fields-optional.js +0 -402
  295. package/test/form/definitions/fields-required.js +0 -402
  296. package/test/form/definitions/file-upload-basic.js +0 -44
  297. package/test/form/definitions/file-upload.js +0 -66
  298. package/test/form/definitions/minimal.js +0 -39
  299. package/test/form/definitions/phase-alpha.json +0 -33
  300. package/test/form/definitions/phase-default.json +0 -26
  301. package/test/form/definitions/radios.json +0 -88
  302. package/test/form/definitions/repeat-mixed.js +0 -54
  303. package/test/form/definitions/repeat.js +0 -70
  304. package/test/form/definitions/status.json +0 -126
  305. package/test/form/definitions/templates.js +0 -183
  306. package/test/form/definitions/test.json +0 -581
  307. package/test/form/definitions/text.json +0 -75
  308. package/test/form/definitions/titles.json +0 -170
  309. package/test/form/definitions.test.js +0 -47
  310. package/test/form/exit-page.test.js +0 -210
  311. package/test/form/feedback.test.js +0 -68
  312. package/test/form/fields-optional.test.js +0 -237
  313. package/test/form/fields-required.test.js +0 -294
  314. package/test/form/file-upload.test.js +0 -313
  315. package/test/form/govuk-notify.test.js +0 -449
  316. package/test/form/journey-basic.test.js +0 -444
  317. package/test/form/persist-files.test.js +0 -227
  318. package/test/form/phase-banner.test.js +0 -71
  319. package/test/form/repeat.test.js +0 -628
  320. package/test/form/summary-submission-email.test.js +0 -95
  321. package/test/form/template.test.js +0 -288
  322. package/test/form/titles.test.js +0 -204
  323. package/test/helpers/component-helpers.js +0 -74
  324. package/test/utils/get-cookie.js +0 -42
  325. package/test/utils/get-form-definitions.js +0 -18
  326. package/tmp.pdf +0 -1
  327. package/tsconfig.json +0 -28
  328. package/webpack.config.js +0 -208
@@ -1,701 +0,0 @@
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
- })