@defra/forms-engine-plugin 0.0.4 → 0.0.5

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 (224) hide show
  1. package/package.json +3 -2
  2. package/src/client/javascripts/application.js +87 -0
  3. package/src/client/javascripts/file-upload.js +386 -0
  4. package/src/client/stylesheets/_code.scss +33 -0
  5. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  6. package/src/client/stylesheets/_prose.scss +56 -0
  7. package/src/client/stylesheets/_service-banner.scss +24 -0
  8. package/src/client/stylesheets/_summary-list.scss +28 -0
  9. package/src/client/stylesheets/_tag-env.scss +24 -0
  10. package/src/client/stylesheets/application.scss +14 -0
  11. package/src/common/cookies.js +58 -0
  12. package/src/common/cookies.test.js +23 -0
  13. package/src/common/types.js +5 -0
  14. package/src/config/index.ts +271 -0
  15. package/src/index.ts +31 -0
  16. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  17. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  18. package/src/server/common/helpers/logging/logger.ts +7 -0
  19. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  20. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  21. package/src/server/common/helpers/redis-client.js +70 -0
  22. package/src/server/constants.js +1 -0
  23. package/src/server/forms/README.md +10 -0
  24. package/src/server/forms/components.json +1015 -0
  25. package/src/server/forms/report-a-terrorist.json +270 -0
  26. package/src/server/forms/runner-components-test.json +365 -0
  27. package/src/server/forms/test.json +581 -0
  28. package/src/server/index.test.ts +582 -0
  29. package/src/server/index.ts +140 -0
  30. package/src/server/plugins/blankie.test.ts +73 -0
  31. package/src/server/plugins/blankie.ts +48 -0
  32. package/src/server/plugins/crumb.ts +20 -0
  33. package/src/server/plugins/engine/README.md +87 -0
  34. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  35. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  36. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  37. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  38. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  39. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  40. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  41. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  42. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  43. package/src/server/plugins/engine/components/Details.ts +30 -0
  44. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  45. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  46. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  47. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  48. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  49. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  50. package/src/server/plugins/engine/components/Html.ts +29 -0
  51. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  52. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  53. package/src/server/plugins/engine/components/List.test.ts +76 -0
  54. package/src/server/plugins/engine/components/List.ts +72 -0
  55. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  56. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  57. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  58. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  59. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  60. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  61. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  62. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  63. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  64. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  65. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  66. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  67. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  68. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  69. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  70. package/src/server/plugins/engine/components/TextField.ts +96 -0
  71. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  72. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  73. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  74. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  75. package/src/server/plugins/engine/components/constants.ts +1 -0
  76. package/src/server/plugins/engine/components/helpers.ts +330 -0
  77. package/src/server/plugins/engine/components/index.ts +24 -0
  78. package/src/server/plugins/engine/components/types.ts +117 -0
  79. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  80. package/src/server/plugins/engine/helpers.test.ts +791 -0
  81. package/src/server/plugins/engine/helpers.ts +379 -0
  82. package/src/server/plugins/engine/index.ts +7 -0
  83. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  84. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  85. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  86. package/src/server/plugins/engine/models/Section.ts +0 -0
  87. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  88. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  89. package/src/server/plugins/engine/models/index.ts +2 -0
  90. package/src/server/plugins/engine/models/types.ts +114 -0
  91. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  92. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  93. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  94. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  95. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  96. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  97. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  98. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  99. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1108 -0
  100. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +446 -0
  101. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  102. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  103. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  104. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +561 -0
  105. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  106. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  107. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  108. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  109. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +50 -0
  110. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +261 -0
  111. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  112. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  113. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  114. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  115. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  116. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  117. package/src/server/plugins/engine/plugin.ts +673 -0
  118. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  119. package/src/server/plugins/engine/services/formsService.js +46 -0
  120. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  121. package/src/server/plugins/engine/services/index.js +3 -0
  122. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  123. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  124. package/src/server/plugins/engine/services/uploadService.js +60 -0
  125. package/src/server/plugins/engine/types.ts +315 -0
  126. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  127. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  128. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  129. package/src/server/plugins/engine/views/components/details.html +6 -0
  130. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  131. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  132. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  133. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  134. package/src/server/plugins/engine/views/components/html.html +3 -0
  135. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  136. package/src/server/plugins/engine/views/components/list.html +36 -0
  137. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  138. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  139. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  140. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  141. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  142. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  143. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  144. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  145. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  146. package/src/server/plugins/engine/views/file-upload.html +45 -0
  147. package/src/server/plugins/engine/views/index.html +39 -0
  148. package/src/server/plugins/engine/views/item-delete.html +56 -0
  149. package/src/server/plugins/engine/views/partials/components.html +6 -0
  150. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  151. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  152. package/src/server/plugins/engine/views/partials/form.html +15 -0
  153. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  154. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  155. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  156. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  157. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  158. package/src/server/plugins/errorPages.ts +58 -0
  159. package/src/server/plugins/nunjucks/context.js +88 -0
  160. package/src/server/plugins/nunjucks/context.test.js +142 -0
  161. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  162. package/src/server/plugins/nunjucks/environment.js +116 -0
  163. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  164. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  165. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  166. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  167. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  168. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  169. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  170. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  171. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  172. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  173. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  174. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  175. package/src/server/plugins/nunjucks/index.js +3 -0
  176. package/src/server/plugins/nunjucks/plugin.js +40 -0
  177. package/src/server/plugins/nunjucks/render.js +42 -0
  178. package/src/server/plugins/nunjucks/types.js +40 -0
  179. package/src/server/plugins/pulse.ts +11 -0
  180. package/src/server/plugins/router.ts +201 -0
  181. package/src/server/plugins/session.ts +28 -0
  182. package/src/server/routes/health.js +13 -0
  183. package/src/server/routes/health.test.js +35 -0
  184. package/src/server/routes/index.test.ts +125 -0
  185. package/src/server/routes/index.ts +2 -0
  186. package/src/server/routes/public.ts +47 -0
  187. package/src/server/routes/types.ts +48 -0
  188. package/src/server/schemas/index.ts +34 -0
  189. package/src/server/secure-context.js +43 -0
  190. package/src/server/services/cacheService.test.ts +276 -0
  191. package/src/server/services/cacheService.ts +131 -0
  192. package/src/server/services/httpService.test.js +491 -0
  193. package/src/server/services/httpService.ts +50 -0
  194. package/src/server/services/index.ts +1 -0
  195. package/src/server/types.ts +54 -0
  196. package/src/server/utils/notify.test.ts +37 -0
  197. package/src/server/utils/notify.ts +50 -0
  198. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  199. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  200. package/src/server/utils/utils.js +24 -0
  201. package/src/server/utils/utils.test.js +54 -0
  202. package/src/server/views/404.html +16 -0
  203. package/src/server/views/500.html +19 -0
  204. package/src/server/views/components/debug/macro.njk +3 -0
  205. package/src/server/views/components/debug/template.njk +13 -0
  206. package/src/server/views/components/service-banner/macro.njk +3 -0
  207. package/src/server/views/components/service-banner/template.njk +20 -0
  208. package/src/server/views/components/service-banner/template.test.js +43 -0
  209. package/src/server/views/components/tag-env/macro.njk +3 -0
  210. package/src/server/views/components/tag-env/template.njk +30 -0
  211. package/src/server/views/components/tag-env/template.test.js +66 -0
  212. package/src/server/views/confirmation.html +19 -0
  213. package/src/server/views/help/accessibility-statement.html +58 -0
  214. package/src/server/views/help/cookie-preferences.html +57 -0
  215. package/src/server/views/help/cookies.html +71 -0
  216. package/src/server/views/help/get-support.html +37 -0
  217. package/src/server/views/help/privacy-notice.html +68 -0
  218. package/src/server/views/help/terms-and-conditions.html +83 -0
  219. package/src/server/views/layout.html +199 -0
  220. package/src/server/views/summary.html +50 -0
  221. package/src/typings/hapi/index.d.ts +95 -0
  222. package/src/typings/hapi-tracing/index.d.ts +6 -0
  223. package/src/typings/index.d.ts +3 -0
  224. package/src/typings/joi/index.d.ts +22 -0
