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
@@ -0,0 +1,35 @@
1
+ import type { PageProps } from "keycloakify/login/pages/PageProps";
2
+ import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
3
+ import type { KcContext } from "../kcContext";
4
+ import type { I18n } from "../i18n";
5
+
6
+ export default function Code(props: PageProps<Extract<KcContext, { pageId: "code.ftl" }>, I18n>) {
7
+ const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
8
+
9
+ const { getClassName } = useGetClassName({
10
+ doUseDefaultCss,
11
+ classes
12
+ });
13
+
14
+ const { code } = kcContext;
15
+
16
+ const { msg } = i18n;
17
+
18
+ return (
19
+ <Template
20
+ {...{ kcContext, i18n, doUseDefaultCss, classes }}
21
+ headerNode={code.success ? msg("codeSuccessTitle") : msg("codeErrorTitle", code.error)}
22
+ >
23
+ <div id="kc-code">
24
+ {code.success ? (
25
+ <>
26
+ <p>{msg("copyCodeInstruction")}</p>
27
+ <input id="code" className={getClassName("kcTextareaClass")} value={code.code} />
28
+ </>
29
+ ) : (
30
+ <p id="error">{code.error}</p>
31
+ )}
32
+ </div>
33
+ </Template>
34
+ );
35
+ }
@@ -0,0 +1,53 @@
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 DeleteAccountConfirm(props: PageProps<Extract<KcContext, { pageId: "delete-account-confirm.ftl" }>, I18n>) {
8
+ const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
9
+
10
+ const { getClassName } = useGetClassName({
11
+ doUseDefaultCss,
12
+ classes
13
+ });
14
+
15
+ const { url, triggered_from_aia } = kcContext;
16
+
17
+ const { msg, msgStr } = i18n;
18
+
19
+ return (
20
+ <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("deleteAccountConfirm")}>
21
+ <form action={url.loginAction} className="form-vertical" method="post">
22
+ <div className="alert alert-warning" style={{ "marginTop": "0", "marginBottom": "30px" }}>
23
+ <span className="pficon pficon-warning-triangle-o"></span>
24
+ {msg("irreversibleAction")}
25
+ </div>
26
+ <p>{msg("deletingImplies")}</p>
27
+ <ul style={{ "color": "#72767b", "listStyle": "disc", "listStylePosition": "inside" }}>
28
+ <li>{msg("loggingOutImmediately")}</li>
29
+ <li>{msg("errasingData")}</li>
30
+ </ul>
31
+ <p className="delete-account-text">{msg("finalDeletionConfirmation")}</p>
32
+ <div id="kc-form-buttons">
33
+ <input
34
+ className={clsx(getClassName("kcButtonClass"), getClassName("kcButtonPrimaryClass"), getClassName("kcButtonLargeClass"))}
35
+ type="submit"
36
+ value={msgStr("doConfirmDelete")}
37
+ />
38
+ {triggered_from_aia && (
39
+ <button
40
+ className={clsx(getClassName("kcButtonClass"), getClassName("kcButtonDefaultClass"), getClassName("kcButtonLargeClass"))}
41
+ style={{ "marginLeft": "calc(100% - 220px)" }}
42
+ type="submit"
43
+ name="cancel-aia"
44
+ value="true"
45
+ >
46
+ {msgStr("doCancel")}
47
+ </button>
48
+ )}
49
+ </div>
50
+ </form>
51
+ </Template>
52
+ );
53
+ }
@@ -0,0 +1,45 @@
1
+ import { clsx } from "keycloakify/tools/clsx";
2
+ import type { PageProps } from "keycloakify/login/pages/PageProps";
3
+ import type { KcContext } from "../kcContext";
4
+ import type { I18n } from "../i18n";
5
+ import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
6
+
7
+ export default function DeleteCredential(props: PageProps<Extract<KcContext, { pageId: "delete-credential.ftl" }>, I18n>) {
8
+ const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
9
+
10
+ const { msgStr, msg } = i18n;
11
+
12
+ const { getClassName } = useGetClassName({
13
+ doUseDefaultCss,
14
+ classes
15
+ });
16
+
17
+ const { url, credentialLabel } = kcContext;
18
+
19
+ return (
20
+ <Template
21
+ {...{ kcContext, i18n, doUseDefaultCss, classes }}
22
+ displayMessage={false}
23
+ headerNode={msg("deleteCredentialTitle", credentialLabel)}
24
+ >
25
+ <div id="kc-delete-text">{msg("deleteCredentialMessage", credentialLabel)}</div>
26
+ <form className="form-actions" action={url.loginAction} method="POST">
27
+ <input
28
+ className={clsx(getClassName("kcButtonClass"), getClassName("kcButtonPrimaryClass"), getClassName("kcButtonLargeClass"))}
29
+ name="accept"
30
+ id="kc-accept"
31
+ type="submit"
32
+ value={msgStr("doConfirmDelete")}
33
+ />
34
+ <input
35
+ className={clsx(getClassName("kcButtonClass"), getClassName("kcButtonDefaultClass"), getClassName("kcButtonLargeClass"))}
36
+ name="cancel-aia"
37
+ value={msgStr("doCancel")}
38
+ id="kc-decline"
39
+ type="submit"
40
+ />
41
+ </form>
42
+ <div className="clearfix" />
43
+ </Template>
44
+ );
45
+ }
@@ -5,7 +5,7 @@ import type { I18n } from "../i18n";
5
5
  export default function Error(props: PageProps<Extract<KcContext, { pageId: "error.ftl" }>, I18n>) {
6
6
  const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
7
7
 
8
- const { message, client } = kcContext;
8
+ const { message, client, skipLink } = kcContext;
9
9
 
10
10
  const { msg } = i18n;
11
11
 
@@ -13,7 +13,7 @@ export default function Error(props: PageProps<Extract<KcContext, { pageId: "err
13
13
  <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} displayMessage={false} headerNode={msg("errorTitle")}>
14
14
  <div id="kc-error-message">
15
15
  <p className="instruction">{message.summary}</p>
16
- {client !== undefined && client.baseUrl !== undefined && (
16
+ {!skipLink && client !== undefined && client.baseUrl !== undefined && (
17
17
  <p>
18
18
  <a id="backToApplication" href={client.baseUrl}>
19
19
  {msg("backToApplication")}
@@ -0,0 +1,41 @@
1
+ import { useEffect } from "react";
2
+ import type { PageProps } from "keycloakify/login/pages/PageProps";
3
+ import type { KcContext } from "../kcContext";
4
+ import type { I18n } from "../i18n";
5
+
6
+ export default function FrontchannelLogout(props: PageProps<Extract<KcContext, { pageId: "frontchannel-logout.ftl" }>, I18n>) {
7
+ const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
8
+
9
+ const { logout } = kcContext;
10
+
11
+ const { msg, msgStr } = i18n;
12
+
13
+ useEffect(() => {
14
+ if (logout.logoutRedirectUri) {
15
+ window.location.replace(logout.logoutRedirectUri);
16
+ }
17
+ }, []);
18
+
19
+ return (
20
+ <Template
21
+ {...{ kcContext, i18n, doUseDefaultCss, classes }}
22
+ documentTitle={msgStr("frontchannel-logout.title")}
23
+ headerNode={msg("frontchannel-logout.title")}
24
+ >
25
+ <p>{msg("frontchannel-logout.message")}</p>
26
+ <ul>
27
+ {logout.clients.map(client => (
28
+ <li key={client.name}>
29
+ {client.name}
30
+ <iframe src={client.frontChannelLogoutUrl} style={{ "display": "none" }} />
31
+ </li>
32
+ ))}
33
+ </ul>
34
+ {logout.logoutRedirectUri && (
35
+ <a id="continue" className="btn btn-primary" href={logout.logoutRedirectUri}>
36
+ {msg("doContinue")}
37
+ </a>
38
+ )}
39
+ </Template>
40
+ );
41
+ }
@@ -1,13 +1,18 @@
1
1
  import { useState } from "react";
2
2
  import { clsx } from "keycloakify/tools/clsx";
3
- import { UserProfileFormFields } from "keycloakify/login/pages/shared/UserProfileFormFields";
4
3
  import type { PageProps } from "keycloakify/login/pages/PageProps";
5
4
  import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
5
+ import type { LazyOrNot } from "keycloakify/tools/LazyOrNot";
6
+ import type { UserProfileFormFieldsProps } from "keycloakify/login/UserProfileFormFields";
6
7
  import type { KcContext } from "../kcContext";
7
8
  import type { I18n } from "../i18n";
8
9
 
9
- export default function IdpReviewUserProfile(props: PageProps<Extract<KcContext, { pageId: "idp-review-user-profile.ftl" }>, I18n>) {
10
- const { kcContext, i18n, doUseDefaultCss, Template, classes } = props;
10
+ type IdpReviewUserProfileProps = PageProps<Extract<KcContext, { pageId: "idp-review-user-profile.ftl" }>, I18n> & {
11
+ UserProfileFormFields: LazyOrNot<(props: UserProfileFormFieldsProps) => JSX.Element>;
12
+ };
13
+
14
+ export default function IdpReviewUserProfile(props: IdpReviewUserProfileProps) {
15
+ const { kcContext, i18n, doUseDefaultCss, Template, classes, UserProfileFormFields } = props;
11
16
 
12
17
  const { getClassName } = useGetClassName({
13
18
  doUseDefaultCss,
@@ -16,12 +21,17 @@ export default function IdpReviewUserProfile(props: PageProps<Extract<KcContext,
16
21
 
17
22
  const { msg, msgStr } = i18n;
18
23
 
19
- const { url } = kcContext;
24
+ const { url, messagesPerField } = kcContext;
20
25
 
21
26
  const [isFomSubmittable, setIsFomSubmittable] = useState(false);
22
27
 
23
28
  return (
24
- <Template {...{ kcContext, i18n, doUseDefaultCss, classes }} headerNode={msg("loginIdpReviewProfileTitle")}>
29
+ <Template
30
+ {...{ kcContext, i18n, doUseDefaultCss, classes }}
31
+ displayMessage={messagesPerField.exists("global")}
32
+ displayRequiredFields={true}
33
+ headerNode={msg("loginIdpReviewProfileTitle")}
34
+ >
25
35
  <form id="kc-idp-review-profile-form" className={getClassName("kcFormClass")} action={url.loginAction} method="post">
26
36
  <UserProfileFormFields
27
37
  kcContext={kcContext}
@@ -24,26 +24,36 @@ export default function Info(props: PageProps<Extract<KcContext, { pageId: "info
24
24
  <div id="kc-info-message">
25
25
  <p className="instruction">
26
26
  {message.summary}
27
-
28
- {requiredActions !== undefined && (
29
- <b>{requiredActions.map(requiredAction => msgStr(`requiredAction.${requiredAction}` as const)).join(",")}</b>
30
- )}
27
+ {requiredActions && <b>{requiredActions.map(requiredAction => msgStr(`requiredAction.${requiredAction}` as const)).join(",")}</b>}
31
28
  </p>
32
- {!skipLink && pageRedirectUri !== undefined ? (
33
- <p>
34
- <a href={pageRedirectUri}>{msg("backToApplication")}</a>
35
- </p>
36
- ) : actionUri !== undefined ? (
37
- <p>
38
- <a href={actionUri}>{msg("proceedWithAction")}</a>
39
- </p>
40
- ) : (
41
- client.baseUrl !== undefined && (
42
- <p>
43
- <a href={client.baseUrl}>{msg("backToApplication")}</a>
44
- </p>
45
- )
46
- )}
29
+ {(() => {
30
+ if (skipLink) {
31
+ return null;
32
+ }
33
+
34
+ if (pageRedirectUri) {
35
+ return (
36
+ <p>
37
+ <a href={pageRedirectUri}>{msg("backToApplication")}</a>
38
+ </p>
39
+ );
40
+ }
41
+ if (actionUri) {
42
+ return (
43
+ <p>
44
+ <a href={actionUri}>{msg("proceedWithAction")}</a>
45
+ </p>
46
+ );
47
+ }
48
+
49
+ if (client.baseUrl) {
50
+ return (
51
+ <p>
52
+ <a href={client.baseUrl}>{msg("backToApplication")}</a>
53
+ </p>
54
+ );
55
+ }
56
+ })()}
47
57
  </div>
48
58
  </Template>
49
59
  );
@@ -1,6 +1,6 @@
1
- import { useState, type FormEventHandler } from "react";
1
+ import { useState, useEffect, useReducer } from "react";
2
+ import { assert } from "tsafe/assert";
2
3
  import { clsx } from "keycloakify/tools/clsx";
3
- import { useConstCallback } from "keycloakify/tools/useConstCallback";
4
4
  import type { PageProps } from "keycloakify/login/pages/PageProps";
5
5
  import { useGetClassName } from "keycloakify/login/lib/useGetClassName";
6
6
  import type { KcContext } from "../kcContext";
@@ -14,115 +14,144 @@ export default function Login(props: PageProps<Extract<KcContext, { pageId: "log
14
14
  classes
15
15
  });
16
16
 
17
- const { social, realm, url, usernameHidden, login, auth, registrationDisabled } = kcContext;
17
+ const { social, realm, url, usernameHidden, login, auth, registrationDisabled, messagesPerField } = kcContext;
18
18
 
19
19
  const { msg, msgStr } = i18n;
20
20
 
21
21
  const [isLoginButtonDisabled, setIsLoginButtonDisabled] = useState(false);
22
22
 
23
- const onSubmit = useConstCallback<FormEventHandler<HTMLFormElement>>(e => {
24
- e.preventDefault();
25
-
26
- setIsLoginButtonDisabled(true);
27
-
28
- const formElement = e.target as HTMLFormElement;
29
-
30
- //NOTE: Even if we login with email Keycloak expect username and password in
31
- //the POST request.
32
- formElement.querySelector("input[name='email']")?.setAttribute("name", "username");
33
-
34
- formElement.submit();
35
- });
36
-
37
23
  return (
38
24
  <Template
39
25
  {...{ kcContext, i18n, doUseDefaultCss, classes }}
26
+ displayMessage={!messagesPerField.existsError("username", "password")}
27
+ headerNode={msg("loginAccountTitle")}
40
28
  displayInfo={realm.password && realm.registrationAllowed && !registrationDisabled}
41
- displayWide={realm.password && social.providers !== undefined}
42
- headerNode={msg("doLogIn")}
43
29
  infoNode={
44
- <div id="kc-registration">
45
- <span>
46
- {msg("noAccount")}
47
- <a tabIndex={6} href={url.registrationUrl}>
48
- {msg("doRegister")}
49
- </a>
50
- </span>
30
+ <div id="kc-registration-container">
31
+ <div id="kc-registration">
32
+ <span>
33
+ {msg("noAccount")}{" "}
34
+ <a tabIndex={8} href={url.registrationUrl}>
35
+ {msg("doRegister")}
36
+ </a>
37
+ </span>
38
+ </div>
51
39
  </div>
52
40
  }
53
- >
54
- <div id="kc-form" className={clsx(realm.password && social.providers !== undefined && getClassName("kcContentWrapperClass"))}>
55
- <div
56
- id="kc-form-wrapper"
57
- className={clsx(
58
- realm.password &&
59
- social.providers && [getClassName("kcFormSocialAccountContentClass"), getClassName("kcFormSocialAccountClass")]
41
+ socialProvidersNode={
42
+ <>
43
+ {realm.password && social.providers?.length && (
44
+ <div id="kc-social-providers" className={getClassName("kcFormSocialAccountSectionClass")}>
45
+ <hr />
46
+ <h2>{msg("identity-provider-login-label")}</h2>
47
+ <ul
48
+ className={clsx(
49
+ getClassName("kcFormSocialAccountListClass"),
50
+ social.providers.length > 3 && getClassName("kcFormSocialAccountListGridClass")
51
+ )}
52
+ >
53
+ {social.providers.map((...[p, , providers]) => (
54
+ <li key={p.alias}>
55
+ <a
56
+ id={`social-${p.alias}`}
57
+ className={clsx(
58
+ getClassName("kcFormSocialAccountListButtonClass"),
59
+ providers.length > 3 && getClassName("kcFormSocialAccountGridItem")
60
+ )}
61
+ type="button"
62
+ href={p.loginUrl}
63
+ >
64
+ {p.iconClasses && (
65
+ <i className={clsx(getClassName("kcCommonLogoIdP"), p.iconClasses)} aria-hidden="true"></i>
66
+ )}
67
+ <span
68
+ className={clsx(getClassName("kcFormSocialAccountNameClass"), p.iconClasses && "kc-social-icon-text")}
69
+ >
70
+ {p.displayName}
71
+ </span>
72
+ </a>
73
+ </li>
74
+ ))}
75
+ </ul>
76
+ </div>
60
77
  )}
61
- >
78
+ </>
79
+ }
80
+ >
81
+ <div id="kc-form">
82
+ <div id="kc-form-wrapper">
62
83
  {realm.password && (
63
- <form id="kc-form-login" onSubmit={onSubmit} action={url.loginAction} method="post">
64
- <div className={getClassName("kcFormGroupClass")}>
65
- {!usernameHidden &&
66
- (() => {
67
- const label = !realm.loginWithEmailAllowed
68
- ? "username"
69
- : realm.registrationEmailAsUsername
70
- ? "email"
71
- : "usernameOrEmail";
72
-
73
- const autoCompleteHelper: typeof label = label === "usernameOrEmail" ? "username" : label;
74
-
75
- return (
76
- <>
77
- <label htmlFor={autoCompleteHelper} className={getClassName("kcLabelClass")}>
78
- {msg(label)}
79
- </label>
80
- <input
81
- tabIndex={1}
82
- id={autoCompleteHelper}
83
- className={getClassName("kcInputClass")}
84
- //NOTE: This is used by Google Chrome auto fill so we use it to tell
85
- //the browser how to pre fill the form but before submit we put it back
86
- //to username because it is what keycloak expects.
87
- name={autoCompleteHelper}
88
- defaultValue={login.username ?? ""}
89
- type="text"
90
- autoFocus={true}
91
- autoComplete="off"
92
- />
93
- </>
94
- );
95
- })()}
96
- </div>
84
+ <form
85
+ id="kc-form-login"
86
+ onSubmit={() => {
87
+ setIsLoginButtonDisabled(true);
88
+ return true;
89
+ }}
90
+ action={url.loginAction}
91
+ method="post"
92
+ >
93
+ {!usernameHidden && (
94
+ <div className={getClassName("kcFormGroupClass")}>
95
+ <label htmlFor="username" className={getClassName("kcLabelClass")}>
96
+ {!realm.loginWithEmailAllowed
97
+ ? msg("username")
98
+ : !realm.registrationEmailAsUsername
99
+ ? msg("usernameOrEmail")
100
+ : msg("email")}
101
+ </label>
102
+ <input
103
+ tabIndex={2}
104
+ id="username"
105
+ className={getClassName("kcInputClass")}
106
+ name="username"
107
+ value={login.username ?? ""}
108
+ type="text"
109
+ autoFocus
110
+ autoComplete="username"
111
+ aria-invalid={messagesPerField.existsError("username", "password")}
112
+ />
113
+ {messagesPerField.existsError("username", "password") && (
114
+ <span id="input-error" className={getClassName("kcInputErrorMessageClass")} aria-live="polite">
115
+ {messagesPerField.getFirstError("username", "password")}
116
+ </span>
117
+ )}
118
+ </div>
119
+ )}
120
+
97
121
  <div className={getClassName("kcFormGroupClass")}>
98
122
  <label htmlFor="password" className={getClassName("kcLabelClass")}>
99
123
  {msg("password")}
100
124
  </label>
101
- <input
102
- tabIndex={2}
103
- id="password"
104
- className={getClassName("kcInputClass")}
105
- name="password"
106
- type="password"
107
- autoComplete="off"
108
- />
125
+ <PasswordWrapper getClassName={getClassName} i18n={i18n} passwordInputId="password">
126
+ <input
127
+ tabIndex={3}
128
+ id="password"
129
+ className={getClassName("kcInputClass")}
130
+ name="password"
131
+ type="password"
132
+ autoComplete="current-password"
133
+ aria-invalid={messagesPerField.existsError("username", "password")}
134
+ />
135
+ </PasswordWrapper>
136
+ {usernameHidden && messagesPerField.existsError("username", "password") && (
137
+ <span id="input-error" className={getClassName("kcInputErrorMessageClass")} aria-live="polite">
138
+ {messagesPerField.getFirstError("username", "password")}
139
+ </span>
140
+ )}
109
141
  </div>
142
+
110
143
  <div className={clsx(getClassName("kcFormGroupClass"), getClassName("kcFormSettingClass"))}>
111
144
  <div id="kc-form-options">
112
145
  {realm.rememberMe && !usernameHidden && (
113
146
  <div className="checkbox">
114
147
  <label>
115
148
  <input
116
- tabIndex={3}
149
+ tabIndex={5}
117
150
  id="rememberMe"
118
151
  name="rememberMe"
119
152
  type="checkbox"
120
- {...(login.rememberMe === "on"
121
- ? {
122
- "checked": true
123
- }
124
- : {})}
125
- />
153
+ defaultChecked={!!login.rememberMe}
154
+ />{" "}
126
155
  {msg("rememberMe")}
127
156
  </label>
128
157
  </div>
@@ -131,26 +160,19 @@ export default function Login(props: PageProps<Extract<KcContext, { pageId: "log
131
160
  <div className={getClassName("kcFormOptionsWrapperClass")}>
132
161
  {realm.resetPasswordAllowed && (
133
162
  <span>
134
- <a tabIndex={5} href={url.loginResetCredentialsUrl}>
163
+ <a tabIndex={6} href={url.loginResetCredentialsUrl}>
135
164
  {msg("doForgotPassword")}
136
165
  </a>
137
166
  </span>
138
167
  )}
139
168
  </div>
140
169
  </div>
170
+
141
171
  <div id="kc-form-buttons" className={getClassName("kcFormGroupClass")}>
172
+ <input type="hidden" id="id-hidden-input" name="credentialId" value={auth.selectedCredential} />
142
173
  <input
143
- type="hidden"
144
- id="id-hidden-input"
145
- name="credentialId"
146
- {...(auth?.selectedCredential !== undefined
147
- ? {
148
- "value": auth.selectedCredential
149
- }
150
- : {})}
151
- />
152
- <input
153
- tabIndex={4}
174
+ tabIndex={7}
175
+ disabled={isLoginButtonDisabled}
154
176
  className={clsx(
155
177
  getClassName("kcButtonClass"),
156
178
  getClassName("kcButtonPrimaryClass"),
@@ -161,34 +183,51 @@ export default function Login(props: PageProps<Extract<KcContext, { pageId: "log
161
183
  id="kc-login"
162
184
  type="submit"
163
185
  value={msgStr("doLogIn")}
164
- disabled={isLoginButtonDisabled}
165
186
  />
166
187
  </div>
167
188
  </form>
168
189
  )}
169
190
  </div>
170
- {realm.password && social.providers !== undefined && (
171
- <div
172
- id="kc-social-providers"
173
- className={clsx(getClassName("kcFormSocialAccountContentClass"), getClassName("kcFormSocialAccountClass"))}
174
- >
175
- <ul
176
- className={clsx(
177
- getClassName("kcFormSocialAccountListClass"),
178
- social.providers.length > 4 && getClassName("kcFormSocialAccountDoubleListClass")
179
- )}
180
- >
181
- {social.providers.map(p => (
182
- <li key={p.providerId} className={getClassName("kcFormSocialAccountListLinkClass")}>
183
- <a href={p.loginUrl} id={`zocial-${p.alias}`} className={clsx("zocial", p.providerId)}>
184
- <span>{p.displayName}</span>
185
- </a>
186
- </li>
187
- ))}
188
- </ul>
189
- </div>
190
- )}
191
191
  </div>
192
192
  </Template>
193
193
  );
194
194
  }
195
+
196
+ function PasswordWrapper(props: {
197
+ getClassName: ReturnType<typeof useGetClassName>["getClassName"];
198
+ i18n: I18n;
199
+ passwordInputId: string;
200
+ children: JSX.Element;
201
+ }) {
202
+ const { getClassName, i18n, passwordInputId, children } = props;
203
+
204
+ const { msgStr } = i18n;
205
+
206
+ const [isPasswordRevealed, toggleIsPasswordRevealed] = useReducer((isPasswordRevealed: boolean) => !isPasswordRevealed, false);
207
+
208
+ useEffect(() => {
209
+ const passwordInputElement = document.getElementById(passwordInputId);
210
+
211
+ assert(passwordInputElement instanceof HTMLInputElement);
212
+
213
+ passwordInputElement.type = isPasswordRevealed ? "text" : "password";
214
+ }, [isPasswordRevealed]);
215
+
216
+ return (
217
+ <div className={getClassName("kcInputGroup")}>
218
+ {children}
219
+ <button
220
+ type="button"
221
+ className={getClassName("kcFormPasswordVisibilityButtonClass")}
222
+ aria-label={msgStr(isPasswordRevealed ? "hidePassword" : "showPassword")}
223
+ aria-controls={passwordInputId}
224
+ onClick={toggleIsPasswordRevealed}
225
+ >
226
+ <i
227
+ className={getClassName(isPasswordRevealed ? "kcFormPasswordVisibilityIconHide" : "kcFormPasswordVisibilityIconShow")}
228
+ aria-hidden
229
+ />
230
+ </button>
231
+ </div>
232
+ );
233
+ }