@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,558 @@
1
+ import {
2
+ ComponentType,
3
+ type MultilineTextFieldComponent
4
+ } from '@defra/forms-model'
5
+
6
+ import { ComponentCollection } from '~/src/server/plugins/engine/components/ComponentCollection.js'
7
+ import { MultilineTextField } from '~/src/server/plugins/engine/components/MultilineTextField.js'
8
+ import {
9
+ getAnswer,
10
+ type Field
11
+ } from '~/src/server/plugins/engine/components/helpers.js'
12
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
13
+ import definition from '~/test/form/definitions/blank.js'
14
+ import { getFormData, getFormState } from '~/test/helpers/component-helpers.js'
15
+
16
+ describe('MultilineTextField', () => {
17
+ let model: FormModel
18
+
19
+ beforeEach(() => {
20
+ model = new FormModel(definition, {
21
+ basePath: 'test'
22
+ })
23
+ })
24
+
25
+ describe('Defaults', () => {
26
+ let def: MultilineTextFieldComponent
27
+ let collection: ComponentCollection
28
+ let field: Field
29
+
30
+ beforeEach(() => {
31
+ def = {
32
+ title: 'Example textarea',
33
+ name: 'myComponent',
34
+ type: ComponentType.MultilineTextField,
35
+ options: {},
36
+ schema: {}
37
+ } satisfies MultilineTextFieldComponent
38
+
39
+ collection = new ComponentCollection([def], { model })
40
+ field = collection.fields[0]
41
+ })
42
+
43
+ describe('Schema', () => {
44
+ it('uses component title as label', () => {
45
+ const { formSchema } = collection
46
+ const { keys } = formSchema.describe()
47
+
48
+ expect(keys).toHaveProperty(
49
+ 'myComponent',
50
+ expect.objectContaining({
51
+ flags: expect.objectContaining({
52
+ label: 'Example textarea'
53
+ })
54
+ })
55
+ )
56
+ })
57
+
58
+ it('uses component name as keys', () => {
59
+ const { formSchema } = collection
60
+ const { keys } = formSchema.describe()
61
+
62
+ expect(field.keys).toEqual(['myComponent'])
63
+ expect(field.collection).toBeUndefined()
64
+
65
+ for (const key of field.keys) {
66
+ expect(keys).toHaveProperty(key)
67
+ }
68
+ })
69
+
70
+ it('is required by default', () => {
71
+ const { formSchema } = collection
72
+ const { keys } = formSchema.describe()
73
+
74
+ expect(keys).toHaveProperty(
75
+ 'myComponent',
76
+ expect.objectContaining({
77
+ flags: expect.objectContaining({
78
+ presence: 'required'
79
+ })
80
+ })
81
+ )
82
+ })
83
+
84
+ it('is optional when configured', () => {
85
+ const collectionOptional = new ComponentCollection(
86
+ [{ ...def, options: { required: false } }],
87
+ { model }
88
+ )
89
+
90
+ const { formSchema } = collectionOptional
91
+ const { keys } = formSchema.describe()
92
+
93
+ expect(keys).toHaveProperty(
94
+ 'myComponent',
95
+ expect.objectContaining({ allow: [''] })
96
+ )
97
+
98
+ const result = collectionOptional.validate(getFormData(''))
99
+ expect(result.errors).toBeUndefined()
100
+ })
101
+
102
+ it('accepts valid values', () => {
103
+ const result1 = collection.validate(getFormData('Text'))
104
+ const result2 = collection.validate(getFormData('Textarea'))
105
+
106
+ expect(result1.errors).toBeUndefined()
107
+ expect(result2.errors).toBeUndefined()
108
+ })
109
+
110
+ it('adds errors for empty value', () => {
111
+ const result = collection.validate(getFormData(''))
112
+
113
+ expect(result.errors).toEqual([
114
+ expect.objectContaining({
115
+ text: 'Enter example textarea'
116
+ })
117
+ ])
118
+ })
119
+
120
+ it('adds errors for invalid values', () => {
121
+ const result1 = collection.validate(getFormData(['invalid']))
122
+ const result2 = collection.validate(
123
+ // @ts-expect-error - Allow invalid param for test
124
+ getFormData({ unknown: 'invalid' })
125
+ )
126
+
127
+ expect(result1.errors).toBeTruthy()
128
+ expect(result2.errors).toBeTruthy()
129
+ })
130
+ })
131
+
132
+ describe('State', () => {
133
+ it('returns text from state', () => {
134
+ const state1 = getFormState('Textarea')
135
+ const state2 = getFormState(null)
136
+
137
+ const answer1 = getAnswer(field, state1)
138
+ const answer2 = getAnswer(field, state2)
139
+
140
+ expect(answer1).toBe('Textarea')
141
+ expect(answer2).toBe('')
142
+ })
143
+
144
+ it('returns payload from state', () => {
145
+ const state1 = getFormState('Textarea')
146
+ const state2 = getFormState(null)
147
+
148
+ const payload1 = field.getFormDataFromState(state1)
149
+ const payload2 = field.getFormDataFromState(state2)
150
+
151
+ expect(payload1).toEqual(getFormData('Textarea'))
152
+ expect(payload2).toEqual(getFormData())
153
+ })
154
+
155
+ it('returns value from state', () => {
156
+ const state1 = getFormState('Textarea')
157
+ const state2 = getFormState(null)
158
+
159
+ const value1 = field.getFormValueFromState(state1)
160
+ const value2 = field.getFormValueFromState(state2)
161
+
162
+ expect(value1).toBe('Textarea')
163
+ expect(value2).toBeUndefined()
164
+ })
165
+
166
+ it('returns context for conditions and form submission', () => {
167
+ const state1 = getFormState('Textarea')
168
+ const state2 = getFormState(null)
169
+
170
+ const value1 = field.getContextValueFromState(state1)
171
+ const value2 = field.getContextValueFromState(state2)
172
+
173
+ expect(value1).toBe('Textarea')
174
+ expect(value2).toBeNull()
175
+ })
176
+
177
+ it('returns state from payload', () => {
178
+ const payload1 = getFormData('Textarea')
179
+ const payload2 = getFormData()
180
+
181
+ const value1 = field.getStateFromValidForm(payload1)
182
+ const value2 = field.getStateFromValidForm(payload2)
183
+
184
+ expect(value1).toEqual(getFormState('Textarea'))
185
+ expect(value2).toEqual(getFormState(null))
186
+ })
187
+ })
188
+
189
+ describe('View model', () => {
190
+ it('sets Nunjucks component defaults', () => {
191
+ const viewModel = field.getViewModel(getFormData('Textarea'))
192
+
193
+ expect(viewModel).toEqual(
194
+ expect.objectContaining({
195
+ label: { text: def.title },
196
+ name: 'myComponent',
197
+ id: 'myComponent',
198
+ value: 'Textarea'
199
+ })
200
+ )
201
+ })
202
+
203
+ it('sets Nunjucks component isCharacterOrWordCount: true', () => {
204
+ const componentCustom1 = new MultilineTextField(
205
+ { ...def, options: { maxWords: 10 } },
206
+ { model }
207
+ )
208
+
209
+ const componentCustom2 = new MultilineTextField(
210
+ { ...def, schema: { max: 10 } },
211
+ { model }
212
+ )
213
+
214
+ const viewModel = field.getViewModel(getFormData('Textarea'))
215
+
216
+ const viewModel1 = componentCustom1.getViewModel(
217
+ getFormData('Textarea custom #1')
218
+ )
219
+
220
+ const viewModel2 = componentCustom2.getViewModel(
221
+ getFormData('Textarea custom #2')
222
+ )
223
+
224
+ expect(viewModel).toEqual(
225
+ expect.objectContaining({ isCharacterOrWordCount: false })
226
+ )
227
+
228
+ expect(viewModel1).toEqual(
229
+ expect.objectContaining({ isCharacterOrWordCount: true })
230
+ )
231
+
232
+ expect(viewModel2).toEqual(
233
+ expect.objectContaining({ isCharacterOrWordCount: true })
234
+ )
235
+ })
236
+ })
237
+ })
238
+
239
+ describe('Validation', () => {
240
+ describe.each([
241
+ {
242
+ description: 'Trim empty spaces',
243
+ component: {
244
+ title: 'Example textarea',
245
+ name: 'myComponent',
246
+ type: ComponentType.MultilineTextField,
247
+ options: {},
248
+ schema: {}
249
+ } satisfies MultilineTextFieldComponent,
250
+ assertions: [
251
+ {
252
+ input: getFormData(' Leading spaces'),
253
+ output: { value: getFormData('Leading spaces') }
254
+ },
255
+ {
256
+ input: getFormData('Trailing spaces '),
257
+ output: { value: getFormData('Trailing spaces') }
258
+ },
259
+ {
260
+ input: getFormData(' Mixed spaces and new lines \n\n'),
261
+ output: { value: getFormData('Mixed spaces and new lines') }
262
+ }
263
+ ]
264
+ },
265
+ {
266
+ description: 'Option max words',
267
+ component: {
268
+ title: 'Example textarea',
269
+ name: 'myComponent',
270
+ type: ComponentType.MultilineTextField,
271
+ options: {
272
+ maxWords: 2
273
+ },
274
+ schema: {}
275
+ } satisfies MultilineTextFieldComponent,
276
+ assertions: [
277
+ {
278
+ input: getFormData('Textarea words'),
279
+ output: {
280
+ value: getFormData('Textarea words')
281
+ }
282
+ },
283
+ {
284
+ input: getFormData('Textarea too many words'),
285
+ output: {
286
+ value: getFormData('Textarea too many words'),
287
+ errors: [
288
+ expect.objectContaining({
289
+ text: 'Example textarea must be 2 words or fewer'
290
+ })
291
+ ]
292
+ }
293
+ }
294
+ ]
295
+ },
296
+ {
297
+ description: 'Schema min and max',
298
+ component: {
299
+ title: 'Example textarea',
300
+ name: 'myComponent',
301
+ type: ComponentType.MultilineTextField,
302
+ options: {},
303
+ schema: {
304
+ min: 5,
305
+ max: 8
306
+ }
307
+ } satisfies MultilineTextFieldComponent,
308
+ assertions: [
309
+ {
310
+ input: getFormData('Text'),
311
+ output: {
312
+ value: getFormData('Text'),
313
+ errors: [
314
+ expect.objectContaining({
315
+ text: 'Example textarea must be 5 characters or more'
316
+ })
317
+ ]
318
+ }
319
+ },
320
+ {
321
+ input: getFormData('Textarea too long'),
322
+ output: {
323
+ value: getFormData('Textarea too long'),
324
+ errors: [
325
+ expect.objectContaining({
326
+ text: 'Example textarea must be 8 characters or less'
327
+ })
328
+ ]
329
+ }
330
+ }
331
+ ]
332
+ },
333
+ {
334
+ description: 'Schema length',
335
+ component: {
336
+ title: 'Example textarea',
337
+ name: 'myComponent',
338
+ type: ComponentType.MultilineTextField,
339
+ options: {},
340
+ schema: {
341
+ length: 4
342
+ }
343
+ } satisfies MultilineTextFieldComponent,
344
+ assertions: [
345
+ {
346
+ input: getFormData('Text'),
347
+ output: { value: getFormData('Text') }
348
+ },
349
+ {
350
+ input: getFormData('Textarea'),
351
+ output: {
352
+ value: getFormData('Textarea'),
353
+ errors: [
354
+ expect.objectContaining({
355
+ text: 'Example textarea length must be 4 characters long'
356
+ })
357
+ ]
358
+ }
359
+ }
360
+ ]
361
+ },
362
+ {
363
+ description: 'Schema regex',
364
+ component: {
365
+ title: 'Example textarea',
366
+ name: 'myComponent',
367
+ type: ComponentType.MultilineTextField,
368
+ options: {},
369
+ schema: {
370
+ regex: '^[a-zA-Z]{1,2}\\d[a-zA-Z\\d]?\\s?\\d[a-zA-Z]{2}$'
371
+ }
372
+ } satisfies MultilineTextFieldComponent,
373
+ assertions: [
374
+ {
375
+ input: getFormData('SW1P'),
376
+ output: {
377
+ value: getFormData('SW1P'),
378
+ errors: [
379
+ expect.objectContaining({
380
+ text: 'Enter a valid example textarea'
381
+ })
382
+ ]
383
+ }
384
+ },
385
+ {
386
+ input: getFormData('SW1P 4DF'),
387
+ output: { value: getFormData('SW1P 4DF') }
388
+ }
389
+ ]
390
+ },
391
+ {
392
+ description: 'Custom validation message',
393
+ component: {
394
+ title: 'Example textarea',
395
+ name: 'myComponent',
396
+ type: ComponentType.MultilineTextField,
397
+ options: {
398
+ customValidationMessage: 'This is a custom error',
399
+ customValidationMessages: {
400
+ 'any.required': 'This is not used',
401
+ 'string.empty': 'This is not used',
402
+ 'string.max': 'This is not used',
403
+ 'string.min': 'This is not used'
404
+ }
405
+ },
406
+ schema: {
407
+ min: 5,
408
+ max: 8
409
+ }
410
+ } satisfies MultilineTextFieldComponent,
411
+ assertions: [
412
+ {
413
+ input: getFormData(),
414
+ output: {
415
+ value: getFormData(''),
416
+ errors: [
417
+ expect.objectContaining({
418
+ text: 'This is a custom error'
419
+ })
420
+ ]
421
+ }
422
+ },
423
+ {
424
+ input: getFormData(''),
425
+ output: {
426
+ value: getFormData(''),
427
+ errors: [
428
+ expect.objectContaining({
429
+ text: 'This is a custom error'
430
+ })
431
+ ]
432
+ }
433
+ },
434
+ {
435
+ input: getFormData('Text'),
436
+ output: {
437
+ value: getFormData('Text'),
438
+ errors: [
439
+ expect.objectContaining({
440
+ text: 'This is a custom error'
441
+ })
442
+ ]
443
+ }
444
+ },
445
+ {
446
+ input: getFormData('Textarea too long'),
447
+ output: {
448
+ value: getFormData('Textarea too long'),
449
+ errors: [
450
+ expect.objectContaining({
451
+ text: 'This is a custom error'
452
+ })
453
+ ]
454
+ }
455
+ }
456
+ ]
457
+ },
458
+ {
459
+ description: 'Custom validation messages (multiple)',
460
+ component: {
461
+ title: 'Example textarea',
462
+ name: 'myComponent',
463
+ type: ComponentType.MultilineTextField,
464
+ options: {
465
+ customValidationMessages: {
466
+ 'any.required': 'This is a custom required error',
467
+ 'string.empty': 'This is a custom empty string error',
468
+ 'string.max': 'This is a custom max length error',
469
+ 'string.min': 'This is a custom min length error'
470
+ }
471
+ },
472
+ schema: {
473
+ min: 5,
474
+ max: 8
475
+ }
476
+ } satisfies MultilineTextFieldComponent,
477
+ assertions: [
478
+ {
479
+ input: getFormData(),
480
+ output: {
481
+ value: getFormData(''),
482
+ errors: [
483
+ expect.objectContaining({
484
+ text: 'This is a custom required error'
485
+ })
486
+ ]
487
+ }
488
+ },
489
+ {
490
+ input: getFormData(''),
491
+ output: {
492
+ value: getFormData(''),
493
+ errors: [
494
+ expect.objectContaining({
495
+ text: 'This is a custom empty string error'
496
+ })
497
+ ]
498
+ }
499
+ },
500
+ {
501
+ input: getFormData('Text'),
502
+ output: {
503
+ value: getFormData('Text'),
504
+ errors: [
505
+ expect.objectContaining({
506
+ text: 'This is a custom min length error'
507
+ })
508
+ ]
509
+ }
510
+ },
511
+ {
512
+ input: getFormData('Textarea too long'),
513
+ output: {
514
+ value: getFormData('Textarea too long'),
515
+ errors: [
516
+ expect.objectContaining({
517
+ text: 'This is a custom max length error'
518
+ })
519
+ ]
520
+ }
521
+ }
522
+ ]
523
+ },
524
+ {
525
+ description: 'Optional field',
526
+ component: {
527
+ title: 'Example textarea',
528
+ name: 'myComponent',
529
+ type: ComponentType.MultilineTextField,
530
+ options: {
531
+ required: false
532
+ },
533
+ schema: {}
534
+ } satisfies MultilineTextFieldComponent,
535
+ assertions: [
536
+ {
537
+ input: getFormData(''),
538
+ output: { value: getFormData('') }
539
+ }
540
+ ]
541
+ }
542
+ ])('$description', ({ component: def, assertions }) => {
543
+ let collection: ComponentCollection
544
+
545
+ beforeEach(() => {
546
+ collection = new ComponentCollection([def], { model })
547
+ })
548
+
549
+ it.each([...assertions])(
550
+ 'validates custom example',
551
+ ({ input, output }) => {
552
+ const result = collection.validate(input)
553
+ expect(result).toEqual(output)
554
+ }
555
+ )
556
+ })
557
+ })
558
+ })
@@ -0,0 +1,138 @@
1
+ import { type MultilineTextFieldComponent } from '@defra/forms-model'
2
+ import Joi, { type CustomValidator, type StringSchema } from 'joi'
3
+
4
+ import { type ComponentBase } from '~/src/server/plugins/engine/components/ComponentBase.js'
5
+ import { FormComponent } from '~/src/server/plugins/engine/components/FormComponent.js'
6
+ import {
7
+ type FormPayload,
8
+ type FormSubmissionError
9
+ } from '~/src/server/plugins/engine/types.js'
10
+
11
+ export class MultilineTextField extends FormComponent {
12
+ declare options: MultilineTextFieldComponent['options']
13
+ declare schema: MultilineTextFieldComponent['schema']
14
+ declare formSchema: StringSchema
15
+ declare stateSchema: StringSchema
16
+
17
+ isCharacterOrWordCount = false
18
+
19
+ constructor(
20
+ def: MultilineTextFieldComponent,
21
+ props: ConstructorParameters<typeof ComponentBase>[1]
22
+ ) {
23
+ super(def, props)
24
+
25
+ const { schema, options, title } = def
26
+
27
+ let formSchema = Joi.string().trim().label(title).required()
28
+
29
+ if (options.required === false) {
30
+ formSchema = formSchema.allow('')
31
+ }
32
+
33
+ if (typeof schema.length !== 'number') {
34
+ if (typeof schema.max === 'number') {
35
+ formSchema = formSchema.max(schema.max)
36
+ this.isCharacterOrWordCount = true
37
+ }
38
+
39
+ if (typeof schema.min === 'number') {
40
+ formSchema = formSchema.min(schema.min)
41
+ }
42
+ } else {
43
+ formSchema = formSchema.length(schema.length)
44
+ }
45
+
46
+ if (typeof options.maxWords === 'number') {
47
+ formSchema = formSchema.custom(
48
+ getValidatorMaxWords(this),
49
+ 'max words validation'
50
+ )
51
+
52
+ this.isCharacterOrWordCount = true
53
+ }
54
+
55
+ if (schema.regex) {
56
+ const pattern = new RegExp(schema.regex)
57
+ formSchema = formSchema.pattern(pattern)
58
+ }
59
+
60
+ if (options.customValidationMessage) {
61
+ const message = options.customValidationMessage
62
+
63
+ formSchema = formSchema.messages({
64
+ 'any.required': message,
65
+ 'string.empty': message,
66
+ 'string.max': message,
67
+ 'string.min': message,
68
+ 'string.length': message,
69
+ 'string.pattern.base': message,
70
+ 'string.maxWords': message
71
+ })
72
+ } else if (options.customValidationMessages) {
73
+ formSchema = formSchema.messages(options.customValidationMessages)
74
+ }
75
+
76
+ this.formSchema = formSchema.default('')
77
+ this.stateSchema = formSchema.default(null).allow(null)
78
+ this.options = options
79
+ this.schema = schema
80
+ }
81
+
82
+ getViewModel(payload: FormPayload, errors?: FormSubmissionError[]) {
83
+ const { schema, options, isCharacterOrWordCount } = this
84
+
85
+ const viewModel = super.getViewModel(payload, errors)
86
+ let { maxlength, maxwords, rows } = viewModel
87
+
88
+ if (schema.max) {
89
+ maxlength = schema.max
90
+ }
91
+
92
+ if (options.maxWords) {
93
+ maxwords = options.maxWords
94
+ }
95
+
96
+ if (options.rows) {
97
+ rows = options.rows
98
+ }
99
+
100
+ return {
101
+ ...viewModel,
102
+ isCharacterOrWordCount,
103
+ maxlength,
104
+ maxwords,
105
+ rows
106
+ }
107
+ }
108
+ }
109
+
110
+ function getValidatorMaxWords(component: MultilineTextField) {
111
+ const validator: CustomValidator = (value: string, helpers) => {
112
+ const { options } = component
113
+
114
+ const {
115
+ customValidationMessage: custom,
116
+ maxWords: limit // See {{#limit}} variable
117
+ } = options
118
+
119
+ if (!limit || count(value) <= limit) {
120
+ return value
121
+ }
122
+
123
+ return custom
124
+ ? helpers.message({ custom }, { limit })
125
+ : helpers.error('string.maxWords', { limit })
126
+ }
127
+
128
+ /**
129
+ * Count the number of words in the given text
130
+ * @see GOV.UK Frontend {@link https://github.com/alphagov/govuk-frontend/blob/v5.4.0/packages/govuk-frontend/src/govuk/components/character-count/character-count.mjs#L343 | Character count `maxwords` implementation}
131
+ */
132
+ function count(text: string) {
133
+ const tokens = text.match(/\S+/g) ?? []
134
+ return tokens.length
135
+ }
136
+
137
+ return validator
138
+ }