@@ -0,0 +1,356 @@
1
+ import {
2
+ ComponentType,
3
+ type TelephoneNumberFieldComponent
4
+ } from '@defra/forms-model'
5
+
6
+ import { ComponentCollection } from '~/src/server/plugins/engine/components/ComponentCollection.js'
7
+ import {
8
+ getAnswer,
9
+ type Field
10
+ } from '~/src/server/plugins/engine/components/helpers.js'
11
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
12
+ import definition from '~/test/form/definitions/blank.js'
13
+ import { getFormData, getFormState } from '~/test/helpers/component-helpers.js'
14
+
15
+ describe('TelephoneNumberField', () => {
16
+ let model: FormModel
17
+
18
+ beforeEach(() => {
19
+ model = new FormModel(definition, {
20
+ basePath: 'test'
21
+ })
22
+ })
23
+
24
+ describe('Defaults', () => {
25
+ let def: TelephoneNumberFieldComponent
26
+ let collection: ComponentCollection
27
+ let field: Field
28
+
29
+ beforeEach(() => {
30
+ def = {
31
+ title: 'Example telephone number field',
32
+ name: 'myComponent',
33
+ type: ComponentType.TelephoneNumberField,
34
+ options: {}
35
+ } satisfies TelephoneNumberFieldComponent
36
+
37
+ collection = new ComponentCollection([def], { model })
38
+ field = collection.fields[0]
39
+ })
40
+
41
+ describe('Schema', () => {
42
+ it('uses component title as label', () => {
43
+ const { formSchema } = collection
44
+ const { keys } = formSchema.describe()
45
+
46
+ expect(keys).toHaveProperty(
47
+ 'myComponent',
48
+ expect.objectContaining({
49
+ flags: expect.objectContaining({
50
+ label: 'Example telephone number field'
51
+ })
52
+ })
53
+ )
54
+ })
55
+
56
+ it('uses component name as keys', () => {
57
+ const { formSchema } = collection
58
+ const { keys } = formSchema.describe()
59
+
60
+ expect(field.keys).toEqual(['myComponent'])
61
+ expect(field.collection).toBeUndefined()
62
+
63
+ for (const key of field.keys) {
64
+ expect(keys).toHaveProperty(key)
65
+ }
66
+ })
67
+
68
+ it('is required by default', () => {
69
+ const { formSchema } = collection
70
+ const { keys } = formSchema.describe()
71
+
72
+ expect(keys).toHaveProperty(
73
+ 'myComponent',
74
+ expect.objectContaining({
75
+ flags: expect.objectContaining({
76
+ presence: 'required'
77
+ })
78
+ })
79
+ )
80
+ })
81
+
82
+ it('is optional when configured', () => {
83
+ const collectionOptional = new ComponentCollection(
84
+ [{ ...def, options: { required: false } }],
85
+ { model }
86
+ )
87
+
88
+ const { formSchema } = collectionOptional
89
+ const { keys } = formSchema.describe()
90
+
91
+ expect(keys).toHaveProperty(
92
+ 'myComponent',
93
+ expect.objectContaining({ allow: [''] })
94
+ )
95
+
96
+ const result = collectionOptional.validate(getFormData(''))
97
+ expect(result.errors).toBeUndefined()
98
+ })
99
+
100
+ it.each([
101
+ '+111-111-11',
102
+ '+111 111 11',
103
+ '+11111111',
104
+ '+44 7930 111 222',
105
+ '07930 111 222',
106
+ '01606 76543',
107
+ '01606 765432',
108
+ '0203 765 443',
109
+ '0800 123 321',
110
+ '(01606) 765432',
111
+ '(01606) 765-432',
112
+ '01606 765-432',
113
+ '+44203-765-443',
114
+ '0800123-321',
115
+ '0800-123-321'
116
+ ])("accepts valid value '%s'", (value) => {
117
+ const result = collection.validate(getFormData(value))
118
+ expect(result.errors).toBeUndefined()
119
+ })
120
+
121
+ it('adds errors for empty value', () => {
122
+ const result = collection.validate(getFormData(''))
123
+
124
+ expect(result.errors).toEqual([
125
+ expect.objectContaining({
126
+ text: 'Enter example telephone number field'
127
+ })
128
+ ])
129
+ })
130
+
131
+ it('adds errors for invalid values', () => {
132
+ const result1 = collection.validate(getFormData('invalid'))
133
+ const result2 = collection.validate(
134
+ // @ts-expect-error - Allow invalid param for test
135
+ getFormData({ unknown: 'invalid' })
136
+ )
137
+
138
+ expect(result1.errors).toBeTruthy()
139
+ expect(result2.errors).toBeTruthy()
140
+ })
141
+ })
142
+
143
+ describe('State', () => {
144
+ it('returns text from state', () => {
145
+ const state1 = getFormState('+447900000000')
146
+ const state2 = getFormState(null)
147
+
148
+ const answer1 = getAnswer(field, state1)
149
+ const answer2 = getAnswer(field, state2)
150
+
151
+ expect(answer1).toBe('+447900000000')
152
+ expect(answer2).toBe('')
153
+ })
154
+
155
+ it('returns payload from state', () => {
156
+ const state1 = getFormState('+447900000000')
157
+ const state2 = getFormState(null)
158
+
159
+ const payload1 = field.getFormDataFromState(state1)
160
+ const payload2 = field.getFormDataFromState(state2)
161
+
162
+ expect(payload1).toEqual(getFormData('+447900000000'))
163
+ expect(payload2).toEqual(getFormData())
164
+ })
165
+
166
+ it('returns value from state', () => {
167
+ const state1 = getFormState('+447900000000')
168
+ const state2 = getFormState(null)
169
+
170
+ const value1 = field.getFormValueFromState(state1)
171
+ const value2 = field.getFormValueFromState(state2)
172
+
173
+ expect(value1).toBe('+447900000000')
174
+ expect(value2).toBeUndefined()
175
+ })
176
+
177
+ it('returns context for conditions and form submission', () => {
178
+ const state1 = getFormState('+447900000000')
179
+ const state2 = getFormState(null)
180
+
181
+ const value1 = field.getContextValueFromState(state1)
182
+ const value2 = field.getContextValueFromState(state2)
183
+
184
+ expect(value1).toBe('+447900000000')
185
+ expect(value2).toBeNull()
186
+ })
187
+
188
+ it('returns state from payload', () => {
189
+ const payload1 = getFormData('+447900000000')
190
+ const payload2 = getFormData()
191
+
192
+ const value1 = field.getStateFromValidForm(payload1)
193
+ const value2 = field.getStateFromValidForm(payload2)
194
+
195
+ expect(value1).toEqual(getFormState('+447900000000'))
196
+ expect(value2).toEqual(getFormState(null))
197
+ })
198
+ })
199
+
200
+ describe('View model', () => {
201
+ it('sets Nunjucks component defaults', () => {
202
+ const viewModel = field.getViewModel(
203
+ getFormData('Telephone number field')
204
+ )
205
+
206
+ expect(viewModel).toEqual(
207
+ expect.objectContaining({
208
+ label: { text: def.title },
209
+ name: 'myComponent',
210
+ id: 'myComponent',
211
+ value: 'Telephone number field',
212
+ attributes: { autocomplete: 'tel' },
213
+ type: 'tel'
214
+ })
215
+ )
216
+ })
217
+ })
218
+ })
219
+
220
+ describe('Validation', () => {
221
+ describe.each([
222
+ {
223
+ description: 'Custom validation message',
224
+ component: {
225
+ title: 'Example telephone number field',
226
+ name: 'myComponent',
227
+ type: ComponentType.TelephoneNumberField,
228
+ options: {
229
+ customValidationMessage: 'This is a custom error',
230
+ customValidationMessages: {
231
+ 'any.required': 'This is not used',
232
+ 'string.empty': 'This is not used',
233
+ 'string.pattern.base': 'This is not used'
234
+ }
235
+ }
236
+ } satisfies TelephoneNumberFieldComponent,
237
+ assertions: [
238
+ {
239
+ input: getFormData(),
240
+ output: {
241
+ value: getFormData(''),
242
+ errors: [
243
+ expect.objectContaining({
244
+ text: 'This is a custom error'
245
+ })
246
+ ]
247
+ }
248
+ },
249
+ {
250
+ input: getFormData(''),
251
+ output: {
252
+ value: getFormData(''),
253
+ errors: [
254
+ expect.objectContaining({
255
+ text: 'This is a custom error'
256
+ })
257
+ ]
258
+ }
259
+ },
260
+ {
261
+ input: getFormData('AA'),
262
+ output: {
263
+ value: getFormData('AA'),
264
+ errors: [
265
+ expect.objectContaining({
266
+ text: 'This is a custom error'
267
+ })
268
+ ]
269
+ }
270
+ }
271
+ ]
272
+ },
273
+ {
274
+ description: 'Custom validation messages (multiple)',
275
+ component: {
276
+ title: 'Example telephone number field',
277
+ name: 'myComponent',
278
+ type: ComponentType.TelephoneNumberField,
279
+ options: {
280
+ customValidationMessages: {
281
+ 'any.required': 'This is a custom required error',
282
+ 'string.empty': 'This is a custom empty string error',
283
+ 'string.pattern.base': 'This is a custom pattern error'
284
+ }
285
+ }
286
+ } satisfies TelephoneNumberFieldComponent,
287
+ assertions: [
288
+ {
289
+ input: getFormData(),
290
+ output: {
291
+ value: getFormData(''),
292
+ errors: [
293
+ expect.objectContaining({
294
+ text: 'This is a custom required error'
295
+ })
296
+ ]
297
+ }
298
+ },
299
+ {
300
+ input: getFormData(''),
301
+ output: {
302
+ value: getFormData(''),
303
+ errors: [
304
+ expect.objectContaining({
305
+ text: 'This is a custom empty string error'
306
+ })
307
+ ]
308
+ }
309
+ },
310
+ {
311
+ input: getFormData('AA'),
312
+ output: {
313
+ value: getFormData('AA'),
314
+ errors: [
315
+ expect.objectContaining({
316
+ text: 'This is a custom pattern error'
317
+ })
318
+ ]
319
+ }
320
+ }
321
+ ]
322
+ },
323
+ {
324
+ description: 'Optional field',
325
+ component: {
326
+ title: 'Example telephone number field',
327
+ name: 'myComponent',
328
+ type: ComponentType.TelephoneNumberField,
329
+ options: {
330
+ required: false
331
+ }
332
+ } satisfies TelephoneNumberFieldComponent,
333
+ assertions: [
334
+ {
335
+ input: getFormData(''),
336
+ output: { value: getFormData('') }
337
+ }
338
+ ]
339
+ }
340
+ ])('$description', ({ component: def, assertions }) => {
341
+ let collection: ComponentCollection
342
+
343
+ beforeEach(() => {
344
+ collection = new ComponentCollection([def], { model })
345
+ })
346
+
347
+ it.each([...assertions])(
348
+ 'validates custom example',
349
+ ({ input, output }) => {
350
+ const result = collection.validate(input)
351
+ expect(result).toEqual(output)
352
+ }
353
+ )
354
+ })
355
+ })
356
+ })
@@ -0,0 +1,67 @@
1
+ import { type TelephoneNumberFieldComponent } from '@defra/forms-model'
2
+ import joi, { type StringSchema } from 'joi'
3
+
4
+ import { FormComponent } from '~/src/server/plugins/engine/components/FormComponent.js'
5
+ import { addClassOptionIfNone } from '~/src/server/plugins/engine/components/helpers.js'
6
+ import {
7
+ type FormPayload,
8
+ type FormSubmissionError
9
+ } from '~/src/server/plugins/engine/types.js'
10
+
11
+ const PATTERN = /^[0-9\\\s+()-]*$/
12
+
13
+ export class TelephoneNumberField extends FormComponent {
14
+ declare options: TelephoneNumberFieldComponent['options']
15
+ declare formSchema: StringSchema
16
+ declare stateSchema: StringSchema
17
+
18
+ constructor(
19
+ def: TelephoneNumberFieldComponent,
20
+ props: ConstructorParameters<typeof FormComponent>[1]
21
+ ) {
22
+ super(def, props)
23
+
24
+ const { options, title } = def
25
+
26
+ let formSchema = joi
27
+ .string()
28
+ .trim()
29
+ .pattern(PATTERN)
30
+ .label(title)
31
+ .required()
32
+
33
+ if (options.required === false) {
34
+ formSchema = formSchema.allow('')
35
+ }
36
+
37
+ if (options.customValidationMessage) {
38
+ const message = options.customValidationMessage
39
+
40
+ formSchema = formSchema.messages({
41
+ 'any.required': message,
42
+ 'string.empty': message,
43
+ 'string.pattern.base': message
44
+ })
45
+ } else if (options.customValidationMessages) {
46
+ formSchema = formSchema.messages(options.customValidationMessages)
47
+ }
48
+
49
+ addClassOptionIfNone(options, 'govuk-input--width-20')
50
+
51
+ this.formSchema = formSchema.default('')
52
+ this.stateSchema = formSchema.default(null).allow(null)
53
+ this.options = options
54
+ }
55
+
56
+ getViewModel(payload: FormPayload, errors?: FormSubmissionError[]) {
57
+ const viewModel = super.getViewModel(payload, errors)
58
+ const { attributes } = viewModel
59
+
60
+ attributes.autocomplete = 'tel'
61
+
62
+ return {
63
+ ...viewModel,
64
+ type: 'tel'
65
+ }
66
+ }
67
+ }