@tinyrack/tinyauth-server 0.5.1 → 0.6.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 (362) hide show
  1. package/dist/entities/oauth-client.entity.d.ts +27 -0
  2. package/dist/entities/oauth-client.entity.d.ts.map +1 -1
  3. package/dist/entities/oauth-code.entity.d.ts +27 -0
  4. package/dist/entities/oauth-code.entity.d.ts.map +1 -1
  5. package/dist/entities/oauth-device-code.entity.d.ts +27 -0
  6. package/dist/entities/oauth-device-code.entity.d.ts.map +1 -1
  7. package/dist/entities/oauth-device-code.entity.js +12 -0
  8. package/dist/entities/oauth-device-code.entity.js.map +1 -1
  9. package/dist/entities/revoked-token.entity.d.ts +27 -0
  10. package/dist/entities/revoked-token.entity.d.ts.map +1 -1
  11. package/dist/entities/user-consent.entity.d.ts +27 -0
  12. package/dist/entities/user-consent.entity.d.ts.map +1 -1
  13. package/dist/entrypoints/app.d.ts +325 -16
  14. package/dist/entrypoints/app.d.ts.map +1 -1
  15. package/dist/entrypoints/app.js +33 -7
  16. package/dist/entrypoints/app.js.map +1 -1
  17. package/dist/entrypoints/database/postgres/compiled-functions.d.ts +16 -7
  18. package/dist/entrypoints/database/postgres/compiled-functions.d.ts.map +1 -1
  19. package/dist/entrypoints/database/postgres/compiled-functions.js +183 -39
  20. package/dist/entrypoints/database/postgres/compiled-functions.js.map +1 -1
  21. package/dist/entrypoints/database/sqlite/compiled-functions.d.ts +16 -7
  22. package/dist/entrypoints/database/sqlite/compiled-functions.d.ts.map +1 -1
  23. package/dist/entrypoints/database/sqlite/compiled-functions.js +183 -39
  24. package/dist/entrypoints/database/sqlite/compiled-functions.js.map +1 -1
  25. package/dist/lib/config/admin.d.ts +9 -0
  26. package/dist/lib/config/admin.d.ts.map +1 -0
  27. package/dist/lib/config/admin.js +13 -0
  28. package/dist/lib/config/admin.js.map +1 -0
  29. package/dist/lib/config/auth.d.ts +85 -0
  30. package/dist/lib/config/auth.d.ts.map +1 -1
  31. package/dist/lib/config/auth.js +61 -0
  32. package/dist/lib/config/auth.js.map +1 -1
  33. package/dist/lib/config/client.d.ts +20 -0
  34. package/dist/lib/config/client.d.ts.map +1 -1
  35. package/dist/lib/config/client.js +14 -0
  36. package/dist/lib/config/client.js.map +1 -1
  37. package/dist/lib/config/index.d.ts +2 -0
  38. package/dist/lib/config/index.d.ts.map +1 -1
  39. package/dist/lib/config/index.js +1 -0
  40. package/dist/lib/config/index.js.map +1 -1
  41. package/dist/lib/config/resolved.d.ts +40 -0
  42. package/dist/lib/config/resolved.d.ts.map +1 -1
  43. package/dist/lib/config/resolved.js +2 -0
  44. package/dist/lib/config/resolved.js.map +1 -1
  45. package/dist/lib/config/server.d.ts.map +1 -1
  46. package/dist/lib/config/server.js +12 -0
  47. package/dist/lib/config/server.js.map +1 -1
  48. package/dist/lib/swagger-tags.d.ts +1 -0
  49. package/dist/lib/swagger-tags.d.ts.map +1 -1
  50. package/dist/lib/swagger-tags.js +1 -0
  51. package/dist/lib/swagger-tags.js.map +1 -1
  52. package/dist/middleware/auth.d.ts +3 -0
  53. package/dist/middleware/auth.d.ts.map +1 -1
  54. package/dist/middleware/auth.js +26 -0
  55. package/dist/middleware/auth.js.map +1 -1
  56. package/dist/middleware/session.d.ts +33 -1
  57. package/dist/middleware/session.d.ts.map +1 -1
  58. package/dist/middleware/session.js +84 -3
  59. package/dist/middleware/session.js.map +1 -1
  60. package/dist/migrations/postgres/Migration20260624190500_add_oauth_device_denied_at.d.ts +6 -0
  61. package/dist/migrations/postgres/Migration20260624190500_add_oauth_device_denied_at.d.ts.map +1 -0
  62. package/dist/migrations/postgres/Migration20260624190500_add_oauth_device_denied_at.js +11 -0
  63. package/dist/migrations/postgres/Migration20260624190500_add_oauth_device_denied_at.js.map +1 -0
  64. package/dist/migrations/postgres/Migration20260624223000_add_oauth_device_poll_state.d.ts +6 -0
  65. package/dist/migrations/postgres/Migration20260624223000_add_oauth_device_poll_state.d.ts.map +1 -0
  66. package/dist/migrations/postgres/Migration20260624223000_add_oauth_device_poll_state.js +14 -0
  67. package/dist/migrations/postgres/Migration20260624223000_add_oauth_device_poll_state.js.map +1 -0
  68. package/dist/migrations/postgres/index.d.ts.map +1 -1
  69. package/dist/migrations/postgres/index.js +4 -0
  70. package/dist/migrations/postgres/index.js.map +1 -1
  71. package/dist/migrations/sqlite/Migration20260624190500_add_oauth_device_denied_at.d.ts +6 -0
  72. package/dist/migrations/sqlite/Migration20260624190500_add_oauth_device_denied_at.d.ts.map +1 -0
  73. package/dist/migrations/sqlite/Migration20260624190500_add_oauth_device_denied_at.js +10 -0
  74. package/dist/migrations/sqlite/Migration20260624190500_add_oauth_device_denied_at.js.map +1 -0
  75. package/dist/migrations/sqlite/Migration20260624223000_add_oauth_device_poll_state.d.ts +6 -0
  76. package/dist/migrations/sqlite/Migration20260624223000_add_oauth_device_poll_state.d.ts.map +1 -0
  77. package/dist/migrations/sqlite/Migration20260624223000_add_oauth_device_poll_state.js +12 -0
  78. package/dist/migrations/sqlite/Migration20260624223000_add_oauth_device_poll_state.js.map +1 -0
  79. package/dist/migrations/sqlite/index.d.ts.map +1 -1
  80. package/dist/migrations/sqlite/index.js +4 -0
  81. package/dist/migrations/sqlite/index.js.map +1 -1
  82. package/dist/repositories/oauth-device-code.repository.d.ts +4 -0
  83. package/dist/repositories/oauth-device-code.repository.d.ts.map +1 -1
  84. package/dist/repositories/oauth-device-code.repository.js +19 -0
  85. package/dist/repositories/oauth-device-code.repository.js.map +1 -1
  86. package/dist/repositories/user.repository.d.ts +4 -2
  87. package/dist/repositories/user.repository.d.ts.map +1 -1
  88. package/dist/repositories/user.repository.js +11 -7
  89. package/dist/repositories/user.repository.js.map +1 -1
  90. package/dist/routes/.well-known/openid-configuration/get.d.ts.map +1 -1
  91. package/dist/routes/.well-known/openid-configuration/get.js +4 -0
  92. package/dist/routes/.well-known/openid-configuration/get.js.map +1 -1
  93. package/dist/routes/api/admin/index.d.ts +185 -0
  94. package/dist/routes/api/admin/index.d.ts.map +1 -0
  95. package/dist/routes/api/admin/index.js +13 -0
  96. package/dist/routes/api/admin/index.js.map +1 -0
  97. package/dist/routes/api/admin/me/get.d.ts +26 -0
  98. package/dist/routes/api/admin/me/get.d.ts.map +1 -0
  99. package/dist/routes/api/admin/me/get.js +29 -0
  100. package/dist/routes/api/admin/me/get.js.map +1 -0
  101. package/dist/routes/api/admin/users/index.d.ts +162 -0
  102. package/dist/routes/api/admin/users/index.d.ts.map +1 -0
  103. package/dist/routes/api/admin/users/index.js +166 -0
  104. package/dist/routes/api/admin/users/index.js.map +1 -0
  105. package/dist/routes/api/auth/accounts/get.d.ts +28 -0
  106. package/dist/routes/api/auth/accounts/get.d.ts.map +1 -0
  107. package/dist/routes/api/auth/accounts/get.js +47 -0
  108. package/dist/routes/api/auth/accounts/get.js.map +1 -0
  109. package/dist/routes/api/auth/accounts/remove.post.d.ts +30 -0
  110. package/dist/routes/api/auth/accounts/remove.post.d.ts.map +1 -0
  111. package/dist/routes/api/auth/accounts/remove.post.js +30 -0
  112. package/dist/routes/api/auth/accounts/remove.post.js.map +1 -0
  113. package/dist/routes/api/auth/accounts/select.post.d.ts +31 -0
  114. package/dist/routes/api/auth/accounts/select.post.d.ts.map +1 -0
  115. package/dist/routes/api/auth/accounts/select.post.js +26 -0
  116. package/dist/routes/api/auth/accounts/select.post.js.map +1 -0
  117. package/dist/routes/api/auth/email/verify/post.d.ts +1 -0
  118. package/dist/routes/api/auth/email/verify/post.d.ts.map +1 -1
  119. package/dist/routes/api/auth/index.d.ts +86 -0
  120. package/dist/routes/api/auth/index.d.ts.map +1 -1
  121. package/dist/routes/api/auth/index.js +6 -0
  122. package/dist/routes/api/auth/index.js.map +1 -1
  123. package/dist/routes/api/auth/login/post.d.ts +1 -0
  124. package/dist/routes/api/auth/login/post.d.ts.map +1 -1
  125. package/dist/routes/api/auth/login/post.js +15 -16
  126. package/dist/routes/api/auth/login/post.js.map +1 -1
  127. package/dist/routes/api/auth/passkey/verify/post.d.ts +1 -0
  128. package/dist/routes/api/auth/passkey/verify/post.d.ts.map +1 -1
  129. package/dist/routes/api/auth/register/post.d.ts +1 -0
  130. package/dist/routes/api/auth/register/post.d.ts.map +1 -1
  131. package/dist/routes/api/auth/totp/recovery/verify/post.d.ts +1 -0
  132. package/dist/routes/api/auth/totp/recovery/verify/post.d.ts.map +1 -1
  133. package/dist/routes/api/auth/totp/verify/post.d.ts +1 -0
  134. package/dist/routes/api/auth/totp/verify/post.d.ts.map +1 -1
  135. package/dist/routes/api/config/get.d.ts +3 -0
  136. package/dist/routes/api/config/get.d.ts.map +1 -1
  137. package/dist/routes/api/config/get.js +7 -1
  138. package/dist/routes/api/config/get.js.map +1 -1
  139. package/dist/routes/api/config/index.d.ts +3 -0
  140. package/dist/routes/api/config/index.d.ts.map +1 -1
  141. package/dist/routes/api/consent/index.d.ts +10 -0
  142. package/dist/routes/api/consent/index.d.ts.map +1 -1
  143. package/dist/routes/api/consent/post.d.ts +10 -0
  144. package/dist/routes/api/consent/post.d.ts.map +1 -1
  145. package/dist/routes/api/consent/post.js +52 -1
  146. package/dist/routes/api/consent/post.js.map +1 -1
  147. package/dist/routes/api/index.d.ts +310 -26
  148. package/dist/routes/api/index.d.ts.map +1 -1
  149. package/dist/routes/api/index.js +2 -0
  150. package/dist/routes/api/index.js.map +1 -1
  151. package/dist/routes/api/user/index.d.ts +3 -0
  152. package/dist/routes/api/user/index.d.ts.map +1 -1
  153. package/dist/routes/api/user/passkeys/register/verify/post.d.ts +1 -0
  154. package/dist/routes/api/user/passkeys/register/verify/post.d.ts.map +1 -1
  155. package/dist/routes/api/user/session/get.d.ts +1 -0
  156. package/dist/routes/api/user/session/get.d.ts.map +1 -1
  157. package/dist/routes/api/user/totp/confirm/post.d.ts +1 -0
  158. package/dist/routes/api/user/totp/confirm/post.d.ts.map +1 -1
  159. package/dist/routes/index.d.ts +303 -16
  160. package/dist/routes/index.d.ts.map +1 -1
  161. package/dist/routes/oauth/.well-known/openid-configuration/get.d.ts.map +1 -1
  162. package/dist/routes/oauth/.well-known/openid-configuration/get.js +1 -0
  163. package/dist/routes/oauth/.well-known/openid-configuration/get.js.map +1 -1
  164. package/dist/routes/oauth/authorize/get.d.ts +2 -0
  165. package/dist/routes/oauth/authorize/get.d.ts.map +1 -1
  166. package/dist/routes/oauth/authorize/get.js +12 -1
  167. package/dist/routes/oauth/authorize/get.js.map +1 -1
  168. package/dist/routes/oauth/client-auth.js +1 -1
  169. package/dist/routes/oauth/client-auth.js.map +1 -1
  170. package/dist/routes/oauth/cors.d.ts.map +1 -1
  171. package/dist/routes/oauth/cors.js +8 -3
  172. package/dist/routes/oauth/cors.js.map +1 -1
  173. package/dist/routes/oauth/device/get-post.d.ts +2 -1
  174. package/dist/routes/oauth/device/get-post.d.ts.map +1 -1
  175. package/dist/routes/oauth/device/get-post.js +15 -9
  176. package/dist/routes/oauth/device/get-post.js.map +1 -1
  177. package/dist/routes/oauth/device-authorization/post.d.ts.map +1 -1
  178. package/dist/routes/oauth/device-authorization/post.js +2 -0
  179. package/dist/routes/oauth/device-authorization/post.js.map +1 -1
  180. package/dist/routes/oauth/end-session/get.d.ts.map +1 -1
  181. package/dist/routes/oauth/end-session/get.js +17 -8
  182. package/dist/routes/oauth/end-session/get.js.map +1 -1
  183. package/dist/routes/oauth/index.d.ts +5 -2
  184. package/dist/routes/oauth/index.d.ts.map +1 -1
  185. package/dist/routes/oauth/revoke/post.d.ts.map +1 -1
  186. package/dist/routes/oauth/revoke/post.js +2 -0
  187. package/dist/routes/oauth/revoke/post.js.map +1 -1
  188. package/dist/routes/oauth/token/post.d.ts +1 -1
  189. package/dist/routes/oauth/token/post.d.ts.map +1 -1
  190. package/dist/routes/oauth/token/post.js +10 -6
  191. package/dist/routes/oauth/token/post.js.map +1 -1
  192. package/dist/routes/oauth/userinfo/get.d.ts.map +1 -1
  193. package/dist/routes/oauth/userinfo/get.js +12 -6
  194. package/dist/routes/oauth/userinfo/get.js.map +1 -1
  195. package/dist/schemas/error.d.ts +56 -6
  196. package/dist/schemas/error.d.ts.map +1 -1
  197. package/dist/schemas/error.js +4 -2
  198. package/dist/schemas/error.js.map +1 -1
  199. package/dist/schemas/field.d.ts +1 -4
  200. package/dist/schemas/field.d.ts.map +1 -1
  201. package/dist/schemas/field.js +3 -1
  202. package/dist/schemas/field.js.map +1 -1
  203. package/dist/schemas/response.d.ts +96 -0
  204. package/dist/schemas/response.d.ts.map +1 -1
  205. package/dist/schemas/response.js +29 -0
  206. package/dist/schemas/response.js.map +1 -1
  207. package/dist/services/account-selection.service.d.ts +39 -0
  208. package/dist/services/account-selection.service.d.ts.map +1 -0
  209. package/dist/services/account-selection.service.js +77 -0
  210. package/dist/services/account-selection.service.js.map +1 -0
  211. package/dist/services/container.d.ts +22 -0
  212. package/dist/services/container.d.ts.map +1 -1
  213. package/dist/services/jwt.service.js +3 -3
  214. package/dist/services/jwt.service.js.map +1 -1
  215. package/dist/services/oauth-authorize.service.d.ts +24 -1
  216. package/dist/services/oauth-authorize.service.d.ts.map +1 -1
  217. package/dist/services/oauth-authorize.service.js +232 -8
  218. package/dist/services/oauth-authorize.service.js.map +1 -1
  219. package/dist/services/oauth-token.service.d.ts +2 -3
  220. package/dist/services/oauth-token.service.d.ts.map +1 -1
  221. package/dist/services/oauth-token.service.js +18 -10
  222. package/dist/services/oauth-token.service.js.map +1 -1
  223. package/dist/services/user.service.d.ts +28 -1
  224. package/dist/services/user.service.d.ts.map +1 -1
  225. package/dist/services/user.service.js +104 -0
  226. package/dist/services/user.service.js.map +1 -1
  227. package/package.json +1 -1
  228. package/public/assets/2fa-BoyBKrjD.js +2 -0
  229. package/public/assets/2fa-BoyBKrjD.js.map +1 -0
  230. package/public/assets/2fa-DfWvDjDW.js +2 -0
  231. package/public/assets/2fa-DfWvDjDW.js.map +1 -0
  232. package/public/assets/2fa-IkQlgUP0.js +2 -0
  233. package/public/assets/2fa-IkQlgUP0.js.map +1 -0
  234. package/public/assets/2fa-SSKfXB7c.js +2 -0
  235. package/public/assets/2fa-SSKfXB7c.js.map +1 -0
  236. package/public/assets/CheckCircle.es-MnJIACCe.js +2 -0
  237. package/public/assets/CheckCircle.es-MnJIACCe.js.map +1 -0
  238. package/public/assets/EnvelopeSimple.es-BZ7u3LYh.js +2 -0
  239. package/public/assets/EnvelopeSimple.es-BZ7u3LYh.js.map +1 -0
  240. package/public/assets/Fingerprint.es-CW755VWD.js +2 -0
  241. package/public/assets/Fingerprint.es-CW755VWD.js.map +1 -0
  242. package/public/assets/IconBase.es-d5KP98Ac.js +2 -0
  243. package/public/assets/IconBase.es-d5KP98Ac.js.map +1 -0
  244. package/public/assets/Key.es-l5aSxw0I.js +2 -0
  245. package/public/assets/Key.es-l5aSxw0I.js.map +1 -0
  246. package/public/assets/Link.es-B-IJS4Q5.js +2 -0
  247. package/public/assets/Link.es-B-IJS4Q5.js.map +1 -0
  248. package/public/assets/Lock.es-Cb_uwQly.js +2 -0
  249. package/public/assets/Lock.es-Cb_uwQly.js.map +1 -0
  250. package/public/assets/ShieldCheck.es-CscPsYbC.js +2 -0
  251. package/public/assets/ShieldCheck.es-CscPsYbC.js.map +1 -0
  252. package/public/assets/Trash.es-BepW9BwV.js +2 -0
  253. package/public/assets/Trash.es-BepW9BwV.js.map +1 -0
  254. package/public/assets/Warning.es-BPpZIJYZ.js +2 -0
  255. package/public/assets/Warning.es-BPpZIJYZ.js.map +1 -0
  256. package/public/assets/X.es-IwdB4hWT.js +2 -0
  257. package/public/assets/X.es-IwdB4hWT.js.map +1 -0
  258. package/public/assets/admin-D2CMlWzS.js +2 -0
  259. package/public/assets/admin-D2CMlWzS.js.map +1 -0
  260. package/public/assets/admin-users-CTAQJl7w.js +2 -0
  261. package/public/assets/admin-users-CTAQJl7w.js.map +1 -0
  262. package/public/assets/alert-CSXqgDVi.js +2 -0
  263. package/public/assets/alert-CSXqgDVi.js.map +1 -0
  264. package/public/assets/consent-C5Qo0iLd.js +2 -0
  265. package/public/assets/consent-C5Qo0iLd.js.map +1 -0
  266. package/public/assets/consent-DwuWkp63.js +2 -0
  267. package/public/assets/consent-DwuWkp63.js.map +1 -0
  268. package/public/assets/email-CIttZRBe.js +2 -0
  269. package/public/assets/email-CIttZRBe.js.map +1 -0
  270. package/public/assets/email-SSKfXB7c.js +2 -0
  271. package/public/assets/email-SSKfXB7c.js.map +1 -0
  272. package/public/assets/error-D60wkdWN.js +2 -0
  273. package/public/assets/error-D60wkdWN.js.map +1 -0
  274. package/public/assets/footer-link-Ib1Hd-fr.js +2 -0
  275. package/public/assets/footer-link-Ib1Hd-fr.js.map +1 -0
  276. package/public/assets/forgot-SSKfXB7c.js +2 -0
  277. package/public/assets/forgot-SSKfXB7c.js.map +1 -0
  278. package/public/assets/forgot-x-UDyHXT.js +2 -0
  279. package/public/assets/forgot-x-UDyHXT.js.map +1 -0
  280. package/public/assets/icon-input-8iU7PNzd.js +2 -0
  281. package/public/assets/icon-input-8iU7PNzd.js.map +1 -0
  282. package/public/assets/index-CsT6OVnP.js +11 -0
  283. package/public/assets/index-CsT6OVnP.js.map +1 -0
  284. package/public/assets/index-D6-1JDnX.css +2 -0
  285. package/public/assets/login-DhbnCudI.js +2 -0
  286. package/public/assets/login-DhbnCudI.js.map +1 -0
  287. package/public/assets/login-SSKfXB7c.js +2 -0
  288. package/public/assets/login-SSKfXB7c.js.map +1 -0
  289. package/public/assets/modal-BjnpKlXn.js +2 -0
  290. package/public/assets/modal-BjnpKlXn.js.map +1 -0
  291. package/public/assets/mutationOptions-Dfvzj6n2.js +2 -0
  292. package/public/assets/mutationOptions-Dfvzj6n2.js.map +1 -0
  293. package/public/assets/page-header-BYMFSGfT.js +2 -0
  294. package/public/assets/page-header-BYMFSGfT.js.map +1 -0
  295. package/public/assets/page-layout-C475gs09.js +2 -0
  296. package/public/assets/page-layout-C475gs09.js.map +1 -0
  297. package/public/assets/passkey-BdISbWr7.js +2 -0
  298. package/public/assets/passkey-BdISbWr7.js.map +1 -0
  299. package/public/assets/passkey-Bv7zPLAZ.js +2 -0
  300. package/public/assets/passkey-Bv7zPLAZ.js.map +1 -0
  301. package/public/assets/passkey-e6uvApHa.js +2 -0
  302. package/public/assets/passkey-e6uvApHa.js.map +1 -0
  303. package/public/assets/password-CkeV4qxb.js +2 -0
  304. package/public/assets/password-CkeV4qxb.js.map +1 -0
  305. package/public/assets/password-SSKfXB7c.js +2 -0
  306. package/public/assets/password-SSKfXB7c.js.map +1 -0
  307. package/public/assets/password-reset-XZJTgJi3.js +2 -0
  308. package/public/assets/password-reset-XZJTgJi3.js.map +1 -0
  309. package/public/assets/pin-input-BM1UizHr.js +2 -0
  310. package/public/assets/pin-input-BM1UizHr.js.map +1 -0
  311. package/public/assets/profile-D2cuVYgE.js +2 -0
  312. package/public/assets/profile-D2cuVYgE.js.map +1 -0
  313. package/public/assets/profile-TKdT20x5.js +2 -0
  314. package/public/assets/profile-TKdT20x5.js.map +1 -0
  315. package/public/assets/promise-OpBtq8tG.js +2 -0
  316. package/public/assets/promise-OpBtq8tG.js.map +1 -0
  317. package/public/assets/recovery-DM8h2gbb.js +2 -0
  318. package/public/assets/recovery-DM8h2gbb.js.map +1 -0
  319. package/public/assets/register-SSKfXB7c.js +2 -0
  320. package/public/assets/register-SSKfXB7c.js.map +1 -0
  321. package/public/assets/register-vWW_43cD.js +2 -0
  322. package/public/assets/register-vWW_43cD.js.map +1 -0
  323. package/public/assets/reset-CgACYrdp.js +2 -0
  324. package/public/assets/reset-CgACYrdp.js.map +1 -0
  325. package/public/assets/reset-SSKfXB7c.js +2 -0
  326. package/public/assets/reset-SSKfXB7c.js.map +1 -0
  327. package/public/assets/routes-CAIeH5mq.js +2 -0
  328. package/public/assets/routes-CAIeH5mq.js.map +1 -0
  329. package/public/assets/select-BCP5fwfB.js +2 -0
  330. package/public/assets/select-BCP5fwfB.js.map +1 -0
  331. package/public/assets/select-SSKfXB7c.js +2 -0
  332. package/public/assets/select-SSKfXB7c.js.map +1 -0
  333. package/public/assets/standard-schema-o4V-s4uY.js +2 -0
  334. package/public/assets/standard-schema-o4V-s4uY.js.map +1 -0
  335. package/public/assets/submit-button-Xx6DwLyh.js +2 -0
  336. package/public/assets/submit-button-Xx6DwLyh.js.map +1 -0
  337. package/public/assets/terms-DPWrbYY2.js +2 -0
  338. package/public/assets/terms-DPWrbYY2.js.map +1 -0
  339. package/public/assets/terms-TKdT20x5.js +2 -0
  340. package/public/assets/terms-TKdT20x5.js.map +1 -0
  341. package/public/assets/terms-checkbox-list-CdrbHxiF.js +2 -0
  342. package/public/assets/terms-checkbox-list-CdrbHxiF.js.map +1 -0
  343. package/public/assets/totp-CKZ6N1NS.js +2 -0
  344. package/public/assets/totp-CKZ6N1NS.js.map +1 -0
  345. package/public/assets/totp-D-PVOsGQ.js +2 -0
  346. package/public/assets/totp-D-PVOsGQ.js.map +1 -0
  347. package/public/assets/totp-NlqqRp4a.js +2 -0
  348. package/public/assets/totp-NlqqRp4a.js.map +1 -0
  349. package/public/assets/use-theme-cVUDAjtt.js +2 -0
  350. package/public/assets/use-theme-cVUDAjtt.js.map +1 -0
  351. package/public/assets/use-totp-setup-BH75uEbE.js +3 -0
  352. package/public/assets/use-totp-setup-BH75uEbE.js.map +1 -0
  353. package/public/assets/useMutation-DVMopbtG.js +2 -0
  354. package/public/assets/useMutation-DVMopbtG.js.map +1 -0
  355. package/public/assets/users-B7ofdp72.js +2 -0
  356. package/public/assets/users-B7ofdp72.js.map +1 -0
  357. package/public/assets/zod-BItJDQBQ.js +66 -0
  358. package/public/assets/zod-BItJDQBQ.js.map +1 -0
  359. package/public/index.html +6 -2
  360. package/public/assets/index-5_9rzim1.css +0 -2
  361. package/public/assets/index-BTGeW26-.js +0 -75
  362. package/public/assets/index-BTGeW26-.js.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mutationOptions-Dfvzj6n2.js","names":[],"sources":["../../../../node_modules/.pnpm/@tanstack+react-query@5.101.0_react@19.2.7/node_modules/@tanstack/react-query/build/modern/mutationOptions.js"],"sourcesContent":["// src/mutationOptions.ts\nfunction mutationOptions(options) {\n return options;\n}\nexport {\n mutationOptions\n};\n//# sourceMappingURL=mutationOptions.js.map"],"x_google_ignoreList":[0],"mappings":"AACA,SAAS,EAAgB,EAAS,CAChC,OAAO,CACT"}
