keycloakify 9.6.6 → 10.0.0-rc.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 (424) hide show
  1. package/README.md +1 -1
  2. package/account/Template.js +34 -14
  3. package/account/Template.js.map +1 -1
  4. package/account/i18n/baseMessages/de.d.ts +1 -1
  5. package/account/i18n/baseMessages/de.js +1 -1
  6. package/account/i18n/baseMessages/de.js.map +1 -1
  7. package/account/i18n/baseMessages/en.d.ts +2 -0
  8. package/account/i18n/baseMessages/en.js +2 -0
  9. package/account/i18n/baseMessages/en.js.map +1 -1
  10. package/account/i18n/baseMessages/es.d.ts +230 -0
  11. package/account/i18n/baseMessages/es.js +231 -1
  12. package/account/i18n/baseMessages/es.js.map +1 -1
  13. package/account/i18n/baseMessages/index.d.ts +353 -407
  14. package/account/i18n/baseMessages/index.js +1 -0
  15. package/account/i18n/baseMessages/index.js.map +1 -1
  16. package/account/i18n/baseMessages/pl.d.ts +115 -1
  17. package/account/i18n/baseMessages/pl.js +115 -1
  18. package/account/i18n/baseMessages/pl.js.map +1 -1
  19. package/account/i18n/baseMessages/tr.d.ts +1 -1
  20. package/account/i18n/baseMessages/tr.js +1 -1
  21. package/account/i18n/baseMessages/tr.js.map +1 -1
  22. package/account/i18n/baseMessages/uk.d.ts +337 -0
  23. package/account/i18n/baseMessages/uk.js +342 -0
  24. package/account/i18n/baseMessages/uk.js.map +1 -0
  25. package/account/i18n/baseMessages/zh-CN.d.ts +10 -0
  26. package/account/i18n/baseMessages/zh-CN.js +10 -0
  27. package/account/i18n/baseMessages/zh-CN.js.map +1 -1
  28. package/bin/constants.d.ts +0 -1
  29. package/bin/constants.js +1 -2
  30. package/bin/constants.js.map +1 -1
  31. package/bin/download-builtin-keycloak-theme.js +12 -2
  32. package/bin/download-builtin-keycloak-theme.js.map +1 -1
  33. package/bin/keycloakify/buildJars/buildJar.d.ts +14 -0
  34. package/bin/keycloakify/buildJars/buildJar.js +216 -0
  35. package/bin/keycloakify/buildJars/buildJar.js.map +1 -0
  36. package/bin/keycloakify/buildJars/buildJars.d.ts +10 -0
  37. package/bin/keycloakify/buildJars/buildJars.js +95 -0
  38. package/bin/keycloakify/buildJars/buildJars.js.map +1 -0
  39. package/bin/keycloakify/buildJars/extensionVersions.d.ts +12 -0
  40. package/bin/keycloakify/buildJars/extensionVersions.js +7 -0
  41. package/bin/keycloakify/buildJars/extensionVersions.js.map +1 -0
  42. package/bin/keycloakify/buildJars/generatePom.d.ts +13 -0
  43. package/bin/keycloakify/buildJars/generatePom.js +94 -0
  44. package/bin/keycloakify/buildJars/generatePom.js.map +1 -0
  45. package/bin/keycloakify/buildJars/getKeycloakVersionRangeForJar.d.ts +6 -0
  46. package/bin/keycloakify/buildJars/getKeycloakVersionRangeForJar.js +35 -0
  47. package/bin/keycloakify/buildJars/getKeycloakVersionRangeForJar.js.map +1 -0
  48. package/bin/keycloakify/buildJars/index.d.ts +1 -0
  49. package/bin/keycloakify/buildJars/index.js +18 -0
  50. package/bin/keycloakify/buildJars/index.js.map +1 -0
  51. package/bin/keycloakify/buildOptions/UserProvidedBuildOptions.d.ts +0 -8
  52. package/bin/keycloakify/buildOptions/UserProvidedBuildOptions.js +1 -3
  53. package/bin/keycloakify/buildOptions/UserProvidedBuildOptions.js.map +1 -1
  54. package/bin/keycloakify/buildOptions/buildOptions.d.ts +0 -2
  55. package/bin/keycloakify/buildOptions/buildOptions.js +3 -5
  56. package/bin/keycloakify/buildOptions/buildOptions.js.map +1 -1
  57. package/bin/keycloakify/buildOptions/getNpmWorkspaceRootDirPath.js +1 -1
  58. package/bin/keycloakify/buildOptions/getNpmWorkspaceRootDirPath.js.map +1 -1
  59. package/bin/keycloakify/generateFtl/ftl_object_to_js_code_declaring_an_object.ftl +103 -232
  60. package/bin/keycloakify/generateFtl/generateFtl.js +17 -39
  61. package/bin/keycloakify/generateFtl/generateFtl.js.map +1 -1
  62. package/bin/keycloakify/generateFtl/pageId.d.ts +1 -1
  63. package/bin/keycloakify/generateFtl/pageId.js +11 -3
  64. package/bin/keycloakify/generateFtl/pageId.js.map +1 -1
  65. package/bin/keycloakify/generateStartKeycloakTestingContainer.d.ts +0 -1
  66. package/bin/keycloakify/generateStartKeycloakTestingContainer.js +3 -6
  67. package/bin/keycloakify/generateStartKeycloakTestingContainer.js.map +1 -1
  68. package/bin/keycloakify/generateTheme/bringInAccountV1.d.ts +1 -1
  69. package/bin/keycloakify/generateTheme/bringInAccountV1.js +5 -9
  70. package/bin/keycloakify/generateTheme/bringInAccountV1.js.map +1 -1
  71. package/bin/keycloakify/generateTheme/generateSrcMainResources.d.ts +21 -0
  72. package/bin/keycloakify/generateTheme/generateSrcMainResources.js +340 -0
  73. package/bin/keycloakify/generateTheme/generateSrcMainResources.js.map +1 -0
  74. package/bin/keycloakify/generateTheme/generateTheme.d.ts +5 -13
  75. package/bin/keycloakify/generateTheme/generateTheme.js +30 -293
  76. package/bin/keycloakify/generateTheme/generateTheme.js.map +1 -1
  77. package/bin/keycloakify/generateTheme/generateThemeVariants.d.ts +5 -0
  78. package/bin/keycloakify/generateTheme/generateThemeVariants.js +67 -0
  79. package/bin/keycloakify/generateTheme/generateThemeVariants.js.map +1 -0
  80. package/bin/keycloakify/generateTheme/readFieldNameUsage.js +54 -18
  81. package/bin/keycloakify/generateTheme/readFieldNameUsage.js.map +1 -1
  82. package/bin/keycloakify/keycloakify.js +28 -104
  83. package/bin/keycloakify/keycloakify.js.map +1 -1
  84. package/lib/useGetClassName.js +2 -2
  85. package/lib/useGetClassName.js.map +1 -1
  86. package/login/Fallback.d.ts +7 -1
  87. package/login/Fallback.js +32 -8
  88. package/login/Fallback.js.map +1 -1
  89. package/login/Template.js +76 -20
  90. package/login/Template.js.map +1 -1
  91. package/login/TemplateProps.d.ts +4 -2
  92. package/login/UserProfileFormFields.d.ts +22 -0
  93. package/login/UserProfileFormFields.js +327 -0
  94. package/login/UserProfileFormFields.js.map +1 -0
  95. package/login/i18n/baseMessages/ar.d.ts +3 -0
  96. package/login/i18n/baseMessages/ar.js +3 -0
  97. package/login/i18n/baseMessages/ar.js.map +1 -1
  98. package/login/i18n/baseMessages/ca.d.ts +1 -0
  99. package/login/i18n/baseMessages/ca.js +1 -0
  100. package/login/i18n/baseMessages/ca.js.map +1 -1
  101. package/login/i18n/baseMessages/cs.d.ts +3 -0
  102. package/login/i18n/baseMessages/cs.js +3 -0
  103. package/login/i18n/baseMessages/cs.js.map +1 -1
  104. package/login/i18n/baseMessages/da.d.ts +1 -0
  105. package/login/i18n/baseMessages/da.js +1 -0
  106. package/login/i18n/baseMessages/da.js.map +1 -1
  107. package/login/i18n/baseMessages/de.d.ts +1 -0
  108. package/login/i18n/baseMessages/de.js +1 -0
  109. package/login/i18n/baseMessages/de.js.map +1 -1
  110. package/login/i18n/baseMessages/el.d.ts +1 -0
  111. package/login/i18n/baseMessages/el.js +1 -0
  112. package/login/i18n/baseMessages/el.js.map +1 -1
  113. package/login/i18n/baseMessages/en.d.ts +7 -0
  114. package/login/i18n/baseMessages/en.js +23 -16
  115. package/login/i18n/baseMessages/en.js.map +1 -1
  116. package/login/i18n/baseMessages/es.d.ts +239 -0
  117. package/login/i18n/baseMessages/es.js +240 -1
  118. package/login/i18n/baseMessages/es.js.map +1 -1
  119. package/login/i18n/baseMessages/fa.d.ts +1 -0
  120. package/login/i18n/baseMessages/fa.js +1 -0
  121. package/login/i18n/baseMessages/fa.js.map +1 -1
  122. package/login/i18n/baseMessages/fi.d.ts +1 -0
  123. package/login/i18n/baseMessages/fi.js +1 -0
  124. package/login/i18n/baseMessages/fi.js.map +1 -1
  125. package/login/i18n/baseMessages/fr.d.ts +1 -0
  126. package/login/i18n/baseMessages/fr.js +2 -1
  127. package/login/i18n/baseMessages/fr.js.map +1 -1
  128. package/login/i18n/baseMessages/hu.d.ts +1 -0
  129. package/login/i18n/baseMessages/hu.js +1 -0
  130. package/login/i18n/baseMessages/hu.js.map +1 -1
  131. package/login/i18n/baseMessages/index.d.ts +211 -416
  132. package/login/i18n/baseMessages/index.js +1 -0
  133. package/login/i18n/baseMessages/index.js.map +1 -1
  134. package/login/i18n/baseMessages/it.d.ts +1 -0
  135. package/login/i18n/baseMessages/it.js +1 -0
  136. package/login/i18n/baseMessages/it.js.map +1 -1
  137. package/login/i18n/baseMessages/ja.d.ts +1 -0
  138. package/login/i18n/baseMessages/ja.js +1 -0
  139. package/login/i18n/baseMessages/ja.js.map +1 -1
  140. package/login/i18n/baseMessages/lt.d.ts +1 -0
  141. package/login/i18n/baseMessages/lt.js +1 -0
  142. package/login/i18n/baseMessages/lt.js.map +1 -1
  143. package/login/i18n/baseMessages/lv.d.ts +1 -0
  144. package/login/i18n/baseMessages/lv.js +1 -0
  145. package/login/i18n/baseMessages/lv.js.map +1 -1
  146. package/login/i18n/baseMessages/nl.d.ts +1 -0
  147. package/login/i18n/baseMessages/nl.js +1 -0
  148. package/login/i18n/baseMessages/nl.js.map +1 -1
  149. package/login/i18n/baseMessages/no.d.ts +1 -0
  150. package/login/i18n/baseMessages/no.js +1 -0
  151. package/login/i18n/baseMessages/no.js.map +1 -1
  152. package/login/i18n/baseMessages/pl.d.ts +1 -0
  153. package/login/i18n/baseMessages/pl.js +1 -0
  154. package/login/i18n/baseMessages/pl.js.map +1 -1
  155. package/login/i18n/baseMessages/pt-BR.d.ts +1 -0
  156. package/login/i18n/baseMessages/pt-BR.js +1 -0
  157. package/login/i18n/baseMessages/pt-BR.js.map +1 -1
  158. package/login/i18n/baseMessages/ru.d.ts +1 -0
  159. package/login/i18n/baseMessages/ru.js +1 -0
  160. package/login/i18n/baseMessages/ru.js.map +1 -1
  161. package/login/i18n/baseMessages/sk.d.ts +2 -0
  162. package/login/i18n/baseMessages/sk.js +3 -1
  163. package/login/i18n/baseMessages/sk.js.map +1 -1
  164. package/login/i18n/baseMessages/sv.d.ts +1 -0
  165. package/login/i18n/baseMessages/sv.js +1 -0
  166. package/login/i18n/baseMessages/sv.js.map +1 -1
  167. package/login/i18n/baseMessages/th.d.ts +1 -0
  168. package/login/i18n/baseMessages/th.js +1 -0
  169. package/login/i18n/baseMessages/th.js.map +1 -1
  170. package/login/i18n/baseMessages/tr.d.ts +1 -0
  171. package/login/i18n/baseMessages/tr.js +1 -0
  172. package/login/i18n/baseMessages/tr.js.map +1 -1
  173. package/login/i18n/baseMessages/uk.d.ts +433 -0
  174. package/login/i18n/baseMessages/uk.js +438 -0
  175. package/login/i18n/baseMessages/uk.js.map +1 -0
  176. package/login/i18n/baseMessages/zh-CN.d.ts +273 -3
  177. package/login/i18n/baseMessages/zh-CN.js +325 -55
  178. package/login/i18n/baseMessages/zh-CN.js.map +1 -1
  179. package/login/i18n/i18n.d.ts +4 -0
  180. package/login/i18n/i18n.js +6 -2
  181. package/login/i18n/i18n.js.map +1 -1
  182. package/login/kcContext/KcContext.d.ts +181 -105
  183. package/login/kcContext/KcContext.js +1 -1
  184. package/login/kcContext/KcContext.js.map +1 -1
  185. package/login/kcContext/createGetKcContext.js +3 -9
  186. package/login/kcContext/createGetKcContext.js.map +1 -1
  187. package/login/kcContext/kcContextMocks.d.ts +1 -1
  188. package/login/kcContext/kcContextMocks.js +92 -198
  189. package/login/kcContext/kcContextMocks.js.map +1 -1
  190. package/login/lib/useDownloadTerms.d.ts +4 -1
  191. package/login/lib/useDownloadTerms.js +9 -4
  192. package/login/lib/useDownloadTerms.js.map +1 -1
  193. package/login/lib/useGetClassName.js +112 -73
  194. package/login/lib/useGetClassName.js.map +1 -1
  195. package/login/lib/useUserProfileForm.d.ts +74 -0
  196. package/login/lib/useUserProfileForm.js +868 -0
  197. package/login/lib/useUserProfileForm.js.map +1 -0
  198. package/login/pages/{UpdateUserProfile.d.ts → Code.d.ts} +2 -2
  199. package/login/pages/Code.js +13 -0
  200. package/login/pages/Code.js.map +1 -0
  201. package/login/pages/DeleteAccountConfirm.d.ts +7 -0
  202. package/login/pages/DeleteAccountConfirm.js +14 -0
  203. package/login/pages/DeleteAccountConfirm.js.map +1 -0
  204. package/login/pages/{RegisterUserProfile.d.ts → DeleteCredential.d.ts} +2 -2
  205. package/login/pages/DeleteCredential.js +14 -0
  206. package/login/pages/DeleteCredential.js.map +1 -0
  207. package/login/pages/Error.js +2 -2
  208. package/login/pages/Error.js.map +1 -1
  209. package/login/pages/FrontchannelLogout.d.ts +7 -0
  210. package/login/pages/FrontchannelLogout.js +14 -0
  211. package/login/pages/FrontchannelLogout.js.map +1 -0
  212. package/login/pages/IdpReviewUserProfile.d.ts +8 -2
  213. package/login/pages/IdpReviewUserProfile.js +3 -4
  214. package/login/pages/IdpReviewUserProfile.js.map +1 -1
  215. package/login/pages/Info.js +14 -1
  216. package/login/pages/Info.js.map +1 -1
  217. package/login/pages/Login.js +24 -38
  218. package/login/pages/Login.js.map +1 -1
  219. package/login/pages/LoginConfigTotp.js +8 -1
  220. package/login/pages/LoginConfigTotp.js.map +1 -1
  221. package/login/pages/{LoginDeviceVerifyUserCode.d.ts → LoginOauth2DeviceVerifyUserCode.d.ts} +1 -1
  222. package/login/pages/{LoginDeviceVerifyUserCode.js → LoginOauth2DeviceVerifyUserCode.js} +2 -2
  223. package/login/pages/LoginOauth2DeviceVerifyUserCode.js.map +1 -0
  224. package/login/pages/LoginOauthGrant.js +3 -2
  225. package/login/pages/LoginOauthGrant.js.map +1 -1
  226. package/login/pages/LoginOtp.js +4 -6
  227. package/login/pages/LoginOtp.js.map +1 -1
  228. package/login/pages/LoginPageExpired.js +1 -1
  229. package/login/pages/LoginPageExpired.js.map +1 -1
  230. package/login/pages/LoginPassword.d.ts +1 -1
  231. package/login/pages/LoginPassword.js +18 -11
  232. package/login/pages/LoginPassword.js.map +1 -1
  233. package/login/pages/LoginRecoveryAuthnCodeConfig.d.ts +7 -0
  234. package/login/pages/LoginRecoveryAuthnCodeConfig.js +147 -0
  235. package/login/pages/LoginRecoveryAuthnCodeConfig.js.map +1 -0
  236. package/login/pages/LoginRecoveryAuthnCodeInput.d.ts +7 -0
  237. package/login/pages/LoginRecoveryAuthnCodeInput.js +14 -0
  238. package/login/pages/LoginRecoveryAuthnCodeInput.js.map +1 -0
  239. package/login/pages/LoginResetOtp.d.ts +7 -0
  240. package/login/pages/LoginResetOtp.js +15 -0
  241. package/login/pages/LoginResetOtp.js.map +1 -0
  242. package/login/pages/LoginResetPassword.js +4 -3
  243. package/login/pages/LoginResetPassword.js.map +1 -1
  244. package/login/pages/LoginUpdatePassword.js +21 -3
  245. package/login/pages/LoginUpdatePassword.js.map +1 -1
  246. package/login/pages/LoginUpdateProfile.d.ts +8 -2
  247. package/login/pages/LoginUpdateProfile.js +11 -5
  248. package/login/pages/LoginUpdateProfile.js.map +1 -1
  249. package/login/pages/LoginUsername.js +11 -35
  250. package/login/pages/LoginUsername.js.map +1 -1
  251. package/login/pages/LoginVerifyEmail.js +1 -1
  252. package/login/pages/LoginVerifyEmail.js.map +1 -1
  253. package/login/pages/LoginX509Info.d.ts +7 -0
  254. package/login/pages/LoginX509Info.js +14 -0
  255. package/login/pages/LoginX509Info.js.map +1 -0
  256. package/login/pages/LogoutConfirm.js +1 -1
  257. package/login/pages/LogoutConfirm.js.map +1 -1
  258. package/login/pages/Register.d.ts +8 -2
  259. package/login/pages/Register.js +26 -4
  260. package/login/pages/Register.js.map +1 -1
  261. package/login/pages/SamlPostForm.js +1 -1
  262. package/login/pages/SamlPostForm.js.map +1 -1
  263. package/login/pages/SelectAuthenticator.js +5 -21
  264. package/login/pages/SelectAuthenticator.js.map +1 -1
  265. package/login/pages/Terms.js +4 -6
  266. package/login/pages/Terms.js.map +1 -1
  267. package/login/pages/UpdateEmail.d.ts +8 -2
  268. package/login/pages/UpdateEmail.js +16 -5
  269. package/login/pages/UpdateEmail.js.map +1 -1
  270. package/login/pages/WebauthnAuthenticate.js +121 -83
  271. package/login/pages/WebauthnAuthenticate.js.map +1 -1
  272. package/login/pages/WebauthnError.d.ts +7 -0
  273. package/login/pages/WebauthnError.js +21 -0
  274. package/login/pages/WebauthnError.js.map +1 -0
  275. package/login/pages/WebauthnRegister.d.ts +7 -0
  276. package/login/pages/WebauthnRegister.js +193 -0
  277. package/login/pages/WebauthnRegister.js.map +1 -0
  278. package/package.json +138 -43
  279. package/src/account/Template.tsx +41 -14
  280. package/src/account/i18n/baseMessages/de.ts +1 -1
  281. package/src/account/i18n/baseMessages/en.ts +2 -0
  282. package/src/account/i18n/baseMessages/es.ts +231 -1
  283. package/src/account/i18n/baseMessages/index.ts +1 -0
  284. package/src/account/i18n/baseMessages/pl.ts +115 -1
  285. package/src/account/i18n/baseMessages/tr.ts +1 -1
  286. package/src/account/i18n/baseMessages/uk.ts +343 -0
  287. package/src/account/i18n/baseMessages/zh-CN.ts +10 -0
  288. package/src/bin/constants.ts +0 -1
  289. package/src/bin/download-builtin-keycloak-theme.ts +14 -1
  290. package/src/bin/keycloakify/buildJars/buildJar.ts +163 -0
  291. package/src/bin/keycloakify/buildJars/buildJars.ts +62 -0
  292. package/src/bin/keycloakify/buildJars/extensionVersions.ts +16 -0
  293. package/src/bin/keycloakify/buildJars/generatePom.ts +86 -0
  294. package/src/bin/keycloakify/buildJars/getKeycloakVersionRangeForJar.ts +37 -0
  295. package/src/bin/keycloakify/buildJars/index.ts +1 -0
  296. package/src/bin/keycloakify/buildOptions/UserProvidedBuildOptions.ts +1 -5
  297. package/src/bin/keycloakify/buildOptions/buildOptions.ts +1 -5
  298. package/src/bin/keycloakify/buildOptions/getNpmWorkspaceRootDirPath.ts +1 -1
  299. package/src/bin/keycloakify/generateFtl/ftl_object_to_js_code_declaring_an_object.ftl +103 -232
  300. package/src/bin/keycloakify/generateFtl/generateFtl.ts +15 -31
  301. package/src/bin/keycloakify/generateFtl/pageId.ts +11 -3
  302. package/src/bin/keycloakify/generateStartKeycloakTestingContainer.ts +4 -8
  303. package/src/bin/keycloakify/generateTheme/bringInAccountV1.ts +5 -11
  304. package/src/bin/keycloakify/generateTheme/generateSrcMainResources.ts +267 -0
  305. package/src/bin/keycloakify/generateTheme/generateTheme.ts +22 -309
  306. package/src/bin/keycloakify/generateTheme/generateThemeVariants.ts +50 -0
  307. package/src/bin/keycloakify/generateTheme/readFieldNameUsage.ts +33 -8
  308. package/src/bin/keycloakify/keycloakify.ts +23 -56
  309. package/src/lib/useGetClassName.ts +2 -2
  310. package/src/login/Fallback.tsx +39 -9
  311. package/src/login/Template.tsx +160 -61
  312. package/src/login/TemplateProps.ts +113 -63
  313. package/src/login/UserProfileFormFields.tsx +750 -0
  314. package/src/login/i18n/baseMessages/ar.ts +3 -0
  315. package/src/login/i18n/baseMessages/ca.ts +1 -0
  316. package/src/login/i18n/baseMessages/cs.ts +3 -0
  317. package/src/login/i18n/baseMessages/da.ts +1 -0
  318. package/src/login/i18n/baseMessages/de.ts +1 -0
  319. package/src/login/i18n/baseMessages/el.ts +1 -0
  320. package/src/login/i18n/baseMessages/en.ts +23 -16
  321. package/src/login/i18n/baseMessages/es.ts +240 -1
  322. package/src/login/i18n/baseMessages/fa.ts +1 -0
  323. package/src/login/i18n/baseMessages/fi.ts +1 -0
  324. package/src/login/i18n/baseMessages/fr.ts +2 -1
  325. package/src/login/i18n/baseMessages/hu.ts +1 -0
  326. package/src/login/i18n/baseMessages/index.ts +1 -0
  327. package/src/login/i18n/baseMessages/it.ts +1 -0
  328. package/src/login/i18n/baseMessages/ja.ts +1 -0
  329. package/src/login/i18n/baseMessages/lt.ts +1 -0
  330. package/src/login/i18n/baseMessages/lv.ts +1 -0
  331. package/src/login/i18n/baseMessages/nl.ts +1 -0
  332. package/src/login/i18n/baseMessages/no.ts +1 -0
  333. package/src/login/i18n/baseMessages/pl.ts +1 -0
  334. package/src/login/i18n/baseMessages/pt-BR.ts +1 -0
  335. package/src/login/i18n/baseMessages/ru.ts +1 -0
  336. package/src/login/i18n/baseMessages/sk.ts +3 -1
  337. package/src/login/i18n/baseMessages/sv.ts +1 -0
  338. package/src/login/i18n/baseMessages/th.ts +1 -0
  339. package/src/login/i18n/baseMessages/tr.ts +1 -0
  340. package/src/login/i18n/baseMessages/uk.ts +439 -0
  341. package/src/login/i18n/baseMessages/zh-CN.ts +325 -55
  342. package/src/login/i18n/i18n.tsx +6 -2
  343. package/src/login/kcContext/KcContext.ts +216 -103
  344. package/src/login/kcContext/createGetKcContext.ts +11 -19
  345. package/src/login/kcContext/kcContextMocks.ts +165 -226
  346. package/src/login/lib/useDownloadTerms.ts +15 -7
  347. package/src/login/lib/useGetClassName.ts +112 -83
  348. package/src/login/lib/useUserProfileForm.tsx +1227 -0
  349. package/src/login/pages/Code.tsx +35 -0
  350. package/src/login/pages/DeleteAccountConfirm.tsx +53 -0
  351. package/src/login/pages/DeleteCredential.tsx +45 -0
  352. package/src/login/pages/Error.tsx +2 -2
  353. package/src/login/pages/FrontchannelLogout.tsx +41 -0
  354. package/src/login/pages/IdpReviewUserProfile.tsx +15 -5
  355. package/src/login/pages/Info.tsx +29 -19
  356. package/src/login/pages/Login.tsx +156 -117
  357. package/src/login/pages/LoginConfigTotp.tsx +26 -1
  358. package/src/login/pages/{LoginDeviceVerifyUserCode.tsx → LoginOauth2DeviceVerifyUserCode.tsx} +3 -1
  359. package/src/login/pages/LoginOauthGrant.tsx +38 -2
  360. package/src/login/pages/LoginOtp.tsx +73 -65
  361. package/src/login/pages/LoginPageExpired.tsx +1 -1
  362. package/src/login/pages/LoginPassword.tsx +77 -28
  363. package/src/login/pages/LoginRecoveryAuthnCodeConfig.tsx +260 -0
  364. package/src/login/pages/LoginRecoveryAuthnCodeInput.tsx +73 -0
  365. package/src/login/pages/LoginResetOtp.tsx +70 -0
  366. package/src/login/pages/LoginResetPassword.tsx +11 -4
  367. package/src/login/pages/LoginUpdatePassword.tsx +120 -81
  368. package/src/login/pages/LoginUpdateProfile.tsx +41 -115
  369. package/src/login/pages/LoginUsername.tsx +93 -101
  370. package/src/login/pages/LoginVerifyEmail.tsx +14 -8
  371. package/src/login/pages/LoginX509Info.tsx +94 -0
  372. package/src/login/pages/LogoutConfirm.tsx +2 -2
  373. package/src/login/pages/Register.tsx +83 -131
  374. package/src/login/pages/SamlPostForm.tsx +1 -1
  375. package/src/login/pages/SelectAuthenticator.tsx +29 -52
  376. package/src/login/pages/Terms.tsx +4 -7
  377. package/src/login/pages/UpdateEmail.tsx +69 -54
  378. package/src/login/pages/WebauthnAuthenticate.tsx +205 -165
  379. package/src/login/pages/WebauthnError.tsx +66 -0
  380. package/src/login/pages/WebauthnRegister.tsx +285 -0
  381. package/src/tools/formatNumber.ts +48 -0
  382. package/src/tools/useInsertLinkTags.ts +82 -0
  383. package/src/tools/useInsertScriptTags.ts +106 -0
  384. package/src/tools/useSetClassName.ts +21 -0
  385. package/tools/formatNumber.d.ts +1 -0
  386. package/tools/formatNumber.js +37 -0
  387. package/tools/formatNumber.js.map +1 -0
  388. package/tools/useInsertLinkTags.d.ts +7 -0
  389. package/tools/useInsertLinkTags.js +57 -0
  390. package/tools/useInsertLinkTags.js.map +1 -0
  391. package/tools/useInsertScriptTags.d.ts +20 -0
  392. package/tools/useInsertScriptTags.js +75 -0
  393. package/tools/useInsertScriptTags.js.map +1 -0
  394. package/tools/useSetClassName.d.ts +4 -0
  395. package/tools/useSetClassName.js +16 -0
  396. package/tools/useSetClassName.js.map +1 -0
  397. package/vite-plugin/tsconfig.tsbuildinfo +1 -1
  398. package/bin/keycloakify/generatePom.d.ts +0 -12
  399. package/bin/keycloakify/generatePom.js +0 -59
  400. package/bin/keycloakify/generatePom.js.map +0 -1
  401. package/lib/usePrepareTemplate.d.ts +0 -11
  402. package/lib/usePrepareTemplate.js +0 -80
  403. package/lib/usePrepareTemplate.js.map +0 -1
  404. package/login/lib/useFormValidation.d.ts +0 -47
  405. package/login/lib/useFormValidation.js +0 -316
  406. package/login/lib/useFormValidation.js.map +0 -1
  407. package/login/pages/LoginDeviceVerifyUserCode.js.map +0 -1
  408. package/login/pages/RegisterUserProfile.js +0 -18
  409. package/login/pages/RegisterUserProfile.js.map +0 -1
  410. package/login/pages/UpdateUserProfile.js +0 -17
  411. package/login/pages/UpdateUserProfile.js.map +0 -1
  412. package/login/pages/shared/UserProfileFormFields.d.ts +0 -18
  413. package/login/pages/shared/UserProfileFormFields.js +0 -58
  414. package/login/pages/shared/UserProfileFormFields.js.map +0 -1
  415. package/src/bin/keycloakify/generatePom.ts +0 -70
  416. package/src/lib/usePrepareTemplate.ts +0 -113
  417. package/src/login/lib/useFormValidation.tsx +0 -474
  418. package/src/login/pages/RegisterUserProfile.tsx +0 -72
  419. package/src/login/pages/UpdateUserProfile.tsx +0 -82
  420. package/src/login/pages/shared/UserProfileFormFields.tsx +0 -177
  421. package/src/tools/headInsert.ts +0 -73
  422. package/tools/headInsert.d.ts +0 -12
  423. package/tools/headInsert.js +0 -50
  424. package/tools/headInsert.js.map +0 -1
