@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,244 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { useCallback, useRef, useState } from 'react'
3
+ import { Button } from '../../../components/forms/button.tsx'
4
+ import { Fieldset } from '../../../components/forms/fieldset.tsx'
5
+ import {
6
+ FormCardAsync,
7
+ FormCardAsyncProps,
8
+ } from '../../../components/forms/form-card-async.tsx'
9
+ import { InputCheckbox } from '../../../components/forms/input-checkbox.tsx'
10
+ import { InputPassword } from '../../../components/forms/input-password.tsx'
11
+ import { InputText } from '../../../components/forms/input-text.tsx'
12
+ import { InputToken } from '../../../components/forms/input-token.tsx'
13
+ import { Admonition } from '../../../components/utils/admonition.tsx'
14
+ import { AtSymbolIcon } from '../../../components/utils/icons.tsx'
15
+ import { AsyncActionController } from '../../../hooks/use-async-action.ts'
16
+ import {
17
+ InvalidCredentialsError,
18
+ SecondAuthenticationFactorRequiredError,
19
+ } from '../../../lib/api.ts'
20
+ import { mergeRefs } from '../../../lib/ref.ts'
21
+ import { Override } from '../../../lib/util.ts'
22
+
23
+ export type SignInFormOutput = {
24
+ username: string
25
+ password: string
26
+ remember?: boolean
27
+ }
28
+
29
+ export type SignInFormProps = Override<
30
+ Omit<FormCardAsyncProps, 'append' | 'onCancel'>,
31
+ {
32
+ usernameDefault?: string
33
+ usernameReadonly?: boolean
34
+ rememberDefault?: boolean
35
+
36
+ onBack?: () => void
37
+ onForgotPassword?: (emailHint?: string) => void
38
+ onSubmit: (
39
+ credentials: SignInFormOutput,
40
+ signal: AbortSignal,
41
+ ) => void | PromiseLike<void>
42
+ }
43
+ >
44
+
45
+ export function SignInForm({
46
+ usernameDefault = '',
47
+ usernameReadonly = false,
48
+ rememberDefault = false,
49
+
50
+ onSubmit,
51
+ onBack,
52
+ onForgotPassword,
53
+
54
+ // FormCardAsync
55
+ ref,
56
+ invalid,
57
+ children,
58
+ ...props
59
+ }: SignInFormProps) {
60
+ const { t } = useLingui()
61
+
62
+ const [username, setUsername] = useState<string>(usernameDefault)
63
+ const [password, setPassword] = useState<string>('')
64
+ const [remember, setRemember] = useState<boolean>(rememberDefault)
65
+ const [otp, setOtp] = useState<string | null>(null)
66
+
67
+ const [secondFactor, setSecondFactor] =
68
+ useState<null | SecondAuthenticationFactorRequiredError>(null)
69
+
70
+ const [loading, setLoading] = useState(false)
71
+
72
+ const formRef = useRef<AsyncActionController>(null)
73
+
74
+ const clearSecondFactor = useCallback(() => {
75
+ setOtp(null)
76
+ setSecondFactor(null)
77
+ }, [setOtp, setSecondFactor])
78
+
79
+ const resetState = useCallback(() => {
80
+ clearSecondFactor()
81
+ formRef.current?.reset()
82
+ }, [clearSecondFactor, formRef])
83
+
84
+ const doSubmit = useCallback(
85
+ async (signal: AbortSignal) => {
86
+ try {
87
+ await onSubmit(
88
+ {
89
+ username,
90
+ password,
91
+ remember,
92
+ ...(secondFactor ? { [secondFactor.type]: otp } : {}),
93
+ },
94
+ signal,
95
+ )
96
+ } catch (err) {
97
+ if (signal.aborted) {
98
+ // If the action was aborted, ignore the error
99
+ return
100
+ }
101
+
102
+ if (err instanceof SecondAuthenticationFactorRequiredError) {
103
+ setSecondFactor(err)
104
+
105
+ // Do not re-throw 2FA required error to prevent the form from from
106
+ // displaying it. Instead, we handle the error by showing the second
107
+ // factor form.
108
+ return
109
+ }
110
+
111
+ if (err instanceof InvalidCredentialsError) {
112
+ // If the username/password are not valid, clear the second factor
113
+ // as valid credentials are a pre-requisite for 2FA.
114
+ clearSecondFactor()
115
+ }
116
+
117
+ // Any thrown err will be displayed through the form's errorRender
118
+ throw err
119
+ }
120
+ },
121
+ [username, password, remember, secondFactor, otp, onSubmit],
122
+ )
123
+
124
+ return (
125
+ <FormCardAsync
126
+ {...props}
127
+ ref={mergeRefs([ref, formRef])}
128
+ onLoading={setLoading}
129
+ onCancel={onBack}
130
+ cancelLabel={t`Back`}
131
+ append={children}
132
+ invalid={
133
+ invalid || !username || !password || (secondFactor != null && !otp)
134
+ }
135
+ submitLabel={secondFactor ? t`Confirm` : t`Sign in`}
136
+ onSubmit={doSubmit}
137
+ >
138
+ <Fieldset disabled={loading} label={<Trans>Account</Trans>}>
139
+ <InputText
140
+ icon={<AtSymbolIcon className="w-5" />}
141
+ name="username"
142
+ type="text"
143
+ placeholder={t`Username or email address`}
144
+ aria-label={t`Username or email address`}
145
+ autoCapitalize="none"
146
+ autoCorrect="off"
147
+ autoComplete="username"
148
+ spellCheck="false"
149
+ dir="auto"
150
+ enterKeyHint="next"
151
+ required
152
+ readOnly={usernameReadonly}
153
+ disabled={usernameReadonly}
154
+ title={t`Valid email address or username`}
155
+ autoFocus
156
+ value={username}
157
+ onChange={(event) => {
158
+ resetState()
159
+ setUsername(event.target.value)
160
+ }}
161
+ />
162
+ </Fieldset>
163
+
164
+ <InputPassword
165
+ name="password"
166
+ onChange={(event) => {
167
+ resetState()
168
+ setPassword(event.target.value)
169
+ }}
170
+ append={
171
+ onForgotPassword && (
172
+ <Button
173
+ className="text-sm"
174
+ type="button"
175
+ onClick={() => {
176
+ onForgotPassword(username?.includes('@') ? username : undefined)
177
+ }}
178
+ aria-label={t`Reset your password`}
179
+ >
180
+ <Trans>Forgot?</Trans>
181
+ </Button>
182
+ )
183
+ }
184
+ enterKeyHint={secondFactor ? 'next' : 'done'}
185
+ disabled={loading}
186
+ required
187
+ />
188
+
189
+ <Admonition role="status">
190
+ <p className="font-bold text-md text-brand pb-1">
191
+ <Trans>Warning</Trans>
192
+ </p>
193
+ <p className="text-sm">
194
+ <Trans>
195
+ Please verify the domain name of the website before entering your
196
+ password. Never enter your password on a domain you do not trust.
197
+ </Trans>
198
+ </p>
199
+ </Admonition>
200
+
201
+ <Fieldset
202
+ key="remember"
203
+ disabled={loading}
204
+ label={<Trans>Session</Trans>}
205
+ >
206
+ <InputCheckbox
207
+ name="remember"
208
+ aria-label={t`Remember this account on this device`}
209
+ enterKeyHint={secondFactor ? 'next' : 'done'}
210
+ checked={remember}
211
+ onChange={(event) => setRemember(event.target.checked)}
212
+ >
213
+ <Trans>Remember this account on this device</Trans>
214
+ </InputCheckbox>
215
+ </Fieldset>
216
+
217
+ {secondFactor && (
218
+ <Fieldset
219
+ key="2fa"
220
+ disabled={loading}
221
+ label={<Trans>2FA Confirmation</Trans>}
222
+ >
223
+ <div>
224
+ <InputToken
225
+ aria-label={t`Confirmation code`}
226
+ enterKeyHint="done"
227
+ required
228
+ autoFocus={true}
229
+ value={otp ?? ''}
230
+ onToken={setOtp}
231
+ />
232
+
233
+ <p className="text-slate-600 dark:text-slate-400 text-sm">
234
+ <Trans>
235
+ Check your {secondFactor.hint} email for a login code and enter
236
+ it here.
237
+ </Trans>
238
+ </p>
239
+ </div>
240
+ </Fieldset>
241
+ )}
242
+ </FormCardAsync>
243
+ )
244
+ }
@@ -0,0 +1,116 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { Account } from '../../../backend-types.ts'
3
+ import { Button } from '../../../components/forms/button.tsx'
4
+ import {
5
+ FormCard,
6
+ FormCardProps,
7
+ } from '../../../components/forms/form-card.tsx'
8
+ import { InputContainer } from '../../../components/forms/input-container.tsx'
9
+ import { AccountImage } from '../../../components/utils/account-image.tsx'
10
+ import {
11
+ AtSymbolIcon,
12
+ CaretRightIcon,
13
+ } from '../../../components/utils/icons.tsx'
14
+ import { Override } from '../../../lib/util.ts'
15
+
16
+ export type SignInPickerProps = Override<
17
+ Omit<FormCardProps, 'cancel' | 'actions' | 'append'>,
18
+ {
19
+ accounts: readonly Account[]
20
+
21
+ onAccount: (account: Account) => void
22
+ onOther?: () => void
23
+ onBack?: () => void
24
+ }
25
+ >
26
+
27
+ export function SignInPicker({
28
+ accounts,
29
+
30
+ onAccount,
31
+ onOther = undefined,
32
+ onBack,
33
+
34
+ // FormCard
35
+ children,
36
+ ...props
37
+ }: SignInPickerProps) {
38
+ const { t } = useLingui()
39
+ return (
40
+ <FormCard
41
+ {...props}
42
+ append={children}
43
+ actions={null}
44
+ cancel={
45
+ onBack && (
46
+ <Button onClick={onBack}>
47
+ <Trans>Back</Trans>
48
+ </Button>
49
+ )
50
+ }
51
+ >
52
+ <p className="text-slate-600 dark:text-slate-400 text-sm font-medium">
53
+ <Trans>Sign in as...</Trans>
54
+ </p>
55
+
56
+ {accounts.map((account) => {
57
+ const [name, identifier] = [
58
+ account.name,
59
+ account.preferred_username,
60
+ account.email,
61
+ account.sub,
62
+ ].filter(Boolean) as [string, string?]
63
+
64
+ return (
65
+ <InputContainer
66
+ tabIndex={0}
67
+ key={account.sub}
68
+ onKeyDown={(event) => {
69
+ if (event.key === 'Enter' || event.key === ' ') {
70
+ onAccount(account)
71
+ }
72
+ }}
73
+ onClick={() => onAccount(account)}
74
+ role="button"
75
+ aria-label={t`Sign in as ${account.name}`}
76
+ icon={<AccountImage src={account.picture} alt={name} />}
77
+ append={<CaretRightIcon aria-hidden className="h-4" />}
78
+ >
79
+ <span className="flex flex-wrap items-center">
80
+ <span className="font-medium truncate mr-2" arial-label={t`Name`}>
81
+ {name}
82
+ </span>
83
+ {identifier && (
84
+ <span
85
+ className="text-sm text-neutral-500 dark:text-neutral-400 truncate"
86
+ arial-label={t`Identifier`}
87
+ >
88
+ {identifier}
89
+ </span>
90
+ )}
91
+ </span>
92
+ </InputContainer>
93
+ )
94
+ })}
95
+
96
+ {onOther && (
97
+ <InputContainer
98
+ key="other"
99
+ tabIndex={0}
100
+ onKeyDown={(event) => {
101
+ if (event.key === 'Enter' || event.key === ' ') onOther()
102
+ }}
103
+ onClick={onOther}
104
+ aria-label={t`Login to account that is not listed`}
105
+ role="button"
106
+ append={<CaretRightIcon aria-hidden className="h-4" />}
107
+ icon={<AtSymbolIcon aria-hidden className="h-4" />}
108
+ >
109
+ <span className="truncate text-slate-700 dark:text-slate-400">
110
+ <Trans>Another account</Trans>
111
+ </span>
112
+ </InputContainer>
113
+ )}
114
+ </FormCard>
115
+ )
116
+ }
@@ -0,0 +1,145 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { useCallback, useEffect, useMemo, useState } from 'react'
3
+ import { Session } 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 { SignInForm, SignInFormOutput } from './sign-in-form.tsx'
10
+ import { SignInPicker } from './sign-in-picker.tsx'
11
+
12
+ export type SignInViewProps = Override<
13
+ LayoutTitlePageProps,
14
+ {
15
+ sessions: readonly Session[]
16
+ selectSub: (sub: string | null) => void
17
+ loginHint?: string
18
+
19
+ onSignIn: (
20
+ credentials: SignInFormOutput,
21
+ signal: AbortSignal,
22
+ ) => void | PromiseLike<void>
23
+ onForgotPassword?: (emailHint?: string) => void
24
+ onBack?: () => void
25
+ }
26
+ >
27
+
28
+ export function SignInView({
29
+ loginHint,
30
+ sessions,
31
+ selectSub,
32
+
33
+ onSignIn,
34
+ onForgotPassword,
35
+ onBack,
36
+
37
+ // LayoutTitlePage
38
+ title,
39
+ subtitle,
40
+ ...props
41
+ }: SignInViewProps) {
42
+ const { t } = useLingui()
43
+ const session = useMemo(() => sessions.find((s) => s.selected), [sessions])
44
+ const clearSession = useCallback(() => selectSub(null), [selectSub])
45
+ const accounts = useMemo(() => sessions.map((s) => s.account), [sessions])
46
+ const [showSignInForm, setShowSignInForm] = useState(sessions.length === 0)
47
+
48
+ title ??= t`Sign in`
49
+
50
+ useEffect(() => {
51
+ // Make sure the "back" action shows the account picker instead of the
52
+ // sign-in form (since the account was added to the list of current
53
+ // sessions).
54
+ if (session) setShowSignInForm(false)
55
+ }, [session])
56
+
57
+ if (session) {
58
+ // All set (parent view will handle the redirect)
59
+ if (!session.loginRequired) return null
60
+
61
+ return (
62
+ <LayoutTitlePage
63
+ {...props}
64
+ title={title}
65
+ subtitle={subtitle ?? <Trans>Confirm your password to continue</Trans>}
66
+ >
67
+ <SignInForm
68
+ onSubmit={onSignIn}
69
+ onForgotPassword={onForgotPassword}
70
+ onBack={clearSession}
71
+ usernameDefault={
72
+ session.account.preferred_username || session.account.sub
73
+ }
74
+ usernameReadonly={true}
75
+ rememberDefault={true}
76
+ />
77
+ </LayoutTitlePage>
78
+ )
79
+ }
80
+
81
+ if (loginHint) {
82
+ return (
83
+ <LayoutTitlePage
84
+ {...props}
85
+ title={title}
86
+ subtitle={subtitle ?? <Trans>Enter your password</Trans>}
87
+ >
88
+ <SignInForm
89
+ onSubmit={onSignIn}
90
+ onForgotPassword={onForgotPassword}
91
+ onBack={onBack}
92
+ usernameDefault={loginHint}
93
+ usernameReadonly={true}
94
+ />
95
+ </LayoutTitlePage>
96
+ )
97
+ }
98
+
99
+ if (sessions.length === 0) {
100
+ return (
101
+ <LayoutTitlePage
102
+ {...props}
103
+ title={title}
104
+ subtitle={subtitle ?? <Trans>Enter your username and password</Trans>}
105
+ >
106
+ <SignInForm
107
+ onSubmit={onSignIn}
108
+ onForgotPassword={onForgotPassword}
109
+ onBack={onBack}
110
+ />
111
+ </LayoutTitlePage>
112
+ )
113
+ }
114
+
115
+ if (showSignInForm) {
116
+ return (
117
+ <LayoutTitlePage
118
+ {...props}
119
+ title={title}
120
+ subtitle={subtitle ?? <Trans>Enter your username and password</Trans>}
121
+ >
122
+ <SignInForm
123
+ onSubmit={onSignIn}
124
+ onForgotPassword={onForgotPassword}
125
+ onBack={() => setShowSignInForm(false)}
126
+ />
127
+ </LayoutTitlePage>
128
+ )
129
+ }
130
+
131
+ return (
132
+ <LayoutTitlePage
133
+ {...props}
134
+ title={title}
135
+ subtitle={subtitle ?? <Trans>Select from an existing account</Trans>}
136
+ >
137
+ <SignInPicker
138
+ accounts={accounts}
139
+ onAccount={(a) => selectSub(a.sub)}
140
+ onOther={() => setShowSignInForm(true)}
141
+ onBack={onBack}
142
+ />
143
+ </LayoutTitlePage>
144
+ )
145
+ }
@@ -0,0 +1,140 @@
1
+ import { Trans, useLingui } from '@lingui/react/macro'
2
+ import { ReactNode, useEffect, useMemo, 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 { InputNewPassword } from '../../../components/forms/input-new-password.tsx'
11
+ import { InputToken } from '../../../components/forms/input-token.tsx'
12
+ import { mergeRefs } from '../../../lib/ref.ts'
13
+ import { Override } from '../../../lib/util.ts'
14
+
15
+ export type SignUpAccountFormOutput = {
16
+ email: string
17
+ password: string
18
+ inviteCode?: string
19
+ }
20
+
21
+ export type SignUpAccountFormProps = Override<
22
+ Omit<
23
+ FormCardAsyncProps,
24
+ 'append' | 'onCancel' | 'onSubmit' | 'submitLabel' | 'cancelLabel'
25
+ >,
26
+ {
27
+ inviteCodeRequired?: boolean
28
+
29
+ credentials?: SignUpAccountFormOutput
30
+ onCredentials?: (credentials?: SignUpAccountFormOutput) => void
31
+
32
+ onNext: (signal: AbortSignal) => void | PromiseLike<void>
33
+ nextLabel?: ReactNode
34
+
35
+ onPrev?: () => void
36
+ prevLabel?: ReactNode
37
+ }
38
+ >
39
+
40
+ export function SignUpAccountForm({
41
+ inviteCodeRequired = true,
42
+
43
+ credentials: creds,
44
+ onCredentials,
45
+
46
+ onNext,
47
+ nextLabel,
48
+
49
+ onPrev,
50
+ prevLabel,
51
+
52
+ // FormCardAsyncProps
53
+ children,
54
+ ref,
55
+ invalid,
56
+ ...props
57
+ }: SignUpAccountFormProps) {
58
+ const { t } = useLingui()
59
+
60
+ const [email, setEmail] = useState(creds?.email)
61
+ const [password, setPassword] = useState(creds?.password)
62
+ const [inviteCode, setInviteCode] = useState(creds?.inviteCode)
63
+
64
+ const formRef = useRef<AsyncActionController>(null)
65
+ const resetForm = () => formRef.current?.reset()
66
+
67
+ const credentials = useMemo(
68
+ () =>
69
+ email && password && (!inviteCodeRequired || inviteCode)
70
+ ? {
71
+ email,
72
+ password,
73
+ inviteCode: inviteCodeRequired ? inviteCode : undefined,
74
+ }
75
+ : undefined,
76
+ [email, password, inviteCode, inviteCodeRequired],
77
+ )
78
+
79
+ useEffect(() => {
80
+ onCredentials?.(credentials)
81
+ }, [credentials, onCredentials])
82
+
83
+ return (
84
+ <FormCardAsync
85
+ {...props}
86
+ ref={mergeRefs([ref, formRef])}
87
+ invalid={invalid || !credentials}
88
+ onCancel={onPrev}
89
+ cancelLabel={prevLabel}
90
+ onSubmit={onNext}
91
+ submitLabel={nextLabel}
92
+ append={children}
93
+ >
94
+ {inviteCodeRequired && (
95
+ <Fieldset label={<Trans>Invite code</Trans>}>
96
+ <InputToken
97
+ autoFocus
98
+ name="inviteCode"
99
+ aria-label={t`Invite code`}
100
+ title={t`Invite code`}
101
+ required
102
+ value={inviteCode}
103
+ onChange={(event) => {
104
+ setInviteCode(event.target.value)
105
+ resetForm()
106
+ }}
107
+ enterKeyHint="next"
108
+ />
109
+ </Fieldset>
110
+ )}
111
+
112
+ <Fieldset label={<Trans>Email</Trans>}>
113
+ <InputEmailAddress
114
+ autoFocus={!inviteCodeRequired}
115
+ name="email"
116
+ enterKeyHint="next"
117
+ required
118
+ defaultValue={email}
119
+ onEmail={(email) => {
120
+ setEmail(email)
121
+ resetForm()
122
+ }}
123
+ />
124
+ </Fieldset>
125
+
126
+ <Fieldset label={<Trans>Password</Trans>}>
127
+ <InputNewPassword
128
+ name="password"
129
+ enterKeyHint="next"
130
+ required
131
+ password={password}
132
+ onPassword={(value) => {
133
+ setPassword(value)
134
+ resetForm()
135
+ }}
136
+ />
137
+ </Fieldset>
138
+ </FormCardAsync>
139
+ )
140
+ }
@@ -0,0 +1,51 @@
1
+ import { Trans } from '@lingui/react/macro'
2
+ import { JSX } from 'react'
3
+ import { LinkDefinition } from '../../../backend-types.ts'
4
+ import { LinkAnchor } from '../../../components/utils/link-anchor.tsx'
5
+ import { clsx } from '../../../lib/clsx.ts'
6
+ import { Override } from '../../../lib/util.ts'
7
+
8
+ export type SignUpDisclaimerProps = Override<
9
+ Omit<JSX.IntrinsicElements['p'], 'children'>,
10
+ {
11
+ links?: readonly LinkDefinition[]
12
+ }
13
+ >
14
+
15
+ export function SignUpDisclaimer({
16
+ links,
17
+
18
+ // p
19
+ className,
20
+ ...attrs
21
+ }: SignUpDisclaimerProps) {
22
+ const tosLink = links?.find((l) => l.rel === 'terms-of-service')
23
+ const ppLink = links?.find((l) => l.rel === 'privacy-policy')
24
+
25
+ return (
26
+ <p
27
+ className={clsx('text-sm text-slate-500 dark:text-slate-400', className)}
28
+ {...attrs}
29
+ >
30
+ <Trans>
31
+ By creating an account you agree to the{' '}
32
+ {tosLink ? (
33
+ <LinkAnchor className="text-brand underline" link={tosLink}>
34
+ <Trans>Terms of Service</Trans>
35
+ </LinkAnchor>
36
+ ) : (
37
+ <Trans>Terms of Service</Trans>
38
+ )}
39
+ {' and the '}
40
+ {ppLink ? (
41
+ <LinkAnchor className="text-brand underline" link={ppLink}>
42
+ <Trans>Privacy Policy</Trans>
43
+ </LinkAnchor>
44
+ ) : (
45
+ <Trans>Privacy Policy</Trans>
46
+ )}{' '}
47
+ of this service.
48
+ </Trans>
49
+ </p>
50
+ )
51
+ }