@toptal/picasso-forms 66.3.2 → 67.0.0

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 (552) hide show
  1. package/{Autocomplete → dist-package/src/Autocomplete}/Autocomplete.d.ts +1 -0
  2. package/dist-package/src/Autocomplete/Autocomplete.d.ts.map +1 -0
  3. package/dist-package/src/Autocomplete/Autocomplete.js.map +1 -0
  4. package/{Autocomplete → dist-package/src/Autocomplete}/index.d.ts +1 -0
  5. package/dist-package/src/Autocomplete/index.d.ts.map +1 -0
  6. package/dist-package/src/Autocomplete/index.js.map +1 -0
  7. package/{AvatarUpload → dist-package/src/AvatarUpload}/AvatarUpload.d.ts +1 -0
  8. package/dist-package/src/AvatarUpload/AvatarUpload.d.ts.map +1 -0
  9. package/dist-package/src/AvatarUpload/AvatarUpload.js.map +1 -0
  10. package/{AvatarUpload → dist-package/src/AvatarUpload}/index.d.ts +1 -0
  11. package/dist-package/src/AvatarUpload/index.d.ts.map +1 -0
  12. package/dist-package/src/AvatarUpload/index.js.map +1 -0
  13. package/{ButtonCheckbox → dist-package/src/ButtonCheckbox}/ButtonCheckbox.d.ts +1 -0
  14. package/dist-package/src/ButtonCheckbox/ButtonCheckbox.d.ts.map +1 -0
  15. package/dist-package/src/ButtonCheckbox/ButtonCheckbox.js.map +1 -0
  16. package/{ButtonCheckbox → dist-package/src/ButtonCheckbox}/index.d.ts +1 -0
  17. package/dist-package/src/ButtonCheckbox/index.d.ts.map +1 -0
  18. package/dist-package/src/ButtonCheckbox/index.js.map +1 -0
  19. package/{ButtonRadio → dist-package/src/ButtonRadio}/ButtonRadio.d.ts +1 -0
  20. package/dist-package/src/ButtonRadio/ButtonRadio.d.ts.map +1 -0
  21. package/dist-package/src/ButtonRadio/ButtonRadio.js.map +1 -0
  22. package/{ButtonRadio → dist-package/src/ButtonRadio}/index.d.ts +1 -0
  23. package/dist-package/src/ButtonRadio/index.d.ts.map +1 -0
  24. package/dist-package/src/ButtonRadio/index.js.map +1 -0
  25. package/{Checkbox → dist-package/src/Checkbox}/Checkbox.d.ts +1 -0
  26. package/dist-package/src/Checkbox/Checkbox.d.ts.map +1 -0
  27. package/dist-package/src/Checkbox/Checkbox.js.map +1 -0
  28. package/dist-package/src/Checkbox/index.d.ts +3 -0
  29. package/dist-package/src/Checkbox/index.d.ts.map +1 -0
  30. package/dist-package/src/Checkbox/index.js.map +1 -0
  31. package/{CheckboxGroup → dist-package/src/CheckboxGroup}/CheckboxGroup.d.ts +1 -0
  32. package/dist-package/src/CheckboxGroup/CheckboxGroup.d.ts.map +1 -0
  33. package/dist-package/src/CheckboxGroup/CheckboxGroup.js.map +1 -0
  34. package/{CheckboxGroup → dist-package/src/CheckboxGroup}/CheckboxGroupContext.d.ts +1 -0
  35. package/dist-package/src/CheckboxGroup/CheckboxGroupContext.d.ts.map +1 -0
  36. package/dist-package/src/CheckboxGroup/CheckboxGroupContext.js.map +1 -0
  37. package/{CheckboxGroup → dist-package/src/CheckboxGroup}/index.d.ts +1 -0
  38. package/dist-package/src/CheckboxGroup/index.d.ts.map +1 -0
  39. package/dist-package/src/CheckboxGroup/index.js.map +1 -0
  40. package/{DatePicker → dist-package/src/DatePicker}/DatePicker.d.ts +1 -0
  41. package/dist-package/src/DatePicker/DatePicker.d.ts.map +1 -0
  42. package/dist-package/src/DatePicker/DatePicker.js.map +1 -0
  43. package/{DatePicker → dist-package/src/DatePicker}/index.d.ts +1 -0
  44. package/dist-package/src/DatePicker/index.d.ts.map +1 -0
  45. package/dist-package/src/DatePicker/index.js.map +1 -0
  46. package/{Dropzone → dist-package/src/Dropzone}/Dropzone.d.ts +1 -0
  47. package/dist-package/src/Dropzone/Dropzone.d.ts.map +1 -0
  48. package/dist-package/src/Dropzone/Dropzone.js.map +1 -0
  49. package/{Dropzone → dist-package/src/Dropzone}/index.d.ts +1 -0
  50. package/dist-package/src/Dropzone/index.d.ts.map +1 -0
  51. package/dist-package/src/Dropzone/index.js.map +1 -0
  52. package/{Field → dist-package/src/Field}/Field.d.ts +1 -0
  53. package/dist-package/src/Field/Field.d.ts.map +1 -0
  54. package/dist-package/src/Field/Field.js.map +1 -0
  55. package/{Field → dist-package/src/Field}/index.d.ts +1 -0
  56. package/dist-package/src/Field/index.d.ts.map +1 -0
  57. package/dist-package/src/Field/index.js.map +1 -0
  58. package/dist-package/src/FieldBase/index.d.ts +2 -0
  59. package/dist-package/src/FieldBase/index.d.ts.map +1 -0
  60. package/dist-package/src/FieldBase/index.js.map +1 -0
  61. package/dist-package/src/FieldBase/types.d.ts +6 -0
  62. package/dist-package/src/FieldBase/types.d.ts.map +1 -0
  63. package/{FieldBase → dist-package/src/FieldBase}/types.js.map +1 -1
  64. package/{FieldLabel → dist-package/src/FieldLabel}/FieldLabel.d.ts +1 -0
  65. package/dist-package/src/FieldLabel/FieldLabel.d.ts.map +1 -0
  66. package/dist-package/src/FieldLabel/FieldLabel.js.map +1 -0
  67. package/{FieldLabel → dist-package/src/FieldLabel}/index.d.ts +1 -0
  68. package/dist-package/src/FieldLabel/index.d.ts.map +1 -0
  69. package/dist-package/src/FieldLabel/index.js.map +1 -0
  70. package/{FieldWrapper → dist-package/src/FieldWrapper}/FieldWrapper.d.ts +1 -0
  71. package/dist-package/src/FieldWrapper/FieldWrapper.d.ts.map +1 -0
  72. package/dist-package/src/FieldWrapper/FieldWrapper.js.map +1 -0
  73. package/{FieldWrapper → dist-package/src/FieldWrapper}/index.d.ts +1 -0
  74. package/dist-package/src/FieldWrapper/index.d.ts.map +1 -0
  75. package/dist-package/src/FieldWrapper/index.js.map +1 -0
  76. package/{FileInput → dist-package/src/FileInput}/FileInput.d.ts +1 -0
  77. package/dist-package/src/FileInput/FileInput.d.ts.map +1 -0
  78. package/dist-package/src/FileInput/FileInput.js.map +1 -0
  79. package/{FileInput → dist-package/src/FileInput}/index.d.ts +1 -0
  80. package/dist-package/src/FileInput/index.d.ts.map +1 -0
  81. package/dist-package/src/FileInput/index.js.map +1 -0
  82. package/{Form → dist-package/src/Form}/Form.d.ts +1 -0
  83. package/dist-package/src/Form/Form.d.ts.map +1 -0
  84. package/{Form → dist-package/src/Form}/Form.js +1 -1
  85. package/dist-package/src/Form/Form.js.map +1 -0
  86. package/{Form → dist-package/src/Form}/FormContext.d.ts +1 -0
  87. package/dist-package/src/Form/FormContext.d.ts.map +1 -0
  88. package/dist-package/src/Form/FormContext.js.map +1 -0
  89. package/{Form → dist-package/src/Form}/FormRenderer.d.ts +1 -0
  90. package/dist-package/src/Form/FormRenderer.d.ts.map +1 -0
  91. package/dist-package/src/Form/FormRenderer.js.map +1 -0
  92. package/{Form → dist-package/src/Form}/index.d.ts +1 -0
  93. package/dist-package/src/Form/index.d.ts.map +1 -0
  94. package/dist-package/src/Form/index.js.map +1 -0
  95. package/{Form → dist-package/src/Form}/mutators/index.d.ts +1 -0
  96. package/dist-package/src/Form/mutators/index.d.ts.map +1 -0
  97. package/dist-package/src/Form/mutators/index.js.map +1 -0
  98. package/{Form → dist-package/src/Form}/mutators/set-active-field-touched.d.ts +1 -0
  99. package/dist-package/src/Form/mutators/set-active-field-touched.d.ts.map +1 -0
  100. package/dist-package/src/Form/mutators/set-active-field-touched.js.map +1 -0
  101. package/{Form → dist-package/src/Form}/mutators/set-has-multiline-counter.d.ts +1 -0
  102. package/dist-package/src/Form/mutators/set-has-multiline-counter.d.ts.map +1 -0
  103. package/dist-package/src/Form/mutators/set-has-multiline-counter.js.map +1 -0
  104. package/{FormCompound → dist-package/src/FormCompound}/index.d.ts +7 -6
  105. package/dist-package/src/FormCompound/index.d.ts.map +1 -0
  106. package/dist-package/src/FormCompound/index.js.map +1 -0
  107. package/{FormConfig → dist-package/src/FormConfig}/FormConfig.d.ts +1 -0
  108. package/dist-package/src/FormConfig/FormConfig.d.ts.map +1 -0
  109. package/dist-package/src/FormConfig/FormConfig.js.map +1 -0
  110. package/dist-package/src/FormConfig/index.d.ts +2 -0
  111. package/dist-package/src/FormConfig/index.d.ts.map +1 -0
  112. package/dist-package/src/FormConfig/index.js.map +1 -0
  113. package/{Input → dist-package/src/Input}/Input.d.ts +1 -0
  114. package/dist-package/src/Input/Input.d.ts.map +1 -0
  115. package/dist-package/src/Input/Input.js.map +1 -0
  116. package/{Input → dist-package/src/Input}/index.d.ts +1 -0
  117. package/dist-package/src/Input/index.d.ts.map +1 -0
  118. package/dist-package/src/Input/index.js.map +1 -0
  119. package/{InputField → dist-package/src/InputField}/InputField.d.ts +1 -0
  120. package/dist-package/src/InputField/InputField.d.ts.map +1 -0
  121. package/dist-package/src/InputField/InputField.js.map +1 -0
  122. package/{InputField → dist-package/src/InputField}/index.d.ts +1 -0
  123. package/dist-package/src/InputField/index.d.ts.map +1 -0
  124. package/dist-package/src/InputField/index.js.map +1 -0
  125. package/{NumberInput → dist-package/src/NumberInput}/NumberInput.d.ts +1 -0
  126. package/dist-package/src/NumberInput/NumberInput.d.ts.map +1 -0
  127. package/dist-package/src/NumberInput/NumberInput.js.map +1 -0
  128. package/dist-package/src/NumberInput/index.d.ts +3 -0
  129. package/dist-package/src/NumberInput/index.d.ts.map +1 -0
  130. package/dist-package/src/NumberInput/index.js.map +1 -0
  131. package/{PasswordInput → dist-package/src/PasswordInput}/FieldRenderer.d.ts +1 -0
  132. package/dist-package/src/PasswordInput/FieldRenderer.d.ts.map +1 -0
  133. package/dist-package/src/PasswordInput/FieldRenderer.js.map +1 -0
  134. package/{PasswordInput → dist-package/src/PasswordInput}/PasswordInput.d.ts +1 -0
  135. package/dist-package/src/PasswordInput/PasswordInput.d.ts.map +1 -0
  136. package/dist-package/src/PasswordInput/PasswordInput.js.map +1 -0
  137. package/{PasswordInput → dist-package/src/PasswordInput}/index.d.ts +1 -0
  138. package/dist-package/src/PasswordInput/index.d.ts.map +1 -0
  139. package/dist-package/src/PasswordInput/index.js.map +1 -0
  140. package/{PasswordInput → dist-package/src/PasswordInput}/validators.d.ts +1 -0
  141. package/dist-package/src/PasswordInput/validators.d.ts.map +1 -0
  142. package/dist-package/src/PasswordInput/validators.js.map +1 -0
  143. package/{Radio → dist-package/src/Radio}/Radio.d.ts +1 -0
  144. package/dist-package/src/Radio/Radio.d.ts.map +1 -0
  145. package/dist-package/src/Radio/Radio.js.map +1 -0
  146. package/{Radio → dist-package/src/Radio}/index.d.ts +1 -0
  147. package/dist-package/src/Radio/index.d.ts.map +1 -0
  148. package/dist-package/src/Radio/index.js.map +1 -0
  149. package/{RadioGroup → dist-package/src/RadioGroup}/RadioGroup.d.ts +1 -0
  150. package/dist-package/src/RadioGroup/RadioGroup.d.ts.map +1 -0
  151. package/dist-package/src/RadioGroup/RadioGroup.js.map +1 -0
  152. package/{RadioGroup → dist-package/src/RadioGroup}/RadioGroupContext.d.ts +1 -0
  153. package/dist-package/src/RadioGroup/RadioGroupContext.d.ts.map +1 -0
  154. package/dist-package/src/RadioGroup/RadioGroupContext.js.map +1 -0
  155. package/{RadioGroup → dist-package/src/RadioGroup}/index.d.ts +1 -0
  156. package/dist-package/src/RadioGroup/index.d.ts.map +1 -0
  157. package/dist-package/src/RadioGroup/index.js.map +1 -0
  158. package/{Rating → dist-package/src/Rating}/Rating.d.ts +1 -0
  159. package/dist-package/src/Rating/Rating.d.ts.map +1 -0
  160. package/dist-package/src/Rating/Rating.js.map +1 -0
  161. package/{Rating → dist-package/src/Rating}/index.d.ts +1 -0
  162. package/dist-package/src/Rating/index.d.ts.map +1 -0
  163. package/dist-package/src/Rating/index.js.map +1 -0
  164. package/{RichTextEditor → dist-package/src/RichTextEditor}/RichTextEditor.d.ts +2 -1
  165. package/dist-package/src/RichTextEditor/RichTextEditor.d.ts.map +1 -0
  166. package/dist-package/src/RichTextEditor/RichTextEditor.js.map +1 -0
  167. package/dist-package/src/RichTextEditor/hooks/index.d.ts +2 -0
  168. package/dist-package/src/RichTextEditor/hooks/index.d.ts.map +1 -0
  169. package/dist-package/src/RichTextEditor/hooks/index.js.map +1 -0
  170. package/{RichTextEditor → dist-package/src/RichTextEditor}/hooks/use-enforce-highlight-autofill.d.ts +1 -0
  171. package/dist-package/src/RichTextEditor/hooks/use-enforce-highlight-autofill.d.ts.map +1 -0
  172. package/dist-package/src/RichTextEditor/hooks/use-enforce-highlight-autofill.js.map +1 -0
  173. package/{RichTextEditor → dist-package/src/RichTextEditor}/index.d.ts +1 -0
  174. package/dist-package/src/RichTextEditor/index.d.ts.map +1 -0
  175. package/dist-package/src/RichTextEditor/index.js.map +1 -0
  176. package/{Select → dist-package/src/Select}/Select.d.ts +1 -0
  177. package/dist-package/src/Select/Select.d.ts.map +1 -0
  178. package/{Select → dist-package/src/Select}/Select.js +1 -1
  179. package/dist-package/src/Select/Select.js.map +1 -0
  180. package/dist-package/src/Select/index.d.ts +3 -0
  181. package/dist-package/src/Select/index.d.ts.map +1 -0
  182. package/dist-package/src/Select/index.js.map +1 -0
  183. package/{SubmitButton → dist-package/src/SubmitButton}/SubmitButton.d.ts +2 -2
  184. package/dist-package/src/SubmitButton/SubmitButton.d.ts.map +1 -0
  185. package/dist-package/src/SubmitButton/SubmitButton.js.map +1 -0
  186. package/{SubmitButton → dist-package/src/SubmitButton}/index.d.ts +1 -0
  187. package/dist-package/src/SubmitButton/index.d.ts.map +1 -0
  188. package/dist-package/src/SubmitButton/index.js.map +1 -0
  189. package/{Switch → dist-package/src/Switch}/Switch.d.ts +1 -0
  190. package/dist-package/src/Switch/Switch.d.ts.map +1 -0
  191. package/dist-package/src/Switch/Switch.js.map +1 -0
  192. package/{Switch → dist-package/src/Switch}/index.d.ts +1 -0
  193. package/dist-package/src/Switch/index.d.ts.map +1 -0
  194. package/dist-package/src/Switch/index.js.map +1 -0
  195. package/{TagSelector → dist-package/src/TagSelector}/TagSelector.d.ts +1 -0
  196. package/dist-package/src/TagSelector/TagSelector.d.ts.map +1 -0
  197. package/dist-package/src/TagSelector/TagSelector.js.map +1 -0
  198. package/dist-package/src/TagSelector/index.d.ts +3 -0
  199. package/dist-package/src/TagSelector/index.d.ts.map +1 -0
  200. package/dist-package/src/TagSelector/index.js.map +1 -0
  201. package/{TimePicker → dist-package/src/TimePicker}/TimePicker.d.ts +1 -0
  202. package/dist-package/src/TimePicker/TimePicker.d.ts.map +1 -0
  203. package/dist-package/src/TimePicker/TimePicker.js.map +1 -0
  204. package/{TimePicker → dist-package/src/TimePicker}/index.d.ts +1 -0
  205. package/dist-package/src/TimePicker/index.d.ts.map +1 -0
  206. package/dist-package/src/TimePicker/index.js.map +1 -0
  207. package/{index.d.ts → dist-package/src/index.d.ts} +10 -0
  208. package/dist-package/src/index.d.ts.map +1 -0
  209. package/{index.js → dist-package/src/index.js} +3 -0
  210. package/dist-package/src/index.js.map +1 -0
  211. package/{utils → dist-package/src/utils}/flat-map.d.ts +1 -0
  212. package/dist-package/src/utils/flat-map.d.ts.map +1 -0
  213. package/dist-package/src/utils/flat-map.js.map +1 -0
  214. package/{utils → dist-package/src/utils}/form-values-change-decorator/form-values-change-decorator.d.ts +1 -0
  215. package/dist-package/src/utils/form-values-change-decorator/form-values-change-decorator.d.ts.map +1 -0
  216. package/dist-package/src/utils/form-values-change-decorator/form-values-change-decorator.js.map +1 -0
  217. package/{utils → dist-package/src/utils}/form-values-change-decorator/index.d.ts +1 -0
  218. package/dist-package/src/utils/form-values-change-decorator/index.d.ts.map +1 -0
  219. package/dist-package/src/utils/form-values-change-decorator/index.js.map +1 -0
  220. package/{utils → dist-package/src/utils}/index.d.ts +1 -0
  221. package/dist-package/src/utils/index.d.ts.map +1 -0
  222. package/dist-package/src/utils/index.js.map +1 -0
  223. package/{utils → dist-package/src/utils}/scroll-to-error-decorator.d.ts +1 -0
  224. package/dist-package/src/utils/scroll-to-error-decorator.d.ts.map +1 -0
  225. package/dist-package/src/utils/scroll-to-error-decorator.js.map +1 -0
  226. package/{utils → dist-package/src/utils}/scroll-to.d.ts +1 -0
  227. package/dist-package/src/utils/scroll-to.d.ts.map +1 -0
  228. package/dist-package/src/utils/scroll-to.js.map +1 -0
  229. package/{utils → dist-package/src/utils}/use-field-validation/index.d.ts +1 -0
  230. package/dist-package/src/utils/use-field-validation/index.d.ts.map +1 -0
  231. package/dist-package/src/utils/use-field-validation/index.js.map +1 -0
  232. package/{utils → dist-package/src/utils}/use-field-validation/use-field-validation.d.ts +1 -0
  233. package/dist-package/src/utils/use-field-validation/use-field-validation.d.ts.map +1 -0
  234. package/dist-package/src/utils/use-field-validation/use-field-validation.js.map +1 -0
  235. package/{utils → dist-package/src/utils}/use-form-auto-save/index.d.ts +1 -0
  236. package/dist-package/src/utils/use-form-auto-save/index.d.ts.map +1 -0
  237. package/dist-package/src/utils/use-form-auto-save/index.js.map +1 -0
  238. package/{utils → dist-package/src/utils}/use-form-auto-save/use-form-auto-save.d.ts +1 -0
  239. package/dist-package/src/utils/use-form-auto-save/use-form-auto-save.d.ts.map +1 -0
  240. package/dist-package/src/utils/use-form-auto-save/use-form-auto-save.js.map +1 -0
  241. package/{utils → dist-package/src/utils}/use-form-input-reset/index.d.ts +1 -0
  242. package/dist-package/src/utils/use-form-input-reset/index.d.ts.map +1 -0
  243. package/dist-package/src/utils/use-form-input-reset/index.js.map +1 -0
  244. package/{utils → dist-package/src/utils}/use-form-input-reset/use-form-input-reset.d.ts +1 -0
  245. package/dist-package/src/utils/use-form-input-reset/use-form-input-reset.d.ts.map +1 -0
  246. package/dist-package/src/utils/use-form-input-reset/use-form-input-reset.js.map +1 -0
  247. package/{utils → dist-package/src/utils}/validators.d.ts +1 -0
  248. package/dist-package/src/utils/validators.d.ts.map +1 -0
  249. package/dist-package/src/utils/validators.js.map +1 -0
  250. package/package.json +24 -11
  251. package/src/Autocomplete/Autocomplete.tsx +37 -0
  252. package/src/Autocomplete/index.ts +1 -0
  253. package/src/AvatarUpload/AvatarUpload.tsx +92 -0
  254. package/src/AvatarUpload/index.ts +1 -0
  255. package/src/ButtonCheckbox/ButtonCheckbox.tsx +63 -0
  256. package/src/ButtonCheckbox/index.ts +1 -0
  257. package/src/ButtonRadio/ButtonRadio.tsx +25 -0
  258. package/src/ButtonRadio/index.ts +1 -0
  259. package/src/Checkbox/Checkbox.tsx +79 -0
  260. package/src/Checkbox/__snapshots__/test.tsx.snap +259 -0
  261. package/src/Checkbox/index.ts +2 -0
  262. package/src/Checkbox/test.tsx +92 -0
  263. package/src/CheckboxGroup/CheckboxGroup.tsx +47 -0
  264. package/src/CheckboxGroup/CheckboxGroupContext.ts +3 -0
  265. package/src/CheckboxGroup/index.ts +3 -0
  266. package/src/CheckboxGroup/test.tsx +36 -0
  267. package/src/DatePicker/DatePicker.tsx +40 -0
  268. package/src/DatePicker/index.ts +1 -0
  269. package/src/Dropzone/Dropzone.tsx +102 -0
  270. package/src/Dropzone/index.ts +6 -0
  271. package/src/Field/Field.tsx +196 -0
  272. package/src/Field/index.ts +2 -0
  273. package/src/FieldBase/index.ts +1 -0
  274. package/src/FieldBase/types.ts +23 -0
  275. package/src/FieldLabel/FieldLabel.tsx +64 -0
  276. package/src/FieldLabel/__snapshots__/test.tsx.snap +21 -0
  277. package/src/FieldLabel/index.ts +2 -0
  278. package/src/FieldLabel/test.tsx +52 -0
  279. package/src/FieldWrapper/FieldWrapper.tsx +48 -0
  280. package/src/FieldWrapper/index.ts +6 -0
  281. package/src/FieldWrapper/story/index.jsx +153 -0
  282. package/src/FieldWrapper/test.tsx +32 -0
  283. package/src/FileInput/FileInput.tsx +85 -0
  284. package/src/FileInput/index.ts +1 -0
  285. package/src/Form/Form.tsx +161 -0
  286. package/src/Form/FormContext.ts +38 -0
  287. package/src/Form/FormRenderer.tsx +38 -0
  288. package/src/Form/__snapshots__/test.tsx.snap +140 -0
  289. package/src/Form/index.ts +2 -0
  290. package/src/Form/mutators/index.ts +2 -0
  291. package/src/Form/mutators/set-active-field-touched.ts +16 -0
  292. package/src/Form/mutators/set-has-multiline-counter.ts +16 -0
  293. package/src/Form/story/AutoSave.example.tsx +120 -0
  294. package/src/Form/story/AvatarUpload.example.tsx +81 -0
  295. package/src/Form/story/BackendCommunication.example.tsx +137 -0
  296. package/src/Form/story/CustomFormLevelConfiguration.example.tsx +35 -0
  297. package/src/Form/story/CustomValidator.example.tsx +53 -0
  298. package/src/Form/story/Default.example.tsx +218 -0
  299. package/src/Form/story/Dropzone.example.tsx +42 -0
  300. package/src/Form/story/FieldRequirements.example.tsx +58 -0
  301. package/src/Form/story/FileInput.example.tsx +43 -0
  302. package/src/Form/story/HighlightAutofill.example.tsx +96 -0
  303. package/src/Form/story/Horizontal.example.tsx +281 -0
  304. package/src/Form/story/HorizontalLabelWidth.example.tsx +26 -0
  305. package/src/Form/story/NoScrolling.example.tsx +42 -0
  306. package/src/Form/story/ParseInput.example.tsx +31 -0
  307. package/src/Form/story/RichTextEditor.example.tsx +32 -0
  308. package/src/Form/story/Status.example.tsx +35 -0
  309. package/src/Form/story/TitleCase.example.tsx +194 -0
  310. package/src/Form/story/ValidateOnSubmit.example.tsx +99 -0
  311. package/src/Form/story/index.jsx +289 -0
  312. package/src/Form/test.tsx +202 -0
  313. package/src/FormCompound/index.ts +51 -0
  314. package/src/FormConfig/FormConfig.ts +35 -0
  315. package/src/FormConfig/index.ts +1 -0
  316. package/src/FormConfig/test.tsx +44 -0
  317. package/src/Input/Input.tsx +80 -0
  318. package/src/Input/index.ts +2 -0
  319. package/src/Input/test.tsx +35 -0
  320. package/src/InputField/InputField.tsx +71 -0
  321. package/src/InputField/index.ts +1 -0
  322. package/src/InputField/test.ts +105 -0
  323. package/src/NumberInput/NumberInput.tsx +58 -0
  324. package/src/NumberInput/index.ts +2 -0
  325. package/src/PasswordInput/FieldRenderer.tsx +36 -0
  326. package/src/PasswordInput/PasswordInput.tsx +144 -0
  327. package/src/PasswordInput/index.ts +1 -0
  328. package/src/PasswordInput/test.ts +45 -0
  329. package/src/PasswordInput/validators.ts +28 -0
  330. package/src/Radio/Radio.tsx +25 -0
  331. package/src/Radio/__snapshots__/test.tsx.snap +235 -0
  332. package/src/Radio/index.ts +1 -0
  333. package/src/Radio/test.tsx +49 -0
  334. package/src/RadioGroup/RadioGroup.tsx +53 -0
  335. package/src/RadioGroup/RadioGroupContext.ts +3 -0
  336. package/src/RadioGroup/index.ts +3 -0
  337. package/src/RadioGroup/test.tsx +36 -0
  338. package/src/Rating/Rating.tsx +95 -0
  339. package/src/Rating/__snapshots__/test.tsx.snap +226 -0
  340. package/src/Rating/index.ts +1 -0
  341. package/src/Rating/test.tsx +171 -0
  342. package/src/RichTextEditor/RichTextEditor.tsx +87 -0
  343. package/{RichTextEditor/hooks/index.d.ts → src/RichTextEditor/hooks/index.ts} +1 -1
  344. package/src/RichTextEditor/hooks/use-enforce-highlight-autofill.test.tsx +67 -0
  345. package/src/RichTextEditor/hooks/use-enforce-highlight-autofill.ts +48 -0
  346. package/src/RichTextEditor/index.ts +1 -0
  347. package/src/Select/Select.tsx +55 -0
  348. package/src/Select/index.ts +2 -0
  349. package/src/SubmitButton/SubmitButton.tsx +71 -0
  350. package/src/SubmitButton/index.ts +6 -0
  351. package/src/SubmitButton/story/ButtonTypes.example.tsx +45 -0
  352. package/src/SubmitButton/story/Default.example.tsx +16 -0
  353. package/src/SubmitButton/story/index.jsx +35 -0
  354. package/src/Switch/Switch.tsx +40 -0
  355. package/src/Switch/index.ts +1 -0
  356. package/src/TagSelector/TagSelector.tsx +41 -0
  357. package/src/TagSelector/index.ts +2 -0
  358. package/src/TimePicker/TimePicker.tsx +43 -0
  359. package/src/TimePicker/index.ts +1 -0
  360. package/src/index.ts +86 -0
  361. package/src/story/Deserialization.example.tsx +40 -0
  362. package/src/story/FormSpy.example.tsx +50 -0
  363. package/src/story/index.jsx +39 -0
  364. package/src/utils/flat-map.ts +4 -0
  365. package/src/utils/form-values-change-decorator/form-values-change-decorator.ts +64 -0
  366. package/src/utils/form-values-change-decorator/index.ts +3 -0
  367. package/src/utils/form-values-change-decorator/test.tsx +77 -0
  368. package/src/utils/index.ts +6 -0
  369. package/src/utils/scroll-to-error-decorator.ts +85 -0
  370. package/src/utils/scroll-to.ts +10 -0
  371. package/src/utils/use-field-validation/index.ts +1 -0
  372. package/src/utils/use-field-validation/test.tsx +73 -0
  373. package/src/utils/use-field-validation/use-field-validation.ts +63 -0
  374. package/src/utils/use-form-auto-save/index.ts +1 -0
  375. package/src/utils/use-form-auto-save/use-form-auto-save.ts +59 -0
  376. package/src/utils/use-form-input-reset/index.ts +1 -0
  377. package/src/utils/use-form-input-reset/test.ts +72 -0
  378. package/src/utils/use-form-input-reset/use-form-input-reset.ts +32 -0
  379. package/src/utils/validators.ts +18 -0
  380. package/utils.d.ts +1 -0
  381. package/utils.js +4 -0
  382. package/Autocomplete/Autocomplete.js.map +0 -1
  383. package/Autocomplete/index.js.map +0 -1
  384. package/AvatarUpload/AvatarUpload.js.map +0 -1
  385. package/AvatarUpload/index.js.map +0 -1
  386. package/ButtonCheckbox/ButtonCheckbox.js.map +0 -1
  387. package/ButtonCheckbox/index.js.map +0 -1
  388. package/ButtonRadio/ButtonRadio.js.map +0 -1
  389. package/ButtonRadio/index.js.map +0 -1
  390. package/Checkbox/Checkbox.js.map +0 -1
  391. package/Checkbox/index.d.ts +0 -1
  392. package/Checkbox/index.js.map +0 -1
  393. package/CheckboxGroup/CheckboxGroup.js.map +0 -1
  394. package/CheckboxGroup/CheckboxGroupContext.js.map +0 -1
  395. package/CheckboxGroup/index.js.map +0 -1
  396. package/DatePicker/DatePicker.js.map +0 -1
  397. package/DatePicker/index.js.map +0 -1
  398. package/Dropzone/Dropzone.js.map +0 -1
  399. package/Dropzone/index.js.map +0 -1
  400. package/Field/Field.js.map +0 -1
  401. package/Field/index.js.map +0 -1
  402. package/FieldBase/index.d.ts +0 -1
  403. package/FieldBase/index.js.map +0 -1
  404. package/FieldBase/types.d.ts +0 -8
  405. package/FieldLabel/FieldLabel.js.map +0 -1
  406. package/FieldLabel/index.js.map +0 -1
  407. package/FieldWrapper/FieldWrapper.js.map +0 -1
  408. package/FieldWrapper/index.js.map +0 -1
  409. package/FileInput/FileInput.js.map +0 -1
  410. package/FileInput/index.js.map +0 -1
  411. package/Form/Form.js.map +0 -1
  412. package/Form/FormContext.js.map +0 -1
  413. package/Form/FormRenderer.js.map +0 -1
  414. package/Form/index.js.map +0 -1
  415. package/Form/mutators/index.js.map +0 -1
  416. package/Form/mutators/set-active-field-touched.js.map +0 -1
  417. package/Form/mutators/set-has-multiline-counter.js.map +0 -1
  418. package/FormCompound/index.js.map +0 -1
  419. package/FormConfig/FormConfig.js.map +0 -1
  420. package/FormConfig/index.d.ts +0 -1
  421. package/FormConfig/index.js.map +0 -1
  422. package/Input/Input.js.map +0 -1
  423. package/Input/index.js.map +0 -1
  424. package/InputField/InputField.js.map +0 -1
  425. package/InputField/index.js.map +0 -1
  426. package/LICENSE +0 -20
  427. package/NumberInput/NumberInput.js.map +0 -1
  428. package/NumberInput/index.d.ts +0 -1
  429. package/NumberInput/index.js.map +0 -1
  430. package/PasswordInput/FieldRenderer.js.map +0 -1
  431. package/PasswordInput/PasswordInput.js.map +0 -1
  432. package/PasswordInput/index.js.map +0 -1
  433. package/PasswordInput/validators.js.map +0 -1
  434. package/Radio/Radio.js.map +0 -1
  435. package/Radio/index.js.map +0 -1
  436. package/RadioGroup/RadioGroup.js.map +0 -1
  437. package/RadioGroup/RadioGroupContext.js.map +0 -1
  438. package/RadioGroup/index.js.map +0 -1
  439. package/Rating/Rating.js.map +0 -1
  440. package/Rating/index.js.map +0 -1
  441. package/RichTextEditor/RichTextEditor.js.map +0 -1
  442. package/RichTextEditor/hooks/index.js.map +0 -1
  443. package/RichTextEditor/hooks/use-enforce-highlight-autofill.js.map +0 -1
  444. package/RichTextEditor/hooks/use-enforce-highlight-autofill.test.d.ts +0 -1
  445. package/RichTextEditor/hooks/use-enforce-highlight-autofill.test.js +0 -51
  446. package/RichTextEditor/hooks/use-enforce-highlight-autofill.test.js.map +0 -1
  447. package/RichTextEditor/index.js.map +0 -1
  448. package/Select/Select.js.map +0 -1
  449. package/Select/index.d.ts +0 -1
  450. package/Select/index.js.map +0 -1
  451. package/SubmitButton/SubmitButton.js.map +0 -1
  452. package/SubmitButton/index.js.map +0 -1
  453. package/Switch/Switch.js.map +0 -1
  454. package/Switch/index.js.map +0 -1
  455. package/TagSelector/TagSelector.js.map +0 -1
  456. package/TagSelector/index.d.ts +0 -1
  457. package/TagSelector/index.js.map +0 -1
  458. package/TimePicker/TimePicker.js.map +0 -1
  459. package/TimePicker/index.js.map +0 -1
  460. package/index.js.map +0 -1
  461. package/utils/flat-map.js.map +0 -1
  462. package/utils/form-values-change-decorator/form-values-change-decorator.js.map +0 -1
  463. package/utils/form-values-change-decorator/index.js.map +0 -1
  464. package/utils/index.js.map +0 -1
  465. package/utils/scroll-to-error-decorator.js.map +0 -1
  466. package/utils/scroll-to.js.map +0 -1
  467. package/utils/use-field-validation/index.js.map +0 -1
  468. package/utils/use-field-validation/use-field-validation.js.map +0 -1
  469. package/utils/use-form-auto-save/index.js.map +0 -1
  470. package/utils/use-form-auto-save/use-form-auto-save.js.map +0 -1
  471. package/utils/use-form-input-reset/index.js.map +0 -1
  472. package/utils/use-form-input-reset/use-form-input-reset.js.map +0 -1
  473. package/utils/validators.js.map +0 -1
  474. /package/{Autocomplete → dist-package/src/Autocomplete}/Autocomplete.js +0 -0
  475. /package/{Autocomplete → dist-package/src/Autocomplete}/index.js +0 -0
  476. /package/{AvatarUpload → dist-package/src/AvatarUpload}/AvatarUpload.js +0 -0
  477. /package/{AvatarUpload → dist-package/src/AvatarUpload}/index.js +0 -0
  478. /package/{ButtonCheckbox → dist-package/src/ButtonCheckbox}/ButtonCheckbox.js +0 -0
  479. /package/{ButtonCheckbox → dist-package/src/ButtonCheckbox}/index.js +0 -0
  480. /package/{ButtonRadio → dist-package/src/ButtonRadio}/ButtonRadio.js +0 -0
  481. /package/{ButtonRadio → dist-package/src/ButtonRadio}/index.js +0 -0
  482. /package/{Checkbox → dist-package/src/Checkbox}/Checkbox.js +0 -0
  483. /package/{Checkbox → dist-package/src/Checkbox}/index.js +0 -0
  484. /package/{CheckboxGroup → dist-package/src/CheckboxGroup}/CheckboxGroup.js +0 -0
  485. /package/{CheckboxGroup → dist-package/src/CheckboxGroup}/CheckboxGroupContext.js +0 -0
  486. /package/{CheckboxGroup → dist-package/src/CheckboxGroup}/index.js +0 -0
  487. /package/{DatePicker → dist-package/src/DatePicker}/DatePicker.js +0 -0
  488. /package/{DatePicker → dist-package/src/DatePicker}/index.js +0 -0
  489. /package/{Dropzone → dist-package/src/Dropzone}/Dropzone.js +0 -0
  490. /package/{Dropzone → dist-package/src/Dropzone}/index.js +0 -0
  491. /package/{Field → dist-package/src/Field}/Field.js +0 -0
  492. /package/{Field → dist-package/src/Field}/index.js +0 -0
  493. /package/{FieldBase → dist-package/src/FieldBase}/index.js +0 -0
  494. /package/{FieldBase → dist-package/src/FieldBase}/types.js +0 -0
  495. /package/{FieldLabel → dist-package/src/FieldLabel}/FieldLabel.js +0 -0
  496. /package/{FieldLabel → dist-package/src/FieldLabel}/index.js +0 -0
  497. /package/{FieldWrapper → dist-package/src/FieldWrapper}/FieldWrapper.js +0 -0
  498. /package/{FieldWrapper → dist-package/src/FieldWrapper}/index.js +0 -0
  499. /package/{FileInput → dist-package/src/FileInput}/FileInput.js +0 -0
  500. /package/{FileInput → dist-package/src/FileInput}/index.js +0 -0
  501. /package/{Form → dist-package/src/Form}/FormContext.js +0 -0
  502. /package/{Form → dist-package/src/Form}/FormRenderer.js +0 -0
  503. /package/{Form → dist-package/src/Form}/index.js +0 -0
  504. /package/{Form → dist-package/src/Form}/mutators/index.js +0 -0
  505. /package/{Form → dist-package/src/Form}/mutators/set-active-field-touched.js +0 -0
  506. /package/{Form → dist-package/src/Form}/mutators/set-has-multiline-counter.js +0 -0
  507. /package/{FormCompound → dist-package/src/FormCompound}/index.js +0 -0
  508. /package/{FormConfig → dist-package/src/FormConfig}/FormConfig.js +0 -0
  509. /package/{FormConfig → dist-package/src/FormConfig}/index.js +0 -0
  510. /package/{Input → dist-package/src/Input}/Input.js +0 -0
  511. /package/{Input → dist-package/src/Input}/index.js +0 -0
  512. /package/{InputField → dist-package/src/InputField}/InputField.js +0 -0
  513. /package/{InputField → dist-package/src/InputField}/index.js +0 -0
  514. /package/{NumberInput → dist-package/src/NumberInput}/NumberInput.js +0 -0
  515. /package/{NumberInput → dist-package/src/NumberInput}/index.js +0 -0
  516. /package/{PasswordInput → dist-package/src/PasswordInput}/FieldRenderer.js +0 -0
  517. /package/{PasswordInput → dist-package/src/PasswordInput}/PasswordInput.js +0 -0
  518. /package/{PasswordInput → dist-package/src/PasswordInput}/index.js +0 -0
  519. /package/{PasswordInput → dist-package/src/PasswordInput}/validators.js +0 -0
  520. /package/{Radio → dist-package/src/Radio}/Radio.js +0 -0
  521. /package/{Radio → dist-package/src/Radio}/index.js +0 -0
  522. /package/{RadioGroup → dist-package/src/RadioGroup}/RadioGroup.js +0 -0
  523. /package/{RadioGroup → dist-package/src/RadioGroup}/RadioGroupContext.js +0 -0
  524. /package/{RadioGroup → dist-package/src/RadioGroup}/index.js +0 -0
  525. /package/{Rating → dist-package/src/Rating}/Rating.js +0 -0
  526. /package/{Rating → dist-package/src/Rating}/index.js +0 -0
  527. /package/{RichTextEditor → dist-package/src/RichTextEditor}/RichTextEditor.js +0 -0
  528. /package/{RichTextEditor → dist-package/src/RichTextEditor}/hooks/index.js +0 -0
  529. /package/{RichTextEditor → dist-package/src/RichTextEditor}/hooks/use-enforce-highlight-autofill.js +0 -0
  530. /package/{RichTextEditor → dist-package/src/RichTextEditor}/index.js +0 -0
  531. /package/{Select → dist-package/src/Select}/index.js +0 -0
  532. /package/{SubmitButton → dist-package/src/SubmitButton}/SubmitButton.js +0 -0
  533. /package/{SubmitButton → dist-package/src/SubmitButton}/index.js +0 -0
  534. /package/{Switch → dist-package/src/Switch}/Switch.js +0 -0
  535. /package/{Switch → dist-package/src/Switch}/index.js +0 -0
  536. /package/{TagSelector → dist-package/src/TagSelector}/TagSelector.js +0 -0
  537. /package/{TagSelector → dist-package/src/TagSelector}/index.js +0 -0
  538. /package/{TimePicker → dist-package/src/TimePicker}/TimePicker.js +0 -0
  539. /package/{TimePicker → dist-package/src/TimePicker}/index.js +0 -0
  540. /package/{utils → dist-package/src/utils}/flat-map.js +0 -0
  541. /package/{utils → dist-package/src/utils}/form-values-change-decorator/form-values-change-decorator.js +0 -0
  542. /package/{utils → dist-package/src/utils}/form-values-change-decorator/index.js +0 -0
  543. /package/{utils → dist-package/src/utils}/index.js +0 -0
  544. /package/{utils → dist-package/src/utils}/scroll-to-error-decorator.js +0 -0
  545. /package/{utils → dist-package/src/utils}/scroll-to.js +0 -0
  546. /package/{utils → dist-package/src/utils}/use-field-validation/index.js +0 -0
  547. /package/{utils → dist-package/src/utils}/use-field-validation/use-field-validation.js +0 -0
  548. /package/{utils → dist-package/src/utils}/use-form-auto-save/index.js +0 -0
  549. /package/{utils → dist-package/src/utils}/use-form-auto-save/use-form-auto-save.js +0 -0
  550. /package/{utils → dist-package/src/utils}/use-form-input-reset/index.js +0 -0
  551. /package/{utils → dist-package/src/utils}/use-form-input-reset/use-form-input-reset.js +0 -0
  552. /package/{utils → dist-package/src/utils}/validators.js +0 -0
