@atproto/oauth-provider 0.3.1 → 0.5.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 (404) hide show
  1. package/.linguirc +57 -0
  2. package/CHANGELOG.md +29 -0
  3. package/LICENSE.txt +1 -1
  4. package/dist/account/account-manager.d.ts +17 -3
  5. package/dist/account/account-manager.d.ts.map +1 -1
  6. package/dist/account/account-manager.js +102 -8
  7. package/dist/account/account-manager.js.map +1 -1
  8. package/dist/account/account-store.d.ts +81 -15
  9. package/dist/account/account-store.d.ts.map +1 -1
  10. package/dist/account/account-store.js +70 -19
  11. package/dist/account/account-store.js.map +1 -1
  12. package/dist/account/sign-in-data.d.ts +28 -0
  13. package/dist/account/sign-in-data.d.ts.map +1 -0
  14. package/dist/account/sign-in-data.js +16 -0
  15. package/dist/account/sign-in-data.js.map +1 -0
  16. package/dist/account/sign-up-data.d.ts +26 -0
  17. package/dist/account/sign-up-data.d.ts.map +1 -0
  18. package/dist/account/sign-up-data.js +11 -0
  19. package/dist/account/sign-up-data.js.map +1 -0
  20. package/dist/assets/app/bundle-manifest.json +598 -6
  21. package/dist/assets/app/index-ItwwtJ8r.js +36 -0
  22. package/dist/assets/app/index-ItwwtJ8r.js.map +1 -0
  23. package/dist/assets/app/main-B_dNxQo_.js +4 -0
  24. package/dist/assets/app/main-B_dNxQo_.js.map +1 -0
  25. package/dist/assets/app/main-CSatvmRR.css +3 -0
  26. package/dist/assets/app/main-CSatvmRR.js +306 -0
  27. package/dist/assets/app/main-CSatvmRR.js.map +1 -0
  28. package/dist/assets/app/messages-BQeltXSF.js +4 -0
  29. package/dist/assets/app/messages-BQeltXSF.js.map +1 -0
  30. package/dist/assets/app/messages-BQkEhfjg.js +4 -0
  31. package/dist/assets/app/messages-BQkEhfjg.js.map +1 -0
  32. package/dist/assets/app/messages-BUjKj_UJ.js +4 -0
  33. package/dist/assets/app/messages-BUjKj_UJ.js.map +1 -0
  34. package/dist/assets/app/messages-BWIQa8fO.js +4 -0
  35. package/dist/assets/app/messages-BWIQa8fO.js.map +1 -0
  36. package/dist/assets/app/messages-BaNVb0bp.js +4 -0
  37. package/dist/assets/app/messages-BaNVb0bp.js.map +1 -0
  38. package/dist/assets/app/messages-BaizVXcF.js +4 -0
  39. package/dist/assets/app/messages-BaizVXcF.js.map +1 -0
  40. package/dist/assets/app/messages-BfoClA1Y.js +4 -0
  41. package/dist/assets/app/messages-BfoClA1Y.js.map +1 -0
  42. package/dist/assets/app/messages-BsKGDZnC.js +4 -0
  43. package/dist/assets/app/messages-BsKGDZnC.js.map +1 -0
  44. package/dist/assets/app/messages-Bu-TJhml.js +4 -0
  45. package/dist/assets/app/messages-Bu-TJhml.js.map +1 -0
  46. package/dist/assets/app/messages-BvOKnBQk.js +4 -0
  47. package/dist/assets/app/messages-BvOKnBQk.js.map +1 -0
  48. package/dist/assets/app/messages-BxDzCiWz.js +4 -0
  49. package/dist/assets/app/messages-BxDzCiWz.js.map +1 -0
  50. package/dist/assets/app/messages-CDgFOy4S.js +4 -0
  51. package/dist/assets/app/messages-CDgFOy4S.js.map +1 -0
  52. package/dist/assets/app/messages-CLbTz0o9.js +4 -0
  53. package/dist/assets/app/messages-CLbTz0o9.js.map +1 -0
  54. package/dist/assets/app/messages-CNwSh0t7.js +4 -0
  55. package/dist/assets/app/messages-CNwSh0t7.js.map +1 -0
  56. package/dist/assets/app/messages-CSMNJ6P8.js +4 -0
  57. package/dist/assets/app/messages-CSMNJ6P8.js.map +1 -0
  58. package/dist/assets/app/messages-CZQUw3mp.js +4 -0
  59. package/dist/assets/app/messages-CZQUw3mp.js.map +1 -0
  60. package/dist/assets/app/messages-CZT41oVp.js +4 -0
  61. package/dist/assets/app/messages-CZT41oVp.js.map +1 -0
  62. package/dist/assets/app/messages-C_b-d3t8.js +4 -0
  63. package/dist/assets/app/messages-C_b-d3t8.js.map +1 -0
  64. package/dist/assets/app/messages-C_u3MTc2.js +4 -0
  65. package/dist/assets/app/messages-C_u3MTc2.js.map +1 -0
  66. package/dist/assets/app/messages-Cn8nHZic.js +4 -0
  67. package/dist/assets/app/messages-Cn8nHZic.js.map +1 -0
  68. package/dist/assets/app/messages-CtDywJUm.js +4 -0
  69. package/dist/assets/app/messages-CtDywJUm.js.map +1 -0
  70. package/dist/assets/app/messages-CurtIjBF.js +4 -0
  71. package/dist/assets/app/messages-CurtIjBF.js.map +1 -0
  72. package/dist/assets/app/messages-Cv6zIbaP.js +4 -0
  73. package/dist/assets/app/messages-Cv6zIbaP.js.map +1 -0
  74. package/dist/assets/app/messages-D1eLQuPE.js +4 -0
  75. package/dist/assets/app/messages-D1eLQuPE.js.map +1 -0
  76. package/dist/assets/app/messages-D8vHEaYW.js +4 -0
  77. package/dist/assets/app/messages-D8vHEaYW.js.map +1 -0
  78. package/dist/assets/app/messages-DJ1Q4GeC.js +4 -0
  79. package/dist/assets/app/messages-DJ1Q4GeC.js.map +1 -0
  80. package/dist/assets/app/messages-DRL3exqd.js +4 -0
  81. package/dist/assets/app/messages-DRL3exqd.js.map +1 -0
  82. package/dist/assets/app/messages-DWLPQRTp.js +4 -0
  83. package/dist/assets/app/messages-DWLPQRTp.js.map +1 -0
  84. package/dist/assets/app/messages-DjVaE9YE.js +4 -0
  85. package/dist/assets/app/messages-DjVaE9YE.js.map +1 -0
  86. package/dist/assets/app/messages-DqpMfFJR.js +4 -0
  87. package/dist/assets/app/messages-DqpMfFJR.js.map +1 -0
  88. package/dist/assets/app/messages-ETjhJBEN.js +4 -0
  89. package/dist/assets/app/messages-ETjhJBEN.js.map +1 -0
  90. package/dist/assets/app/messages-EUKrgrGn.js +4 -0
  91. package/dist/assets/app/messages-EUKrgrGn.js.map +1 -0
  92. package/dist/assets/app/messages-QQrOUcPW.js +4 -0
  93. package/dist/assets/app/messages-QQrOUcPW.js.map +1 -0
  94. package/dist/assets/app/messages-e2QGqFL6.js +4 -0
  95. package/dist/assets/app/messages-e2QGqFL6.js.map +1 -0
  96. package/dist/assets/app/messages-p61py7gD.js +4 -0
  97. package/dist/assets/app/messages-p61py7gD.js.map +1 -0
  98. package/dist/assets/asset.d.ts +1 -0
  99. package/dist/assets/asset.d.ts.map +1 -1
  100. package/dist/assets/assets-middleware.d.ts.map +1 -1
  101. package/dist/assets/assets-middleware.js +12 -7
  102. package/dist/assets/assets-middleware.js.map +1 -1
  103. package/dist/assets/index.d.ts +3 -2
  104. package/dist/assets/index.d.ts.map +1 -1
  105. package/dist/assets/index.js +13 -1
  106. package/dist/assets/index.js.map +1 -1
  107. package/dist/client/client-store.d.ts +3 -3
  108. package/dist/client/client-store.d.ts.map +1 -1
  109. package/dist/client/client-store.js +6 -5
  110. package/dist/client/client-store.js.map +1 -1
  111. package/dist/device/device-manager.d.ts +12 -13
  112. package/dist/device/device-manager.d.ts.map +1 -1
  113. package/dist/device/device-manager.js +5 -3
  114. package/dist/device/device-manager.js.map +1 -1
  115. package/dist/device/device-store.d.ts +3 -3
  116. package/dist/device/device-store.d.ts.map +1 -1
  117. package/dist/device/device-store.js +10 -9
  118. package/dist/device/device-store.js.map +1 -1
  119. package/dist/dpop/dpop-manager.d.ts +15 -7
  120. package/dist/dpop/dpop-manager.d.ts.map +1 -1
  121. package/dist/dpop/dpop-manager.js +17 -3
  122. package/dist/dpop/dpop-manager.js.map +1 -1
  123. package/dist/dpop/dpop-nonce.d.ts +11 -5
  124. package/dist/dpop/dpop-nonce.d.ts.map +1 -1
  125. package/dist/dpop/dpop-nonce.js +47 -38
  126. package/dist/dpop/dpop-nonce.js.map +1 -1
  127. package/dist/errors/handle-unavailable-error.d.ts +11 -0
  128. package/dist/errors/handle-unavailable-error.d.ts.map +1 -0
  129. package/dist/errors/handle-unavailable-error.js +19 -0
  130. package/dist/errors/handle-unavailable-error.js.map +1 -0
  131. package/dist/errors/invalid-request-error.d.ts +6 -8
  132. package/dist/errors/invalid-request-error.d.ts.map +1 -1
  133. package/dist/errors/invalid-request-error.js +10 -8
  134. package/dist/errors/invalid-request-error.js.map +1 -1
  135. package/dist/lib/csp/index.d.ts +18 -0
  136. package/dist/lib/csp/index.d.ts.map +1 -0
  137. package/dist/lib/csp/index.js +72 -0
  138. package/dist/lib/csp/index.js.map +1 -0
  139. package/dist/lib/hcaptcha.d.ts +177 -0
  140. package/dist/lib/hcaptcha.d.ts.map +1 -0
  141. package/dist/lib/hcaptcha.js +155 -0
  142. package/dist/lib/hcaptcha.js.map +1 -0
  143. package/dist/lib/html/build-document.d.ts +11 -3
  144. package/dist/lib/html/build-document.d.ts.map +1 -1
  145. package/dist/lib/html/build-document.js +51 -15
  146. package/dist/lib/html/build-document.js.map +1 -1
  147. package/dist/lib/http/middleware.d.ts.map +1 -1
  148. package/dist/lib/http/middleware.js +4 -1
  149. package/dist/lib/http/middleware.js.map +1 -1
  150. package/dist/lib/http/request.d.ts +18 -3
  151. package/dist/lib/http/request.d.ts.map +1 -1
  152. package/dist/lib/http/request.js +56 -23
  153. package/dist/lib/http/request.js.map +1 -1
  154. package/dist/lib/http/response.d.ts +4 -2
  155. package/dist/lib/http/response.d.ts.map +1 -1
  156. package/dist/lib/http/response.js +23 -5
  157. package/dist/lib/http/response.js.map +1 -1
  158. package/dist/lib/locale.d.ts +15 -0
  159. package/dist/lib/locale.d.ts.map +1 -0
  160. package/dist/lib/locale.js +17 -0
  161. package/dist/lib/locale.js.map +1 -0
  162. package/dist/lib/util/function.d.ts +2 -2
  163. package/dist/lib/util/function.d.ts.map +1 -1
  164. package/dist/lib/util/function.js.map +1 -1
  165. package/dist/lib/util/type.d.ts +88 -1
  166. package/dist/lib/util/type.d.ts.map +1 -1
  167. package/dist/lib/util/type.js +41 -0
  168. package/dist/lib/util/type.js.map +1 -1
  169. package/dist/metadata/build-metadata.d.ts +2 -2
  170. package/dist/metadata/build-metadata.d.ts.map +1 -1
  171. package/dist/metadata/build-metadata.js.map +1 -1
  172. package/dist/oauth-errors.d.ts +1 -0
  173. package/dist/oauth-errors.d.ts.map +1 -1
  174. package/dist/oauth-errors.js +3 -1
  175. package/dist/oauth-errors.js.map +1 -1
  176. package/dist/oauth-hooks.d.ts +60 -3
  177. package/dist/oauth-hooks.d.ts.map +1 -1
  178. package/dist/oauth-hooks.js +3 -3
  179. package/dist/oauth-hooks.js.map +1 -1
  180. package/dist/oauth-provider.d.ts +28 -22
  181. package/dist/oauth-provider.d.ts.map +1 -1
  182. package/dist/oauth-provider.js +212 -211
  183. package/dist/oauth-provider.js.map +1 -1
  184. package/dist/oauth-verifier.d.ts +1 -1
  185. package/dist/oauth-verifier.d.ts.map +1 -1
  186. package/dist/oauth-verifier.js +2 -1
  187. package/dist/oauth-verifier.js.map +1 -1
  188. package/dist/output/build-authorize-data.d.ts +0 -1
  189. package/dist/output/build-authorize-data.d.ts.map +1 -1
  190. package/dist/output/build-authorize-data.js +0 -1
  191. package/dist/output/build-authorize-data.js.map +1 -1
  192. package/dist/output/build-customization-data.d.ts +232 -0
  193. package/dist/output/build-customization-data.d.ts.map +1 -0
  194. package/dist/output/build-customization-data.js +145 -0
  195. package/dist/output/build-customization-data.js.map +1 -0
  196. package/dist/output/output-manager.d.ts +16 -9
  197. package/dist/output/output-manager.d.ts.map +1 -1
  198. package/dist/output/output-manager.js +78 -42
  199. package/dist/output/output-manager.js.map +1 -1
  200. package/dist/output/send-authorize-redirect.d.ts +9 -6
  201. package/dist/output/send-authorize-redirect.d.ts.map +1 -1
  202. package/dist/output/send-authorize-redirect.js +20 -14
  203. package/dist/output/send-authorize-redirect.js.map +1 -1
  204. package/dist/output/send-web-page.d.ts +7 -2
  205. package/dist/output/send-web-page.d.ts.map +1 -1
  206. package/dist/output/send-web-page.js +37 -21
  207. package/dist/output/send-web-page.js.map +1 -1
  208. package/dist/request/request-manager.d.ts +1 -1
  209. package/dist/request/request-manager.d.ts.map +1 -1
  210. package/dist/request/request-manager.js +4 -4
  211. package/dist/request/request-manager.js.map +1 -1
  212. package/dist/request/request-store.d.ts +3 -3
  213. package/dist/request/request-store.d.ts.map +1 -1
  214. package/dist/request/request-store.js +11 -10
  215. package/dist/request/request-store.js.map +1 -1
  216. package/dist/token/token-store.d.ts +4 -4
  217. package/dist/token/token-store.d.ts.map +1 -1
  218. package/dist/token/token-store.js +13 -12
  219. package/dist/token/token-store.js.map +1 -1
  220. package/package.json +46 -21
  221. package/rollup.config.js +61 -17
  222. package/src/account/account-manager.ts +159 -8
  223. package/src/account/account-store.ts +127 -32
  224. package/src/account/sign-in-data.ts +15 -0
  225. package/src/account/sign-up-data.ts +11 -0
  226. package/src/assets/app/app.tsx +31 -16
  227. package/src/assets/app/backend-data.ts +15 -60
  228. package/src/assets/app/backend-types.ts +66 -0
  229. package/src/assets/app/components/forms/button-toggle-visibility.tsx +43 -0
  230. package/src/assets/app/components/forms/button.tsx +60 -0
  231. package/src/assets/app/components/forms/fieldset.tsx +55 -0
  232. package/src/assets/app/components/forms/form-card-async.tsx +103 -0
  233. package/src/assets/app/components/forms/form-card.tsx +49 -0
  234. package/src/assets/app/components/forms/input-checkbox.tsx +73 -0
  235. package/src/assets/app/components/forms/input-container.tsx +107 -0
  236. package/src/assets/app/components/forms/input-email-address.tsx +66 -0
  237. package/src/assets/app/components/forms/input-new-password.tsx +62 -0
  238. package/src/assets/app/components/forms/input-password.tsx +88 -0
  239. package/src/assets/app/components/forms/input-text.tsx +76 -0
  240. package/src/assets/app/components/forms/input-token.tsx +94 -0
  241. package/src/assets/app/components/forms/wizard-card.tsx +116 -0
  242. package/src/assets/app/components/layouts/layout-title-page.tsx +77 -0
  243. package/src/assets/app/components/layouts/layout-welcome.tsx +73 -0
  244. package/src/assets/app/components/utils/account-identifier.tsx +23 -0
  245. package/src/assets/app/components/utils/account-image.tsx +33 -0
  246. package/src/assets/app/components/utils/admonition.tsx +52 -0
  247. package/src/assets/app/components/utils/client-name.tsx +45 -0
  248. package/src/assets/app/components/utils/error-card.tsx +93 -0
  249. package/src/assets/app/components/utils/error-message.tsx +62 -0
  250. package/src/assets/app/components/utils/help-card.tsx +46 -0
  251. package/src/assets/app/components/utils/icons.tsx +88 -0
  252. package/src/assets/app/components/utils/link-anchor.tsx +28 -0
  253. package/src/assets/app/components/utils/link-title.tsx +26 -0
  254. package/src/assets/app/components/utils/multi-lang-string.tsx +56 -0
  255. package/src/assets/app/components/utils/password-strength-label.tsx +37 -0
  256. package/src/assets/app/components/utils/password-strength-meter.tsx +58 -0
  257. package/src/assets/app/components/{url-viewer.tsx → utils/url-viewer.tsx} +9 -6
  258. package/src/assets/app/hooks/use-api.ts +128 -55
  259. package/src/assets/app/hooks/use-async-action.ts +120 -0
  260. package/src/assets/app/hooks/use-browser-color-scheme.ts +31 -0
  261. package/src/assets/app/hooks/use-csrf-token.ts +1 -1
  262. package/src/assets/app/hooks/use-random-string.ts +37 -0
  263. package/src/assets/app/hooks/use-stepper.ts +87 -0
  264. package/src/assets/app/index.html +182 -0
  265. package/src/assets/app/lib/api.ts +248 -79
  266. package/src/assets/app/lib/clsx.ts +5 -8
  267. package/src/assets/app/lib/json-client.ts +94 -0
  268. package/src/assets/app/lib/password.ts +98 -0
  269. package/src/assets/app/lib/ref.ts +17 -0
  270. package/src/assets/app/locales/an/messages.po +492 -0
  271. package/src/assets/app/locales/ast/messages.po +492 -0
  272. package/src/assets/app/locales/ca/messages.po +492 -0
  273. package/src/assets/app/locales/da/messages.po +492 -0
  274. package/src/assets/app/locales/de/messages.po +492 -0
  275. package/src/assets/app/locales/el/messages.po +492 -0
  276. package/src/assets/app/locales/en/messages.po +492 -0
  277. package/src/assets/app/locales/en-GB/messages.po +492 -0
  278. package/src/assets/app/locales/es/messages.po +492 -0
  279. package/src/assets/app/locales/eu/messages.po +492 -0
  280. package/src/assets/app/locales/fi/messages.po +492 -0
  281. package/src/assets/app/locales/fr/messages.po +492 -0
  282. package/src/assets/app/locales/ga/messages.po +492 -0
  283. package/src/assets/app/locales/gl/messages.po +492 -0
  284. package/src/assets/app/locales/hi/messages.po +492 -0
  285. package/src/assets/app/locales/hu/messages.po +492 -0
  286. package/src/assets/app/locales/ia/messages.po +492 -0
  287. package/src/assets/app/locales/id/messages.po +492 -0
  288. package/src/assets/app/locales/it/messages.po +492 -0
  289. package/src/assets/app/locales/ja/messages.po +492 -0
  290. package/src/assets/app/locales/km/messages.po +492 -0
  291. package/src/assets/app/locales/ko/messages.po +492 -0
  292. package/src/assets/app/locales/load.ts +8 -0
  293. package/src/assets/app/locales/locale-context.ts +19 -0
  294. package/src/assets/app/locales/locale-provider.tsx +112 -0
  295. package/src/assets/app/locales/locale-selector.tsx +58 -0
  296. package/src/assets/app/locales/locales.ts +168 -0
  297. package/src/assets/app/locales/ne/messages.po +492 -0
  298. package/src/assets/app/locales/nl/messages.po +492 -0
  299. package/src/assets/app/locales/pl/messages.po +492 -0
  300. package/src/assets/app/locales/pt-BR/messages.po +492 -0
  301. package/src/assets/app/locales/ro/messages.po +492 -0
  302. package/src/assets/app/locales/ru/messages.po +492 -0
  303. package/src/assets/app/locales/sv/messages.po +492 -0
  304. package/src/assets/app/locales/th/messages.po +492 -0
  305. package/src/assets/app/locales/tr/messages.po +492 -0
  306. package/src/assets/app/locales/uk/messages.po +492 -0
  307. package/src/assets/app/locales/vi/messages.po +492 -0
  308. package/src/assets/app/locales/zh-CN/messages.po +492 -0
  309. package/src/assets/app/locales/zh-HK/messages.po +492 -0
  310. package/src/assets/app/locales/zh-TW/messages.po +492 -0
  311. package/src/assets/app/main.css +23 -2
  312. package/src/assets/app/main.tsx +24 -8
  313. package/src/assets/app/views/authorize/accept/accept-form.tsx +150 -0
  314. package/src/assets/app/views/authorize/accept/accept-view.tsx +70 -0
  315. package/src/assets/app/views/authorize/authorize-view.tsx +180 -0
  316. package/src/assets/app/views/authorize/reset-password/reset-password-confirm-form.tsx +88 -0
  317. package/src/assets/app/views/authorize/reset-password/reset-password-request-form.tsx +80 -0
  318. package/src/assets/app/views/authorize/reset-password/reset-password-view.tsx +127 -0
  319. package/src/assets/app/views/authorize/sign-in/sign-in-form.tsx +244 -0
  320. package/src/assets/app/views/authorize/sign-in/sign-in-picker.tsx +116 -0
  321. package/src/assets/app/views/authorize/sign-in/sign-in-view.tsx +145 -0
  322. package/src/assets/app/views/authorize/sign-up/sign-up-account-form.tsx +140 -0
  323. package/src/assets/app/views/authorize/sign-up/sign-up-disclaimer.tsx +51 -0
  324. package/src/assets/app/views/authorize/sign-up/sign-up-handle-form.tsx +289 -0
  325. package/src/assets/app/views/authorize/sign-up/sign-up-hcaptcha-form.tsx +108 -0
  326. package/src/assets/app/views/authorize/sign-up/sign-up-view.tsx +158 -0
  327. package/src/assets/app/views/authorize/welcome/welcome-view.tsx +56 -0
  328. package/src/assets/app/views/error/error-view.tsx +31 -0
  329. package/src/assets/asset.ts +1 -0
  330. package/src/assets/assets-middleware.ts +13 -8
  331. package/src/assets/index.ts +15 -2
  332. package/src/client/client-store.ts +10 -12
  333. package/src/device/device-manager.ts +14 -15
  334. package/src/device/device-store.ts +9 -15
  335. package/src/dpop/dpop-manager.ts +20 -8
  336. package/src/dpop/dpop-nonce.ts +58 -40
  337. package/src/errors/handle-unavailable-error.ts +18 -0
  338. package/src/errors/invalid-request-error.ts +10 -8
  339. package/src/lib/csp/index.ts +98 -0
  340. package/src/lib/hcaptcha.ts +182 -0
  341. package/src/lib/html/build-document.ts +60 -16
  342. package/src/lib/http/middleware.ts +4 -3
  343. package/src/lib/http/request.ts +81 -28
  344. package/src/lib/http/response.ts +22 -9
  345. package/src/lib/locale.ts +21 -0
  346. package/src/lib/util/function.ts +0 -3
  347. package/src/lib/util/type.ts +130 -1
  348. package/src/metadata/build-metadata.ts +2 -1
  349. package/src/oauth-errors.ts +1 -0
  350. package/src/oauth-hooks.ts +69 -3
  351. package/src/oauth-provider.ts +410 -315
  352. package/src/oauth-verifier.ts +3 -1
  353. package/src/output/build-authorize-data.ts +1 -3
  354. package/src/output/build-customization-data.ts +189 -0
  355. package/src/output/output-manager.ts +111 -48
  356. package/src/output/send-authorize-redirect.ts +43 -36
  357. package/src/output/send-web-page.ts +40 -26
  358. package/src/request/request-manager.ts +4 -4
  359. package/src/request/request-store.ts +12 -16
  360. package/src/token/token-store.ts +14 -18
  361. package/tailwind.config.js +5 -0
  362. package/tsconfig.backend.tsbuildinfo +1 -1
  363. package/tsconfig.frontend.tsbuildinfo +1 -1
  364. package/tsconfig.tools.tsbuildinfo +1 -1
  365. package/vite.config.mjs +16 -0
  366. package/.postcssrc.yml +0 -3
  367. package/dist/assets/app/main.css +0 -3
  368. package/dist/assets/app/main.js +0 -20
  369. package/dist/assets/app/main.js.map +0 -1
  370. package/dist/output/customization.d.ts +0 -27
  371. package/dist/output/customization.d.ts.map +0 -1
  372. package/dist/output/customization.js +0 -88
  373. package/dist/output/customization.js.map +0 -1
  374. package/src/assets/app/components/accept-form.tsx +0 -137
  375. package/src/assets/app/components/account-identifier.tsx +0 -18
  376. package/src/assets/app/components/account-picker.tsx +0 -127
  377. package/src/assets/app/components/button.tsx +0 -34
  378. package/src/assets/app/components/client-name.tsx +0 -37
  379. package/src/assets/app/components/fieldset.tsx +0 -26
  380. package/src/assets/app/components/form-card.tsx +0 -47
  381. package/src/assets/app/components/help-card.tsx +0 -42
  382. package/src/assets/app/components/icons/alert-icon.tsx +0 -5
  383. package/src/assets/app/components/icons/at-symbol-icon.tsx +0 -5
  384. package/src/assets/app/components/icons/caret-right-icon.tsx +0 -5
  385. package/src/assets/app/components/icons/lock-icon.tsx +0 -5
  386. package/src/assets/app/components/icons/token-icon.tsx +0 -5
  387. package/src/assets/app/components/icons/util.tsx +0 -17
  388. package/src/assets/app/components/info-card.tsx +0 -45
  389. package/src/assets/app/components/input-checkbox.tsx +0 -47
  390. package/src/assets/app/components/input-container.tsx +0 -37
  391. package/src/assets/app/components/input-layout.tsx +0 -47
  392. package/src/assets/app/components/input-text.tsx +0 -69
  393. package/src/assets/app/components/layout-title-page.tsx +0 -60
  394. package/src/assets/app/components/layout-welcome.tsx +0 -74
  395. package/src/assets/app/components/sign-in-form.tsx +0 -337
  396. package/src/assets/app/components/sign-up-account-form.tsx +0 -194
  397. package/src/assets/app/components/sign-up-disclaimer.tsx +0 -44
  398. package/src/assets/app/views/accept-view.tsx +0 -55
  399. package/src/assets/app/views/authorize-view.tsx +0 -106
  400. package/src/assets/app/views/error-view.tsx +0 -36
  401. package/src/assets/app/views/sign-in-view.tsx +0 -111
  402. package/src/assets/app/views/sign-up-view.tsx +0 -86
  403. package/src/assets/app/views/welcome-view.tsx +0 -54
  404. package/src/output/customization.ts +0 -118