@@ -1,7 +1,5 @@
1
- import type { FormEventHandler } from "react";
2
1
  import { useState } from "react";
3
2
  import { clsx } from "keycloakify/tools/clsx";
4
- import { useConstCallback } from "keycloakify/tools/useConstCallback";
5
3
  import type { PageProps } from "keycloakify/login/pages/PageProps";
6
4
  import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
7
5
  import type { KcContext } from "../kcContext";
@@ -15,90 +13,108 @@ export default function LoginUsername(props: PageProps<Extract<KcContext, { page
15
13
  classes
16
14
  });
17
15
 
18
- const { social, realm, url, usernameHidden, login, registrationDisabled } = kcContext;
16
+ const { social, realm, url, usernameHidden, login, registrationDisabled, messagesPerField } = kcContext;
19
17
 
20
18
  const { msg, msgStr } = i18n;
21
19
 
22
20
  const [isLoginButtonDisabled, setIsLoginButtonDisabled] = useState(false);
23
21
 
24
- const onSubmit = useConstCallback<FormEventHandler<HTMLFormElement>>(e => {
25
- e.preventDefault();
26
-
27
- setIsLoginButtonDisabled(true);
28
-
29
- const formElement = e.target as HTMLFormElement;
30
-
31
- //NOTE: Even if we login with email Keycloak expect username and password in
32
- //the POST request.
33
- formElement.querySelector("input[name='email']")?.setAttribute("name", "username");
34
-
35
- formElement.submit();
36
- });
37
-
38
22
  return (
39
23
  <Template
40
24
  {...{ kcContext, i18n, doUseDefaultCss, classes }}
41
- displayInfo={social.displayInfo}
42
- displayWide={realm.password && social.providers !== undefined}
43
- headerNode={msg("doLogIn")}
25
+ displayMessage={!messagesPerField.existsError("username")}
26
+ displayInfo={realm.password && realm.registrationAllowed && !registrationDisabled}
44
27
  infoNode={
45
- realm.password &&
46
- realm.registrationAllowed &&
47
- !registrationDisabled && (
48
- <div id="kc-registration">
49
- <span>
50
- {msg("noAccount")}
51
- <a tabIndex={6} href={url.registrationUrl}>
52
- {msg("doRegister")}
53
- </a>
54
- </span>
55
- </div>
56
- )
28
+ <div id="kc-registration">
29
+ <span>
30
+ {msg("noAccount")}
31
+ <a tabIndex={6} href={url.registrationUrl}>
32
+ {msg("doRegister")}
33
+ </a>
34
+ </span>
35
+ </div>
57
36
  }
58
- >
59
- <div id="kc-form" className={clsx(realm.password && social.providers !== undefined && getClassName("kcContentWrapperClass"))}>
60
- <div
61
- id="kc-form-wrapper"
62
- className={clsx(
63
- realm.password &&
64
- social.providers && [getClassName("kcFormSocialAccountContentClass"), getClassName("kcFormSocialAccountClass")]
37
+ headerNode={msg("doLogIn")}
38
+ socialProvidersNode={
39
+ <>
40
+ {realm.password && social.providers?.length && (
41
+ <div id="kc-social-providers" className={getClassName("kcFormSocialAccountSectionClass")}>
42
+ <hr />
43
+ <h2>{msg("identity-provider-login-label")}</h2>
44
+ <ul
45
+ className={clsx(
46
+ getClassName("kcFormSocialAccountListClass"),
47
+ social.providers.length > 3 && getClassName("kcFormSocialAccountListGridClass")
48
+ )}
49
+ >
50
+ {social.providers.map((...[p, , providers]) => (
51
+ <li key={p.alias}>
52
+ <a
53
+ id={`social-${p.alias}`}
54
+ className={clsx(
55
+ getClassName("kcFormSocialAccountListButtonClass"),
56
+ providers.length > 3 && getClassName("kcFormSocialAccountGridItem")
57
+ )}
58
+ type="button"
59
+ href={p.loginUrl}
60
+ >
61
+ {p.iconClasses && (
62
+ <i className={clsx(getClassName("kcCommonLogoIdP"), p.iconClasses)} aria-hidden="true"></i>
63
+ )}
64
+ <span
65
+ className={clsx(getClassName("kcFormSocialAccountNameClass"), p.iconClasses && "kc-social-icon-text")}
66
+ >
67
+ {p.displayName}
68
+ </span>
69
+ </a>
70
+ </li>
71
+ ))}
72
+ </ul>
73
+ </div>
65
74
  )}
66
- >
75
+ </>
76
+ }
77
+ >
78
+ <div id="kc-form">
79
+ <div id="kc-form-wrapper">
67
80
  {realm.password && (
68
- <form id="kc-form-login" onSubmit={onSubmit} action={url.loginAction} method="post">
69
- <div className={getClassName("kcFormGroupClass")}>
70
- {!usernameHidden &&
71
- (() => {
72
- const label = !realm.loginWithEmailAllowed
73
- ? "username"
74
- : realm.registrationEmailAsUsername
75
- ? "email"
76
- : "usernameOrEmail";
77
-
78
- const autoCompleteHelper: typeof label = label === "usernameOrEmail" ? "username" : label;
81
+ <form
82
+ id="kc-form-login"
83
+ onSubmit={() => {
84
+ setIsLoginButtonDisabled(true);
85
+ return true;
86
+ }}
87
+ action={url.loginAction}
88
+ method="post"
89
+ >
90
+ {!usernameHidden && (
91
+ <div className={getClassName("kcFormGroupClass")}>
92
+ <label htmlFor="username" className={getClassName("kcLabelClass")}>
93
+ {!realm.loginWithEmailAllowed
94
+ ? msg("username")
95
+ : !realm.registrationEmailAsUsername
96
+ ? msg("usernameOrEmail")
97
+ : msg("email")}
98
+ </label>
99
+ <input
100
+ tabIndex={2}
101
+ id="username"
102
+ className={getClassName("kcInputClass")}
103
+ name="username"
104
+ value={login.username ?? ""}
105
+ type="text"
106
+ autoFocus
107
+ autoComplete="off"
108
+ aria-invalid={messagesPerField.existsError("username")}
109
+ />
110
+ {messagesPerField.existsError("username") && (
111
+ <span id="input-error" className={getClassName("kcInputErrorMessageClass")} aria-live="polite">
112
+ {messagesPerField.getFirstError("username")}
113
+ </span>
114
+ )}
115
+ </div>
116
+ )}
79
117
 
80
- return (
81
- <>
82
- <label htmlFor={autoCompleteHelper} className={getClassName("kcLabelClass")}>
83
- {msg(label)}
84
- </label>
85
- <input
86
- tabIndex={1}
87
- id={autoCompleteHelper}
88
- className={getClassName("kcInputClass")}
89
- //NOTE: This is used by Google Chrome auto fill so we use it to tell
90
- //the browser how to pre fill the form but before submit we put it back
91
- //to username because it is what keycloak expects.
92
- name={autoCompleteHelper}
93
- defaultValue={login.username ?? ""}
94
- type="text"
95
- autoFocus={true}
96
- autoComplete="off"
97
- />
98
- </>
99
- );
100
- })()}
101
- </div>
102
118
  <div className={clsx(getClassName("kcFormGroupClass"), getClassName("kcFormSettingClass"))}>
103
119
  <div id="kc-form-options">
104
120
  {realm.rememberMe && !usernameHidden && (
@@ -109,21 +125,19 @@ export default function LoginUsername(props: PageProps<Extract<KcContext, { page
109
125
  id="rememberMe"
110
126
  name="rememberMe"
111
127
  type="checkbox"
112
- {...(login.rememberMe === "on"
113
- ? {
114
- "checked": true
115
- }
116
- : {})}
117
- />
128
+ defaultChecked={!!login.rememberMe}
129
+ />{" "}
118
130
  {msg("rememberMe")}
119
131
  </label>
120
132
  </div>
121
133
  )}
122
134
  </div>
123
135
  </div>
136
+
124
137
  <div id="kc-form-buttons" className={getClassName("kcFormGroupClass")}>
125
138
  <input
126
139
  tabIndex={4}
140
+ disabled={isLoginButtonDisabled}
127
141
  className={clsx(
128
142
  getClassName("kcButtonClass"),
129
143
  getClassName("kcButtonPrimaryClass"),
@@ -134,33 +148,11 @@ export default function LoginUsername(props: PageProps<Extract<KcContext, { page
134
148
  id="kc-login"
135
149
  type="submit"
136
150
  value={msgStr("doLogIn")}
137
- disabled={isLoginButtonDisabled}
138
151
  />
139
152
  </div>
140
153
  </form>
141
154
  )}
142
155
  </div>
143
- {realm.password && social.providers !== undefined && (
144
- <div
145
- id="kc-social-providers"
146
- className={clsx(getClassName("kcFormSocialAccountContentClass"), getClassName("kcFormSocialAccountClass"))}
147
- >
148
- <ul
149
- className={clsx(
150
- getClassName("kcFormSocialAccountListClass"),
151
- social.providers.length > 4 && getClassName("kcFormSocialAccountDoubleListClass")
152
- )}
153
- >
154
- {social.providers.map(p => (
155
- <li key={p.providerId} className={getClassName("kcFormSocialAccountListLinkClass")}>
156
- <a href={p.loginUrl} id={`zocial-${p.alias}`} className={clsx("zocial", p.providerId)}>
157
- <span>{p.displayName}</span>
158
- </a>
159
- </li>
160
- ))}
161
- </ul>
162
- </div>
163
- )}
164
156
  </div>
165
157
  </Template>
166
158
  );
@@ -10,15 +10,21 @@ export default function LoginVerifyEmail(props: PageProps<Extract<KcContext, { p
10
10
  const { url, user } = kcContext;
11
11
 
12
12
  return (
13
- <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} displayMessage={false} headerNode={msg("emailVerifyTitle")}>
13
+ <Template
14
+ {...{ kcContext, i18n, doUseDefaultCss, classes }}
15
+ displayInfo
16
+ headerNode={msg("emailVerifyTitle")}
17
+ infoNode={
18
+ <p className="instruction">
19
+ {msg("emailVerifyInstruction2")}
20
+ <br />
21
+ <a href={url.loginAction}>{msg("doClickHere")}</a>
22
+ &nbsp;
23
+ {msg("emailVerifyInstruction3")}
24
+ </p>
25
+ }
26
+ >
14
27
  <p className="instruction">{msg("emailVerifyInstruction1", user?.email ?? "")}</p>
15
- <p className="instruction">
16
- {msg("emailVerifyInstruction2")}
17
- <br />
18
- <a href={url.loginAction}>{msg("doClickHere")}</a>
19
- &nbsp;
20
- {msg("emailVerifyInstruction3")}
21
- </p>
22
28
  </Template>
23
29
  );
24
30
  }
@@ -0,0 +1,94 @@
1
+ import { clsx } from "keycloakify/tools/clsx";
2
+ import type { PageProps } from "keycloakify/login/pages/PageProps";
3
+ import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
4
+ import type { KcContext } from "../kcContext";
5
+ import type { I18n } from "../i18n";
6
+
7
+ export default function LoginX509Info(props: PageProps<Extract<KcContext, { pageId: "login-x509-info.ftl" }>, I18n>) {
8
+ const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
9
+
10
+ const { getClassName } = useGetClassName({
11
+ doUseDefaultCss,
12
+ classes
13
+ });
14
+
15
+ const { url, x509 } = kcContext;
16
+
17
+ const { msg, msgStr } = i18n;
18
+
19
+ return (
20
+ <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("doLogIn")}>
21
+ <form id="kc-x509-login-info" className={getClassName("kcFormClass")} action={url.loginAction} method="post">
22
+ <div className={getClassName("kcFormGroupClass")}>
23
+ <div className={getClassName("kcLabelWrapperClass")}>
24
+ <label htmlFor="certificate_subjectDN" className={getClassName("kcLabelClass")}>
25
+ {msg("clientCertificate")}
26
+ </label>
27
+ </div>
28
+ {x509.formData.subjectDN ? (
29
+ <div className={getClassName("kcLabelWrapperClass")}>
30
+ <label id="certificate_subjectDN" className={getClassName("kcLabelClass")}>
31
+ {x509.formData.subjectDN}
32
+ </label>
33
+ </div>
34
+ ) : (
35
+ <div className={getClassName("kcLabelWrapperClass")}>
36
+ <label id="certificate_subjectDN" className={getClassName("kcLabelClass")}>
37
+ {msg("noCertificate")}
38
+ </label>
39
+ </div>
40
+ )}
41
+ </div>
42
+ <div className={getClassName("kcFormGroupClass")}>
43
+ {x509.formData.isUserEnabled && (
44
+ <>
45
+ <div className={getClassName("kcLabelWrapperClass")}>
46
+ <label htmlFor="username" className={getClassName("kcLabelClass")}>
47
+ {msg("doX509Login")}
48
+ </label>
49
+ </div>
50
+ <div className={getClassName("kcLabelWrapperClass")}>
51
+ <label id="username" className={getClassName("kcLabelClass")}>
52
+ {x509.formData.username}
53
+ </label>
54
+ </div>
55
+ </>
56
+ )}
57
+ </div>
58
+ <div className={getClassName("kcFormGroupClass")}>
59
+ <div id="kc-form-options" className={getClassName("kcFormOptionsClass")}>
60
+ <div className={getClassName("kcFormOptionsWrapperClass")} />
61
+ </div>
62
+ <div id="kc-form-buttons" className={getClassName("kcFormButtonsClass")}>
63
+ <div className={getClassName("kcFormButtonsWrapperClass")}>
64
+ <input
65
+ className={clsx(
66
+ getClassName("kcButtonClass"),
67
+ getClassName("kcButtonPrimaryClass"),
68
+ getClassName("kcButtonLargeClass")
69
+ )}
70
+ name="login"
71
+ id="kc-login"
72
+ type="submit"
73
+ value={msgStr("doContinue")}
74
+ />
75
+ {x509.formData.isUserEnabled && (
76
+ <input
77
+ className={clsx(
78
+ getClassName("kcButtonClass"),
79
+ getClassName("kcButtonDefaultClass"),
80
+ getClassName("kcButtonLargeClass")
81
+ )}
82
+ name="cancel"
83
+ id="kc-cancel"
84
+ type="submit"
85
+ value={msgStr("doIgnore")}
86
+ />
87
+ )}
88
+ </div>
89
+ </div>
90
+ </div>
91
+ </form>
92
+ </Template>
93
+ );
94
+ }
@@ -17,7 +17,7 @@ export default function LogoutConfirm(props: PageProps<Extract<KcContext, { page
17
17
  const { msg, msgStr } = i18n;
18
18
 
19
19
  return (
20
- <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} displayMessage={false} headerNode={msg("logoutConfirmTitle")}>
20
+ <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("logoutConfirmTitle")}>
21
21
  <div id="kc-logout-confirm" className="content-area">
22
22
  <p className="instruction">{msg("logoutConfirmHeader")}</p>
23
23
  <form className="form-actions" action={url.logoutConfirmAction} method="POST">
@@ -46,7 +46,7 @@ export default function LogoutConfirm(props: PageProps<Extract<KcContext, { page
46
46
  <div id="kc-info-message">
47
47
  {!logoutConfirm.skipLink && client.baseUrl && (
48
48
  <p>
49
- <a href={client.baseUrl} dangerouslySetInnerHTML={{ __html: msgStr("backToApplication") }} />
49
+ <a href={client.baseUrl}>{msg("backToApplication")}</a>
50
50
  </p>
51
51
  )}
52
52
  </div>
@@ -1,151 +1,52 @@
1
+ import { useState } from "react";
1
2
  import { clsx } from "keycloakify/tools/clsx";
2
3
  import type { PageProps } from "keycloakify/login/pages/PageProps";
3
4
  import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
5
+ import type { LazyOrNot } from "keycloakify/tools/LazyOrNot";
6
+ import { useTermsMarkdown } from "keycloakify/login/lib/useDownloadTerms";
7
+ import type { UserProfileFormFieldsProps } from "keycloakify/login/UserProfileFormFields";
8
+ import { Markdown } from "keycloakify/tools/Markdown";
4
9
  import type { KcContext } from "../kcContext";
5
10
  import type { I18n } from "../i18n";
6
11
 
7
- export default function Register(props: PageProps<Extract<KcContext, { pageId: "register.ftl" }>, I18n>) {
8
- const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
12
+ type RegisterProps = PageProps<Extract<KcContext, { pageId: "register.ftl" }>, I18n> & {
13
+ UserProfileFormFields: LazyOrNot<(props: UserProfileFormFieldsProps) => JSX.Element>;
14
+ };
15
+
16
+ export default function Register(props: RegisterProps) {
17
+ const { kcContext, i18n, doUseDefaultCss, Template, classes, UserProfileFormFields } = props;
9
18
 
10
19
  const { getClassName } = useGetClassName({
11
20
  doUseDefaultCss,
12
21
  classes
13
22
  });
14
23
 
15
- const { url, messagesPerField, register, realm, passwordRequired, recaptchaRequired, recaptchaSiteKey } = kcContext;
24
+ const { url, messagesPerField, recaptchaRequired, recaptchaSiteKey, termsAcceptanceRequired } = kcContext;
16
25
 
17
26
  const { msg, msgStr } = i18n;
18
27
 
28
+ const [isFormSubmittable, setIsFormSubmittable] = useState(false);
29
+
19
30
  return (
20
31
  <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("registerTitle")}>
21
32
  <form id="kc-register-form" className={getClassName("kcFormClass")} action={url.registrationAction} method="post">
22
- <div
23
- className={clsx(
24
- getClassName("kcFormGroupClass"),
25
- messagesPerField.printIfExists("firstName", getClassName("kcFormGroupErrorClass"))
26
- )}
27
- >
28
- <div className={getClassName("kcLabelWrapperClass")}>
29
- <label htmlFor="firstName" className={getClassName("kcLabelClass")}>
30
- {msg("firstName")}
31
- </label>
32
- </div>
33
- <div className={getClassName("kcInputWrapperClass")}>
34
- <input
35
- type="text"
36
- id="firstName"
37
- className={getClassName("kcInputClass")}
38
- name="firstName"
39
- defaultValue={register.formData.firstName ?? ""}
40
- />
41
- </div>
42
- </div>
43
-
44
- <div
45
- className={clsx(
46
- getClassName("kcFormGroupClass"),
47
- messagesPerField.printIfExists("lastName", getClassName("kcFormGroupErrorClass"))
48
- )}
49
- >
50
- <div className={getClassName("kcLabelWrapperClass")}>
51
- <label htmlFor="lastName" className={getClassName("kcLabelClass")}>
52
- {msg("lastName")}
53
- </label>
54
- </div>
55
- <div className={getClassName("kcInputWrapperClass")}>
56
- <input
57
- type="text"
58
- id="lastName"
59
- className={getClassName("kcInputClass")}
60
- name="lastName"
61
- defaultValue={register.formData.lastName ?? ""}
62
- />
63
- </div>
64
- </div>
65
-
66
- <div
67
- className={clsx(getClassName("kcFormGroupClass"), messagesPerField.printIfExists("email", getClassName("kcFormGroupErrorClass")))}
68
- >
69
- <div className={getClassName("kcLabelWrapperClass")}>
70
- <label htmlFor="email" className={getClassName("kcLabelClass")}>
71
- {msg("email")}
72
- </label>
73
- </div>
74
- <div className={getClassName("kcInputWrapperClass")}>
75
- <input
76
- type="text"
77
- id="email"
78
- className={getClassName("kcInputClass")}
79
- name="email"
80
- defaultValue={register.formData.email ?? ""}
81
- autoComplete="email"
82
- />
83
- </div>
84
- </div>
85
- {!realm.registrationEmailAsUsername && (
86
- <div
87
- className={clsx(
88
- getClassName("kcFormGroupClass"),
89
- messagesPerField.printIfExists("username", getClassName("kcFormGroupErrorClass"))
90
- )}
91
- >
92
- <div className={getClassName("kcLabelWrapperClass")}>
93
- <label htmlFor="username" className={getClassName("kcLabelClass")}>
94
- {msg("username")}
95
- </label>
96
- </div>
97
- <div className={getClassName("kcInputWrapperClass")}>
98
- <input
99
- type="text"
100
- id="username"
101
- className={getClassName("kcInputClass")}
102
- name="username"
103
- defaultValue={register.formData.username ?? ""}
104
- autoComplete="username"
105
- />
106
- </div>
107
- </div>
108
- )}
109
- {passwordRequired && (
110
- <>
111
- <div
112
- className={clsx(
113
- getClassName("kcFormGroupClass"),
114
- messagesPerField.printIfExists("password", getClassName("kcFormGroupErrorClass"))
115
- )}
116
- >
117
- <div className={getClassName("kcLabelWrapperClass")}>
118
- <label htmlFor="password" className={getClassName("kcLabelClass")}>
119
- {msg("password")}
120
- </label>
121
- </div>
122
- <div className={getClassName("kcInputWrapperClass")}>
123
- <input
124
- type="password"
125
- id="password"
126
- className={getClassName("kcInputClass")}
127
- name="password"
128
- autoComplete="new-password"
129
- />
130
- </div>
131
- </div>
132
-
133
- <div
134
- className={clsx(
135
- getClassName("kcFormGroupClass"),
136
- messagesPerField.printIfExists("password-confirm", getClassName("kcFormGroupErrorClass"))
137
- )}
138
- >
139
- <div className={getClassName("kcLabelWrapperClass")}>
140
- <label htmlFor="password-confirm" className={getClassName("kcLabelClass")}>
141
- {msg("passwordConfirm")}
142
- </label>
143
- </div>
144
- <div className={getClassName("kcInputWrapperClass")}>
145
- <input type="password" id="password-confirm" className={getClassName("kcInputClass")} name="password-confirm" />
146
- </div>
147
- </div>
148
- </>
33
+ <UserProfileFormFields
34
+ {...{
35
+ kcContext,
36
+ i18n,
37
+ getClassName,
38
+ messagesPerField
39
+ }}
40
+ onIsFormSubmittableValueChange={setIsFormSubmittable}
41
+ />
42
+ {termsAcceptanceRequired && (
43
+ <TermsAcceptance
44
+ {...{
45
+ i18n,
46
+ getClassName,
47
+ messagesPerField
48
+ }}
49
+ />
149
50
  )}
150
51
  {recaptchaRequired && (
151
52
  <div className="form-group">
@@ -162,9 +63,9 @@ export default function Register(props: PageProps<Extract<KcContext, { pageId: "
162
63
  </span>
163
64
  </div>
164
65
  </div>
165
-
166
66
  <div id="kc-form-buttons" className={getClassName("kcFormButtonsClass")}>
167
67
  <input
68
+ disabled={!isFormSubmittable}
168
69
  className={clsx(
169
70
  getClassName("kcButtonClass"),
170
71
  getClassName("kcButtonPrimaryClass"),
@@ -180,3 +81,54 @@ export default function Register(props: PageProps<Extract<KcContext, { pageId: "
180
81
  </Template>
181
82
  );
182
83
  }
84
+
85
+ function TermsAcceptance(props: {
86
+ i18n: I18n;
87
+ getClassName: ReturnType<typeof useGetClassName>["getClassName"];
88
+ messagesPerField: Pick<KcContext.Common["messagesPerField"], "existsError" | "get">;
89
+ }) {
90
+ const { i18n, getClassName, messagesPerField } = props;
91
+
92
+ const { msg } = i18n;
93
+
94
+ // NOTE: Refer to https://docs.keycloakify.dev/terms-and-conditions to load your terms and conditions.
95
+ const { termsMarkdown } = useTermsMarkdown();
96
+
97
+ if (termsMarkdown === undefined) {
98
+ return null;
99
+ }
100
+
101
+ return (
102
+ <>
103
+ <div className="form-group">
104
+ <div className={getClassName("kcInputWrapperClass")}>
105
+ {msg("termsTitle")}
106
+ <div id="kc-registration-terms-text">
107
+ <Markdown>{termsMarkdown}</Markdown>
108
+ </div>
109
+ </div>
110
+ </div>
111
+ <div className="form-group">
112
+ <div className={getClassName("kcLabelWrapperClass")}>
113
+ <input
114
+ type="checkbox"
115
+ id="termsAccepted"
116
+ name="termsAccepted"
117
+ className={getClassName("kcCheckboxInputClass")}
118
+ aria-invalid={messagesPerField.existsError("termsAccepted")}
119
+ />
120
+ <label htmlFor="termsAccepted" className={getClassName("kcLabelClass")}>
121
+ {msg("acceptTerms")}
122
+ </label>
123
+ </div>
124
+ {messagesPerField.existsError("termsAccepted") && (
125
+ <div className={getClassName("kcLabelWrapperClass")}>
126
+ <span id="input-error-terms-accepted" className={getClassName("kcInputErrorMessageClass")} aria-live="polite">
127
+ {messagesPerField.get("termsAccepted")}
128
+ </span>
129
+ </div>
130
+ )}
131
+ </div>
132
+ </>
133
+ );
134
+ }
@@ -26,7 +26,7 @@ export default function SamlPostForm(props: PageProps<Extract<KcContext, { pageI
26
26
  htmlFormElement.submit();
27
27
  }, [htmlFormElement]);
28
28
  return (
29
- <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} displayMessage={false} headerNode={msg("saml.post-form.title")}>
29
+ <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("saml.post-form.title")}>
30
30
  <p>{msg("saml.post-form.message")}</p>
31
31
  <form name="saml-post-binding" method="post" action={samlPost.url} ref={setHtmlFormElement}>
32
32
  {samlPost.SAMLRequest && <input type="hidden" name="SAMLRequest" value={samlPost.SAMLRequest} />}