@@ -0,0 +1,161 @@
1
+ import type { ReactNode } from 'react'
2
+ import React, { useMemo, useRef } from 'react'
3
+ import type { FormProps as FinalFormProps } from 'react-final-form'
4
+ import { Form as FinalForm } from 'react-final-form'
5
+ import type { FormApi, SubmissionErrors, AnyObject } from 'final-form'
6
+ import { getIn, setIn } from 'final-form'
7
+ import { useNotifications } from '@toptal/picasso-notification'
8
+
9
+ import { createScrollToErrorDecorator } from '../utils'
10
+ import type { Validators, FormContextProps } from './FormContext'
11
+ import { FormContext, createFormContext } from './FormContext'
12
+ import type { Props as FormProps } from './FormRenderer'
13
+ import FormRenderer from './FormRenderer'
14
+ import { setActiveFieldTouched, setHasMultilineCounter } from './mutators'
15
+
16
+ export type Props<T = AnyObject> = FinalFormProps<T> & {
17
+ disableScrollOnError?: boolean
18
+ autoComplete?: HTMLFormElement['autocomplete']
19
+ successSubmitMessage?: ReactNode
20
+ failedSubmitMessage?: ReactNode
21
+ scrollOffsetTop?: number
22
+ layout?: 'horizontal' | 'vertical'
23
+ labelWidth?: FormProps['labelWidth']
24
+ 'data-testid'?: string
25
+ }
26
+
27
+ const getValidationErrors = (
28
+ validators: Validators,
29
+ formValues: any,
30
+ form: FormApi<any>
31
+ ): SubmissionErrors | void => {
32
+ let errors: SubmissionErrors
33
+
34
+ Object.entries(validators).forEach(([key, validator]) => {
35
+ const fieldValue = getIn(formValues, key)
36
+ const fieldMetaState = form.getFieldState(key)
37
+
38
+ if (!validator) {
39
+ return
40
+ }
41
+
42
+ const error = validator(fieldValue, formValues, fieldMetaState)
43
+
44
+ if (error) {
45
+ errors = setIn(errors || {}, key, error)
46
+ }
47
+ })
48
+
49
+ return errors
50
+ }
51
+
52
+ export const Form = <T extends AnyObject = AnyObject>(props: Props<T>) => {
53
+ const {
54
+ autoComplete,
55
+ children,
56
+ disableScrollOnError,
57
+ onSubmit,
58
+ successSubmitMessage,
59
+ failedSubmitMessage,
60
+ decorators = [],
61
+ mutators = {},
62
+ validateOnBlur,
63
+ 'data-testid': dataTestId,
64
+ layout,
65
+ labelWidth,
66
+ ...rest
67
+ } = props
68
+ const { showSuccess, showError } = useNotifications()
69
+ const scrollToErrorDecorator = useMemo(
70
+ () =>
71
+ createScrollToErrorDecorator({
72
+ disableScrollOnError,
73
+ }),
74
+ [disableScrollOnError]
75
+ )
76
+
77
+ const validationObject = useRef<FormContextProps>(createFormContext())
78
+
79
+ const showSuccessNotification = () => {
80
+ if (!successSubmitMessage) {
81
+ return
82
+ }
83
+
84
+ showSuccess(successSubmitMessage)
85
+ }
86
+
87
+ const showErrorNotification = (errors: SubmissionErrors) => {
88
+ if (typeof errors === 'string') {
89
+ showError(errors, undefined, { persist: true })
90
+
91
+ return
92
+ }
93
+
94
+ if (!failedSubmitMessage) {
95
+ return
96
+ }
97
+
98
+ showError(failedSubmitMessage, undefined, { persist: true })
99
+ }
100
+
101
+ const handleSubmit = async (
102
+ values: T,
103
+ form: FormApi<T>,
104
+ callback?: (errors?: SubmissionErrors) => void
105
+ ) => {
106
+ const validationErrors = getValidationErrors(
107
+ validationObject.current.getValidators(),
108
+ values,
109
+ form
110
+ )
111
+
112
+ if (validationErrors) {
113
+ return validationErrors
114
+ }
115
+
116
+ const submissionErrors = await onSubmit(values, form, callback)
117
+
118
+ if (!submissionErrors) {
119
+ showSuccessNotification()
120
+ } else {
121
+ showErrorNotification(submissionErrors)
122
+ }
123
+
124
+ return submissionErrors
125
+ }
126
+
127
+ return (
128
+ <FormContext.Provider value={validationObject}>
129
+ <FinalForm
130
+ render={({ form, handleSubmit: handleFormRendererSubmit }) => (
131
+ <FormRenderer
132
+ autoComplete={autoComplete}
133
+ data-testid={dataTestId}
134
+ onSubmit={handleFormRendererSubmit}
135
+ validateOnBlur={validateOnBlur}
136
+ setActiveFieldTouched={form.mutators.setActiveFieldTouched}
137
+ labelWidth={labelWidth}
138
+ layout={layout}
139
+ >
140
+ {children}
141
+ </FormRenderer>
142
+ )}
143
+ onSubmit={handleSubmit}
144
+ decorators={[...decorators, scrollToErrorDecorator]}
145
+ mutators={{
146
+ ...mutators,
147
+ setActiveFieldTouched,
148
+ setHasMultilineCounter,
149
+ }}
150
+ validateOnBlur={validateOnBlur}
151
+ {...rest}
152
+ />
153
+ </FormContext.Provider>
154
+ )
155
+ }
156
+
157
+ Form.defaultProps = {}
158
+
159
+ Form.displayName = 'Form'
160
+
161
+ export default Form
@@ -0,0 +1,38 @@
1
+ import type { MutableRefObject } from 'react'
2
+ import { createContext, useContext } from 'react'
3
+ import type { FieldValidator } from 'final-form'
4
+
5
+ export type Validators = Record<string, FieldValidator<any>>
6
+
7
+ export type FormContextProps = {
8
+ getValidators: () => Validators
9
+ setValidators: (fieldName: string, validator: FieldValidator<any>) => void
10
+ clearValidators: (fieldName: string) => void
11
+ }
12
+
13
+ export const createFormContext = (): FormContextProps => {
14
+ const validators: Validators = {}
15
+
16
+ return {
17
+ getValidators: () => validators,
18
+ setValidators: (fieldName, validator) => {
19
+ validators[fieldName] = validator
20
+ },
21
+ clearValidators: fieldName => {
22
+ delete validators[fieldName]
23
+ },
24
+ }
25
+ }
26
+
27
+ export const FormContext =
28
+ createContext<MutableRefObject<FormContextProps> | null>(null)
29
+
30
+ export const useFormContext = () => {
31
+ const context = useContext(FormContext)
32
+
33
+ if (!context) {
34
+ throw new Error('Form Field cannot be rendered outside Form component')
35
+ }
36
+
37
+ return context.current
38
+ }
@@ -0,0 +1,38 @@
1
+ import React, { useCallback } from 'react'
2
+ import type { FormProps } from '@toptal/picasso'
3
+ import { Form, Container } from '@toptal/picasso'
4
+
5
+ export interface Props extends FormProps {
6
+ setActiveFieldTouched: () => void
7
+ validateOnBlur?: boolean
8
+ }
9
+ export const FormRenderer = (props: Props) => {
10
+ const { onSubmit, setActiveFieldTouched, children, validateOnBlur, ...rest } =
11
+ props
12
+
13
+ const handleNativeSubmit = useCallback(
14
+ (event: React.SyntheticEvent<HTMLFormElement>) => {
15
+ if (validateOnBlur) {
16
+ // force validation for active field
17
+ setActiveFieldTouched()
18
+ }
19
+
20
+ onSubmit?.(event)
21
+ },
22
+ [onSubmit, setActiveFieldTouched, validateOnBlur]
23
+ )
24
+
25
+ return (
26
+ <Container>
27
+ <Form {...rest} onSubmit={handleNativeSubmit}>
28
+ {children}
29
+ </Form>
30
+ </Container>
31
+ )
32
+ }
33
+
34
+ FormRenderer.defaultProps = {}
35
+
36
+ FormRenderer.displayName = 'FormRenderer'
37
+
38
+ export default FormRenderer
@@ -0,0 +1,140 @@
1
+ // Jest Snapshot v1, https://goo.gl/fbAQLP
2
+
3
+ exports[`Form renders 1`] = `
4
+ <div>
5
+ <div
6
+ class="Picasso-root"
7
+ >
8
+ <div
9
+ class=""
10
+ >
11
+ <form
12
+ class=""
13
+ data-testid="form"
14
+ >
15
+ <div
16
+ class="PicassoFormField-root"
17
+ data-field-has-error="false"
18
+ >
19
+ <div
20
+ class="MuiInputBase-root MuiOutlinedInput-root PicassoOutlinedInput-root PicassoInput-root PicassoOutlinedInput-rootAuto PicassoOutlinedInput-rootMedium MuiInputBase-adornedStart MuiOutlinedInput-adornedStart MuiInputBase-adornedEnd MuiOutlinedInput-adornedEnd"
21
+ >
22
+ <input
23
+ aria-invalid="false"
24
+ autocomplete="none"
25
+ class="MuiInputBase-input MuiOutlinedInput-input PicassoOutlinedInput-input PicassoOutlinedInput-inputMedium MuiInputBase-inputAdornedStart MuiOutlinedInput-inputAdornedStart MuiInputBase-inputAdornedEnd MuiOutlinedInput-inputAdornedEnd"
26
+ id="test"
27
+ name="test"
28
+ placeholder="test input"
29
+ type="text"
30
+ value=""
31
+ />
32
+ <fieldset
33
+ aria-hidden="true"
34
+ class="PrivateNotchedOutline-root MuiOutlinedInput-notchedOutline PicassoOutlinedInput-notchedOutline"
35
+ style="padding-left: 8px;"
36
+ >
37
+ <legend
38
+ class="PrivateNotchedOutline-legend"
39
+ style="width: 0.01px;"
40
+ >
41
+ <span>
42
+
43
+ </span>
44
+ </legend>
45
+ </fieldset>
46
+ </div>
47
+ </div>
48
+ <button
49
+ class="MuiButtonBase-root PicassoButton-medium PicassoButton-primary PicassoButton-root"
50
+ data-component-type="button"
51
+ tabindex="0"
52
+ type="submit"
53
+ >
54
+ <span
55
+ class="PicassoContainer-centerAlignItems PicassoContainer-flex PicassoContainer-inline PicassoButton-content"
56
+ >
57
+ Submit
58
+ </span>
59
+ </button>
60
+ </form>
61
+ </div>
62
+ </div>
63
+ </div>
64
+ `;
65
+
66
+ exports[`Form renders with an error 1`] = `
67
+ <div>
68
+ <div
69
+ class="Picasso-root"
70
+ >
71
+ <div
72
+ class=""
73
+ >
74
+ <form
75
+ class=""
76
+ data-testid="form"
77
+ >
78
+ <div
79
+ class="PicassoFormField-root"
80
+ data-field-has-error="true"
81
+ >
82
+ <div
83
+ class="MuiInputBase-root MuiOutlinedInput-root PicassoOutlinedInput-root PicassoInput-root PicassoOutlinedInput-rootAuto PicassoOutlinedInput-rootMedium Mui-error Mui-error MuiInputBase-adornedStart MuiOutlinedInput-adornedStart MuiInputBase-adornedEnd MuiOutlinedInput-adornedEnd"
84
+ >
85
+ <input
86
+ aria-invalid="true"
87
+ autocomplete="none"
88
+ class="MuiInputBase-input MuiOutlinedInput-input PicassoOutlinedInput-input PicassoOutlinedInput-inputMedium MuiInputBase-inputAdornedStart MuiOutlinedInput-inputAdornedStart MuiInputBase-inputAdornedEnd MuiOutlinedInput-inputAdornedEnd"
89
+ id="test"
90
+ name="test"
91
+ placeholder="test input"
92
+ type="text"
93
+ value=""
94
+ />
95
+ <fieldset
96
+ aria-hidden="true"
97
+ class="PrivateNotchedOutline-root MuiOutlinedInput-notchedOutline PicassoOutlinedInput-notchedOutline"
98
+ style="padding-left: 8px;"
99
+ >
100
+ <legend
101
+ class="PrivateNotchedOutline-legend"
102
+ style="width: 0.01px;"
103
+ >
104
+ <span>
105
+
106
+ </span>
107
+ </legend>
108
+ </fieldset>
109
+ </div>
110
+ <div
111
+ class=""
112
+ >
113
+ <div
114
+ class="PicassoFormError-root PicassoFormField-error"
115
+ >
116
+ <p
117
+ class="MuiTypography-root PicassoTypography-bodyXxsmall PicassoTypography-red PicassoFormError-error MuiTypography-body1"
118
+ >
119
+ Please complete this field.
120
+ </p>
121
+ </div>
122
+ </div>
123
+ </div>
124
+ <button
125
+ class="MuiButtonBase-root PicassoButton-medium PicassoButton-primary PicassoButton-root"
126
+ data-component-type="button"
127
+ tabindex="0"
128
+ type="submit"
129
+ >
130
+ <span
131
+ class="PicassoContainer-centerAlignItems PicassoContainer-flex PicassoContainer-inline PicassoButton-content"
132
+ >
133
+ Submit
134
+ </span>
135
+ </button>
136
+ </form>
137
+ </div>
138
+ </div>
139
+ </div>
140
+ `;
@@ -0,0 +1,2 @@
1
+ export { default } from './Form'
2
+ export * from './Form'
@@ -0,0 +1,2 @@
1
+ export * from './set-active-field-touched'
2
+ export * from './set-has-multiline-counter'
@@ -0,0 +1,16 @@
1
+ import type { MutableState } from 'final-form'
2
+
3
+ export const setActiveFieldTouched = <
4
+ FormValues = object,
5
+ InitialFormValues = Partial<FormValues>
6
+ >(
7
+ _: any[],
8
+ state: MutableState<FormValues, InitialFormValues>
9
+ ) => {
10
+ const activeFieldName = state.formState.active
11
+ const field = activeFieldName && state.fields[activeFieldName]
12
+
13
+ if (field) {
14
+ field.touched = true
15
+ }
16
+ }
@@ -0,0 +1,16 @@
1
+ import type { MutableState } from 'final-form'
2
+
3
+ export const setHasMultilineCounter = <
4
+ FormValues = object,
5
+ InitialFormValues = Partial<FormValues>
6
+ >(
7
+ args: [name: string, hasCounter: boolean],
8
+ state: MutableState<FormValues, InitialFormValues>
9
+ ) => {
10
+ const [name, hasCounter] = args
11
+ const field = state.fields[name]
12
+
13
+ if (field) {
14
+ field.data = { ...field.data, hasMultilineCounter: hasCounter }
15
+ }
16
+ }
@@ -0,0 +1,120 @@
1
+ import React, { useCallback, useState } from 'react'
2
+ import { Container, FormAutoSaveIndicator, Typography } from '@toptal/picasso'
3
+ import { SPACING_6, SPACING_4 } from '@toptal/picasso-utils'
4
+ import type { ChangedFields } from '@toptal/picasso-forms'
5
+ import {
6
+ FormNonCompound as Form,
7
+ useFormAutoSave,
8
+ Input,
9
+ RichTextEditor,
10
+ SubmitButton,
11
+ } from '@toptal/picasso-forms'
12
+
13
+ // the emulation of the api call
14
+ const saveWithDelay = async () =>
15
+ new Promise(resolve => setTimeout(() => resolve('success'), 2000))
16
+
17
+ interface FormData {
18
+ 'autoSave-firstName'?: string
19
+ 'autoSave-lastName'?: string
20
+ 'autoSave-about'?: string
21
+ 'autoSave-bio'?: string
22
+ }
23
+
24
+ const autoSaveSubscribedFields: (keyof FormData)[] = [
25
+ 'autoSave-about',
26
+ 'autoSave-bio',
27
+ ]
28
+
29
+ const Example = () => {
30
+ const [autoSaveValues, setAutoSaveValues] = useState<FormData>({
31
+ 'autoSave-firstName': undefined,
32
+ 'autoSave-lastName': undefined,
33
+ 'autoSave-about': undefined,
34
+ 'autoSave-bio': undefined,
35
+ })
36
+
37
+ const handleFormValuesChange = useCallback(
38
+ async (changedFields: ChangedFields<FormData>, values: FormData) => {
39
+ await saveWithDelay()
40
+
41
+ setAutoSaveValues(values)
42
+ },
43
+ []
44
+ )
45
+
46
+ const { autoSaveDecorator, savingFields } = useFormAutoSave({
47
+ subscribedFields: autoSaveSubscribedFields,
48
+ onFormValuesChange: handleFormValuesChange,
49
+ })
50
+
51
+ return (
52
+ <Form<FormData>
53
+ onSubmit={values => window.alert(JSON.stringify(values, undefined, 2))}
54
+ decorators={[autoSaveDecorator]}
55
+ >
56
+ <Container flex direction='row' gap={SPACING_6}>
57
+ <Container>
58
+ <Input
59
+ required
60
+ name='autoSave-firstName'
61
+ label='First name'
62
+ placeholder='e.g. Bruce'
63
+ />
64
+ <Input
65
+ required
66
+ name='autoSave-lastName'
67
+ label='Last name'
68
+ placeholder='e.g. Wayne'
69
+ />
70
+ <Input
71
+ required
72
+ name='autoSave-about'
73
+ multiline
74
+ limit={100}
75
+ rows={5}
76
+ label='About'
77
+ hint='Tell us about yourself'
78
+ placeholder='Please tell us about yourself'
79
+ autoSaveIndicator={
80
+ <FormAutoSaveIndicator
81
+ saving={savingFields?.['autoSave-about']}
82
+ />
83
+ }
84
+ />
85
+ <RichTextEditor
86
+ id='autoSave-rich-text-editor'
87
+ label='Bio'
88
+ required
89
+ name='autoSave-bio'
90
+ hint='Write a short bio'
91
+ placeholder='Write a short bio'
92
+ minLength={5}
93
+ maxLength={25}
94
+ autoSaveIndicator={
95
+ <FormAutoSaveIndicator saving={savingFields?.['autoSave-bio']} />
96
+ }
97
+ />
98
+ </Container>
99
+ <Container variant='grey' padded={SPACING_6}>
100
+ <Typography size='small'>
101
+ Values should be updated only after subscribed fields changes.
102
+ </Typography>
103
+ <pre style={{ width: 500 }}>
104
+ Saved values: {JSON.stringify(autoSaveValues, undefined, 2)}
105
+ </pre>
106
+ {(savingFields?.['autoSave-bio'] ||
107
+ savingFields?.['autoSave-about']) && (
108
+ <Typography size='medium'>Saving...</Typography>
109
+ )}
110
+ </Container>
111
+ </Container>
112
+
113
+ <Container top={SPACING_4}>
114
+ <SubmitButton>Submit</SubmitButton>
115
+ </Container>
116
+ </Form>
117
+ )
118
+ }
119
+
120
+ export default Example
@@ -0,0 +1,81 @@
1
+ import React, { useState } from 'react'
2
+ import type {
3
+ AvatarUploadFileUpload,
4
+ AvatarUploadFileRejection,
5
+ } from '@toptal/picasso'
6
+ import { SPACING_4 } from '@toptal/picasso-utils'
7
+ import { Container } from '@toptal/picasso'
8
+ import {
9
+ FormNonCompound as Form,
10
+ useForm,
11
+ AvatarUpload,
12
+ SubmitButton,
13
+ } from '@toptal/picasso-forms'
14
+
15
+ type FormType = {
16
+ avatarUpload: AvatarUploadFileUpload
17
+ }
18
+
19
+ const FormRenderer = () => {
20
+ const { change } = useForm()
21
+ const [uploading, setUploading] = useState<boolean>(false)
22
+
23
+ const handleDrop = (
24
+ acceptedFile: File | null,
25
+ fileRejection: AvatarUploadFileRejection | null
26
+ ) => {
27
+ if (acceptedFile) {
28
+ // simulate upload with external upload service
29
+ const reader = new FileReader()
30
+
31
+ reader.readAsDataURL(acceptedFile)
32
+
33
+ reader.onload = () => {
34
+ setUploading(true)
35
+
36
+ setTimeout(() => {
37
+ setUploading(false)
38
+
39
+ // set result of upload to form
40
+ change('avatarUpload', { src: reader.result as string })
41
+ }, 1000)
42
+ }
43
+
44
+ reader.onerror = error => {
45
+ console.log('Error: upload failed, ', error)
46
+ }
47
+ } else if (fileRejection) {
48
+ // file rejected
49
+ console.log(fileRejection.errors.join(', '))
50
+ }
51
+ }
52
+
53
+ return (
54
+ <>
55
+ <AvatarUpload
56
+ required
57
+ name='avatarUpload'
58
+ onDrop={handleDrop}
59
+ uploading={uploading}
60
+ />
61
+
62
+ <Container top={SPACING_4}>
63
+ <SubmitButton>Submit</SubmitButton>
64
+ </Container>
65
+ </>
66
+ )
67
+ }
68
+
69
+ const Example = () => {
70
+ const handleSubmit = ({ avatarUpload }: FormType) => {
71
+ window.alert(`src: ${avatarUpload.src}`)
72
+ }
73
+
74
+ return (
75
+ <Form<FormType> autoComplete='off' onSubmit={handleSubmit}>
76
+ <FormRenderer />
77
+ </Form>
78
+ )
79
+ }
80
+
81
+ export default Example