@@ -0,0 +1,150 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import type { OAuthClientMetadata } from '@atproto/oauth-types'
3
+ import { Account, ScopeDetail } from '../../../backend-types.ts'
4
+ import { Button } from '../../../components/forms/button.tsx'
5
+ import {
6
+ FormCard,
7
+ FormCardProps,
8
+ } from '../../../components/forms/form-card.tsx'
9
+ import { AccountIdentifier } from '../../../components/utils/account-identifier.tsx'
10
+ import { ClientName } from '../../../components/utils/client-name.tsx'
11
+ import { Override } from '../../../lib/util.ts'
12
+
13
+ export type AcceptFormProps = Override<
14
+ Omit<FormCardProps, 'onSubmit' | 'cancel' | 'actions' | 'children'>,
15
+ {
16
+ clientId: string
17
+ clientMetadata: OAuthClientMetadata
18
+ clientTrusted: boolean
19
+
20
+ account: Account
21
+ scopeDetails?: ScopeDetail[]
22
+
23
+ onAccept: () => void
24
+ onReject: () => void
25
+ onBack?: () => void
26
+ }
27
+ >
28
+
29
+ export function AcceptForm({
30
+ clientId,
31
+ clientMetadata,
32
+ clientTrusted,
33
+
34
+ account,
35
+ scopeDetails,
36
+
37
+ onAccept,
38
+ onReject,
39
+ onBack,
40
+
41
+ // FormCardProps
42
+ ...props
43
+ }: AcceptFormProps) {
44
+ const { t } = useLingui()
45
+ return (
46
+ <FormCard
47
+ {...props}
48
+ onSubmit={(event) => {
49
+ event.preventDefault()
50
+ onAccept()
51
+ }}
52
+ cancel={onBack && <Button onClick={onBack}>Back</Button>}
53
+ actions={
54
+ <>
55
+ <Button type="submit" color="brand">
56
+ <Trans>Authorize</Trans>
57
+ </Button>
58
+
59
+ <Button onClick={onReject}>
60
+ <Trans>Deny access</Trans>
61
+ </Button>
62
+ </>
63
+ }
64
+ >
65
+ {clientTrusted && clientMetadata.logo_uri && (
66
+ <div key="logo" className="flex items-center justify-center">
67
+ <img
68
+ crossOrigin="anonymous"
69
+ src={clientMetadata.logo_uri}
70
+ alt={clientMetadata.client_name}
71
+ className="w-16 h-16 rounded-full"
72
+ />
73
+ </div>
74
+ )}
75
+ <p>
76
+ <Trans>
77
+ <ClientName
78
+ clientId={clientId}
79
+ clientMetadata={clientMetadata}
80
+ clientTrusted={clientTrusted}
81
+ />{' '}
82
+ is asking for permission to access your account (
83
+ <AccountIdentifier account={account} />
84
+ ).
85
+ </Trans>
86
+ </p>
87
+
88
+ <p>
89
+ <Trans>
90
+ By clicking{' '}
91
+ <b>
92
+ <Trans>Authorize</Trans>
93
+ </b>
94
+ , you allow this application to perform the following actions in
95
+ accordance with their{' '}
96
+ <a
97
+ role="link"
98
+ href={clientMetadata.tos_uri}
99
+ rel="nofollow noopener"
100
+ target="_blank"
101
+ className="text-brand underline"
102
+ >
103
+ <Trans>terms of service</Trans>
104
+ </a>
105
+ {' and '}
106
+ <a
107
+ role="link"
108
+ href={clientMetadata.policy_uri}
109
+ rel="nofollow noopener"
110
+ target="_blank"
111
+ className="text-brand underline"
112
+ >
113
+ <Trans>privacy policy</Trans>
114
+ </a>
115
+ :
116
+ </Trans>
117
+ </p>
118
+
119
+ {scopeDetails?.length ? (
120
+ <ul
121
+ className="list-disc list-inside"
122
+ key="scopes"
123
+ aria-label={t`Requested permissions`}
124
+ >
125
+ {scopeDetails.map(({ scope, description }) => (
126
+ <li key={scope}>
127
+ {description || <ScopeDescription scope={scope} />}
128
+ </li>
129
+ ))}
130
+ </ul>
131
+ ) : null}
132
+ </FormCard>
133
+ )
134
+ }
135
+
136
+ type ScopeDescriptionProps = {
137
+ scope: string
138
+ }
139
+ function ScopeDescription({ scope }: ScopeDescriptionProps) {
140
+ switch (scope) {
141
+ case 'atproto':
142
+ return <Trans>Uniquely identify you</Trans>
143
+ case 'transition:generic':
144
+ return <Trans>Access your account data (except chat messages)</Trans>
145
+ case 'transition:chat.bsky':
146
+ return <Trans>Access your chat messages</Trans>
147
+ default:
148
+ return scope
149
+ }
150
+ }
@@ -0,0 +1,70 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import type { OAuthClientMetadata } from '@atproto/oauth-types'
3
+ import { Account, ScopeDetail } from '../../../backend-types.ts'
4
+ import {
5
+ LayoutTitlePage,
6
+ LayoutTitlePageProps,
7
+ } from '../../../components/layouts/layout-title-page.tsx'
8
+ import { Override } from '../../../lib/util.ts'
9
+ import { AcceptForm } from './accept-form.tsx'
10
+
11
+ export type AcceptViewProps = Override<
12
+ LayoutTitlePageProps,
13
+ {
14
+ clientId: string
15
+ clientMetadata: OAuthClientMetadata
16
+ clientTrusted: boolean
17
+
18
+ account: Account
19
+ scopeDetails?: ScopeDetail[]
20
+
21
+ onAccept: () => void
22
+ onReject: () => void
23
+ onBack?: () => void
24
+ }
25
+ >
26
+
27
+ export function AcceptView({
28
+ clientId,
29
+ clientMetadata,
30
+ clientTrusted,
31
+ account,
32
+ scopeDetails,
33
+ onAccept,
34
+ onReject,
35
+ onBack,
36
+
37
+ // LayoutTitlePage
38
+ title,
39
+ subtitle = (
40
+ <Trans>
41
+ Grant access to your{' '}
42
+ <b className="text-slate-800 dark:text-slate-200">
43
+ {account.preferred_username || account.email || account.sub}
44
+ </b>{' '}
45
+ account
46
+ </Trans>
47
+ ),
48
+ ...props
49
+ }: AcceptViewProps) {
50
+ const { t } = useLingui()
51
+
52
+ return (
53
+ <LayoutTitlePage
54
+ {...props}
55
+ title={title ?? t`Authorize`}
56
+ subtitle={subtitle}
57
+ >
58
+ <AcceptForm
59
+ clientId={clientId}
60
+ clientMetadata={clientMetadata}
61
+ clientTrusted={clientTrusted}
62
+ account={account}
63
+ scopeDetails={scopeDetails}
64
+ onBack={onBack}
65
+ onAccept={onAccept}
66
+ onReject={onReject}
67
+ />
68
+ </LayoutTitlePage>
69
+ )
70
+ }
@@ -0,0 +1,180 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { useEffect, useState } from 'react'
3
+ import type { AuthorizeData, CustomizationData } from '../../backend-types.ts'
4
+ import {
5
+ LayoutTitlePage,
6
+ LayoutTitlePageProps,
7
+ } from '../../components/layouts/layout-title-page.tsx'
8
+ import { useApi } from '../../hooks/use-api.ts'
9
+ import { useBoundDispatch } from '../../hooks/use-bound-dispatch.ts'
10
+ import { Override } from '../../lib/util.ts'
11
+ import { AcceptView } from './accept/accept-view.tsx'
12
+ import { ResetPasswordView } from './reset-password/reset-password-view.tsx'
13
+ import { SignInView } from './sign-in/sign-in-view.tsx'
14
+ import { SignUpView } from './sign-up/sign-up-view.tsx'
15
+ import { WelcomeView } from './welcome/welcome-view.tsx'
16
+
17
+ export type AuthorizeViewProps = Override<
18
+ LayoutTitlePageProps,
19
+ {
20
+ customizationData?: CustomizationData
21
+ authorizeData: AuthorizeData
22
+ }
23
+ >
24
+
25
+ enum View {
26
+ Welcome,
27
+ SignUp,
28
+ SignIn,
29
+ ResetPassword,
30
+ Accept,
31
+ Done,
32
+ }
33
+
34
+ export function AuthorizeView({
35
+ authorizeData,
36
+ customizationData,
37
+
38
+ // LayoutTitlePage
39
+ ...props
40
+ }: AuthorizeViewProps) {
41
+ const { t } = useLingui()
42
+
43
+ const forceSignIn = authorizeData?.loginHint != null
44
+
45
+ const initialView = forceSignIn ? View.SignIn : View.Welcome
46
+ const [view, setView] = useState<View>(initialView)
47
+
48
+ const showDone = useBoundDispatch(setView, View.Done)
49
+ const showSignIn = useBoundDispatch(setView, View.SignIn)
50
+ const showResetPassword = useBoundDispatch(setView, View.ResetPassword)
51
+ const showSignUp = useBoundDispatch(setView, View.SignUp)
52
+ const showAccept = useBoundDispatch(setView, View.Accept)
53
+ const showWelcome = useBoundDispatch(setView, View.Welcome)
54
+
55
+ const [resetPasswordHint, setResetPasswordHint] = useState<
56
+ string | undefined
57
+ >(undefined)
58
+
59
+ const {
60
+ sessions,
61
+ selectSub,
62
+ doValidateNewHandle,
63
+ doSignUp,
64
+ doSignIn,
65
+ doInitiatePasswordReset,
66
+ doConfirmResetPassword,
67
+ doAccept,
68
+ doReject,
69
+ } = useApi({ ...authorizeData, onRedirected: showDone })
70
+
71
+ // Navigate when the user signs-in (selects a new session)
72
+ const session = sessions.find((s) => s.selected && !s.loginRequired)
73
+ useEffect(() => {
74
+ if (session) {
75
+ if (session.consentRequired) showAccept()
76
+ else doAccept(session.account)
77
+ }
78
+ }, [session, doAccept, showAccept])
79
+
80
+ const canSignUp =
81
+ Boolean(customizationData?.availableUserDomains?.length) &&
82
+ !authorizeData.loginHint
83
+
84
+ // Fool-proofing
85
+ const resetNeeded =
86
+ (view === View.SignUp && !canSignUp) || (view === View.Accept && !session)
87
+ useEffect(() => {
88
+ if (resetNeeded) showWelcome()
89
+ }, [resetNeeded, showWelcome])
90
+
91
+ if (view === View.Welcome) {
92
+ return (
93
+ <WelcomeView
94
+ {...props}
95
+ customizationData={customizationData}
96
+ onSignIn={showSignIn}
97
+ onSignUp={canSignUp ? showSignUp : undefined}
98
+ onCancel={doReject}
99
+ />
100
+ )
101
+ }
102
+
103
+ if (view === View.SignUp) {
104
+ return (
105
+ <SignUpView
106
+ {...props}
107
+ customizationData={customizationData}
108
+ onValidateNewHandle={doValidateNewHandle}
109
+ onBack={showWelcome}
110
+ onDone={doSignUp}
111
+ />
112
+ )
113
+ }
114
+
115
+ if (view === View.ResetPassword) {
116
+ return (
117
+ <ResetPasswordView
118
+ {...props}
119
+ emailDefault={resetPasswordHint}
120
+ onresetPasswordRequest={doInitiatePasswordReset}
121
+ onResetPasswordConfirm={doConfirmResetPassword}
122
+ onBack={forceSignIn ? showSignIn : showWelcome}
123
+ />
124
+ )
125
+ }
126
+
127
+ if (view === View.SignIn) {
128
+ return (
129
+ <SignInView
130
+ {...props}
131
+ loginHint={authorizeData.loginHint}
132
+ sessions={sessions}
133
+ selectSub={selectSub}
134
+ onSignIn={doSignIn}
135
+ onBack={forceSignIn ? doReject : showWelcome}
136
+ onForgotPassword={(email) => {
137
+ showResetPassword()
138
+ setResetPasswordHint(email)
139
+ }}
140
+ />
141
+ )
142
+ }
143
+
144
+ if (view === View.Accept) {
145
+ // TypeSafety: should never be null here
146
+ if (!session) return null
147
+
148
+ return (
149
+ <AcceptView
150
+ {...props}
151
+ clientId={authorizeData.clientId}
152
+ clientMetadata={authorizeData.clientMetadata}
153
+ clientTrusted={authorizeData.clientTrusted}
154
+ account={session.account}
155
+ scopeDetails={authorizeData.scopeDetails}
156
+ onAccept={() => doAccept(session.account)}
157
+ onReject={doReject}
158
+ onBack={
159
+ forceSignIn
160
+ ? undefined
161
+ : () => {
162
+ selectSub(null)
163
+ setView(sessions.length ? View.SignIn : View.Welcome)
164
+ }
165
+ }
166
+ />
167
+ )
168
+ }
169
+
170
+ if (view === View.Done) {
171
+ return (
172
+ <LayoutTitlePage {...props} title={props.title ?? t`Login complete`}>
173
+ <Trans>You are being redirected...</Trans>
174
+ </LayoutTitlePage>
175
+ )
176
+ }
177
+
178
+ // Fool-proofing
179
+ throw new Error('Unexpected application state')
180
+ }
@@ -0,0 +1,88 @@
1
+ import { Trans } from '@lingui/react/macro'
2
+ import { useRef, useState } from 'react'
3
+ import { Fieldset } from '../../../components/forms/fieldset.tsx'
4
+ import {
5
+ FormCardAsync,
6
+ FormCardAsyncProps,
7
+ } from '../../../components/forms/form-card-async.tsx'
8
+ import { InputNewPassword } from '../../../components/forms/input-new-password.tsx'
9
+ import { InputToken } from '../../../components/forms/input-token.tsx'
10
+ import { Admonition } from '../../../components/utils/admonition.tsx'
11
+ import { useRandomString } from '../../../hooks/use-random-string.ts'
12
+ import { Override } from '../../../lib/util.ts'
13
+
14
+ export type ResetPasswordConfirmFormProps = Override<
15
+ FormCardAsyncProps,
16
+ {
17
+ onSubmit: (
18
+ data: {
19
+ token: string
20
+ password: string
21
+ },
22
+ signal: AbortSignal,
23
+ ) => void | PromiseLike<void>
24
+
25
+ tokenPattern?: string
26
+ tokenFormat?: string
27
+ tokenParseValue?: (value: string) => string | false
28
+ }
29
+ >
30
+
31
+ export function ResetPasswordConfirmForm({
32
+ onSubmit,
33
+
34
+ // FormCardAsyncProps
35
+ invalid,
36
+ ...props
37
+ }: ResetPasswordConfirmFormProps) {
38
+ const tokenAriaId = useRandomString({ prefix: 'reset-pwd-email-' })
39
+ const passwordRef = useRef<HTMLInputElement>(null)
40
+
41
+ const [token, setToken] = useState<string | null>(null)
42
+ const [password, setPassword] = useState<string | undefined>(undefined)
43
+
44
+ return (
45
+ <FormCardAsync
46
+ {...props}
47
+ onSubmit={(signal) => {
48
+ if (token && password) return onSubmit({ token, password }, signal)
49
+ }}
50
+ invalid={invalid || !token || !password}
51
+ >
52
+ <Admonition role="info">
53
+ <p id={tokenAriaId} className="text-md">
54
+ <Trans>
55
+ You will receive an email with a "reset code". Enter that code here
56
+ then enter your new password.
57
+ </Trans>
58
+ </p>
59
+ </Admonition>
60
+
61
+ <Fieldset label={<Trans>Reset code</Trans>}>
62
+ <InputToken
63
+ name="code"
64
+ aria-label={tokenAriaId}
65
+ enterKeyHint="next"
66
+ required
67
+ autoFocus={true}
68
+ onToken={(token) => {
69
+ setToken(token)
70
+ // Auto-focus next field when token is complete
71
+ if (token) passwordRef.current?.focus()
72
+ }}
73
+ />
74
+ </Fieldset>
75
+
76
+ <Fieldset label={<Trans>New password</Trans>}>
77
+ <InputNewPassword
78
+ ref={passwordRef}
79
+ name="password"
80
+ enterKeyHint="done"
81
+ required
82
+ password={password}
83
+ onPassword={setPassword}
84
+ />
85
+ </Fieldset>
86
+ </FormCardAsync>
87
+ )
88
+ }
@@ -0,0 +1,80 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { useCallback, useRef, useState } from 'react'
3
+ import { Fieldset } from '../../../components/forms/fieldset.tsx'
4
+ import {
5
+ AsyncActionController,
6
+ FormCardAsync,
7
+ FormCardAsyncProps,
8
+ } from '../../../components/forms/form-card-async.tsx'
9
+ import { InputEmailAddress } from '../../../components/forms/input-email-address.tsx'
10
+ import { Admonition } from '../../../components/utils/admonition.tsx'
11
+ import { useRandomString } from '../../../hooks/use-random-string.ts'
12
+ import { mergeRefs } from '../../../lib/ref.ts'
13
+ import { Override } from '../../../lib/util.ts'
14
+
15
+ export type ResetPasswordRequestFormProps = Override<
16
+ Omit<FormCardAsyncProps, 'children'>,
17
+ {
18
+ emailDefault?: string
19
+ onSubmit: (
20
+ data: { email: string },
21
+ signal: AbortSignal,
22
+ ) => void | PromiseLike<void>
23
+ }
24
+ >
25
+
26
+ export function ResetPasswordRequestForm({
27
+ emailDefault,
28
+ onSubmit,
29
+
30
+ // FormCardAsyncProps
31
+ invalid,
32
+ ref,
33
+ ...props
34
+ }: ResetPasswordRequestFormProps) {
35
+ const { t } = useLingui()
36
+ const emailAriaId = useRandomString({ prefix: 'reset-pwd-email-' })
37
+ const [email, setEmail] = useState(emailDefault)
38
+
39
+ const ctrlRef = useRef<AsyncActionController>(null)
40
+
41
+ const doSubmit = useCallback(
42
+ (signal: AbortSignal) => {
43
+ if (email) return onSubmit({ email }, signal)
44
+ },
45
+ [email, onSubmit],
46
+ )
47
+
48
+ return (
49
+ <FormCardAsync
50
+ {...props}
51
+ ref={mergeRefs([ref, ctrlRef])}
52
+ invalid={invalid || !email}
53
+ onSubmit={doSubmit}
54
+ >
55
+ <Fieldset label={<Trans>Email address</Trans>}>
56
+ <InputEmailAddress
57
+ name="email"
58
+ placeholder={t`Enter your email address`}
59
+ aria-labelledby={emailAriaId}
60
+ title={t`Email address`}
61
+ required
62
+ autoFocus={true}
63
+ value={email}
64
+ onEmail={(email) => {
65
+ ctrlRef.current?.reset()
66
+ setEmail(email)
67
+ }}
68
+ />
69
+ <Admonition role="info">
70
+ <p id={emailAriaId} className="">
71
+ <Trans>
72
+ Enter the email you used to create your account. We'll send you a
73
+ "reset code" so you can set a new password.
74
+ </Trans>
75
+ </p>
76
+ </Admonition>
77
+ </Fieldset>
78
+ </FormCardAsync>
79
+ )
80
+ }
@@ -0,0 +1,127 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { useState } from 'react'
3
+ import { Button } from '../../../components/forms/button.tsx'
4
+ import {
5
+ LayoutTitlePage,
6
+ LayoutTitlePageProps,
7
+ } from '../../../components/layouts/layout-title-page.tsx'
8
+ import { Override } from '../../../lib/util.ts'
9
+ import { ResetPasswordConfirmForm } from './reset-password-confirm-form.tsx'
10
+ import { ResetPasswordRequestForm } from './reset-password-request-form.tsx'
11
+
12
+ export type ResetPasswordViewProps = Override<
13
+ LayoutTitlePageProps,
14
+ {
15
+ emailDefault?: string
16
+ onresetPasswordRequest: (
17
+ data: { email: string },
18
+ signal: AbortSignal,
19
+ ) => void | PromiseLike<void>
20
+ onResetPasswordConfirm: (
21
+ data: {
22
+ token: string
23
+ password: string
24
+ },
25
+ signal: AbortSignal,
26
+ ) => void | PromiseLike<void>
27
+ onBack: () => void
28
+ }
29
+ >
30
+
31
+ enum View {
32
+ RequestReset,
33
+ ConfirmReset,
34
+ PasswordUpdated,
35
+ }
36
+
37
+ export function ResetPasswordView({
38
+ emailDefault,
39
+ onresetPasswordRequest,
40
+ onResetPasswordConfirm,
41
+ onBack,
42
+
43
+ // LayoutTitlePage
44
+ ...props
45
+ }: ResetPasswordViewProps) {
46
+ const { t } = useLingui()
47
+ const [view, setView] = useState<View>(View.RequestReset)
48
+
49
+ if (view === View.RequestReset) {
50
+ return (
51
+ <LayoutTitlePage
52
+ {...props}
53
+ title={props.title || t`Forgot Password`}
54
+ subtitle={
55
+ props.subtitle || <Trans>Let's get your password reset!</Trans>
56
+ }
57
+ >
58
+ <ResetPasswordRequestForm
59
+ emailDefault={emailDefault}
60
+ submitLabel={<Trans>Next</Trans>}
61
+ onSubmit={async (data, signal) => {
62
+ await onresetPasswordRequest(data, signal)
63
+ if (!signal.aborted) setView(View.ConfirmReset)
64
+ }}
65
+ cancelLabel={<Trans>Back</Trans>}
66
+ onCancel={onBack}
67
+ />
68
+ <hr className="my-5 border-gray-300 dark:border-gray-700" />
69
+ <center>
70
+ <Button transparent onClick={() => setView(View.ConfirmReset)}>
71
+ <Trans>Already have a code?</Trans>
72
+ </Button>
73
+ </center>
74
+ </LayoutTitlePage>
75
+ )
76
+ }
77
+
78
+ if (view === View.ConfirmReset) {
79
+ return (
80
+ <LayoutTitlePage
81
+ {...props}
82
+ title={props.title || t`Reset Password`}
83
+ subtitle={
84
+ props.subtitle || (
85
+ <Trans>Enter the code you received to reset your password.</Trans>
86
+ )
87
+ }
88
+ >
89
+ <ResetPasswordConfirmForm
90
+ submitLabel={<Trans>Next</Trans>}
91
+ onSubmit={async (data, signal) => {
92
+ await onResetPasswordConfirm(data, signal)
93
+ if (!signal.aborted) setView(View.PasswordUpdated)
94
+ }}
95
+ cancelLabel={<Trans>Back</Trans>}
96
+ onCancel={onBack}
97
+ />
98
+ </LayoutTitlePage>
99
+ )
100
+ }
101
+
102
+ if (view === View.PasswordUpdated) {
103
+ return (
104
+ <LayoutTitlePage
105
+ {...props}
106
+ title={props.title || t`Password Updated`}
107
+ subtitle={
108
+ props.subtitle || <Trans>Your password has been updated!</Trans>
109
+ }
110
+ >
111
+ <center>
112
+ <h2 className="text-xl font-bold pb-2">
113
+ <Trans>Password updated!</Trans>
114
+ </h2>
115
+ <p className="pb-4">
116
+ <Trans>You can now sign in with your new password.</Trans>
117
+ </p>
118
+ <Button color="brand" onClick={onBack}>
119
+ <Trans>Okay</Trans>
120
+ </Button>
121
+ </center>
122
+ </LayoutTitlePage>
123
+ )
124
+ }
125
+
126
+ throw new Error(`Invalid view: ${view}`)
127
+ }