@@ -0,0 +1,2 @@
1
+ import{o as e}from"./IconBase.es-d5KP98Ac.js";var t=e();function n({title:e,subtitle:n,iconUrl:r,className:i=``}){return(0,t.jsxs)(`div`,{className:i,children:[r&&(0,t.jsx)(`div`,{className:`mb-4 flex justify-center`,children:(0,t.jsx)(`img`,{alt:``,className:`h-12 w-12 object-contain`,src:r})}),(0,t.jsx)(`h1`,{className:`text-center font-bold text-2xl ${n?`mb-0`:`mb-6`}`,children:e}),n&&(0,t.jsx)(`p`,{className:`mb-6 text-center text-base-content/60 text-lg`,children:n})]})}export{n as t};
2
+ //# sourceMappingURL=page-header-BYMFSGfT.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"page-header-BYMFSGfT.js","names":[],"sources":["../../../frontend/src/components/auth/page-header.tsx"],"sourcesContent":["type PageHeaderProps = {\n title: string;\n subtitle?: string;\n iconUrl?: string;\n className?: string;\n};\n\nexport function PageHeader({\n title,\n subtitle,\n iconUrl,\n className = '',\n}: PageHeaderProps) {\n return (\n <div className={className}>\n {iconUrl && (\n <div className=\"mb-4 flex justify-center\">\n <img alt=\"\" className=\"h-12 w-12 object-contain\" src={iconUrl} />\n </div>\n )}\n <h1\n className={`text-center font-bold text-2xl ${\n subtitle ? 'mb-0' : 'mb-6'\n }`}\n >\n {title}\n </h1>\n {subtitle && (\n <p className=\"mb-6 text-center text-base-content/60 text-lg\">\n {subtitle}\n </p>\n )}\n </div>\n );\n}\n"],"mappings":"wDAOA,SAAgB,EAAW,CACzB,QACA,WACA,UACA,YAAY,IACM,CAClB,OACE,EAAA,EAAA,MAAC,MAAD,CAAgB,qBAAhB,CACG,IACC,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,qCACb,EAAA,EAAA,KAAC,MAAD,CAAK,IAAI,GAAG,UAAU,2BAA2B,IAAK,CAAU,CAAA,CAC7D,CAAA,GAEP,EAAA,EAAA,KAAC,KAAD,CACE,UAAW,kCACT,EAAW,OAAS,kBAGrB,CACC,CAAA,EACH,IACC,EAAA,EAAA,KAAC,IAAD,CAAG,UAAU,yDACV,CACA,CAAA,CAEF,GAET"}
@@ -0,0 +1,2 @@
1
+ import{o as e,r as t,s as n,t as r,u as i}from"./IconBase.es-d5KP98Ac.js";import{Q as a,c as o,l as s,n as c,o as l,s as u,t as d}from"./use-theme-cVUDAjtt.js";var f=i(n(),1),p=new Map([[`bold`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M128,20A108,108,0,1,0,236,128,108.12,108.12,0,0,0,128,20Zm12,24.87a83.53,83.53,0,0,1,24,7.25V203.88a83.53,83.53,0,0,1-24,7.25ZM44,128a84.12,84.12,0,0,1,72-83.13V211.13A84.12,84.12,0,0,1,44,128Zm144,58.71V69.29a83.81,83.81,0,0,1,0,117.42Z`}))],[`duotone`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M224,128a96,96,0,0,1-96,96V32A96,96,0,0,1,224,128Z`,opacity:`0.2`}),f.createElement(`path`,{d:`M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24ZM40,128a88.11,88.11,0,0,1,80-87.63V215.63A88.11,88.11,0,0,1,40,128Zm96,87.63V40.37a88,88,0,0,1,0,175.26Z`}))],[`fill`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24ZM40,128a88.1,88.1,0,0,1,88-88V216A88.1,88.1,0,0,1,40,128Z`}))],[`light`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M128,26A102,102,0,1,0,230,128,102.12,102.12,0,0,0,128,26Zm6,12.2a89.86,89.86,0,0,1,20,3.63V214.17a89.86,89.86,0,0,1-20,3.63Zm32,8.23a90.48,90.48,0,0,1,20,12.81V196.76a90.48,90.48,0,0,1-20,12.81ZM38,128a90.12,90.12,0,0,1,84-89.8V217.8A90.12,90.12,0,0,1,38,128Zm160,56.5V71.5a89.81,89.81,0,0,1,0,113Z`}))],[`regular`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24Zm8,16.37a86.4,86.4,0,0,1,16,3V212.67a86.4,86.4,0,0,1-16,3Zm32,9.26a87.81,87.81,0,0,1,16,10.54V195.83a87.81,87.81,0,0,1-16,10.54ZM40,128a88.11,88.11,0,0,1,80-87.63V215.63A88.11,88.11,0,0,1,40,128Zm160,50.54V77.46a87.82,87.82,0,0,1,0,101.08Z`}))],[`thin`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M128,28A100,100,0,1,0,228,128,100.11,100.11,0,0,0,128,28Zm4,8.09a91.58,91.58,0,0,1,24,4.27V215.64a91.58,91.58,0,0,1-24,4.27Zm32,7.25a92.21,92.21,0,0,1,24,15V197.69a92.21,92.21,0,0,1-24,15ZM36,128a92.11,92.11,0,0,1,88-91.91V219.91A92.11,92.11,0,0,1,36,128Zm160,61.9V66.1a91.83,91.83,0,0,1,0,123.8Z`}))]]),m=new Map([[`bold`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M236.37,139.4a12,12,0,0,0-12-3A84.07,84.07,0,0,1,119.6,31.59a12,12,0,0,0-15-15A108.86,108.86,0,0,0,49.69,55.07,108,108,0,0,0,136,228a107.09,107.09,0,0,0,64.93-21.69,108.86,108.86,0,0,0,38.44-54.94A12,12,0,0,0,236.37,139.4Zm-49.88,47.74A84,84,0,0,1,68.86,69.51,84.93,84.93,0,0,1,92.27,48.29Q92,52.13,92,56A108.12,108.12,0,0,0,200,164q3.87,0,7.71-.27A84.79,84.79,0,0,1,186.49,187.14Z`}))],[`duotone`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M227.89,147.89A96,96,0,1,1,108.11,28.11,96.09,96.09,0,0,0,227.89,147.89Z`,opacity:`0.2`}),f.createElement(`path`,{d:`M233.54,142.23a8,8,0,0,0-8-2,88.08,88.08,0,0,1-109.8-109.8,8,8,0,0,0-10-10,104.84,104.84,0,0,0-52.91,37A104,104,0,0,0,136,224a103.09,103.09,0,0,0,62.52-20.88,104.84,104.84,0,0,0,37-52.91A8,8,0,0,0,233.54,142.23ZM188.9,190.34A88,88,0,0,1,65.66,67.11a89,89,0,0,1,31.4-26A106,106,0,0,0,96,56,104.11,104.11,0,0,0,200,160a106,106,0,0,0,14.92-1.06A89,89,0,0,1,188.9,190.34Z`}))],[`fill`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M235.54,150.21a104.84,104.84,0,0,1-37,52.91A104,104,0,0,1,32,120,103.09,103.09,0,0,1,52.88,57.48a104.84,104.84,0,0,1,52.91-37,8,8,0,0,1,10,10,88.08,88.08,0,0,0,109.8,109.8,8,8,0,0,1,10,10Z`}))],[`light`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M232.13,143.64a6,6,0,0,0-6-1.49A90.07,90.07,0,0,1,113.86,29.85a6,6,0,0,0-7.49-7.48A102.88,102.88,0,0,0,54.48,58.68,102,102,0,0,0,197.32,201.52a102.88,102.88,0,0,0,36.31-51.89A6,6,0,0,0,232.13,143.64Zm-42,48.29a90,90,0,0,1-126-126A90.9,90.9,0,0,1,99.65,37.66,102.06,102.06,0,0,0,218.34,156.35,90.9,90.9,0,0,1,190.1,191.93Z`}))],[`regular`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M233.54,142.23a8,8,0,0,0-8-2,88.08,88.08,0,0,1-109.8-109.8,8,8,0,0,0-10-10,104.84,104.84,0,0,0-52.91,37A104,104,0,0,0,136,224a103.09,103.09,0,0,0,62.52-20.88,104.84,104.84,0,0,0,37-52.91A8,8,0,0,0,233.54,142.23ZM188.9,190.34A88,88,0,0,1,65.66,67.11a89,89,0,0,1,31.4-26A106,106,0,0,0,96,56,104.11,104.11,0,0,0,200,160a106,106,0,0,0,14.92-1.06A89,89,0,0,1,188.9,190.34Z`}))],[`thin`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M230.72,145.06a4,4,0,0,0-4-1A92.08,92.08,0,0,1,111.94,29.27a4,4,0,0,0-5-5A100.78,100.78,0,0,0,56.08,59.88a100,100,0,0,0,140,140,100.78,100.78,0,0,0,35.59-50.87A4,4,0,0,0,230.72,145.06ZM191.3,193.53A92,92,0,0,1,62.47,64.7a93,93,0,0,1,39.88-30.35,100.09,100.09,0,0,0,119.3,119.3A93,93,0,0,1,191.3,193.53Z`}))]]),h=new Map([[`bold`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M116,36V20a12,12,0,0,1,24,0V36a12,12,0,0,1-24,0Zm80,92a68,68,0,1,1-68-68A68.07,68.07,0,0,1,196,128Zm-24,0a44,44,0,1,0-44,44A44.05,44.05,0,0,0,172,128ZM51.51,68.49a12,12,0,1,0,17-17l-12-12a12,12,0,0,0-17,17Zm0,119-12,12a12,12,0,0,0,17,17l12-12a12,12,0,1,0-17-17ZM196,72a12,12,0,0,0,8.49-3.51l12-12a12,12,0,0,0-17-17l-12,12A12,12,0,0,0,196,72Zm8.49,115.51a12,12,0,0,0-17,17l12,12a12,12,0,0,0,17-17ZM48,128a12,12,0,0,0-12-12H20a12,12,0,0,0,0,24H36A12,12,0,0,0,48,128Zm80,80a12,12,0,0,0-12,12v16a12,12,0,0,0,24,0V220A12,12,0,0,0,128,208Zm108-92H220a12,12,0,0,0,0,24h16a12,12,0,0,0,0-24Z`}))],[`duotone`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M184,128a56,56,0,1,1-56-56A56,56,0,0,1,184,128Z`,opacity:`0.2`}),f.createElement(`path`,{d:`M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm72,88a64,64,0,1,1-64-64A64.07,64.07,0,0,1,192,128Zm-16,0a48,48,0,1,0-48,48A48.05,48.05,0,0,0,176,128ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z`}))],[`fill`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm8,24a64,64,0,1,0,64,64A64.07,64.07,0,0,0,128,64ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z`}))],[`light`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M122,40V16a6,6,0,0,1,12,0V40a6,6,0,0,1-12,0Zm68,88a62,62,0,1,1-62-62A62.07,62.07,0,0,1,190,128Zm-12,0a50,50,0,1,0-50,50A50.06,50.06,0,0,0,178,128ZM59.76,68.24a6,6,0,1,0,8.48-8.48l-16-16a6,6,0,0,0-8.48,8.48Zm0,119.52-16,16a6,6,0,1,0,8.48,8.48l16-16a6,6,0,1,0-8.48-8.48ZM192,70a6,6,0,0,0,4.24-1.76l16-16a6,6,0,0,0-8.48-8.48l-16,16A6,6,0,0,0,192,70Zm4.24,117.76a6,6,0,0,0-8.48,8.48l16,16a6,6,0,0,0,8.48-8.48ZM46,128a6,6,0,0,0-6-6H16a6,6,0,0,0,0,12H40A6,6,0,0,0,46,128Zm82,82a6,6,0,0,0-6,6v24a6,6,0,0,0,12,0V216A6,6,0,0,0,128,210Zm112-88H216a6,6,0,0,0,0,12h24a6,6,0,0,0,0-12Z`}))],[`regular`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm72,88a64,64,0,1,1-64-64A64.07,64.07,0,0,1,192,128Zm-16,0a48,48,0,1,0-48,48A48.05,48.05,0,0,0,176,128ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z`}))],[`thin`,f.createElement(f.Fragment,null,f.createElement(`path`,{d:`M124,40V16a4,4,0,0,1,8,0V40a4,4,0,0,1-8,0Zm64,88a60,60,0,1,1-60-60A60.07,60.07,0,0,1,188,128Zm-8,0a52,52,0,1,0-52,52A52.06,52.06,0,0,0,180,128ZM61.17,66.83a4,4,0,0,0,5.66-5.66l-16-16a4,4,0,0,0-5.66,5.66Zm0,122.34-16,16a4,4,0,0,0,5.66,5.66l16-16a4,4,0,0,0-5.66-5.66ZM192,68a4,4,0,0,0,2.83-1.17l16-16a4,4,0,1,0-5.66-5.66l-16,16A4,4,0,0,0,192,68Zm2.83,121.17a4,4,0,0,0-5.66,5.66l16,16a4,4,0,0,0,5.66-5.66ZM40,124H16a4,4,0,0,0,0,8H40a4,4,0,0,0,0-8Zm88,88a4,4,0,0,0-4,4v24a4,4,0,0,0,8,0V216A4,4,0,0,0,128,212Zm112-88H216a4,4,0,0,0,0,8h24a4,4,0,0,0,0-8Z`}))]]),g=f.forwardRef((e,t)=>f.createElement(r,{ref:t,...e,weights:p}));g.displayName=`CircleHalfIcon`;var _=f.forwardRef((e,t)=>f.createElement(r,{ref:t,...e,weights:m}));_.displayName=`MoonIcon`;var v=f.forwardRef((e,t)=>f.createElement(r,{ref:t,...e,weights:h}));v.displayName=`SunIcon`;var y=e();function b({themeMode:e,isAutoMode:n,detectedTheme:r,darkTheme:i,onCycle:a,className:o=`fixed start-3 top-3 z-50 sm:absolute sm:start-4 sm:top-4`}){let{t:s}=t();return(0,y.jsx)(`div`,{className:`tooltip tooltip-right ${o}`,"data-tip":(()=>{if(e===`system`||n){let e=s(r===i?`common.theme.dark`:`common.theme.light`);return`${s(`common.theme.auto`)} (${e})`}return s(e===`light`?`common.theme.light`:`common.theme.dark`)})(),children:(0,y.jsx)(`button`,{"aria-label":s(`common.theme.select`),className:`btn btn-circle btn-sm`,"data-testid":`theme-toggle`,onClick:a,type:`button`,children:e===`system`||n?(0,y.jsx)(g,{className:`size-4`,weight:`fill`}):e===`light`?(0,y.jsx)(v,{className:`size-4`,weight:`fill`}):(0,y.jsx)(_,{className:`size-4`,weight:`fill`})})})}function x(e){let t=t=>{t.key===`tinyauth-language`&&e()};return window.addEventListener(`storage`,t),window.addEventListener(`language-storage-change`,e),()=>{window.removeEventListener(`storage`,t),window.removeEventListener(`language-storage-change`,e)}}function S(){return localStorage.getItem(u)}function C(){window.dispatchEvent(new Event(`language-storage-change`))}function w(){let{i18n:e}=t(),{data:n}=a(c),r=n.i18n.supported_languages,i=s(r),l=(0,f.useSyncExternalStore)(x,S,S)===null,d=o(i,n.i18n.fallback_language),p=(0,f.useCallback)(t=>{i.includes(t)&&(e.changeLanguage(t),localStorage.setItem(u,t),C())},[i,e]),m=(0,f.useCallback)(()=>{localStorage.removeItem(u),C(),e.changeLanguage(d)},[d,e]);return{language:e.language,languages:i,setLanguage:p,setAutoLanguage:m,isAutoMode:l,detectedLanguage:d,showLanguageSelector:i.length>1}}function T({className:e=``}){let{t:n}=t(),{language:r,languages:i,setLanguage:a,setAutoLanguage:o,isAutoMode:s,detectedLanguage:c,showLanguageSelector:u}=w();if(!u)return null;let d=l[c]||c,f=`${n(`common.language.auto`)} (${d})`,p=e=>{let t=e.target.value;t===`auto`?o():a(t)},m=s?`auto`:r;return(0,y.jsx)(`div`,{className:e,children:(0,y.jsxs)(`select`,{"aria-label":n(`common.language.select`),className:`select select-ghost select-sm text-base-content/60`,"data-testid":`language-selector`,onChange:p,value:m,children:[(0,y.jsx)(`option`,{value:`auto`,children:f}),i.map(e=>(0,y.jsx)(`option`,{value:e,children:l[e]||e},e))]})})}var E={sm:`max-w-sm`,md:`max-w-md`,lg:`max-w-lg`,xl:`max-w-xl`,"2xl":`max-w-2xl`,100:`max-w-100`};function D({children:e,maxWidth:t=`100`,cardPadding:n=!1,responsivePadding:r=!1}){let{themeMode:i,darkTheme:o,canToggleTheme:s,cycleThemeMode:l,isAutoMode:u,detectedTheme:f}=d(),{data:p}=a(c),m=p.branding.background_url,h=r?`p-4 md:p-8`:`p-4`,g=`${E[t]}${n?` p-10`:``}`;return(0,y.jsxs)(`div`,{className:`relative flex min-h-screen flex-col bg-base-200 bg-cover ${h}`,style:m?{backgroundImage:`url(${m})`}:void 0,children:[s&&(0,y.jsx)(b,{className:`fixed start-3 top-3 z-50 sm:absolute sm:start-4 sm:top-4`,darkTheme:o,detectedTheme:f,isAutoMode:u,onCycle:l,themeMode:i}),(0,y.jsx)(`div`,{className:`flex flex-1 items-center justify-center`,children:(0,y.jsx)(`div`,{className:`card w-full border border-base-200 bg-base-100 shadow-lg ${g}`,children:e})}),(0,y.jsx)(T,{className:`mx-auto mt-4 pb-2`})]})}export{D as t};
2
+ //# sourceMappingURL=page-layout-C475gs09.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"page-layout-C475gs09.js","names":["a","e","c","t","e","n","n","a"],"sources":["../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/defs/CircleHalf.es.js","../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/defs/Moon.es.js","../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/defs/Sun.es.js","../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/csr/CircleHalf.es.js","../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/csr/Moon.es.js","../../../../node_modules/.pnpm/@phosphor-icons+react@2.1.10_react-dom@19.2.7_react@19.2.7__react@19.2.7/node_modules/@phosphor-icons/react/dist/csr/Sun.es.js","../../../frontend/src/components/ui/theme-toggle.tsx","../../../frontend/src/hooks/use-language.ts","../../../frontend/src/features/layout/language-selector.tsx","../../../frontend/src/features/layout/page-layout.tsx"],"sourcesContent":["import * as e from \"react\";\nconst a = /* @__PURE__ */ new Map([\n [\n \"bold\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,20A108,108,0,1,0,236,128,108.12,108.12,0,0,0,128,20Zm12,24.87a83.53,83.53,0,0,1,24,7.25V203.88a83.53,83.53,0,0,1-24,7.25ZM44,128a84.12,84.12,0,0,1,72-83.13V211.13A84.12,84.12,0,0,1,44,128Zm144,58.71V69.29a83.81,83.81,0,0,1,0,117.42Z\" }))\n ],\n [\n \"duotone\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\n \"path\",\n {\n d: \"M224,128a96,96,0,0,1-96,96V32A96,96,0,0,1,224,128Z\",\n opacity: \"0.2\"\n }\n ), /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24ZM40,128a88.11,88.11,0,0,1,80-87.63V215.63A88.11,88.11,0,0,1,40,128Zm96,87.63V40.37a88,88,0,0,1,0,175.26Z\" }))\n ],\n [\n \"fill\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24ZM40,128a88.1,88.1,0,0,1,88-88V216A88.1,88.1,0,0,1,40,128Z\" }))\n ],\n [\n \"light\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,26A102,102,0,1,0,230,128,102.12,102.12,0,0,0,128,26Zm6,12.2a89.86,89.86,0,0,1,20,3.63V214.17a89.86,89.86,0,0,1-20,3.63Zm32,8.23a90.48,90.48,0,0,1,20,12.81V196.76a90.48,90.48,0,0,1-20,12.81ZM38,128a90.12,90.12,0,0,1,84-89.8V217.8A90.12,90.12,0,0,1,38,128Zm160,56.5V71.5a89.81,89.81,0,0,1,0,113Z\" }))\n ],\n [\n \"regular\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,24A104,104,0,1,0,232,128,104.11,104.11,0,0,0,128,24Zm8,16.37a86.4,86.4,0,0,1,16,3V212.67a86.4,86.4,0,0,1-16,3Zm32,9.26a87.81,87.81,0,0,1,16,10.54V195.83a87.81,87.81,0,0,1-16,10.54ZM40,128a88.11,88.11,0,0,1,80-87.63V215.63A88.11,88.11,0,0,1,40,128Zm160,50.54V77.46a87.82,87.82,0,0,1,0,101.08Z\" }))\n ],\n [\n \"thin\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M128,28A100,100,0,1,0,228,128,100.11,100.11,0,0,0,128,28Zm4,8.09a91.58,91.58,0,0,1,24,4.27V215.64a91.58,91.58,0,0,1-24,4.27Zm32,7.25a92.21,92.21,0,0,1,24,15V197.69a92.21,92.21,0,0,1-24,15ZM36,128a92.11,92.11,0,0,1,88-91.91V219.91A92.11,92.11,0,0,1,36,128Zm160,61.9V66.1a91.83,91.83,0,0,1,0,123.8Z\" }))\n ]\n]);\nexport {\n a as default\n};\n","import * as e from \"react\";\nconst a = /* @__PURE__ */ new Map([\n [\n \"bold\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M236.37,139.4a12,12,0,0,0-12-3A84.07,84.07,0,0,1,119.6,31.59a12,12,0,0,0-15-15A108.86,108.86,0,0,0,49.69,55.07,108,108,0,0,0,136,228a107.09,107.09,0,0,0,64.93-21.69,108.86,108.86,0,0,0,38.44-54.94A12,12,0,0,0,236.37,139.4Zm-49.88,47.74A84,84,0,0,1,68.86,69.51,84.93,84.93,0,0,1,92.27,48.29Q92,52.13,92,56A108.12,108.12,0,0,0,200,164q3.87,0,7.71-.27A84.79,84.79,0,0,1,186.49,187.14Z\" }))\n ],\n [\n \"duotone\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\n \"path\",\n {\n d: \"M227.89,147.89A96,96,0,1,1,108.11,28.11,96.09,96.09,0,0,0,227.89,147.89Z\",\n opacity: \"0.2\"\n }\n ), /* @__PURE__ */ e.createElement(\"path\", { d: \"M233.54,142.23a8,8,0,0,0-8-2,88.08,88.08,0,0,1-109.8-109.8,8,8,0,0,0-10-10,104.84,104.84,0,0,0-52.91,37A104,104,0,0,0,136,224a103.09,103.09,0,0,0,62.52-20.88,104.84,104.84,0,0,0,37-52.91A8,8,0,0,0,233.54,142.23ZM188.9,190.34A88,88,0,0,1,65.66,67.11a89,89,0,0,1,31.4-26A106,106,0,0,0,96,56,104.11,104.11,0,0,0,200,160a106,106,0,0,0,14.92-1.06A89,89,0,0,1,188.9,190.34Z\" }))\n ],\n [\n \"fill\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M235.54,150.21a104.84,104.84,0,0,1-37,52.91A104,104,0,0,1,32,120,103.09,103.09,0,0,1,52.88,57.48a104.84,104.84,0,0,1,52.91-37,8,8,0,0,1,10,10,88.08,88.08,0,0,0,109.8,109.8,8,8,0,0,1,10,10Z\" }))\n ],\n [\n \"light\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M232.13,143.64a6,6,0,0,0-6-1.49A90.07,90.07,0,0,1,113.86,29.85a6,6,0,0,0-7.49-7.48A102.88,102.88,0,0,0,54.48,58.68,102,102,0,0,0,197.32,201.52a102.88,102.88,0,0,0,36.31-51.89A6,6,0,0,0,232.13,143.64Zm-42,48.29a90,90,0,0,1-126-126A90.9,90.9,0,0,1,99.65,37.66,102.06,102.06,0,0,0,218.34,156.35,90.9,90.9,0,0,1,190.1,191.93Z\" }))\n ],\n [\n \"regular\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M233.54,142.23a8,8,0,0,0-8-2,88.08,88.08,0,0,1-109.8-109.8,8,8,0,0,0-10-10,104.84,104.84,0,0,0-52.91,37A104,104,0,0,0,136,224a103.09,103.09,0,0,0,62.52-20.88,104.84,104.84,0,0,0,37-52.91A8,8,0,0,0,233.54,142.23ZM188.9,190.34A88,88,0,0,1,65.66,67.11a89,89,0,0,1,31.4-26A106,106,0,0,0,96,56,104.11,104.11,0,0,0,200,160a106,106,0,0,0,14.92-1.06A89,89,0,0,1,188.9,190.34Z\" }))\n ],\n [\n \"thin\",\n /* @__PURE__ */ e.createElement(e.Fragment, null, /* @__PURE__ */ e.createElement(\"path\", { d: \"M230.72,145.06a4,4,0,0,0-4-1A92.08,92.08,0,0,1,111.94,29.27a4,4,0,0,0-5-5A100.78,100.78,0,0,0,56.08,59.88a100,100,0,0,0,140,140,100.78,100.78,0,0,0,35.59-50.87A4,4,0,0,0,230.72,145.06ZM191.3,193.53A92,92,0,0,1,62.47,64.7a93,93,0,0,1,39.88-30.35,100.09,100.09,0,0,0,119.3,119.3A93,93,0,0,1,191.3,193.53Z\" }))\n ]\n]);\nexport {\n a as default\n};\n","import * as a from \"react\";\nconst e = /* @__PURE__ */ new Map([\n [\n \"bold\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M116,36V20a12,12,0,0,1,24,0V36a12,12,0,0,1-24,0Zm80,92a68,68,0,1,1-68-68A68.07,68.07,0,0,1,196,128Zm-24,0a44,44,0,1,0-44,44A44.05,44.05,0,0,0,172,128ZM51.51,68.49a12,12,0,1,0,17-17l-12-12a12,12,0,0,0-17,17Zm0,119-12,12a12,12,0,0,0,17,17l12-12a12,12,0,1,0-17-17ZM196,72a12,12,0,0,0,8.49-3.51l12-12a12,12,0,0,0-17-17l-12,12A12,12,0,0,0,196,72Zm8.49,115.51a12,12,0,0,0-17,17l12,12a12,12,0,0,0,17-17ZM48,128a12,12,0,0,0-12-12H20a12,12,0,0,0,0,24H36A12,12,0,0,0,48,128Zm80,80a12,12,0,0,0-12,12v16a12,12,0,0,0,24,0V220A12,12,0,0,0,128,208Zm108-92H220a12,12,0,0,0,0,24h16a12,12,0,0,0,0-24Z\" }))\n ],\n [\n \"duotone\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M184,128a56,56,0,1,1-56-56A56,56,0,0,1,184,128Z\", opacity: \"0.2\" }), /* @__PURE__ */ a.createElement(\"path\", { d: \"M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm72,88a64,64,0,1,1-64-64A64.07,64.07,0,0,1,192,128Zm-16,0a48,48,0,1,0-48,48A48.05,48.05,0,0,0,176,128ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z\" }))\n ],\n [\n \"fill\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm8,24a64,64,0,1,0,64,64A64.07,64.07,0,0,0,128,64ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z\" }))\n ],\n [\n \"light\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M122,40V16a6,6,0,0,1,12,0V40a6,6,0,0,1-12,0Zm68,88a62,62,0,1,1-62-62A62.07,62.07,0,0,1,190,128Zm-12,0a50,50,0,1,0-50,50A50.06,50.06,0,0,0,178,128ZM59.76,68.24a6,6,0,1,0,8.48-8.48l-16-16a6,6,0,0,0-8.48,8.48Zm0,119.52-16,16a6,6,0,1,0,8.48,8.48l16-16a6,6,0,1,0-8.48-8.48ZM192,70a6,6,0,0,0,4.24-1.76l16-16a6,6,0,0,0-8.48-8.48l-16,16A6,6,0,0,0,192,70Zm4.24,117.76a6,6,0,0,0-8.48,8.48l16,16a6,6,0,0,0,8.48-8.48ZM46,128a6,6,0,0,0-6-6H16a6,6,0,0,0,0,12H40A6,6,0,0,0,46,128Zm82,82a6,6,0,0,0-6,6v24a6,6,0,0,0,12,0V216A6,6,0,0,0,128,210Zm112-88H216a6,6,0,0,0,0,12h24a6,6,0,0,0,0-12Z\" }))\n ],\n [\n \"regular\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M120,40V16a8,8,0,0,1,16,0V40a8,8,0,0,1-16,0Zm72,88a64,64,0,1,1-64-64A64.07,64.07,0,0,1,192,128Zm-16,0a48,48,0,1,0-48,48A48.05,48.05,0,0,0,176,128ZM58.34,69.66A8,8,0,0,0,69.66,58.34l-16-16A8,8,0,0,0,42.34,53.66Zm0,116.68-16,16a8,8,0,0,0,11.32,11.32l16-16a8,8,0,0,0-11.32-11.32ZM192,72a8,8,0,0,0,5.66-2.34l16-16a8,8,0,0,0-11.32-11.32l-16,16A8,8,0,0,0,192,72Zm5.66,114.34a8,8,0,0,0-11.32,11.32l16,16a8,8,0,0,0,11.32-11.32ZM48,128a8,8,0,0,0-8-8H16a8,8,0,0,0,0,16H40A8,8,0,0,0,48,128Zm80,80a8,8,0,0,0-8,8v24a8,8,0,0,0,16,0V216A8,8,0,0,0,128,208Zm112-88H216a8,8,0,0,0,0,16h24a8,8,0,0,0,0-16Z\" }))\n ],\n [\n \"thin\",\n /* @__PURE__ */ a.createElement(a.Fragment, null, /* @__PURE__ */ a.createElement(\"path\", { d: \"M124,40V16a4,4,0,0,1,8,0V40a4,4,0,0,1-8,0Zm64,88a60,60,0,1,1-60-60A60.07,60.07,0,0,1,188,128Zm-8,0a52,52,0,1,0-52,52A52.06,52.06,0,0,0,180,128ZM61.17,66.83a4,4,0,0,0,5.66-5.66l-16-16a4,4,0,0,0-5.66,5.66Zm0,122.34-16,16a4,4,0,0,0,5.66,5.66l16-16a4,4,0,0,0-5.66-5.66ZM192,68a4,4,0,0,0,2.83-1.17l16-16a4,4,0,1,0-5.66-5.66l-16,16A4,4,0,0,0,192,68Zm2.83,121.17a4,4,0,0,0-5.66,5.66l16,16a4,4,0,0,0,5.66-5.66ZM40,124H16a4,4,0,0,0,0,8H40a4,4,0,0,0,0-8Zm88,88a4,4,0,0,0-4,4v24a4,4,0,0,0,8,0V216A4,4,0,0,0,128,212Zm112-88H216a4,4,0,0,0,0,8h24a4,4,0,0,0,0-8Z\" }))\n ]\n]);\nexport {\n e as default\n};\n","import * as e from \"react\";\nimport c from \"../lib/IconBase.es.js\";\nimport t from \"../defs/CircleHalf.es.js\";\nconst o = e.forwardRef((r, a) => /* @__PURE__ */ e.createElement(c, { ref: a, ...r, weights: t }));\no.displayName = \"CircleHalfIcon\";\nconst l = o;\nexport {\n l as CircleHalf,\n o as CircleHalfIcon\n};\n","import * as o from \"react\";\nimport n from \"../lib/IconBase.es.js\";\nimport a from \"../defs/Moon.es.js\";\nconst e = o.forwardRef((r, t) => /* @__PURE__ */ o.createElement(n, { ref: t, ...r, weights: a }));\ne.displayName = \"MoonIcon\";\nconst s = e;\nexport {\n s as Moon,\n e as MoonIcon\n};\n","import * as o from \"react\";\nimport n from \"../lib/IconBase.es.js\";\nimport a from \"../defs/Sun.es.js\";\nconst e = o.forwardRef((r, t) => /* @__PURE__ */ o.createElement(n, { ref: t, ...r, weights: a }));\ne.displayName = \"SunIcon\";\nconst s = e;\nexport {\n s as Sun,\n e as SunIcon\n};\n","import { CircleHalfIcon, MoonIcon, SunIcon } from '@phosphor-icons/react';\nimport { useTranslation } from 'react-i18next';\nimport type { Theme, ThemeMode } from '#frontend/queries/config.ts';\n\ntype ThemeToggleProps = {\n themeMode: ThemeMode;\n isAutoMode: boolean;\n detectedTheme: Theme;\n darkTheme: Theme;\n onCycle: () => void;\n className?: string;\n};\n\nexport function ThemeToggle({\n themeMode,\n isAutoMode,\n detectedTheme,\n darkTheme,\n onCycle,\n className = 'fixed start-3 top-3 z-50 sm:absolute sm:start-4 sm:top-4',\n}: ThemeToggleProps) {\n const { t } = useTranslation();\n\n // Determine which icon to show based on theme mode\n const renderIcon = () => {\n if (themeMode === 'system' || isAutoMode) {\n return <CircleHalfIcon className=\"size-4\" weight=\"fill\" />;\n }\n if (themeMode === 'light') {\n return <SunIcon className=\"size-4\" weight=\"fill\" />;\n }\n return <MoonIcon className=\"size-4\" weight=\"fill\" />;\n };\n\n // Build tooltip label\n const getTooltipLabel = () => {\n if (themeMode === 'system' || isAutoMode) {\n const detectedLabel =\n detectedTheme === darkTheme\n ? t('common.theme.dark')\n : t('common.theme.light');\n return `${t('common.theme.auto')} (${detectedLabel})`;\n }\n if (themeMode === 'light') {\n return t('common.theme.light');\n }\n return t('common.theme.dark');\n };\n\n return (\n <div\n className={`tooltip tooltip-right ${className}`}\n data-tip={getTooltipLabel()}\n >\n <button\n aria-label={t('common.theme.select')}\n className=\"btn btn-circle btn-sm\"\n data-testid=\"theme-toggle\"\n onClick={onCycle}\n type=\"button\"\n >\n {renderIcon()}\n </button>\n </div>\n );\n}\n","import { useSuspenseQuery } from '@tanstack/react-query';\nimport { useCallback, useSyncExternalStore } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport {\n detectBrowserLanguage,\n getAvailableLanguages,\n LANGUAGE_STORAGE_KEY,\n} from '#frontend/i18n/index.ts';\nimport { appConfigQueryOptions } from '#frontend/queries/config.ts';\n\n/**\n * Subscribe to localStorage changes for language preference\n */\nfunction subscribeToLanguageStorage(callback: () => void) {\n const handleStorage = (e: StorageEvent) => {\n if (e.key === LANGUAGE_STORAGE_KEY) {\n callback();\n }\n };\n window.addEventListener('storage', handleStorage);\n\n // Custom event for same-tab updates\n window.addEventListener('language-storage-change', callback);\n\n return () => {\n window.removeEventListener('storage', handleStorage);\n window.removeEventListener('language-storage-change', callback);\n };\n}\n\nfunction getLanguageStorageSnapshot() {\n return localStorage.getItem(LANGUAGE_STORAGE_KEY);\n}\n\nfunction dispatchLanguageStorageChange() {\n window.dispatchEvent(new Event('language-storage-change'));\n}\n\nexport function useLanguage() {\n const { i18n } = useTranslation();\n const { data: config } = useSuspenseQuery(appConfigQueryOptions);\n\n // Get supported languages from server config\n const supportedLanguages = config.i18n.supported_languages;\n\n // Filter to only languages we have translations for\n const availableLanguages = getAvailableLanguages(supportedLanguages);\n\n // Sync with localStorage using useSyncExternalStore for proper reactivity\n const storedLanguage = useSyncExternalStore(\n subscribeToLanguageStorage,\n getLanguageStorageSnapshot,\n getLanguageStorageSnapshot,\n );\n\n // Check if user is in auto mode (no localStorage preference)\n const isAutoMode = storedLanguage === null;\n\n // Detect browser language for display in auto mode\n const detectedLanguage = detectBrowserLanguage(\n availableLanguages,\n config.i18n.fallback_language,\n );\n\n const setLanguage = useCallback(\n (lang: string) => {\n if (\n !availableLanguages.includes(\n lang as (typeof availableLanguages)[number],\n )\n ) {\n return;\n }\n i18n.changeLanguage(lang);\n localStorage.setItem(LANGUAGE_STORAGE_KEY, lang);\n dispatchLanguageStorageChange();\n },\n [availableLanguages, i18n],\n );\n\n const setAutoLanguage = useCallback(() => {\n localStorage.removeItem(LANGUAGE_STORAGE_KEY);\n dispatchLanguageStorageChange();\n // Immediately switch to detected browser language\n i18n.changeLanguage(detectedLanguage);\n }, [detectedLanguage, i18n]);\n\n return {\n language: i18n.language,\n languages: availableLanguages,\n setLanguage,\n setAutoLanguage,\n isAutoMode,\n detectedLanguage,\n // Show language selector only if more than one language is supported\n showLanguageSelector: availableLanguages.length > 1,\n };\n}\n","import { useTranslation } from 'react-i18next';\nimport { useLanguage } from '#frontend/hooks/use-language.ts';\nimport { LANGUAGE_LABELS } from '#frontend/i18n/index.ts';\n\ntype LanguageSelectorProps = {\n className?: string;\n};\n\nexport function LanguageSelector({ className = '' }: LanguageSelectorProps) {\n const { t } = useTranslation();\n const {\n language,\n languages,\n setLanguage,\n setAutoLanguage,\n isAutoMode,\n detectedLanguage,\n showLanguageSelector,\n } = useLanguage();\n\n if (!showLanguageSelector) {\n return null;\n }\n\n // Build auto label with detected language name\n const detectedLanguageName =\n LANGUAGE_LABELS[detectedLanguage] || detectedLanguage;\n const autoLabel = `${t('common.language.auto')} (${detectedLanguageName})`;\n\n const handleChange = (e: React.ChangeEvent<HTMLSelectElement>) => {\n const value = e.target.value;\n if (value === 'auto') {\n setAutoLanguage();\n } else {\n setLanguage(value);\n }\n };\n\n const currentValue = isAutoMode ? 'auto' : language;\n\n return (\n <div className={className}>\n <select\n aria-label={t('common.language.select')}\n className=\"select select-ghost select-sm text-base-content/60\"\n data-testid=\"language-selector\"\n onChange={handleChange}\n value={currentValue}\n >\n <option value=\"auto\">{autoLabel}</option>\n {languages.map((lang) => (\n <option key={lang} value={lang}>\n {LANGUAGE_LABELS[lang] || lang}\n </option>\n ))}\n </select>\n </div>\n );\n}\n","import { useSuspenseQuery } from '@tanstack/react-query';\nimport { ThemeToggle } from '#frontend/components/ui/theme-toggle.tsx';\nimport { LanguageSelector } from '#frontend/features/layout/language-selector.tsx';\nimport { useTheme } from '#frontend/hooks/use-theme.ts';\nimport { appConfigQueryOptions } from '#frontend/queries/config.ts';\n\ntype PageLayoutProps = {\n children: React.ReactNode;\n maxWidth?: 'sm' | 'md' | 'lg' | 'xl' | '2xl' | '100';\n cardPadding?: boolean;\n responsivePadding?: boolean;\n};\n\nconst maxWidthClasses: Record<\n NonNullable<PageLayoutProps['maxWidth']>,\n string\n> = {\n sm: 'max-w-sm',\n md: 'max-w-md',\n lg: 'max-w-lg',\n xl: 'max-w-xl',\n '2xl': 'max-w-2xl',\n '100': 'max-w-100',\n};\n\nexport function PageLayout({\n children,\n maxWidth = '100',\n cardPadding = false,\n responsivePadding = false,\n}: PageLayoutProps) {\n const {\n themeMode,\n darkTheme,\n canToggleTheme,\n cycleThemeMode,\n isAutoMode,\n detectedTheme,\n } = useTheme();\n const { data: configData } = useSuspenseQuery(appConfigQueryOptions);\n\n const backgroundUrl = configData.branding.background_url;\n\n const containerClass = responsivePadding ? 'p-4 md:p-8' : 'p-4';\n const cardClass = `${maxWidthClasses[maxWidth]}${cardPadding ? ' p-10' : ''}`;\n\n return (\n <div\n className={`relative flex min-h-screen flex-col bg-base-200 bg-cover ${containerClass}`}\n style={\n backgroundUrl\n ? {\n backgroundImage: `url(${backgroundUrl})`,\n }\n : undefined\n }\n >\n {canToggleTheme && (\n <ThemeToggle\n className=\"fixed start-3 top-3 z-50 sm:absolute sm:start-4 sm:top-4\"\n darkTheme={darkTheme}\n detectedTheme={detectedTheme}\n isAutoMode={isAutoMode}\n onCycle={cycleThemeMode}\n themeMode={themeMode}\n />\n )}\n <div className=\"flex flex-1 items-center justify-center\">\n <div\n className={`card w-full border border-base-200 bg-base-100 shadow-lg ${cardClass}`}\n >\n {children}\n </div>\n </div>\n <LanguageSelector className=\"mx-auto mt-4 pb-2\" />\n </div>\n );\n}\n"],"x_google_ignoreList":[0,1,2,3,4,5],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import{o as e,r as t,s as n,u as r}from"./IconBase.es-d5KP98Ac.js";import{f as i,h as a,ut as o}from"./use-theme-cVUDAjtt.js";import{t as s}from"./useMutation-DVMopbtG.js";import{t as c}from"./page-layout-C475gs09.js";import{t as l}from"./Fingerprint.es-CW755VWD.js";import{t as u}from"./ShieldCheck.es-CscPsYbC.js";import{c as d,o as f}from"./zod-BItJDQBQ.js";import{A as p,M as m,P as h,S as g,j as _,u as v}from"./index-CsT6OVnP.js";import{t as y}from"./page-header-BYMFSGfT.js";import{t as b}from"./alert-CSXqgDVi.js";import{t as x}from"./promise-OpBtq8tG.js";import{i as S}from"./passkey-e6uvApHa.js";import{r as C,t as w}from"./standard-schema-o4V-s4uY.js";import{t as T}from"./footer-link-Ib1Hd-fr.js";import{t as E}from"./submit-button-Xx6DwLyh.js";var D=r(n()),O=e();function k(){let{t:e}=t(),n=a(),r=o(),k=v.useSearch(),[A,j]=(0,D.useState)(k.passkey_name?`registering`:`form`),[M,N]=(0,D.useState)(``),P=(0,D.useRef)(!1),{register:F,handleSubmit:I,formState:{errors:L}}=C({defaultValues:{name:``},resolver:w((0,D.useMemo)(()=>f({name:d().max(100,e(`validation.passkey.name.max`))}),[e]))}),R=s({...S,onSuccess:async e=>{e.second_factor_setup_completed&&e.user?(r.setQueryData(g.queryKey,{user:e.user}),await x(),m(k)?window.location.href=p(k):n.navigate({to:`/profile`})):(r.invalidateQueries({queryKey:g.queryKey}),n.navigate({to:`/profile`}))},onError:t=>{j(`error`),t instanceof Error&&t.name===`NotAllowedError`?N(e(`setupPasskey.error.cancelled`)):N(e(`setupPasskey.error.failed`))},onSettled:()=>{r.invalidateQueries({queryKey:g.queryKey})}});return(0,D.useEffect)(()=>{k.passkey_name&&!P.current&&(P.current=!0,R.mutate({name:k.passkey_name}))},[R,k.passkey_name]),A===`registering`?(0,O.jsxs)(c,{cardPadding:!0,maxWidth:`100`,children:[(0,O.jsx)(y,{subtitle:e(`setupPasskey.subtitle`),title:e(`setupPasskey.title`)}),(0,O.jsxs)(`div`,{className:`flex flex-col items-center gap-4 py-8`,children:[(0,O.jsx)(l,{className:`size-16 animate-pulse text-primary`}),(0,O.jsx)(`p`,{className:`text-center text-base-content/70`,children:e(`setupPasskey.waiting`)})]})]}):A===`error`?(0,O.jsxs)(c,{cardPadding:!0,maxWidth:`100`,children:[(0,O.jsx)(y,{subtitle:e(`setupPasskey.subtitle`),title:e(`setupPasskey.title`)}),(0,O.jsx)(b,{icon:h,type:`error`,children:M}),(0,O.jsx)(`button`,{className:`btn btn-primary btn-block mt-4`,onClick:()=>{k.passkey_name?(j(`registering`),N(``),R.mutate({name:k.passkey_name})):j(`form`)},type:`button`,children:e(`setupPasskey.retry`)}),(0,O.jsx)(T,{as:i,linkText:e(`setupPasskey.backToLogin`),search:_(k),text:``,to:`/login`})]}):(0,O.jsxs)(c,{cardPadding:!0,maxWidth:`100`,children:[(0,O.jsx)(y,{subtitle:e(`setupPasskey.subtitle`),title:e(`setupPasskey.title`)}),(0,O.jsx)(b,{icon:u,type:`info`,children:e(`setupPasskey.required`)}),(0,O.jsx)(`p`,{className:`mt-4 text-center text-base-content/60 text-sm`,children:e(`setupPasskey.description`)}),(0,O.jsxs)(`form`,{className:`mt-4 flex flex-col gap-4`,onSubmit:I(e=>{j(`registering`),N(``),R.mutate({name:e.name||void 0})}),children:[(0,O.jsxs)(`div`,{className:`form-control`,children:[(0,O.jsx)(`label`,{className:`label`,htmlFor:`passkey-name`,children:(0,O.jsx)(`span`,{className:`label-text`,children:e(`setupPasskey.name.label`)})}),(0,O.jsx)(`input`,{className:`input input-bordered ${L.name?`input-error`:``}`,id:`passkey-name`,placeholder:e(`setupPasskey.name.placeholder`),type:`text`,...F(`name`)}),(0,O.jsx)(`span`,{className:`label-text-alt mt-1 text-base-content/50`,children:e(`setupPasskey.name.hint`)}),L.name&&(0,O.jsx)(`span`,{className:`label-text-alt mt-1 text-error`,children:L.name.message})]}),(0,O.jsx)(E,{className:`mt-2`,isPending:R.isPending,pendingText:e(`setupPasskey.registering`),children:e(`setupPasskey.continue`)})]}),(0,O.jsx)(T,{as:i,linkText:e(`setupPasskey.backToLogin`),search:_(k),text:``,to:`/login`})]})}export{k as component};
2
+ //# sourceMappingURL=passkey-BdISbWr7.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passkey-BdISbWr7.js","names":["standardSchemaResolver","FingerprintIcon","ShieldCheckIcon","WarningCircleIcon","useMutation","useQueryClient","Link","useRouter","useEffect","useMemo","useRef","useState","useForm","useTranslation","z","FooterLink","PageHeader","SubmitButton","Alert","PageLayout","buildAuthorizeUrl","extractOAuthParams","isOAuthFlow","tick","registerPasskeyMutationOptions","getSessionQueryOptions","Route","SetupStep","SetupPasskey","t","router","queryClient","search","useSearch","step","setStep","passkey_name","errorMessage","setErrorMessage","autoRegisterCalledRef","formSchema","object","name","string","max","FormValues","infer","register","handleSubmit","formState","errors","defaultValues","resolver","registerMutation","onSuccess","data","second_factor_setup_completed","user","setQueryData","queryKey","window","location","href","navigate","to","invalidateQueries","onError","error","Error","onSettled","current","mutate","onSubmit","values","undefined","message","isPending","component"],"sources":["../../../frontend/src/routes/setup/passkey/index.tsx?tsr-split=component"],"sourcesContent":["import { standardSchemaResolver } from '@hookform/resolvers/standard-schema';\nimport {\n FingerprintIcon,\n ShieldCheckIcon,\n WarningCircleIcon,\n} from '@phosphor-icons/react';\nimport { useMutation, useQueryClient } from '@tanstack/react-query';\nimport { createFileRoute, Link, useRouter } from '@tanstack/react-router';\nimport { useEffect, useMemo, useRef, useState } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { useTranslation } from 'react-i18next';\nimport { z } from 'zod';\nimport { FooterLink } from '#frontend/components/auth/footer-link.tsx';\nimport { PageHeader } from '#frontend/components/auth/page-header.tsx';\nimport { SubmitButton } from '#frontend/components/auth/submit-button.tsx';\nimport { Alert } from '#frontend/components/ui/alert.tsx';\nimport { PageLayout } from '#frontend/features/layout/page-layout.tsx';\nimport {\n buildAuthorizeUrl,\n extractOAuthParams,\n isOAuthFlow,\n OAuthSearchSchema,\n} from '#frontend/libs/oauth-search.ts';\nimport { tick } from '#frontend/libs/promise.ts';\nimport { registerPasskeyMutationOptions } from '#frontend/queries/passkey.ts';\nimport { getSessionQueryOptions } from '#frontend/queries/session.ts';\n\nconst SearchSchema = OAuthSearchSchema.extend({\n passkey_name: z.string().optional(),\n});\n\nexport const Route = createFileRoute('/setup/passkey/')({\n component: SetupPasskey,\n validateSearch: SearchSchema,\n});\n\ntype SetupStep = 'form' | 'registering' | 'error';\n\nfunction SetupPasskey() {\n const { t } = useTranslation();\n const router = useRouter();\n const queryClient = useQueryClient();\n const search = Route.useSearch();\n\n const [step, setStep] = useState<SetupStep>(\n search.passkey_name ? 'registering' : 'form',\n );\n const [errorMessage, setErrorMessage] = useState<string>('');\n const autoRegisterCalledRef = useRef(false);\n\n const formSchema = useMemo(\n () =>\n z.object({\n name: z.string().max(100, t('validation.passkey.name.max')),\n }),\n [t],\n );\n\n type FormValues = z.infer<typeof formSchema>;\n\n const {\n register,\n handleSubmit,\n formState: { errors },\n } = useForm<FormValues>({\n defaultValues: { name: '' },\n resolver: standardSchemaResolver(formSchema),\n });\n\n const registerMutation = useMutation({\n ...registerPasskeyMutationOptions,\n onSuccess: async (data) => {\n if (data.second_factor_setup_completed && data.user) {\n queryClient.setQueryData(getSessionQueryOptions.queryKey, {\n user: data.user,\n });\n await tick();\n\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n router.navigate({ to: '/profile' });\n }\n } else {\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n router.navigate({ to: '/profile' });\n }\n },\n onError: (error) => {\n setStep('error');\n if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n setErrorMessage(t('setupPasskey.error.cancelled'));\n } else {\n setErrorMessage(t('setupPasskey.error.failed'));\n }\n } else {\n setErrorMessage(t('setupPasskey.error.failed'));\n }\n },\n onSettled: () => {\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n });\n\n useEffect(() => {\n if (search.passkey_name && !autoRegisterCalledRef.current) {\n autoRegisterCalledRef.current = true;\n registerMutation.mutate({ name: search.passkey_name });\n }\n }, [registerMutation, search.passkey_name]);\n\n const onSubmit = (values: FormValues) => {\n setStep('registering');\n setErrorMessage('');\n registerMutation.mutate({ name: values.name || undefined });\n };\n\n // Registering state - waiting for WebAuthn\n if (step === 'registering') {\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n subtitle={t('setupPasskey.subtitle')}\n title={t('setupPasskey.title')}\n />\n <div className=\"flex flex-col items-center gap-4 py-8\">\n <FingerprintIcon className=\"size-16 animate-pulse text-primary\" />\n <p className=\"text-center text-base-content/70\">\n {t('setupPasskey.waiting')}\n </p>\n </div>\n </PageLayout>\n );\n }\n\n // Error state\n if (step === 'error') {\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n subtitle={t('setupPasskey.subtitle')}\n title={t('setupPasskey.title')}\n />\n <Alert icon={WarningCircleIcon} type=\"error\">\n {errorMessage}\n </Alert>\n <button\n className=\"btn btn-primary btn-block mt-4\"\n onClick={() => {\n if (search.passkey_name) {\n setStep('registering');\n setErrorMessage('');\n registerMutation.mutate({ name: search.passkey_name });\n } else {\n setStep('form');\n }\n }}\n type=\"button\"\n >\n {t('setupPasskey.retry')}\n </button>\n <FooterLink\n as={Link}\n linkText={t('setupPasskey.backToLogin')}\n search={extractOAuthParams(search)}\n text=\"\"\n to=\"/login\"\n />\n </PageLayout>\n );\n }\n\n // Form state\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n subtitle={t('setupPasskey.subtitle')}\n title={t('setupPasskey.title')}\n />\n\n <Alert icon={ShieldCheckIcon} type=\"info\">\n {t('setupPasskey.required')}\n </Alert>\n\n <p className=\"mt-4 text-center text-base-content/60 text-sm\">\n {t('setupPasskey.description')}\n </p>\n\n <form\n className=\"mt-4 flex flex-col gap-4\"\n onSubmit={handleSubmit(onSubmit)}\n >\n <div className=\"form-control\">\n <label className=\"label\" htmlFor=\"passkey-name\">\n <span className=\"label-text\">{t('setupPasskey.name.label')}</span>\n </label>\n <input\n className={`input input-bordered ${\n errors.name ? 'input-error' : ''\n }`}\n id=\"passkey-name\"\n placeholder={t('setupPasskey.name.placeholder')}\n type=\"text\"\n {...register('name')}\n />\n <span className=\"label-text-alt mt-1 text-base-content/50\">\n {t('setupPasskey.name.hint')}\n </span>\n {errors.name && (\n <span className=\"label-text-alt mt-1 text-error\">\n {errors.name.message}\n </span>\n )}\n </div>\n\n <SubmitButton\n className=\"mt-2\"\n isPending={registerMutation.isPending}\n pendingText={t('setupPasskey.registering')}\n >\n {t('setupPasskey.continue')}\n </SubmitButton>\n </form>\n\n <FooterLink\n as={Link}\n linkText={t('setupPasskey.backToLogin')}\n search={extractOAuthParams(search)}\n text=\"\"\n to=\"/login\"\n />\n </PageLayout>\n );\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import{n as e,o as t,r as n,s as r,u as i}from"./IconBase.es-d5KP98Ac.js";import{f as a,h as o,ut as s}from"./use-theme-cVUDAjtt.js";import{t as c}from"./useMutation-DVMopbtG.js";import{t as l}from"./page-layout-C475gs09.js";import{t as u}from"./Fingerprint.es-CW755VWD.js";import{A as d,M as f,P as p,S as m,m as h}from"./index-CsT6OVnP.js";import{t as g}from"./page-header-BYMFSGfT.js";import{t as _}from"./alert-CSXqgDVi.js";import{t as v}from"./promise-OpBtq8tG.js";import{t as y}from"./passkey-e6uvApHa.js";import{t as b}from"./footer-link-Ib1Hd-fr.js";var x=i(r()),S=t();function C(){let{t}=n(),r=o(),i=s(),C=h.useSearch(),[w,T]=(0,x.useState)(null),E=(0,x.useRef)(!1),D=c({...y,onSuccess:async e=>{e.user&&(i.setQueryData(m.queryKey,{user:e.user}),await v(),f(C)?window.location.href=d(C):r.navigate({to:`/profile`}))},onError:async n=>{if(n instanceof e){if(n.code===`SECOND_FACTOR_SESSION_EXPIRED`){T(t(`verifyPasskey.error.expired`));return}if(n.code===`PASSKEY_USER_MISMATCH`){T(t(`verifyPasskey.error.userMismatch`));return}}else{T(t(`verifyPasskey.error.failed`));return}},onSettled:()=>{i.invalidateQueries({queryKey:m.queryKey})}});return(0,x.useEffect)(()=>{E.current||(E.current=!0,D.mutate())},[D]),(0,S.jsxs)(l,{cardPadding:!0,maxWidth:`100`,children:[(0,S.jsx)(g,{subtitle:t(`verifyPasskey.subtitle`),title:t(`verifyPasskey.title`)}),(0,S.jsxs)(`div`,{className:`flex flex-col items-center gap-6`,children:[D.isPending&&(0,S.jsxs)(`div`,{className:`flex flex-col items-center gap-4`,children:[(0,S.jsx)(`div`,{className:`flex size-20 items-center justify-center rounded-full bg-base-200`,children:(0,S.jsx)(u,{className:`size-10 animate-pulse text-primary`})}),(0,S.jsx)(`p`,{className:`text-center text-base-content/70 text-sm`,children:t(`verifyPasskey.waiting`)})]}),w&&(0,S.jsxs)(S.Fragment,{children:[(0,S.jsx)(_,{className:`w-full`,icon:p,type:`error`,children:w}),(0,S.jsxs)(`button`,{className:`btn btn-primary btn-block`,onClick:()=>{T(null),D.mutate()},type:`button`,children:[(0,S.jsx)(u,{className:`size-5`,weight:`regular`}),t(`verifyPasskey.retry`)]})]})]}),(0,S.jsx)(b,{as:a,linkText:t(`verifyPasskey.backToLogin`),search:C,text:``,to:`/login`})]})}export{C as component};
2
+ //# sourceMappingURL=passkey-Bv7zPLAZ.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passkey-Bv7zPLAZ.js","names":["FingerprintIcon","WarningCircleIcon","useMutation","useQueryClient","Link","useRouter","useEffect","useRef","useState","useTranslation","FooterLink","PageHeader","Alert","PageLayout","TinyAuthError","buildAuthorizeUrl","isOAuthFlow","tick","authenticateWithPasskeyMutationOptions","getSessionQueryOptions","Route","VerifyPasskey","t","router","queryClient","search","useSearch","error","setError","hasStarted","verifyMutation","onSuccess","data","user","setQueryData","queryKey","window","location","href","navigate","to","onError","err","code","onSettled","invalidateQueries","current","mutate","isPending","component"],"sources":["../../../frontend/src/routes/verify/passkey/index.tsx?tsr-split=component"],"sourcesContent":["import { FingerprintIcon, WarningCircleIcon } from '@phosphor-icons/react';\nimport { useMutation, useQueryClient } from '@tanstack/react-query';\nimport { createFileRoute, Link, useRouter } from '@tanstack/react-router';\nimport { useEffect, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { FooterLink } from '#frontend/components/auth/footer-link.tsx';\nimport { PageHeader } from '#frontend/components/auth/page-header.tsx';\nimport { Alert } from '#frontend/components/ui/alert.tsx';\nimport { PageLayout } from '#frontend/features/layout/page-layout.tsx';\nimport { TinyAuthError } from '#frontend/libs/error.ts';\nimport {\n buildAuthorizeUrl,\n isOAuthFlow,\n OAuthSearchSchema,\n} from '#frontend/libs/oauth-search.ts';\nimport { tick } from '#frontend/libs/promise.ts';\nimport { authenticateWithPasskeyMutationOptions } from '#frontend/queries/passkey.ts';\nimport { getSessionQueryOptions } from '#frontend/queries/session.ts';\n\nconst SearchSchema = OAuthSearchSchema;\n\nexport const Route = createFileRoute('/verify/passkey/')({\n component: VerifyPasskey,\n validateSearch: SearchSchema,\n});\n\nfunction VerifyPasskey() {\n const { t } = useTranslation();\n const router = useRouter();\n const queryClient = useQueryClient();\n const search = Route.useSearch();\n const [error, setError] = useState<string | null>(null);\n const hasStarted = useRef(false);\n\n const verifyMutation = useMutation({\n ...authenticateWithPasskeyMutationOptions,\n onSuccess: async (data) => {\n if (data.user) {\n queryClient.setQueryData(getSessionQueryOptions.queryKey, {\n user: data.user,\n });\n await tick();\n\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n router.navigate({ to: '/profile' });\n }\n }\n },\n onError: async (err) => {\n if (err instanceof TinyAuthError) {\n if (err.code === 'SECOND_FACTOR_SESSION_EXPIRED') {\n setError(t('verifyPasskey.error.expired'));\n return;\n }\n if (err.code === 'PASSKEY_USER_MISMATCH') {\n setError(t('verifyPasskey.error.userMismatch'));\n return;\n }\n } else {\n setError(t('verifyPasskey.error.failed'));\n return;\n }\n },\n onSettled: () => {\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n });\n\n // Auto-start passkey authentication on mount (with guard for StrictMode)\n useEffect(() => {\n if (hasStarted.current) return;\n hasStarted.current = true;\n verifyMutation.mutate();\n }, [verifyMutation]);\n\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n subtitle={t('verifyPasskey.subtitle')}\n title={t('verifyPasskey.title')}\n />\n\n <div className=\"flex flex-col items-center gap-6\">\n {verifyMutation.isPending && (\n <div className=\"flex flex-col items-center gap-4\">\n <div className=\"flex size-20 items-center justify-center rounded-full bg-base-200\">\n <FingerprintIcon className=\"size-10 animate-pulse text-primary\" />\n </div>\n <p className=\"text-center text-base-content/70 text-sm\">\n {t('verifyPasskey.waiting')}\n </p>\n </div>\n )}\n\n {error && (\n <>\n <Alert className=\"w-full\" icon={WarningCircleIcon} type=\"error\">\n {error}\n </Alert>\n <button\n className=\"btn btn-primary btn-block\"\n onClick={() => {\n setError(null);\n verifyMutation.mutate();\n }}\n type=\"button\"\n >\n <FingerprintIcon className=\"size-5\" weight=\"regular\" />\n {t('verifyPasskey.retry')}\n </button>\n </>\n )}\n </div>\n\n <FooterLink\n as={Link}\n linkText={t('verifyPasskey.backToLogin')}\n search={search}\n text=\"\"\n to=\"/login\"\n />\n </PageLayout>\n );\n}\n"],"mappings":"ikBA0BA,SAASqB,GAAgB,CACvB,GAAM,CAAEC,GAAMb,EAAe,EACvBc,EAASlB,EAAU,EACnBmB,EAAcrB,EAAe,EAC7BsB,EAASL,EAAMM,UAAU,EACzB,CAACC,EAAOC,IAAAA,EAAAA,EAAAA,UAAoC,IAAI,EAChDC,GAAAA,EAAAA,EAAAA,QAAoB,EAAK,EAEzBC,EAAiB5B,EAAY,CACjC,GAAGgB,EACHa,UAAW,KAAOC,IAAS,CACrBA,EAAKC,OACPT,EAAYU,aAAaf,EAAuBgB,SAAU,CACxDF,KAAMD,EAAKC,IACb,CAAC,EACD,MAAMhB,EAAK,EAEPD,EAAYS,CAAM,EACpBW,OAAOC,SAASC,KAAOvB,EAAkBU,CAAM,EAE/CF,EAAOgB,SAAS,CAAEC,GAAI,UAAW,CAAC,EAGxC,EACAC,QAAS,KAAOC,IAAQ,CACtB,GAAIA,aAAe5B,EAAe,CAChC,GAAI4B,EAAIC,OAAS,gCAAiC,CAChDf,EAASN,EAAE,6BAA6B,CAAC,EACzC,MACF,CACA,GAAIoB,EAAIC,OAAS,wBAAyB,CACxCf,EAASN,EAAE,kCAAkC,CAAC,EAC9C,MACF,CACF,KAAO,CACLM,EAASN,EAAE,4BAA4B,CAAC,EACxC,MACF,CACF,EACAsB,cAAiB,CACfpB,EAAYqB,kBAAkB,CAC5BV,SAAUhB,EAAuBgB,QACnC,CAAC,CACH,CACF,CAAC,EASD,OANA7B,EAAAA,EAAAA,eAAgB,CACVuB,EAAWiB,UACfjB,EAAWiB,QAAU,GACrBhB,EAAeiB,OAAO,EACxB,EAAG,CAACjB,CAAc,CAAC,GAGjB,EAAA,EAAA,MAAC,EAAD,CAAY,YAAA,GAAY,SAAS,eAAjC,EACE,EAAA,EAAA,KAAC,EAAD,CACE,SAAUR,EAAE,wBAAwB,EACpC,MAAOA,EAAE,qBAAqB,CAAE,CAAA,GAGlC,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,4CAAf,CACGQ,EAAekB,YACd,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,4CAAf,EACE,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,8EACb,EAAA,EAAA,KAAC,EAAD,CAAiB,UAAU,oCAAoC,CAAA,CAC5D,CAAA,GACL,EAAA,EAAA,KAAC,IAAD,CAAG,UAAU,oDACV1B,EAAE,uBAAuB,CACzB,CAAA,CACA,IAGNK,IACC,EAAA,EAAA,MAAA,EAAA,SAAA,CAAA,SAAA,EACE,EAAA,EAAA,KAAC,EAAD,CAAO,UAAU,SAAS,KAAM1B,EAAmB,KAAK,iBACrD0B,CACI,CAAA,GACP,EAAA,EAAA,MAAC,SAAD,CACE,UAAU,4BACV,YAAe,CACbC,EAAS,IAAI,EACbE,EAAeiB,OAAO,CACxB,EACA,KAAK,kBANP,EAQE,EAAA,EAAA,KAAC,EAAD,CAAiB,UAAU,SAAS,OAAO,SAAS,CAAA,EACnDzB,EAAE,qBAAqB,CAClB,GACV,CAAA,CAAA,CAEC,KAEL,EAAA,EAAA,KAAC,EAAD,CACE,GAAIlB,EACJ,SAAUkB,EAAE,2BAA2B,EAC/BG,SACR,KAAK,GACL,GAAG,QAAQ,CAAA,CAEH,GAEhB"}
@@ -0,0 +1,2 @@
1
+ import{Z as e,a as t,i as n,r}from"./use-theme-cVUDAjtt.js";import{t as i}from"./mutationOptions-Dfvzj6n2.js";function a(e){let t=new Uint8Array(e),n=``;for(let e of t)n+=String.fromCharCode(e);return btoa(n).replace(/\+/g,`-`).replace(/\//g,`_`).replace(/=/g,``)}function o(e){let t=e.replace(/-/g,`+`).replace(/_/g,`/`),n=(4-t.length%4)%4,r=t.padEnd(t.length+n,`=`),i=atob(r),a=new ArrayBuffer(i.length),o=new Uint8Array(a);for(let e=0;e<i.length;e++)o[e]=i.charCodeAt(e);return a}function s(){return c.stubThis(globalThis?.PublicKeyCredential!==void 0&&typeof globalThis.PublicKeyCredential==`function`)}var c={stubThis:e=>e};function l(e){let{id:t}=e;return{...e,id:o(t),transports:e.transports}}function u(e){return e===`localhost`||/^((xn--[a-z0-9-]+|[a-z0-9]+(-[a-z0-9]+)*)\.)+([a-z]{2,}|xn--[a-z0-9-]+)$/i.test(e)}var d=class extends Error{constructor({message:e,code:t,cause:n,name:r}){super(e,{cause:n}),Object.defineProperty(this,"code",{enumerable:!0,configurable:!0,writable:!0,value:void 0}),this.name=r??n.name,this.code=t}};function f({error:e,options:t}){let{publicKey:n}=t;if(!n)throw Error(`options was missing required publicKey property`);if(e.name===`AbortError`){if(t.signal instanceof AbortSignal)return new d({message:`Registration ceremony was sent an abort signal`,code:`ERROR_CEREMONY_ABORTED`,cause:e})}else if(e.name===`ConstraintError`){if(n.authenticatorSelection?.requireResidentKey===!0)return new d({message:`Discoverable credentials were required but no available authenticator supported it`,code:`ERROR_AUTHENTICATOR_MISSING_DISCOVERABLE_CREDENTIAL_SUPPORT`,cause:e});if(t.mediation===`conditional`&&n.authenticatorSelection?.userVerification===`required`)return new d({message:`User verification was required during automatic registration but it could not be performed`,code:`ERROR_AUTO_REGISTER_USER_VERIFICATION_FAILURE`,cause:e});if(n.authenticatorSelection?.userVerification===`required`)return new d({message:`User verification was required but no available authenticator supported it`,code:`ERROR_AUTHENTICATOR_MISSING_USER_VERIFICATION_SUPPORT`,cause:e})}else if(e.name===`InvalidStateError`)return new d({message:`The authenticator was previously registered`,code:`ERROR_AUTHENTICATOR_PREVIOUSLY_REGISTERED`,cause:e});else if(e.name===`NotAllowedError`)return new d({message:e.message,code:`ERROR_PASSTHROUGH_SEE_CAUSE_PROPERTY`,cause:e});else if(e.name===`NotSupportedError`)return n.pubKeyCredParams.filter(e=>e.type===`public-key`).length===0?new d({message:`No entry in pubKeyCredParams was of type "public-key"`,code:`ERROR_MALFORMED_PUBKEYCREDPARAMS`,cause:e}):new d({message:`No available authenticator supported any of the specified pubKeyCredParams algorithms`,code:`ERROR_AUTHENTICATOR_NO_SUPPORTED_PUBKEYCREDPARAMS_ALG`,cause:e});else if(e.name===`SecurityError`){let t=globalThis.location.hostname;if(!u(t))return new d({message:`${globalThis.location.hostname} is an invalid domain`,code:`ERROR_INVALID_DOMAIN`,cause:e});if(n.rp.id!==t)return new d({message:`The RP ID "${n.rp.id}" is invalid for this domain`,code:`ERROR_INVALID_RP_ID`,cause:e})}else if(e.name===`TypeError`){if(n.user.id.byteLength<1||n.user.id.byteLength>64)return new d({message:`User ID was not between 1 and 64 characters`,code:`ERROR_INVALID_USER_ID_LENGTH`,cause:e})}else if(e.name===`UnknownError`)return new d({message:`The authenticator was unable to process the specified options, or could not create a new credential`,code:`ERROR_AUTHENTICATOR_GENERAL_ERROR`,cause:e});return e}var p=new class{constructor(){Object.defineProperty(this,"controller",{enumerable:!0,configurable:!0,writable:!0,value:void 0})}createNewAbortSignal(){if(this.controller){let e=Error(`Cancelling existing WebAuthn API call for new one`);e.name=`AbortError`,this.controller.abort(e)}let e=new AbortController;return this.controller=e,e.signal}cancelCeremony(){if(this.controller){let e=Error(`Manually cancelling existing WebAuthn API call`);e.name=`AbortError`,this.controller.abort(e),this.controller=void 0}}},m=[`cross-platform`,`platform`];function h(e){if(e&&!(m.indexOf(e)<0))return e}async function g(e){!e.optionsJSON&&e.challenge&&(console.warn(`startRegistration() was not called correctly. It will try to continue with the provided options, but this call should be refactored to use the expected call structure instead. See https://simplewebauthn.dev/docs/packages/browser#typeerror-cannot-read-properties-of-undefined-reading-challenge for more information.`),e={optionsJSON:e});let{optionsJSON:t,useAutoRegister:n=!1}=e;if(!s())throw Error(`WebAuthn is not supported in this browser`);let r={...t,challenge:o(t.challenge),user:{...t.user,id:o(t.user.id)},excludeCredentials:t.excludeCredentials?.map(l)},i={};n&&(i.mediation=`conditional`),i.publicKey=r,i.signal=p.createNewAbortSignal();let c;try{c=await navigator.credentials.create(i)}catch(e){throw f({error:e,options:i})}if(!c)throw Error(`Registration was not completed`);let{id:u,rawId:d,response:m,type:g}=c,v;typeof m.getTransports==`function`&&(v=m.getTransports());let y;if(typeof m.getPublicKeyAlgorithm==`function`)try{y=m.getPublicKeyAlgorithm()}catch(e){_(`getPublicKeyAlgorithm()`,e)}let b;if(typeof m.getPublicKey==`function`)try{let e=m.getPublicKey();e!==null&&(b=a(e))}catch(e){_(`getPublicKey()`,e)}let x;if(typeof m.getAuthenticatorData==`function`)try{x=a(m.getAuthenticatorData())}catch(e){_(`getAuthenticatorData()`,e)}return{id:u,rawId:a(d),response:{attestationObject:a(m.attestationObject),clientDataJSON:a(m.clientDataJSON),transports:v,publicKeyAlgorithm:y,publicKey:b,authenticatorData:x},type:g,clientExtensionResults:c.getClientExtensionResults(),authenticatorAttachment:h(c.authenticatorAttachment)}}function _(e,t){console.warn(`The browser extension that intercepted this WebAuthn API call incorrectly implemented ${e}. You should report this error to them.\n`,t)}function v(){if(!s())return y.stubThis(new Promise(e=>e(!1)));let e=globalThis.PublicKeyCredential;return e?.isConditionalMediationAvailable===void 0?y.stubThis(new Promise(e=>e(!1))):y.stubThis(e.isConditionalMediationAvailable())}var y={stubThis:e=>e};function b({error:e,options:t}){let{publicKey:n}=t;if(!n)throw Error(`options was missing required publicKey property`);if(e.name===`AbortError`){if(t.signal instanceof AbortSignal)return new d({message:`Authentication ceremony was sent an abort signal`,code:`ERROR_CEREMONY_ABORTED`,cause:e})}else if(e.name===`NotAllowedError`)return new d({message:e.message,code:`ERROR_PASSTHROUGH_SEE_CAUSE_PROPERTY`,cause:e});else if(e.name===`SecurityError`){let t=globalThis.location.hostname;if(!u(t))return new d({message:`${globalThis.location.hostname} is an invalid domain`,code:`ERROR_INVALID_DOMAIN`,cause:e});if(n.rpId!==t)return new d({message:`The RP ID "${n.rpId}" is invalid for this domain`,code:`ERROR_INVALID_RP_ID`,cause:e})}else if(e.name===`UnknownError`)return new d({message:`The authenticator was unable to process the specified options, or could not create a new assertion signature`,code:`ERROR_AUTHENTICATOR_GENERAL_ERROR`,cause:e});return e}async function x(e){!e.optionsJSON&&e.challenge&&(console.warn(`startAuthentication() was not called correctly. It will try to continue with the provided options, but this call should be refactored to use the expected call structure instead. See https://simplewebauthn.dev/docs/packages/browser#typeerror-cannot-read-properties-of-undefined-reading-challenge for more information.`),e={optionsJSON:e});let{optionsJSON:t,useBrowserAutofill:n=!1,verifyBrowserAutofillInput:r=!0}=e;if(!s())throw Error(`WebAuthn is not supported in this browser`);let i;t.allowCredentials?.length!==0&&(i=t.allowCredentials?.map(l));let c={...t,challenge:o(t.challenge),allowCredentials:i},u={};if(n){if(!await v())throw Error(`Browser does not support WebAuthn autofill`);if(document.querySelectorAll(`input[autocomplete$='webauthn']`).length<1&&r)throw Error('No <input> with "webauthn" as the only or last value in its `autocomplete` attribute was detected');u.mediation=`conditional`,c.allowCredentials=[]}u.publicKey=c,u.signal=p.createNewAbortSignal();let d;try{d=await navigator.credentials.get(u)}catch(e){throw b({error:e,options:u})}if(!d)throw Error(`Authentication was not completed`);let{id:f,rawId:m,response:g,type:_}=d,y;return g.userHandle&&(y=a(g.userHandle)),{id:f,rawId:a(m),response:{authenticatorData:a(g.authenticatorData),clientDataJSON:a(g.clientDataJSON),signature:a(g.signature),userHandle:y},type:_,clientExtensionResults:d.getClientExtensionResults(),authenticatorAttachment:h(d.authenticatorAttachment)}}var S=e({queryKey:r.passkeys(),queryFn:async()=>t(await n.api.user.passkeys.$get())}),C=i({mutationFn:async e=>{let{options:r}=await t(await n.api.user.passkeys.register.options.$post()),i=await g({optionsJSON:r});return t(await n.api.user.passkeys.register.verify.$post({json:{response:i,name:e.name}}))}}),w=i({mutationFn:async e=>t(await n.api.user.passkeys[`:id`].$delete({param:{id:e.id}}))}),T=i({mutationFn:async e=>t(await n.api.user.passkeys[`:id`].$patch({param:{id:e.id},json:{name:e.name}}))}),E=i({mutationFn:async()=>{let{options:e}=await t(await n.api.auth.passkey.options.$post()),r=await x({optionsJSON:e});return t(await n.api.auth.passkey.verify.$post({json:{response:{...r,clientExtensionResults:{...r.clientExtensionResults}}}}))}}),D=async(e,r)=>{try{let{options:i}=await t(await n.api.auth.passkey.options.$post({},{init:{signal:r}})),a=await x({optionsJSON:i,useBrowserAutofill:!0});e(await t(await n.api.auth.passkey.verify.$post({json:{response:{...a,clientExtensionResults:{...a.clientExtensionResults}}}},{init:{signal:r}})))}catch(e){e instanceof Error&&e.name}};export{T as a,C as i,w as n,D as o,S as r,E as t};
2
+ //# sourceMappingURL=passkey-e6uvApHa.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passkey-e6uvApHa.js","names":[],"sources":["../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/bufferToBase64URLString.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/base64URLStringToBuffer.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/browserSupportsWebAuthn.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/toPublicKeyCredentialDescriptor.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/isValidDomain.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/webAuthnError.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/identifyRegistrationError.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/webAuthnAbortService.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/toAuthenticatorAttachment.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/methods/startRegistration.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/browserSupportsWebAuthnAutofill.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/helpers/identifyAuthenticationError.js","../../../../node_modules/.pnpm/@simplewebauthn+browser@13.3.0/node_modules/@simplewebauthn/browser/esm/methods/startAuthentication.js","../../../frontend/src/queries/passkey.ts"],"sourcesContent":["/**\n * Convert the given array buffer into a Base64URL-encoded string. Ideal for converting various\n * credential response ArrayBuffers to string for sending back to the server as JSON.\n *\n * Helper method to compliment `base64URLStringToBuffer`\n */\nexport function bufferToBase64URLString(buffer) {\n const bytes = new Uint8Array(buffer);\n let str = '';\n for (const charCode of bytes) {\n str += String.fromCharCode(charCode);\n }\n const base64String = btoa(str);\n return base64String.replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=/g, '');\n}\n","/**\n * Convert from a Base64URL-encoded string to an Array Buffer. Best used when converting a\n * credential ID from a JSON string to an ArrayBuffer, like in allowCredentials or\n * excludeCredentials\n *\n * Helper method to compliment `bufferToBase64URLString`\n */\nexport function base64URLStringToBuffer(base64URLString) {\n // Convert from Base64URL to Base64\n const base64 = base64URLString.replace(/-/g, '+').replace(/_/g, '/');\n /**\n * Pad with '=' until it's a multiple of four\n * (4 - (85 % 4 = 1) = 3) % 4 = 3 padding\n * (4 - (86 % 4 = 2) = 2) % 4 = 2 padding\n * (4 - (87 % 4 = 3) = 1) % 4 = 1 padding\n * (4 - (88 % 4 = 0) = 4) % 4 = 0 padding\n */\n const padLength = (4 - (base64.length % 4)) % 4;\n const padded = base64.padEnd(base64.length + padLength, '=');\n // Convert to a binary string\n const binary = atob(padded);\n // Convert binary string to buffer\n const buffer = new ArrayBuffer(binary.length);\n const bytes = new Uint8Array(buffer);\n for (let i = 0; i < binary.length; i++) {\n bytes[i] = binary.charCodeAt(i);\n }\n return buffer;\n}\n","/**\n * Determine if the browser is capable of Webauthn\n */\nexport function browserSupportsWebAuthn() {\n return _browserSupportsWebAuthnInternals.stubThis(globalThis?.PublicKeyCredential !== undefined &&\n typeof globalThis.PublicKeyCredential === 'function');\n}\n/**\n * Make it possible to stub the return value during testing\n * @ignore Don't include this in docs output\n */\nexport const _browserSupportsWebAuthnInternals = {\n stubThis: (value) => value,\n};\n","import { base64URLStringToBuffer } from './base64URLStringToBuffer.js';\nexport function toPublicKeyCredentialDescriptor(descriptor) {\n const { id } = descriptor;\n return {\n ...descriptor,\n id: base64URLStringToBuffer(id),\n /**\n * `descriptor.transports` is an array of our `AuthenticatorTransportFuture` that includes newer\n * transports that TypeScript's DOM lib is ignorant of. Convince TS that our list of transports\n * are fine to pass to WebAuthn since browsers will recognize the new value.\n */\n transports: descriptor.transports,\n };\n}\n","/**\n * A simple test to determine if a hostname is a properly-formatted domain name\n *\n * A \"valid domain\" is defined here: https://url.spec.whatwg.org/#valid-domain\n *\n * Regex was originally sourced from here, then remixed to add punycode support:\n * https://www.oreilly.com/library/view/regular-expressions-cookbook/9781449327453/ch08s15.html\n */\nexport function isValidDomain(hostname) {\n return (\n // Consider localhost valid as well since it's okay wrt Secure Contexts\n hostname === 'localhost' ||\n // Support punycode (ACE) or ascii labels and domains\n /^((xn--[a-z0-9-]+|[a-z0-9]+(-[a-z0-9]+)*)\\.)+([a-z]{2,}|xn--[a-z0-9-]+)$/i.test(hostname));\n}\n","/**\n * A custom Error used to return a more nuanced error detailing _why_ one of the eight documented\n * errors in the spec was raised after calling `navigator.credentials.create()` or\n * `navigator.credentials.get()`:\n *\n * - `AbortError`\n * - `ConstraintError`\n * - `InvalidStateError`\n * - `NotAllowedError`\n * - `NotSupportedError`\n * - `SecurityError`\n * - `TypeError`\n * - `UnknownError`\n *\n * Error messages were determined through investigation of the spec to determine under which\n * scenarios a given error would be raised.\n */\nexport class WebAuthnError extends Error {\n constructor({ message, code, cause, name, }) {\n // @ts-ignore: help Rollup understand that `cause` is okay to set\n super(message, { cause });\n Object.defineProperty(this, \"code\", {\n enumerable: true,\n configurable: true,\n writable: true,\n value: void 0\n });\n this.name = name ?? cause.name;\n this.code = code;\n }\n}\n","import { isValidDomain } from './isValidDomain.js';\nimport { WebAuthnError } from './webAuthnError.js';\n/**\n * Attempt to intuit _why_ an error was raised after calling `navigator.credentials.create()`\n */\nexport function identifyRegistrationError({ error, options, }) {\n const { publicKey } = options;\n if (!publicKey) {\n throw Error('options was missing required publicKey property');\n }\n if (error.name === 'AbortError') {\n if (options.signal instanceof AbortSignal) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 16)\n return new WebAuthnError({\n message: 'Registration ceremony was sent an abort signal',\n code: 'ERROR_CEREMONY_ABORTED',\n cause: error,\n });\n }\n }\n else if (error.name === 'ConstraintError') {\n if (publicKey.authenticatorSelection?.requireResidentKey === true) {\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 4)\n return new WebAuthnError({\n message: 'Discoverable credentials were required but no available authenticator supported it',\n code: 'ERROR_AUTHENTICATOR_MISSING_DISCOVERABLE_CREDENTIAL_SUPPORT',\n cause: error,\n });\n }\n else if (\n // @ts-ignore: `mediation` doesn't yet exist on CredentialCreationOptions but it's possible as of Sept 2024\n options.mediation === 'conditional' &&\n publicKey.authenticatorSelection?.userVerification === 'required') {\n // https://w3c.github.io/webauthn/#sctn-createCredential (Step 22.4)\n return new WebAuthnError({\n message: 'User verification was required during automatic registration but it could not be performed',\n code: 'ERROR_AUTO_REGISTER_USER_VERIFICATION_FAILURE',\n cause: error,\n });\n }\n else if (publicKey.authenticatorSelection?.userVerification === 'required') {\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 5)\n return new WebAuthnError({\n message: 'User verification was required but no available authenticator supported it',\n code: 'ERROR_AUTHENTICATOR_MISSING_USER_VERIFICATION_SUPPORT',\n cause: error,\n });\n }\n }\n else if (error.name === 'InvalidStateError') {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 20)\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 3)\n return new WebAuthnError({\n message: 'The authenticator was previously registered',\n code: 'ERROR_AUTHENTICATOR_PREVIOUSLY_REGISTERED',\n cause: error,\n });\n }\n else if (error.name === 'NotAllowedError') {\n /**\n * Pass the error directly through. Platforms are overloading this error beyond what the spec\n * defines and we don't want to overwrite potentially useful error messages.\n */\n return new WebAuthnError({\n message: error.message,\n code: 'ERROR_PASSTHROUGH_SEE_CAUSE_PROPERTY',\n cause: error,\n });\n }\n else if (error.name === 'NotSupportedError') {\n const validPubKeyCredParams = publicKey.pubKeyCredParams.filter((param) => param.type === 'public-key');\n if (validPubKeyCredParams.length === 0) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 10)\n return new WebAuthnError({\n message: 'No entry in pubKeyCredParams was of type \"public-key\"',\n code: 'ERROR_MALFORMED_PUBKEYCREDPARAMS',\n cause: error,\n });\n }\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 2)\n return new WebAuthnError({\n message: 'No available authenticator supported any of the specified pubKeyCredParams algorithms',\n code: 'ERROR_AUTHENTICATOR_NO_SUPPORTED_PUBKEYCREDPARAMS_ALG',\n cause: error,\n });\n }\n else if (error.name === 'SecurityError') {\n const effectiveDomain = globalThis.location.hostname;\n if (!isValidDomain(effectiveDomain)) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 7)\n return new WebAuthnError({\n message: `${globalThis.location.hostname} is an invalid domain`,\n code: 'ERROR_INVALID_DOMAIN',\n cause: error,\n });\n }\n else if (publicKey.rp.id !== effectiveDomain) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 8)\n return new WebAuthnError({\n message: `The RP ID \"${publicKey.rp.id}\" is invalid for this domain`,\n code: 'ERROR_INVALID_RP_ID',\n cause: error,\n });\n }\n }\n else if (error.name === 'TypeError') {\n if (publicKey.user.id.byteLength < 1 || publicKey.user.id.byteLength > 64) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 5)\n return new WebAuthnError({\n message: 'User ID was not between 1 and 64 characters',\n code: 'ERROR_INVALID_USER_ID_LENGTH',\n cause: error,\n });\n }\n }\n else if (error.name === 'UnknownError') {\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 1)\n // https://www.w3.org/TR/webauthn-2/#sctn-op-make-cred (Step 8)\n return new WebAuthnError({\n message: 'The authenticator was unable to process the specified options, or could not create a new credential',\n code: 'ERROR_AUTHENTICATOR_GENERAL_ERROR',\n cause: error,\n });\n }\n return error;\n}\n","class BaseWebAuthnAbortService {\n constructor() {\n Object.defineProperty(this, \"controller\", {\n enumerable: true,\n configurable: true,\n writable: true,\n value: void 0\n });\n }\n createNewAbortSignal() {\n // Abort any existing calls to navigator.credentials.create() or navigator.credentials.get()\n if (this.controller) {\n const abortError = new Error('Cancelling existing WebAuthn API call for new one');\n abortError.name = 'AbortError';\n this.controller.abort(abortError);\n }\n const newController = new AbortController();\n this.controller = newController;\n return newController.signal;\n }\n cancelCeremony() {\n if (this.controller) {\n const abortError = new Error('Manually cancelling existing WebAuthn API call');\n abortError.name = 'AbortError';\n this.controller.abort(abortError);\n this.controller = undefined;\n }\n }\n}\n/**\n * A service singleton to help ensure that only a single WebAuthn ceremony is active at a time.\n *\n * Users of **@simplewebauthn/browser** shouldn't typically need to use this, but it can help e.g.\n * developers building projects that use client-side routing to better control the behavior of\n * their UX in response to router navigation events.\n */\nexport const WebAuthnAbortService = new BaseWebAuthnAbortService();\n","const attachments = ['cross-platform', 'platform'];\n/**\n * If possible coerce a `string` value into a known `AuthenticatorAttachment`\n */\nexport function toAuthenticatorAttachment(attachment) {\n if (!attachment) {\n return;\n }\n if (attachments.indexOf(attachment) < 0) {\n return;\n }\n return attachment;\n}\n","import { bufferToBase64URLString } from '../helpers/bufferToBase64URLString.js';\nimport { base64URLStringToBuffer } from '../helpers/base64URLStringToBuffer.js';\nimport { browserSupportsWebAuthn } from '../helpers/browserSupportsWebAuthn.js';\nimport { toPublicKeyCredentialDescriptor } from '../helpers/toPublicKeyCredentialDescriptor.js';\nimport { identifyRegistrationError } from '../helpers/identifyRegistrationError.js';\nimport { WebAuthnAbortService } from '../helpers/webAuthnAbortService.js';\nimport { toAuthenticatorAttachment } from '../helpers/toAuthenticatorAttachment.js';\n/**\n * Begin authenticator \"registration\" via WebAuthn attestation\n *\n * @param optionsJSON Output from **@simplewebauthn/server**'s `generateRegistrationOptions()`\n * @param useAutoRegister (Optional) Try to silently create a passkey with the password manager that the user just signed in with. Defaults to `false`.\n */\nexport async function startRegistration(options) {\n // @ts-ignore: Intentionally check for old call structure to warn about improper API call\n if (!options.optionsJSON && options.challenge) {\n console.warn('startRegistration() was not called correctly. It will try to continue with the provided options, but this call should be refactored to use the expected call structure instead. See https://simplewebauthn.dev/docs/packages/browser#typeerror-cannot-read-properties-of-undefined-reading-challenge for more information.');\n // @ts-ignore: Reassign the options, passed in as a positional argument, to the expected variable\n options = { optionsJSON: options };\n }\n const { optionsJSON, useAutoRegister = false } = options;\n if (!browserSupportsWebAuthn()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n // We need to convert some values to Uint8Arrays before passing the credentials to the navigator\n const publicKey = {\n ...optionsJSON,\n challenge: base64URLStringToBuffer(optionsJSON.challenge),\n user: {\n ...optionsJSON.user,\n id: base64URLStringToBuffer(optionsJSON.user.id),\n },\n excludeCredentials: optionsJSON.excludeCredentials?.map(toPublicKeyCredentialDescriptor),\n };\n // Prepare options for `.create()`\n const createOptions = {};\n /**\n * Try to use conditional create to register a passkey for the user with the password manager\n * the user just used to authenticate with. The user won't be shown any prominent UI by the\n * browser.\n */\n if (useAutoRegister) {\n // @ts-ignore: `mediation` doesn't yet exist on CredentialCreationOptions but it's possible as of Sept 2024\n createOptions.mediation = 'conditional';\n }\n // Finalize options\n createOptions.publicKey = publicKey;\n // Set up the ability to cancel this request if the user attempts another\n createOptions.signal = WebAuthnAbortService.createNewAbortSignal();\n // Wait for the user to complete attestation\n let credential;\n try {\n credential = (await navigator.credentials.create(createOptions));\n }\n catch (err) {\n throw identifyRegistrationError({ error: err, options: createOptions });\n }\n if (!credential) {\n throw new Error('Registration was not completed');\n }\n const { id, rawId, response, type } = credential;\n // Continue to play it safe with `getTransports()` for now, even when L3 types say it's required\n let transports = undefined;\n if (typeof response.getTransports === 'function') {\n transports = response.getTransports();\n }\n // L3 says this is required, but browser and webview support are still not guaranteed.\n let responsePublicKeyAlgorithm = undefined;\n if (typeof response.getPublicKeyAlgorithm === 'function') {\n try {\n responsePublicKeyAlgorithm = response.getPublicKeyAlgorithm();\n }\n catch (error) {\n warnOnBrokenImplementation('getPublicKeyAlgorithm()', error);\n }\n }\n let responsePublicKey = undefined;\n if (typeof response.getPublicKey === 'function') {\n try {\n const _publicKey = response.getPublicKey();\n if (_publicKey !== null) {\n responsePublicKey = bufferToBase64URLString(_publicKey);\n }\n }\n catch (error) {\n warnOnBrokenImplementation('getPublicKey()', error);\n }\n }\n // L3 says this is required, but browser and webview support are still not guaranteed.\n let responseAuthenticatorData;\n if (typeof response.getAuthenticatorData === 'function') {\n try {\n responseAuthenticatorData = bufferToBase64URLString(response.getAuthenticatorData());\n }\n catch (error) {\n warnOnBrokenImplementation('getAuthenticatorData()', error);\n }\n }\n return {\n id,\n rawId: bufferToBase64URLString(rawId),\n response: {\n attestationObject: bufferToBase64URLString(response.attestationObject),\n clientDataJSON: bufferToBase64URLString(response.clientDataJSON),\n transports,\n publicKeyAlgorithm: responsePublicKeyAlgorithm,\n publicKey: responsePublicKey,\n authenticatorData: responseAuthenticatorData,\n },\n type,\n clientExtensionResults: credential.getClientExtensionResults(),\n authenticatorAttachment: toAuthenticatorAttachment(credential.authenticatorAttachment),\n };\n}\n/**\n * Visibly warn when we detect an issue related to a passkey provider intercepting WebAuthn API\n * calls\n */\nfunction warnOnBrokenImplementation(methodName, cause) {\n console.warn(`The browser extension that intercepted this WebAuthn API call incorrectly implemented ${methodName}. You should report this error to them.\\n`, cause);\n}\n","import { browserSupportsWebAuthn } from './browserSupportsWebAuthn.js';\n/**\n * Determine if the browser supports conditional UI, so that WebAuthn credentials can\n * be shown to the user in the browser's typical password autofill popup.\n */\nexport function browserSupportsWebAuthnAutofill() {\n if (!browserSupportsWebAuthn()) {\n return _browserSupportsWebAuthnAutofillInternals.stubThis(new Promise((resolve) => resolve(false)));\n }\n /**\n * I don't like the `as unknown` here but there's a `declare var PublicKeyCredential` in\n * TS' DOM lib that's making it difficult for me to just go `as PublicKeyCredentialFuture` as I\n * want. I think I'm fine with this for now since it's _supposed_ to be temporary, until TS types\n * have a chance to catch up.\n */\n const globalPublicKeyCredential = globalThis\n .PublicKeyCredential;\n if (globalPublicKeyCredential?.isConditionalMediationAvailable === undefined) {\n return _browserSupportsWebAuthnAutofillInternals.stubThis(new Promise((resolve) => resolve(false)));\n }\n return _browserSupportsWebAuthnAutofillInternals.stubThis(globalPublicKeyCredential.isConditionalMediationAvailable());\n}\n// Make it possible to stub the return value during testing\nexport const _browserSupportsWebAuthnAutofillInternals = {\n stubThis: (value) => value,\n};\n","import { isValidDomain } from './isValidDomain.js';\nimport { WebAuthnError } from './webAuthnError.js';\n/**\n * Attempt to intuit _why_ an error was raised after calling `navigator.credentials.get()`\n */\nexport function identifyAuthenticationError({ error, options, }) {\n const { publicKey } = options;\n if (!publicKey) {\n throw Error('options was missing required publicKey property');\n }\n if (error.name === 'AbortError') {\n if (options.signal instanceof AbortSignal) {\n // https://www.w3.org/TR/webauthn-2/#sctn-createCredential (Step 16)\n return new WebAuthnError({\n message: 'Authentication ceremony was sent an abort signal',\n code: 'ERROR_CEREMONY_ABORTED',\n cause: error,\n });\n }\n }\n else if (error.name === 'NotAllowedError') {\n /**\n * Pass the error directly through. Platforms are overloading this error beyond what the spec\n * defines and we don't want to overwrite potentially useful error messages.\n */\n return new WebAuthnError({\n message: error.message,\n code: 'ERROR_PASSTHROUGH_SEE_CAUSE_PROPERTY',\n cause: error,\n });\n }\n else if (error.name === 'SecurityError') {\n const effectiveDomain = globalThis.location.hostname;\n if (!isValidDomain(effectiveDomain)) {\n // https://www.w3.org/TR/webauthn-2/#sctn-discover-from-external-source (Step 5)\n return new WebAuthnError({\n message: `${globalThis.location.hostname} is an invalid domain`,\n code: 'ERROR_INVALID_DOMAIN',\n cause: error,\n });\n }\n else if (publicKey.rpId !== effectiveDomain) {\n // https://www.w3.org/TR/webauthn-2/#sctn-discover-from-external-source (Step 6)\n return new WebAuthnError({\n message: `The RP ID \"${publicKey.rpId}\" is invalid for this domain`,\n code: 'ERROR_INVALID_RP_ID',\n cause: error,\n });\n }\n }\n else if (error.name === 'UnknownError') {\n // https://www.w3.org/TR/webauthn-2/#sctn-op-get-assertion (Step 1)\n // https://www.w3.org/TR/webauthn-2/#sctn-op-get-assertion (Step 12)\n return new WebAuthnError({\n message: 'The authenticator was unable to process the specified options, or could not create a new assertion signature',\n code: 'ERROR_AUTHENTICATOR_GENERAL_ERROR',\n cause: error,\n });\n }\n return error;\n}\n","import { bufferToBase64URLString } from '../helpers/bufferToBase64URLString.js';\nimport { base64URLStringToBuffer } from '../helpers/base64URLStringToBuffer.js';\nimport { browserSupportsWebAuthn } from '../helpers/browserSupportsWebAuthn.js';\nimport { browserSupportsWebAuthnAutofill } from '../helpers/browserSupportsWebAuthnAutofill.js';\nimport { toPublicKeyCredentialDescriptor } from '../helpers/toPublicKeyCredentialDescriptor.js';\nimport { identifyAuthenticationError } from '../helpers/identifyAuthenticationError.js';\nimport { WebAuthnAbortService } from '../helpers/webAuthnAbortService.js';\nimport { toAuthenticatorAttachment } from '../helpers/toAuthenticatorAttachment.js';\n/**\n * Begin authenticator \"login\" via WebAuthn assertion\n *\n * @param optionsJSON Output from **@simplewebauthn/server**'s `generateAuthenticationOptions()`\n * @param useBrowserAutofill (Optional) Initialize conditional UI to enable logging in via browser autofill prompts. Defaults to `false`.\n * @param verifyBrowserAutofillInput (Optional) Ensure a suitable `<input>` element is present when `useBrowserAutofill` is `true`. Defaults to `true`.\n */\nexport async function startAuthentication(options) {\n // @ts-ignore: Intentionally check for old call structure to warn about improper API call\n if (!options.optionsJSON && options.challenge) {\n console.warn('startAuthentication() was not called correctly. It will try to continue with the provided options, but this call should be refactored to use the expected call structure instead. See https://simplewebauthn.dev/docs/packages/browser#typeerror-cannot-read-properties-of-undefined-reading-challenge for more information.');\n // @ts-ignore: Reassign the options, passed in as a positional argument, to the expected variable\n options = { optionsJSON: options };\n }\n const { optionsJSON, useBrowserAutofill = false, verifyBrowserAutofillInput = true, } = options;\n if (!browserSupportsWebAuthn()) {\n throw new Error('WebAuthn is not supported in this browser');\n }\n // We need to avoid passing empty array to avoid blocking retrieval\n // of public key\n let allowCredentials;\n if (optionsJSON.allowCredentials?.length !== 0) {\n allowCredentials = optionsJSON.allowCredentials?.map(toPublicKeyCredentialDescriptor);\n }\n // We need to convert some values to Uint8Arrays before passing the credentials to the navigator\n const publicKey = {\n ...optionsJSON,\n challenge: base64URLStringToBuffer(optionsJSON.challenge),\n allowCredentials,\n };\n // Prepare options for `.get()`\n const getOptions = {};\n /**\n * Set up the page to prompt the user to select a credential for authentication via the browser's\n * input autofill mechanism.\n */\n if (useBrowserAutofill) {\n if (!(await browserSupportsWebAuthnAutofill())) {\n throw Error('Browser does not support WebAuthn autofill');\n }\n // Check for an <input> with \"webauthn\" in its `autocomplete` attribute\n const eligibleInputs = document.querySelectorAll(\"input[autocomplete$='webauthn']\");\n // WebAuthn autofill requires at least one valid input\n if (eligibleInputs.length < 1 && verifyBrowserAutofillInput) {\n throw Error('No <input> with \"webauthn\" as the only or last value in its `autocomplete` attribute was detected');\n }\n // `CredentialMediationRequirement` doesn't know about \"conditional\" yet as of\n // typescript@4.6.3\n getOptions.mediation = 'conditional';\n // Conditional UI requires an empty allow list\n publicKey.allowCredentials = [];\n }\n // Finalize options\n getOptions.publicKey = publicKey;\n // Set up the ability to cancel this request if the user attempts another\n getOptions.signal = WebAuthnAbortService.createNewAbortSignal();\n // Wait for the user to complete assertion\n let credential;\n try {\n credential = (await navigator.credentials.get(getOptions));\n }\n catch (err) {\n throw identifyAuthenticationError({ error: err, options: getOptions });\n }\n if (!credential) {\n throw new Error('Authentication was not completed');\n }\n const { id, rawId, response, type } = credential;\n let userHandle = undefined;\n if (response.userHandle) {\n userHandle = bufferToBase64URLString(response.userHandle);\n }\n // Convert values to base64 to make it easier to send back to the server\n return {\n id,\n rawId: bufferToBase64URLString(rawId),\n response: {\n authenticatorData: bufferToBase64URLString(response.authenticatorData),\n clientDataJSON: bufferToBase64URLString(response.clientDataJSON),\n signature: bufferToBase64URLString(response.signature),\n userHandle,\n },\n type,\n clientExtensionResults: credential.getClientExtensionResults(),\n authenticatorAttachment: toAuthenticatorAttachment(credential.authenticatorAttachment),\n };\n}\n","import {\n startAuthentication,\n startRegistration,\n} from '@simplewebauthn/browser';\nimport { mutationOptions, queryOptions } from '@tanstack/react-query';\nimport type { InferResponseType } from 'hono/client';\nimport { client, jsonOk } from '#frontend/libs/api.ts';\nimport { queryKeys } from './keys';\nimport type { AuthResponse } from './session';\n\nexport type PasskeysResponse = InferResponseType<\n (typeof client.api.user.passkeys)['$get'],\n 200\n>;\n\nexport type PasskeyInfo = PasskeysResponse['passkeys'][number];\n\nexport type PasskeySetupVerifyResponse = InferResponseType<\n (typeof client.api.user.passkeys.register.verify)['$post'],\n 200\n>;\n\n/**\n * Get all passkeys for current user\n */\nexport const getPasskeysQueryOptions = queryOptions({\n queryKey: queryKeys.passkeys(),\n queryFn: async () => {\n const res = await client.api.user.passkeys.$get();\n return jsonOk(res);\n },\n});\n\n/**\n * Register a new passkey\n */\nexport type RegisterPasskeyParams = {\n name?: string;\n};\n\nexport const registerPasskeyMutationOptions = mutationOptions({\n mutationFn: async (params: RegisterPasskeyParams) => {\n // Step 1: Get registration options from server\n const optionsRes = await client.api.user.passkeys.register.options.$post();\n const { options } = await jsonOk(optionsRes);\n\n // Step 2: Start WebAuthn registration in browser\n const registrationResponse = await startRegistration({\n optionsJSON: options,\n });\n\n // Step 3: Send registration response to server\n const verifyRes = await client.api.user.passkeys.register.verify.$post({\n json: {\n response: registrationResponse,\n name: params.name,\n },\n });\n return jsonOk(verifyRes);\n },\n});\n\n/**\n * Delete a passkey\n */\nexport type DeletePasskeyParams = {\n id: string;\n};\n\nexport const deletePasskeyMutationOptions = mutationOptions({\n mutationFn: async (params: DeletePasskeyParams) => {\n const res = await client.api.user.passkeys[':id'].$delete({\n param: { id: params.id },\n });\n return jsonOk(res);\n },\n});\n\n/**\n * Rename a passkey\n */\nexport type RenamePasskeyParams = {\n id: string;\n name: string;\n};\n\nexport const renamePasskeyMutationOptions = mutationOptions({\n mutationFn: async (params: RenamePasskeyParams) => {\n const res = await client.api.user.passkeys[':id'].$patch({\n param: { id: params.id },\n json: { name: params.name },\n });\n return jsonOk(res);\n },\n});\n\n/**\n * Authenticate with passkey (supports both passwordless\n * and 2FA)\n *\n * The server automatically determines the mode based on\n * session state:\n * - If pending2FAUser session exists: 2FA mode\n * - Otherwise: Passwordless mode (discoverable\n * credentials)\n */\nexport const authenticateWithPasskeyMutationOptions = mutationOptions({\n mutationFn: async () => {\n // Step 1: Get authentication options from server\n const optionsRes = await client.api.auth.passkey.options.$post();\n const { options } = await jsonOk(optionsRes);\n\n // Step 2: Start WebAuthn authentication in browser\n const authenticationResponse = await startAuthentication({\n optionsJSON: options,\n });\n\n // Step 3: Send authentication response to server\n const verifyRes = await client.api.auth.passkey.verify.$post({\n json: {\n response: {\n ...authenticationResponse,\n clientExtensionResults: {\n ...authenticationResponse.clientExtensionResults,\n },\n },\n },\n });\n return jsonOk(verifyRes);\n },\n});\n\n/**\n * Start conditional passkey authentication (browser\n * autofill / Conditional UI)\n *\n * This enables the browser to suggest passkeys in the\n * autofill dropdown when the user focuses on an input\n * with autocomplete=\"username webauthn\".\n *\n * Should be called on page load and runs in the background\n * until:\n * - User selects a passkey from the autofill dropdown\n * - The AbortController is aborted (e.g., on component\n * unmount)\n *\n * @param onSuccess - Callback when passkey authentication\n * succeeds\n * @param abortSignal - AbortSignal to cancel the\n * conditional authentication\n */\nexport const startConditionalPasskeyAuth = async (\n onSuccess: (data: AuthResponse) => void,\n abortSignal: AbortSignal,\n): Promise<void> => {\n try {\n // Step 1: Get authentication options from server\n const optionsRes = await client.api.auth.passkey.options.$post(\n {},\n { init: { signal: abortSignal } },\n );\n const { options } = await jsonOk(optionsRes);\n\n // Step 2: Start WebAuthn with conditional mediation\n const authenticationResponse = await startAuthentication({\n optionsJSON: options,\n useBrowserAutofill: true,\n });\n\n // Step 3: Send authentication response to server\n const verifyRes = await client.api.auth.passkey.verify.$post(\n {\n json: {\n response: {\n ...authenticationResponse,\n clientExtensionResults: {\n ...authenticationResponse.clientExtensionResults,\n },\n },\n },\n },\n { init: { signal: abortSignal } },\n );\n const data = await jsonOk(verifyRes);\n onSuccess(data);\n } catch (error) {\n // AbortError is expected when component unmounts\n if (error instanceof Error && error.name !== 'AbortError') {\n // Non-abort error - could be handled here in the future\n }\n }\n};\n"],"x_google_ignoreList":[0,1,2,3,4,5,6,7,8,9,10,11,12],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import{o as e,r as t,s as n,u as r}from"./IconBase.es-d5KP98Ac.js";import{Q as i,a,f as o,h as s,i as c,n as l,ut as u}from"./use-theme-cVUDAjtt.js";import{t as d}from"./useMutation-DVMopbtG.js";import{t as f}from"./mutationOptions-Dfvzj6n2.js";import{t as p}from"./page-layout-C475gs09.js";import{t as m}from"./EnvelopeSimple.es-BZ7u3LYh.js";import{t as h}from"./Lock.es-Cb_uwQly.js";import{c as g,i as _,o as v}from"./zod-BItJDQBQ.js";import{A as y,M as b,S as x,j as S,s as C}from"./index-CsT6OVnP.js";import{t as w}from"./page-header-BYMFSGfT.js";import{t as T}from"./promise-OpBtq8tG.js";import{o as E}from"./passkey-e6uvApHa.js";import{r as D,t as O}from"./standard-schema-o4V-s4uY.js";import{t as k}from"./footer-link-Ib1Hd-fr.js";import{t as A}from"./icon-input-8iU7PNzd.js";import{t as j}from"./submit-button-Xx6DwLyh.js";var M=r(n()),N=f({mutationFn:async e=>a(await c.api.auth.login.$post({json:e}))}),P=e();function F(){let{t:e,i18n:n}=t(),r=s(),a=u(),c=C.useSearch(),f=c.lang??n.language,{data:F}=i(l),I=F.branding.title?.[f]??F.branding.title?.[F.i18n.fallback_language],L=F.branding.subtitle?.[f]??F.branding.subtitle?.[F.i18n.fallback_language],R=F.auth.password.enabled,z=F.auth.passkey.enabled,B=(0,M.useMemo)(()=>v({email:_(e(`validation.email.invalid`)),password:g().min(1,e(`validation.password.required`))}),[e]),V=d({...N,onSuccess:async(e,t)=>{let n=e.user;if(n.email_verification_required&&!n.email_verified){r.navigate({to:`/verify/email`,search:{email:t.email,...S(c)}});return}a.setQueryData(x.queryKey,{user:n}),await T();let i=[];if(F.auth.password.totp.enabled&&n.totp_registered&&i.push(`totp`),F.auth.passkey.enabled&&n.passkey_count>0&&i.push(`passkey`),n.second_factor_required&&i.length===0){if(F.available_2fa_setup_methods.length>1)return r.navigate({to:`/setup/2fa`,search:S(c)});if(F.available_2fa_setup_methods.length===1)return F.available_2fa_setup_methods[0]===`totp`?r.navigate({to:`/setup/totp`,search:S(c)}):r.navigate({to:`/setup/passkey`,search:{...S(c),passkey_name:`default`}})}if(i.length>1)return r.navigate({to:`/verify/2fa`,search:S(c)});if(i.length===1)return i[0]===`totp`?r.navigate({to:`/verify/totp`,search:S(c)}):r.navigate({to:`/verify/passkey`,search:S(c)});if(b(c))window.location.href=y(c);else return r.navigate({to:`/profile`})},onError:t=>{G(`email`,{type:`manual`,message:e(`login.error.failed`)})},onSettled:()=>{a.invalidateQueries({queryKey:x.queryKey})}}),H=(0,M.useCallback)(async e=>{e.user&&(a.setQueryData(x.queryKey,e),await T(),b(c)?window.location.href=y(c):r.navigate({to:`/profile`})),a.invalidateQueries({queryKey:x.queryKey})},[a,r,c]),U=(0,M.useRef)(null);(0,M.useEffect)(()=>{if(!(!z||!R))return U.current=new AbortController,E(H,U.current.signal),()=>{U.current?.abort()}},[z,R,H]);let{register:W,setError:G,handleSubmit:K,formState:{errors:q}}=D({defaultValues:{email:``,password:``},resolver:O(B)});return(0,P.jsxs)(p,{cardPadding:!0,maxWidth:`100`,children:[(0,P.jsx)(w,{iconUrl:F.branding.icon_url,subtitle:L??e(`login.selectMethod.subtitle`),title:I??e(`login.title`)}),R&&(0,P.jsxs)(`form`,{className:`flex flex-col gap-4`,onSubmit:K(async e=>{V.mutate(e)}),children:[(0,P.jsx)(A,{autoComplete:`username webauthn`,error:q.email,icon:m,placeholder:e(`login.email.placeholder`),...W(`email`),type:`email`}),(0,P.jsx)(A,{autoComplete:`current-password`,error:q.password,icon:h,placeholder:e(`login.password.placeholder`),...W(`password`),type:`password`}),F.email.enabled&&(0,P.jsx)(`div`,{className:`flex items-center justify-end`,children:(0,P.jsx)(o,{className:`link text-sm`,to:`/password/forgot`,children:e(`login.link.forgotPassword`)})}),(0,P.jsx)(j,{className:`mt-2`,isPending:V.isPending,pendingText:e(`login.submitting`),children:e(`login.submit`)})]}),F.registration.public_registration&&(0,P.jsx)(k,{as:o,linkText:e(`login.link.register`),search:S(c),text:e(`login.footer.noAccount`),to:`/register`})]})}export{F as component};
2
+ //# sourceMappingURL=password-CkeV4qxb.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password-CkeV4qxb.js","names":["standardSchemaResolver","EnvelopeSimpleIcon","LockIcon","useMutation","useQueryClient","useSuspenseQuery","Link","useRouter","useCallback","useEffect","useMemo","useRef","useForm","useTranslation","z","FooterLink","IconInput","PageHeader","SubmitButton","PageLayout","buildAuthorizeUrl","extractOAuthParams","isOAuthFlow","tick","appConfigQueryOptions","loginMutationOptions","startConditionalPasskeyAuth","getSessionQueryOptions","Route","LoginPassword","t","i18n","router","queryClient","search","useSearch","lang","language","data","configData","customTitle","branding","title","fallback_language","customSubtitle","subtitle","isPasswordAuthEnabled","auth","password","enabled","isPasskeyEnabled","passkey","loginSchema","object","email","string","min","loginMutation","onSuccess","params","user","email_verification_required","email_verified","navigate","to","setQueryData","queryKey","registered_2fa_methods","SecondFactorMethod","totp","totp_registered","push","passkey_count","second_factor_required","length","available_2fa_setup_methods","method","passkey_name","window","location","href","onError","_error","setError","type","message","onSettled","invalidateQueries","handlePasskeySuccess","AuthResponse","abortControllerRef","AbortController","current","signal","abort","register","handleSubmit","formState","errors","defaultValues","resolver","onSubmit","values","infer","mutate","icon_url","isPending","registration","public_registration","component"],"sources":["../../../frontend/src/queries/login.ts","../../../frontend/src/routes/login/password/index.tsx?tsr-split=component"],"sourcesContent":["import { mutationOptions } from '@tanstack/react-query';\nimport type { InferRequestType, InferResponseType } from 'hono/client';\nimport { client, jsonOk } from '#frontend/libs/api.ts';\n\nexport type LoginParams = InferRequestType<\n (typeof client.api.auth.login)['$post']\n>['json'];\n\nexport type LoginResponse = InferResponseType<\n (typeof client.api.auth.login)['$post'],\n 200\n>;\n\nexport const loginMutationOptions = mutationOptions({\n mutationFn: async (values: LoginParams) => {\n const res = await client.api.auth.login.$post({\n json: values,\n });\n return jsonOk(res);\n },\n});\n","import { standardSchemaResolver } from '@hookform/resolvers/standard-schema';\nimport { EnvelopeSimpleIcon, LockIcon } from '@phosphor-icons/react';\nimport {\n useMutation,\n useQueryClient,\n useSuspenseQuery,\n} from '@tanstack/react-query';\nimport { createFileRoute, Link, useRouter } from '@tanstack/react-router';\nimport { useCallback, useEffect, useMemo, useRef } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { useTranslation } from 'react-i18next';\nimport { z } from 'zod';\nimport { FooterLink } from '#frontend/components/auth/footer-link.tsx';\nimport { IconInput } from '#frontend/components/auth/icon-input.tsx';\nimport { PageHeader } from '#frontend/components/auth/page-header.tsx';\nimport { SubmitButton } from '#frontend/components/auth/submit-button.tsx';\nimport { RouteErrorFallback } from '#frontend/components/ui/route-error-fallback.tsx';\nimport { PageLayout } from '#frontend/features/layout/page-layout.tsx';\nimport {\n buildAuthorizeUrl,\n extractOAuthParams,\n isOAuthFlow,\n OAuthSearchSchema,\n type SecondFactorMethod,\n} from '#frontend/libs/oauth-search.ts';\nimport { tick } from '#frontend/libs/promise.ts';\nimport { appConfigQueryOptions } from '#frontend/queries/config.ts';\nimport { loginMutationOptions } from '#frontend/queries/login.ts';\nimport { startConditionalPasskeyAuth } from '#frontend/queries/passkey.ts';\nimport {\n type AuthResponse,\n getSessionQueryOptions,\n} from '#frontend/queries/session.ts';\n\nconst SearchSchema = OAuthSearchSchema;\n\nexport const Route = createFileRoute('/login/password/')({\n component: LoginPassword,\n errorComponent: RouteErrorFallback,\n validateSearch: SearchSchema,\n loader: async ({ context }) => {\n await context.queryClient.ensureQueryData(appConfigQueryOptions);\n },\n});\n\nfunction LoginPassword() {\n const { t, i18n } = useTranslation();\n const router = useRouter();\n const queryClient = useQueryClient();\n const search = Route.useSearch();\n const lang = search.lang ?? i18n.language;\n\n const { data: configData } = useSuspenseQuery(appConfigQueryOptions);\n\n const customTitle =\n configData.branding.title?.[lang] ??\n configData.branding.title?.[configData.i18n.fallback_language];\n const customSubtitle =\n configData.branding.subtitle?.[lang] ??\n configData.branding.subtitle?.[configData.i18n.fallback_language];\n const isPasswordAuthEnabled = configData.auth.password.enabled;\n const isPasskeyEnabled = configData.auth.passkey.enabled;\n\n const loginSchema = useMemo(\n () =>\n z.object({\n email: z.email(t('validation.email.invalid')),\n password: z.string().min(1, t('validation.password.required')),\n }),\n [t],\n );\n\n const loginMutation = useMutation({\n ...loginMutationOptions,\n onSuccess: async (data, params) => {\n const user = data.user;\n\n if (user.email_verification_required && !user.email_verified) {\n router.navigate({\n to: '/verify/email',\n search: {\n email: params.email,\n ...extractOAuthParams(search),\n },\n });\n return;\n }\n\n queryClient.setQueryData(getSessionQueryOptions.queryKey, {\n user: user,\n });\n await tick();\n\n const registered_2fa_methods: SecondFactorMethod[] = [];\n if (configData.auth.password.totp.enabled && user.totp_registered) {\n registered_2fa_methods.push('totp');\n }\n if (configData.auth.passkey.enabled && user.passkey_count > 0) {\n registered_2fa_methods.push('passkey');\n }\n\n if (user.second_factor_required && registered_2fa_methods.length === 0) {\n if (configData.available_2fa_setup_methods.length > 1) {\n return router.navigate({\n to: '/setup/2fa',\n search: extractOAuthParams(search),\n });\n } else if (configData.available_2fa_setup_methods.length === 1) {\n const method = configData.available_2fa_setup_methods[0];\n if (method === 'totp') {\n return router.navigate({\n to: '/setup/totp',\n search: extractOAuthParams(search),\n });\n } else {\n return router.navigate({\n to: '/setup/passkey',\n search: {\n ...extractOAuthParams(search),\n passkey_name: 'default',\n },\n });\n }\n }\n }\n\n if (registered_2fa_methods.length > 1) {\n return router.navigate({\n to: '/verify/2fa',\n search: extractOAuthParams(search),\n });\n } else if (registered_2fa_methods.length === 1) {\n const method = registered_2fa_methods[0];\n if (method === 'totp') {\n return router.navigate({\n to: '/verify/totp',\n search: extractOAuthParams(search),\n });\n } else {\n return router.navigate({\n to: '/verify/passkey',\n search: extractOAuthParams(search),\n });\n }\n } else {\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n return router.navigate({ to: '/profile' });\n }\n }\n },\n onError: (_error) => {\n setError('email', {\n type: 'manual',\n message: t('login.error.failed'),\n });\n },\n onSettled: () => {\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n });\n\n const handlePasskeySuccess = useCallback(\n async (data: AuthResponse) => {\n if (data.user) {\n queryClient.setQueryData(getSessionQueryOptions.queryKey, data);\n await tick();\n\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n router.navigate({ to: '/profile' });\n }\n }\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n [queryClient, router, search],\n );\n\n // Conditional UI: Start passkey autofill on page load\n const abortControllerRef = useRef<AbortController | null>(null);\n\n useEffect(() => {\n if (!isPasskeyEnabled || !isPasswordAuthEnabled) {\n return;\n }\n\n abortControllerRef.current = new AbortController();\n startConditionalPasskeyAuth(\n handlePasskeySuccess,\n abortControllerRef.current.signal,\n );\n\n return () => {\n abortControllerRef.current?.abort();\n };\n }, [isPasskeyEnabled, isPasswordAuthEnabled, handlePasskeySuccess]);\n\n const {\n register,\n setError,\n handleSubmit,\n formState: { errors },\n } = useForm({\n defaultValues: {\n email: '',\n password: '',\n },\n resolver: standardSchemaResolver(loginSchema),\n });\n\n const onSubmit = async (values: z.infer<typeof loginSchema>) => {\n loginMutation.mutate(values);\n };\n\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n iconUrl={configData.branding.icon_url}\n subtitle={customSubtitle ?? t('login.selectMethod.subtitle')}\n title={customTitle ?? t('login.title')}\n />\n\n {isPasswordAuthEnabled && (\n <form className=\"flex flex-col gap-4\" onSubmit={handleSubmit(onSubmit)}>\n <IconInput\n autoComplete=\"username webauthn\"\n error={errors.email}\n icon={EnvelopeSimpleIcon}\n placeholder={t('login.email.placeholder')}\n {...register('email')}\n type=\"email\"\n />\n\n <IconInput\n autoComplete=\"current-password\"\n error={errors.password}\n icon={LockIcon}\n placeholder={t('login.password.placeholder')}\n {...register('password')}\n type=\"password\"\n />\n\n {configData.email.enabled && (\n <div className=\"flex items-center justify-end\">\n <Link className=\"link text-sm\" to=\"/password/forgot\">\n {t('login.link.forgotPassword')}\n </Link>\n </div>\n )}\n\n <SubmitButton\n className=\"mt-2\"\n isPending={loginMutation.isPending}\n pendingText={t('login.submitting')}\n >\n {t('login.submit')}\n </SubmitButton>\n </form>\n )}\n\n {configData.registration.public_registration && (\n <FooterLink\n as={Link}\n linkText={t('login.link.register')}\n search={extractOAuthParams(search)}\n text={t('login.footer.noAccount')}\n to=\"/register\"\n />\n )}\n\n {/* <div className=\"flex flex-col items-center gap-2\">\n <Link\n to=\"/login\"\n search={extractOAuthParams(search)}\n className=\"link text-sm\"\n >\n {t('login.password.backToMethods')}\n </Link>\n </div> */}\n </PageLayout>\n );\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import{N as e}from"./index-CsT6OVnP.js";var t=e;export{t as errorComponent};
2
+ //# sourceMappingURL=password-SSKfXB7c.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password-SSKfXB7c.js","names":["RouteErrorFallback","SplitErrorComponent","errorComponent"],"sources":["../../../frontend/src/routes/login/password/index.tsx?tsr-split=errorComponent"],"sourcesContent":["import { standardSchemaResolver } from '@hookform/resolvers/standard-schema';\nimport { EnvelopeSimpleIcon, LockIcon } from '@phosphor-icons/react';\nimport {\n useMutation,\n useQueryClient,\n useSuspenseQuery,\n} from '@tanstack/react-query';\nimport { createFileRoute, Link, useRouter } from '@tanstack/react-router';\nimport { useCallback, useEffect, useMemo, useRef } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { useTranslation } from 'react-i18next';\nimport { z } from 'zod';\nimport { FooterLink } from '#frontend/components/auth/footer-link.tsx';\nimport { IconInput } from '#frontend/components/auth/icon-input.tsx';\nimport { PageHeader } from '#frontend/components/auth/page-header.tsx';\nimport { SubmitButton } from '#frontend/components/auth/submit-button.tsx';\nimport { RouteErrorFallback } from '#frontend/components/ui/route-error-fallback.tsx';\nimport { PageLayout } from '#frontend/features/layout/page-layout.tsx';\nimport {\n buildAuthorizeUrl,\n extractOAuthParams,\n isOAuthFlow,\n OAuthSearchSchema,\n type SecondFactorMethod,\n} from '#frontend/libs/oauth-search.ts';\nimport { tick } from '#frontend/libs/promise.ts';\nimport { appConfigQueryOptions } from '#frontend/queries/config.ts';\nimport { loginMutationOptions } from '#frontend/queries/login.ts';\nimport { startConditionalPasskeyAuth } from '#frontend/queries/passkey.ts';\nimport {\n type AuthResponse,\n getSessionQueryOptions,\n} from '#frontend/queries/session.ts';\n\nconst SearchSchema = OAuthSearchSchema;\n\nexport const Route = createFileRoute('/login/password/')({\n component: LoginPassword,\n errorComponent: RouteErrorFallback,\n validateSearch: SearchSchema,\n loader: async ({ context }) => {\n await context.queryClient.ensureQueryData(appConfigQueryOptions);\n },\n});\n\nfunction LoginPassword() {\n const { t, i18n } = useTranslation();\n const router = useRouter();\n const queryClient = useQueryClient();\n const search = Route.useSearch();\n const lang = search.lang ?? i18n.language;\n\n const { data: configData } = useSuspenseQuery(appConfigQueryOptions);\n\n const customTitle =\n configData.branding.title?.[lang] ??\n configData.branding.title?.[configData.i18n.fallback_language];\n const customSubtitle =\n configData.branding.subtitle?.[lang] ??\n configData.branding.subtitle?.[configData.i18n.fallback_language];\n const isPasswordAuthEnabled = configData.auth.password.enabled;\n const isPasskeyEnabled = configData.auth.passkey.enabled;\n\n const loginSchema = useMemo(\n () =>\n z.object({\n email: z.email(t('validation.email.invalid')),\n password: z.string().min(1, t('validation.password.required')),\n }),\n [t],\n );\n\n const loginMutation = useMutation({\n ...loginMutationOptions,\n onSuccess: async (data, params) => {\n const user = data.user;\n\n if (user.email_verification_required && !user.email_verified) {\n router.navigate({\n to: '/verify/email',\n search: {\n email: params.email,\n ...extractOAuthParams(search),\n },\n });\n return;\n }\n\n queryClient.setQueryData(getSessionQueryOptions.queryKey, {\n user: user,\n });\n await tick();\n\n const registered_2fa_methods: SecondFactorMethod[] = [];\n if (configData.auth.password.totp.enabled && user.totp_registered) {\n registered_2fa_methods.push('totp');\n }\n if (configData.auth.passkey.enabled && user.passkey_count > 0) {\n registered_2fa_methods.push('passkey');\n }\n\n if (user.second_factor_required && registered_2fa_methods.length === 0) {\n if (configData.available_2fa_setup_methods.length > 1) {\n return router.navigate({\n to: '/setup/2fa',\n search: extractOAuthParams(search),\n });\n } else if (configData.available_2fa_setup_methods.length === 1) {\n const method = configData.available_2fa_setup_methods[0];\n if (method === 'totp') {\n return router.navigate({\n to: '/setup/totp',\n search: extractOAuthParams(search),\n });\n } else {\n return router.navigate({\n to: '/setup/passkey',\n search: {\n ...extractOAuthParams(search),\n passkey_name: 'default',\n },\n });\n }\n }\n }\n\n if (registered_2fa_methods.length > 1) {\n return router.navigate({\n to: '/verify/2fa',\n search: extractOAuthParams(search),\n });\n } else if (registered_2fa_methods.length === 1) {\n const method = registered_2fa_methods[0];\n if (method === 'totp') {\n return router.navigate({\n to: '/verify/totp',\n search: extractOAuthParams(search),\n });\n } else {\n return router.navigate({\n to: '/verify/passkey',\n search: extractOAuthParams(search),\n });\n }\n } else {\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n return router.navigate({ to: '/profile' });\n }\n }\n },\n onError: (_error) => {\n setError('email', {\n type: 'manual',\n message: t('login.error.failed'),\n });\n },\n onSettled: () => {\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n });\n\n const handlePasskeySuccess = useCallback(\n async (data: AuthResponse) => {\n if (data.user) {\n queryClient.setQueryData(getSessionQueryOptions.queryKey, data);\n await tick();\n\n if (isOAuthFlow(search)) {\n window.location.href = buildAuthorizeUrl(search);\n } else {\n router.navigate({ to: '/profile' });\n }\n }\n queryClient.invalidateQueries({\n queryKey: getSessionQueryOptions.queryKey,\n });\n },\n [queryClient, router, search],\n );\n\n // Conditional UI: Start passkey autofill on page load\n const abortControllerRef = useRef<AbortController | null>(null);\n\n useEffect(() => {\n if (!isPasskeyEnabled || !isPasswordAuthEnabled) {\n return;\n }\n\n abortControllerRef.current = new AbortController();\n startConditionalPasskeyAuth(\n handlePasskeySuccess,\n abortControllerRef.current.signal,\n );\n\n return () => {\n abortControllerRef.current?.abort();\n };\n }, [isPasskeyEnabled, isPasswordAuthEnabled, handlePasskeySuccess]);\n\n const {\n register,\n setError,\n handleSubmit,\n formState: { errors },\n } = useForm({\n defaultValues: {\n email: '',\n password: '',\n },\n resolver: standardSchemaResolver(loginSchema),\n });\n\n const onSubmit = async (values: z.infer<typeof loginSchema>) => {\n loginMutation.mutate(values);\n };\n\n return (\n <PageLayout cardPadding maxWidth=\"100\">\n <PageHeader\n iconUrl={configData.branding.icon_url}\n subtitle={customSubtitle ?? t('login.selectMethod.subtitle')}\n title={customTitle ?? t('login.title')}\n />\n\n {isPasswordAuthEnabled && (\n <form className=\"flex flex-col gap-4\" onSubmit={handleSubmit(onSubmit)}>\n <IconInput\n autoComplete=\"username webauthn\"\n error={errors.email}\n icon={EnvelopeSimpleIcon}\n placeholder={t('login.email.placeholder')}\n {...register('email')}\n type=\"email\"\n />\n\n <IconInput\n autoComplete=\"current-password\"\n error={errors.password}\n icon={LockIcon}\n placeholder={t('login.password.placeholder')}\n {...register('password')}\n type=\"password\"\n />\n\n {configData.email.enabled && (\n <div className=\"flex items-center justify-end\">\n <Link className=\"link text-sm\" to=\"/password/forgot\">\n {t('login.link.forgotPassword')}\n </Link>\n </div>\n )}\n\n <SubmitButton\n className=\"mt-2\"\n isPending={loginMutation.isPending}\n pendingText={t('login.submitting')}\n >\n {t('login.submit')}\n </SubmitButton>\n </form>\n )}\n\n {configData.registration.public_registration && (\n <FooterLink\n as={Link}\n linkText={t('login.link.register')}\n search={extractOAuthParams(search)}\n text={t('login.footer.noAccount')}\n to=\"/register\"\n />\n )}\n\n {/* <div className=\"flex flex-col items-center gap-2\">\n <Link\n to=\"/login\"\n search={extractOAuthParams(search)}\n className=\"link text-sm\"\n >\n {t('login.password.backToMethods')}\n </Link>\n </div> */}\n </PageLayout>\n );\n}\n"],"mappings":"wCAgBsF,IAAAC,EAA7ED"}
@@ -0,0 +1,2 @@
1
+ import{a as e,i as t}from"./use-theme-cVUDAjtt.js";import{t as n}from"./mutationOptions-Dfvzj6n2.js";var r=n({mutationFn:async n=>e(await t.api.auth.password.forgot.$post({json:n,header:{}}))}),i=n({mutationFn:async n=>e(await t.api.auth.password.reset.$post({json:n}))});export{i as n,r as t};
2
+ //# sourceMappingURL=password-reset-XZJTgJi3.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"password-reset-XZJTgJi3.js","names":[],"sources":["../../../frontend/src/queries/password-reset.ts"],"sourcesContent":["import { mutationOptions } from '@tanstack/react-query';\nimport type { InferRequestType, InferResponseType } from 'hono/client';\nimport { client, jsonOk } from '#frontend/libs/api.ts';\n\nexport type ForgotPasswordParams = InferRequestType<\n (typeof client.api.auth.password.forgot)['$post']\n>['json'];\n\nexport type ForgotPasswordResponse = InferResponseType<\n (typeof client.api.auth.password.forgot)['$post'],\n 200\n>;\n\nexport const forgotPasswordMutationOptions = mutationOptions({\n mutationFn: async (values: ForgotPasswordParams) => {\n const res = await client.api.auth.password.forgot.$post({\n json: values,\n header: {},\n });\n return jsonOk(res);\n },\n});\n\nexport type ResetPasswordParams = InferRequestType<\n (typeof client.api.auth.password.reset)['$post']\n>['json'];\n\nexport type ResetPasswordResponse = InferResponseType<\n (typeof client.api.auth.password.reset)['$post'],\n 200\n>;\n\nexport const resetPasswordMutationOptions = mutationOptions({\n mutationFn: async (values: ResetPasswordParams) => {\n const res = await client.api.auth.password.reset.$post({\n json: values,\n });\n return jsonOk(res);\n },\n});\n"],"mappings":"qGAaA,IAAa,EAAgC,EAAgB,CAC3D,WAAY,KAAO,IAKV,EAAO,MAJI,EAAO,IAAI,KAAK,SAAS,OAAO,MAAM,CACtD,KAAM,EACN,OAAQ,CAAC,CACX,CAAC,CACgB,CAErB,CAAC,EAWY,EAA+B,EAAgB,CAC1D,WAAY,KAAO,IAIV,EAAO,MAHI,EAAO,IAAI,KAAK,SAAS,MAAM,MAAM,CACrD,KAAM,CACR,CAAC,CACgB,CAErB,CAAC"}
@@ -0,0 +1,2 @@
1
+ import{o as e,s as t,u as n}from"./IconBase.es-d5KP98Ac.js";var r=n(t(),1),i=e(),a=({length:e=6,error:t,disabled:n=!1,onComplete:a,onChange:o,value:s=``,autoFocus:c=!1,className:l=``,ref:u})=>{let[d,f]=(0,r.useState)(()=>s.split(``).concat(Array(e).fill(``)).slice(0,e)),p=(0,r.useRef)([]);(0,r.useEffect)(()=>{f(s.split(``).concat(Array(e).fill(``)).slice(0,e))},[s,e]),(0,r.useImperativeHandle)(u,()=>({focus:()=>{p.current[0]?.focus()},clear:()=>{f(Array(e).fill(``)),p.current[0]?.focus()}}),[e]),(0,r.useEffect)(()=>{c&&!n&&p.current[0]?.focus()},[c,n]);let m=(0,r.useCallback)(t=>{let n=t.join(``);o?.(n),t.every(e=>e!==``)&&n.length===e&&a?.(n)},[o,a,e]),h=(0,r.useCallback)((t,n)=>{let r=n.replace(/\D/g,``).slice(-1),i=[...d];i[t]=r,f(i),m(i),r&&t<e-1&&p.current[t+1]?.focus()},[d,e,m]),g=(0,r.useCallback)((t,n)=>{switch(n.key){case`Backspace`:if(n.preventDefault(),d[t]){let e=[...d];e[t]=``,f(e),m(e)}else if(t>0){let e=[...d];e[t-1]=``,f(e),m(e),p.current[t-1]?.focus()}break;case`ArrowLeft`:n.preventDefault(),t>0&&p.current[t-1]?.focus();break;case`ArrowRight`:n.preventDefault(),t<e-1&&p.current[t+1]?.focus();break;case`Delete`:n.preventDefault();{let e=[...d];e[t]=``,f(e),m(e)}break}},[d,e,m]),_=(0,r.useCallback)(t=>{t.preventDefault();let n=t.clipboardData.getData(`text`).replace(/\D/g,``).slice(0,e);if(n){let t=n.split(``).concat(Array(e).fill(``)).slice(0,e);f(t),m(t);let r=t.indexOf(``),i=r===-1?e-1:r;p.current[i]?.focus()}},[e,m]),v=(0,r.useCallback)(e=>{e.target.select()},[]);return(0,i.jsxs)(`div`,{className:l,children:[(0,i.jsx)(`div`,{className:`flex justify-center gap-1.5 sm:gap-2`,children:d.map((r,a)=>(0,i.jsx)(`input`,{"aria-label":`Digit ${a+1} of ${e}`,autoComplete:a===0?`one-time-code`:`off`,className:`input input-bordered h-12 w-10 text-center font-mono text-xl sm:h-14 sm:w-12 sm:text-2xl ${t?`input-error`:``} ${n?`input-disabled`:``}`,disabled:n,inputMode:`numeric`,maxLength:1,onChange:e=>h(a,e.target.value),onFocus:v,onKeyDown:e=>g(a,e),onPaste:_,pattern:`[0-9]*`,ref:e=>{p.current[a]=e},type:`text`,value:r},a))}),t&&(0,i.jsx)(`p`,{className:`mt-2 text-center text-error text-sm`,"data-testid":`pin-input-error`,children:t.message})]})};export{a as t};
2
+ //# sourceMappingURL=pin-input-BM1UizHr.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pin-input-BM1UizHr.js","names":[],"sources":["../../../frontend/src/components/ui/pin-input.tsx"],"sourcesContent":["import {\n useCallback,\n useEffect,\n useImperativeHandle,\n useRef,\n useState,\n} from 'react';\nimport type { FieldError } from 'react-hook-form';\n\nexport type PinInputRef = {\n focus: () => void;\n clear: () => void;\n};\n\ntype PinInputProps = {\n /** Number of digits (default: 6) */\n length?: number;\n /** Error state from react-hook-form */\n error?: FieldError;\n /** Whether the input is disabled */\n disabled?: boolean;\n /** Called when all digits are filled */\n onComplete?: (value: string) => void;\n /** Called when the value changes */\n onChange?: (value: string) => void;\n /** Current value (controlled) */\n value?: string;\n /** Auto focus the first input on mount */\n autoFocus?: boolean;\n /** Additional class name for the container */\n className?: string;\n /** Ref for imperative handle */\n ref?: React.Ref<PinInputRef>;\n};\n\nexport const PinInput = ({\n length = 6,\n error,\n disabled = false,\n onComplete,\n onChange,\n value = '',\n autoFocus = false,\n className = '',\n ref,\n}: PinInputProps) => {\n const [digits, setDigits] = useState<string[]>(() =>\n value.split('').concat(Array(length).fill('')).slice(0, length),\n );\n const inputRefs = useRef<(HTMLInputElement | null)[]>([]);\n\n // Sync internal state with controlled value\n useEffect(() => {\n const newDigits = value\n .split('')\n .concat(Array(length).fill(''))\n .slice(0, length);\n setDigits(newDigits);\n }, [value, length]);\n\n // Expose methods via ref\n useImperativeHandle(\n ref,\n () => ({\n focus: () => {\n inputRefs.current[0]?.focus();\n },\n clear: () => {\n setDigits(Array(length).fill(''));\n inputRefs.current[0]?.focus();\n },\n }),\n [length],\n );\n\n // Auto focus on mount\n useEffect(() => {\n if (autoFocus && !disabled) {\n inputRefs.current[0]?.focus();\n }\n }, [autoFocus, disabled]);\n\n const updateValue = useCallback(\n (newDigits: string[]) => {\n const newValue = newDigits.join('');\n onChange?.(newValue);\n\n if (newDigits.every((d) => d !== '') && newValue.length === length) {\n onComplete?.(newValue);\n }\n },\n [onChange, onComplete, length],\n );\n\n const handleChange = useCallback(\n (index: number, inputValue: string) => {\n // Only accept digits\n const digit = inputValue.replace(/\\D/g, '').slice(-1);\n\n const newDigits = [...digits];\n newDigits[index] = digit;\n setDigits(newDigits);\n updateValue(newDigits);\n\n // Move to next input if digit was entered\n if (digit && index < length - 1) {\n inputRefs.current[index + 1]?.focus();\n }\n },\n [digits, length, updateValue],\n );\n\n const handleKeyDown = useCallback(\n (index: number, e: React.KeyboardEvent<HTMLInputElement>) => {\n switch (e.key) {\n case 'Backspace':\n e.preventDefault();\n if (digits[index]) {\n // Clear current digit\n const newDigits = [...digits];\n newDigits[index] = '';\n setDigits(newDigits);\n updateValue(newDigits);\n } else if (index > 0) {\n // Move to previous input and clear it\n const newDigits = [...digits];\n newDigits[index - 1] = '';\n setDigits(newDigits);\n updateValue(newDigits);\n inputRefs.current[index - 1]?.focus();\n }\n break;\n\n case 'ArrowLeft':\n e.preventDefault();\n if (index > 0) {\n inputRefs.current[index - 1]?.focus();\n }\n break;\n\n case 'ArrowRight':\n e.preventDefault();\n if (index < length - 1) {\n inputRefs.current[index + 1]?.focus();\n }\n break;\n\n case 'Delete':\n e.preventDefault();\n {\n const newDigits = [...digits];\n newDigits[index] = '';\n setDigits(newDigits);\n updateValue(newDigits);\n }\n break;\n }\n },\n [digits, length, updateValue],\n );\n\n const handlePaste = useCallback(\n (e: React.ClipboardEvent<HTMLInputElement>) => {\n e.preventDefault();\n const pastedData = e.clipboardData\n .getData('text')\n .replace(/\\D/g, '')\n .slice(0, length);\n\n if (pastedData) {\n const newDigits = pastedData\n .split('')\n .concat(Array(length).fill(''))\n .slice(0, length);\n setDigits(newDigits);\n updateValue(newDigits);\n\n // Focus the next empty input or the last one\n const nextEmptyIndex = newDigits.indexOf('');\n const focusIndex = nextEmptyIndex === -1 ? length - 1 : nextEmptyIndex;\n inputRefs.current[focusIndex]?.focus();\n }\n },\n [length, updateValue],\n );\n\n const handleFocus = useCallback((e: React.FocusEvent<HTMLInputElement>) => {\n e.target.select();\n }, []);\n\n return (\n <div className={className}>\n <div className=\"flex justify-center gap-1.5 sm:gap-2\">\n {digits.map((digit, index) => (\n <input\n aria-label={`Digit ${index + 1} of ${length}`}\n autoComplete={index === 0 ? 'one-time-code' : 'off'}\n className={`input input-bordered h-12 w-10 text-center font-mono text-xl sm:h-14 sm:w-12 sm:text-2xl ${\n error ? 'input-error' : ''\n } ${disabled ? 'input-disabled' : ''}`}\n disabled={disabled}\n inputMode=\"numeric\"\n key={index}\n maxLength={1}\n onChange={(e) => handleChange(index, e.target.value)}\n onFocus={handleFocus}\n onKeyDown={(e) => handleKeyDown(index, e)}\n onPaste={handlePaste}\n pattern=\"[0-9]*\"\n ref={(el) => {\n inputRefs.current[index] = el;\n }}\n type=\"text\"\n value={digit}\n />\n ))}\n </div>\n {error && (\n <p\n className=\"mt-2 text-center text-error text-sm\"\n data-testid=\"pin-input-error\"\n >\n {error.message}\n </p>\n )}\n </div>\n );\n};\n"],"mappings":"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"}