@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,103 @@
1
+ import { Trans } from '@lingui/react/macro'
2
+ import { FormEvent, ReactNode, useCallback } from 'react'
3
+ import {
4
+ UseAsyncActionOptions,
5
+ useAsyncAction,
6
+ } from '../../hooks/use-async-action.ts'
7
+ import { Override } from '../../lib/util.ts'
8
+ import { ErrorCard } from '../utils/error-card.tsx'
9
+ import { Button } from './button.tsx'
10
+ import { FormCard, FormCardProps } from './form-card.tsx'
11
+
12
+ export type { AsyncActionController } from '../../hooks/use-async-action.ts'
13
+
14
+ export type ErrorRender = (data: { error: Error }) => ReactNode
15
+ export const errorRenderDefault: ErrorRender = ({ error }) => (
16
+ <ErrorCard error={error} />
17
+ )
18
+
19
+ export type FormCardAsyncProps = Override<
20
+ Override<
21
+ Omit<FormCardProps, 'cancel' | 'actions' | 'prepend'>,
22
+ Pick<UseAsyncActionOptions, 'ref' | 'onLoading' | 'onError'>
23
+ >,
24
+ {
25
+ invalid?: boolean
26
+ disabled?: boolean
27
+
28
+ onSubmit: (signal: AbortSignal) => void | PromiseLike<void>
29
+ submitLabel?: ReactNode
30
+
31
+ onCancel?: () => void
32
+ cancelLabel?: ReactNode
33
+
34
+ errorRender?: ErrorRender
35
+ }
36
+ >
37
+
38
+ export function FormCardAsync({
39
+ invalid,
40
+ disabled,
41
+
42
+ onSubmit,
43
+ submitLabel,
44
+
45
+ onCancel = undefined,
46
+ cancelLabel,
47
+
48
+ errorRender = errorRenderDefault,
49
+
50
+ // UseAsyncActionOptions
51
+ ref,
52
+ onLoading,
53
+ onError,
54
+
55
+ // FormCardProps
56
+ children,
57
+ ...props
58
+ }: FormCardAsyncProps) {
59
+ const { run, loading, error } = useAsyncAction(onSubmit, {
60
+ ref,
61
+ onError,
62
+ onLoading,
63
+ })
64
+
65
+ const doSubmit = useCallback(
66
+ (event: FormEvent<HTMLFormElement>) => {
67
+ event.preventDefault()
68
+
69
+ if (!event.currentTarget.reportValidity()) return
70
+
71
+ if (!disabled && !invalid) void run()
72
+ },
73
+ [disabled, invalid, run],
74
+ )
75
+
76
+ return (
77
+ <FormCard
78
+ {...props}
79
+ onSubmit={doSubmit}
80
+ disabled={disabled || loading}
81
+ prepend={error != null ? errorRender({ error }) : undefined}
82
+ cancel={
83
+ onCancel && (
84
+ <Button onClick={onCancel}>
85
+ {cancelLabel || <Trans>Cancel</Trans>}
86
+ </Button>
87
+ )
88
+ }
89
+ actions={
90
+ <Button
91
+ color="brand"
92
+ type="submit"
93
+ loading={loading}
94
+ disabled={disabled}
95
+ >
96
+ {submitLabel || <Trans>Submit</Trans>}
97
+ </Button>
98
+ }
99
+ >
100
+ {children}
101
+ </FormCard>
102
+ )
103
+ }
@@ -0,0 +1,49 @@
1
+ import { JSX, ReactNode } from 'react'
2
+ import { Override } from '../../lib/util.ts'
3
+
4
+ export type FormCardProps = Override<
5
+ JSX.IntrinsicElements['form'],
6
+ {
7
+ disabled?: boolean
8
+ append?: ReactNode
9
+ prepend?: ReactNode
10
+ cancel?: ReactNode
11
+ actions?: ReactNode
12
+ }
13
+ >
14
+
15
+ export function FormCard({
16
+ actions,
17
+ cancel,
18
+ append,
19
+ children,
20
+ prepend,
21
+ disabled,
22
+
23
+ // form
24
+ inert = disabled,
25
+ ...props
26
+ }: FormCardProps) {
27
+ return (
28
+ <form {...props} inert={inert} className="flex flex-col space-y-4">
29
+ {prepend && <div key="prepend">{prepend}</div>}
30
+
31
+ <div key="children" className="space-y-4">
32
+ {children}
33
+ </div>
34
+
35
+ {append && <div key="append">{append}</div>}
36
+
37
+ {(actions || cancel) && (
38
+ <div
39
+ key="buttons"
40
+ className="flex flex-wrap flex-row-reverse items-center justify-end space-x-reverse space-x-2"
41
+ >
42
+ {actions}
43
+ <div className="flex-auto" />
44
+ {cancel}
45
+ </div>
46
+ )}
47
+ </form>
48
+ )
49
+ }
@@ -0,0 +1,73 @@
1
+ import { JSX, ReactNode, useContext, useRef } from 'react'
2
+ import { useRandomString } from '../../hooks/use-random-string.ts'
3
+ import { clsx } from '../../lib/clsx.ts'
4
+ import { mergeRefs } from '../../lib/ref.ts'
5
+ import { Override } from '../../lib/util.ts'
6
+ import { FieldsetContext } from './fieldset.tsx'
7
+ import { InputContainer } from './input-container.tsx'
8
+
9
+ export type InputCheckboxProps = Override<
10
+ Omit<JSX.IntrinsicElements['input'], 'className' | 'type' | 'children'>,
11
+ {
12
+ className?: string
13
+ children?: ReactNode
14
+ }
15
+ >
16
+
17
+ export function InputCheckbox({
18
+ className,
19
+ children,
20
+
21
+ // input
22
+ id,
23
+ ref,
24
+ disabled,
25
+ 'aria-labelledby': ariaLabelledBy,
26
+ ...props
27
+ }: InputCheckboxProps) {
28
+ const htmlFor = useRandomString('input-checkbox-')
29
+ const containerRef = useRef<HTMLDivElement>(null)
30
+ const inputRef = useRef<HTMLInputElement>(null)
31
+ const ctx = useContext(FieldsetContext)
32
+
33
+ const inputId = id ?? htmlFor
34
+
35
+ return (
36
+ <InputContainer
37
+ ref={containerRef}
38
+ className={clsx('cursor-pointer', className)}
39
+ icon={
40
+ <input
41
+ {...props}
42
+ disabled={disabled ?? ctx.disabled}
43
+ aria-labelledby={
44
+ children
45
+ ? // Prefer the local "<label>" element (through "htmlFor") over the wrapping "<fieldset>" to describe the checkbox.
46
+ undefined
47
+ : ariaLabelledBy ?? ctx.labelId
48
+ }
49
+ ref={mergeRefs([ref, inputRef])}
50
+ id={inputId}
51
+ className="accent-brand outline-none"
52
+ type="checkbox"
53
+ />
54
+ }
55
+ tabIndex={-1}
56
+ onClick={(event) => {
57
+ if (event.target === containerRef.current && !event.defaultPrevented) {
58
+ inputRef.current?.click()
59
+ inputRef.current?.focus()
60
+ }
61
+ }}
62
+ >
63
+ {children && (
64
+ <label
65
+ htmlFor={inputId}
66
+ className="block w-full leading-[1.6] select-none cursor-pointer"
67
+ >
68
+ {children}
69
+ </label>
70
+ )}
71
+ </InputContainer>
72
+ )
73
+ }
@@ -0,0 +1,107 @@
1
+ import { JSX, ReactNode, useState } from 'react'
2
+ import { clsx } from '../../lib/clsx.ts'
3
+ import { Override } from '../../lib/util.ts'
4
+
5
+ export type InputContainerProps = Override<
6
+ JSX.IntrinsicElements['div'],
7
+ {
8
+ icon: ReactNode
9
+ append?: ReactNode
10
+ bellow?: ReactNode
11
+ }
12
+ >
13
+
14
+ export function InputContainer({
15
+ icon,
16
+ append,
17
+ bellow,
18
+
19
+ // div
20
+ className,
21
+ children,
22
+ onFocus,
23
+ onBlur,
24
+ ...props
25
+ }: InputContainerProps) {
26
+ const [hasFocus, setHasFocus] = useState(false)
27
+
28
+ return (
29
+ <div
30
+ {...props}
31
+ onFocus={(event) => {
32
+ onFocus?.(event)
33
+ if (!event.defaultPrevented) setHasFocus(true)
34
+ }}
35
+ onBlur={(event) => {
36
+ onBlur?.(event)
37
+ if (!event.defaultPrevented) setHasFocus(false)
38
+ }}
39
+ className={clsx(
40
+ // Layout
41
+ 'min-h-12',
42
+ 'max-w-full',
43
+ 'overflow-hidden',
44
+ // Border
45
+ 'rounded-lg',
46
+ className,
47
+ )}
48
+ >
49
+ <div
50
+ className={clsx(
51
+ // Layout
52
+ 'px-1',
53
+ 'w-full min-h-12',
54
+ 'flex items-center justify-stretch',
55
+ // Border
56
+ 'rounded-lg',
57
+ bellow ? 'rounded-br-none rounded-bl-none' : undefined,
58
+ 'outline-none',
59
+ 'border-solid border-2 border-transparent',
60
+ 'focus:border-brand has-[:focus]:border-brand',
61
+ 'hover:border-gray-400 hover:focus:border-gray-400',
62
+ 'dark:hover:border-gray-500 dark:hover:focus:border-gray-500',
63
+ // Background
64
+ 'bg-gray-100 focus:bg-slate-200 has-[:focus]:bg-slate-200',
65
+ 'dark:bg-slate-800 dark:focus:bg-slate-700 dark:has-[:focus]:bg-slate-700',
66
+ // Font
67
+ 'text-slate-600 dark:text-slate-300',
68
+ 'accent-brand',
69
+ )}
70
+ >
71
+ {icon && (
72
+ <div
73
+ className={clsx(
74
+ 'shrink-0 grow-0',
75
+ 'mx-1',
76
+ hasFocus ? 'text-brand' : 'text-slate-500',
77
+ )}
78
+ >
79
+ {icon}
80
+ </div>
81
+ )}
82
+
83
+ {children}
84
+
85
+ <div className="ml-1 grow-0 shrink-0 flex items-center">{append}</div>
86
+ </div>
87
+ {bellow && (
88
+ <div
89
+ className={clsx(
90
+ // Layout
91
+ 'px-3 py-2 space-x-2',
92
+ 'flex flex-row items-center gap-1',
93
+ // Border
94
+ 'rounded-br-2 rounded-bl-2',
95
+ // Background
96
+ 'bg-gray-200 dark:bg-slate-700',
97
+ // Font
98
+ 'text-gray-700 dark:text-gray-300',
99
+ 'text-sm italic',
100
+ )}
101
+ >
102
+ {bellow}
103
+ </div>
104
+ )}
105
+ </div>
106
+ )
107
+ }
@@ -0,0 +1,66 @@
1
+ import { useLingui } from '@lingui/react/macro'
2
+ import { ChangeEvent, useCallback, useState } from 'react'
3
+ import { Override } from '../../lib/util.ts'
4
+ import { AtSymbolIcon } from '../utils/icons.tsx'
5
+ import { InputText, InputTextProps } from './input-text.tsx'
6
+
7
+ export type InputEmailAddressProps = Override<
8
+ Omit<InputTextProps, 'type'>,
9
+ {
10
+ onEmail?: (email: string | undefined) => void
11
+ }
12
+ >
13
+
14
+ export function InputEmailAddress({
15
+ onEmail,
16
+
17
+ // InputTextProps
18
+ autoCapitalize = 'none',
19
+ autoComplete = 'email',
20
+ autoCorrect = 'off',
21
+ dir = 'auto',
22
+ icon = <AtSymbolIcon className="w-5" />,
23
+ onBlur,
24
+ onChange,
25
+ pattern = '^[^@]+@[^@]+\\.[^@]+$',
26
+ spellCheck = 'false',
27
+ value,
28
+ defaultValue = value,
29
+ ...props
30
+ }: InputEmailAddressProps) {
31
+ const { t } = useLingui()
32
+ const [email, setEmail] = useState<string>(
33
+ typeof defaultValue === 'string' ? defaultValue : '',
34
+ )
35
+
36
+ const doChange = useCallback(
37
+ (event: ChangeEvent<HTMLInputElement>) => {
38
+ const email = event.target.value.toLowerCase()
39
+
40
+ setEmail(email)
41
+ onChange?.(event)
42
+ onEmail?.(event.target.validity.valid ? email : undefined)
43
+ },
44
+ [onChange, onEmail],
45
+ )
46
+
47
+ return (
48
+ <InputText
49
+ aria-label={t`Email`}
50
+ placeholder={t`Email`}
51
+ title={t`Email`}
52
+ {...props}
53
+ type="email"
54
+ autoCapitalize={autoCapitalize}
55
+ autoCorrect={autoCorrect}
56
+ dir={dir}
57
+ spellCheck={spellCheck}
58
+ icon={icon}
59
+ pattern={pattern}
60
+ autoComplete={autoComplete}
61
+ value={email}
62
+ onChange={doChange}
63
+ onBlur={onBlur}
64
+ />
65
+ )
66
+ }
@@ -0,0 +1,62 @@
1
+ import { useLingui } from '@lingui/react/macro'
2
+ import { ChangeEvent, useCallback, useState } from 'react'
3
+ import { MIN_PASSWORD_LENGTH } from '../../lib/password.ts'
4
+ import { Override } from '../../lib/util.ts'
5
+ import { PasswordStrengthLabel } from '../utils/password-strength-label.tsx'
6
+ import { PasswordStrengthMeter } from '../utils/password-strength-meter.tsx'
7
+ import { InputPassword, InputPasswordProps } from './input-password.tsx'
8
+
9
+ export type InputNewPasswordProps = Override<
10
+ Omit<InputPasswordProps, 'value' | 'defaultValue'>,
11
+ {
12
+ password?: string
13
+ onPassword?: (password: undefined | string) => void
14
+ }
15
+ >
16
+
17
+ export function InputNewPassword({
18
+ password: passwordInit = '',
19
+ onPassword,
20
+
21
+ // InputPasswordProps
22
+ onChange,
23
+ autoComplete = 'new-password',
24
+ minLength = MIN_PASSWORD_LENGTH,
25
+ ...props
26
+ }: InputNewPasswordProps) {
27
+ const { t } = useLingui()
28
+ const [password, setPassword] = useState<string>(passwordInit)
29
+
30
+ const doChange = useCallback(
31
+ (event: ChangeEvent<HTMLInputElement>) => {
32
+ const { value } = event.target
33
+ onChange?.(event)
34
+ if (event.defaultPrevented) return
35
+ setPassword(value)
36
+ onPassword?.(event.target.validity.valid ? value : undefined)
37
+ },
38
+ [onChange, onPassword],
39
+ )
40
+
41
+ return (
42
+ <InputPassword
43
+ {...props}
44
+ placeholder={t`Enter a password`}
45
+ aria-label={t`Enter your new password`}
46
+ title={t`Password with at least ${MIN_PASSWORD_LENGTH} characters`}
47
+ minLength={minLength}
48
+ onChange={doChange}
49
+ value={password}
50
+ autoComplete={autoComplete}
51
+ bellow={
52
+ <>
53
+ <PasswordStrengthMeter password={password} />
54
+ <PasswordStrengthLabel
55
+ className="grow-1 min-w-max text-xs text-gray-500 dark:text-gray-400"
56
+ password={password}
57
+ />
58
+ </>
59
+ }
60
+ />
61
+ )
62
+ }
@@ -0,0 +1,88 @@
1
+ import { useLingui } from '@lingui/react/macro'
2
+ import { ChangeEvent, useCallback, useRef, useState } from 'react'
3
+ import { mergeRefs } from '../../lib/ref.ts'
4
+ import { Override } from '../../lib/util.ts'
5
+ import { LockIcon } from '../utils/icons.tsx'
6
+ import { ButtonToggleVisibility } from './button-toggle-visibility.tsx'
7
+ import { InputText, InputTextProps } from './input-text.tsx'
8
+
9
+ export type InputPasswordProps = Override<
10
+ Omit<InputTextProps, 'type' | 'children'>,
11
+ {
12
+ autoHide?: boolean
13
+ }
14
+ >
15
+
16
+ export function InputPassword({
17
+ autoHide = true,
18
+
19
+ // InputTextProps
20
+ onBlur,
21
+ onChange,
22
+ append,
23
+ autoComplete = 'current-password',
24
+ icon = <LockIcon className="w-5" />,
25
+ value,
26
+ defaultValue = value,
27
+ ref,
28
+ dir = 'auto',
29
+ autoCapitalize = 'none',
30
+ autoCorrect = 'off',
31
+ spellCheck = 'false',
32
+ ...props
33
+ }: InputPasswordProps) {
34
+ const { t } = useLingui()
35
+ const inputRef = useRef<HTMLInputElement>(null)
36
+ const [visible, setVisible] = useState<boolean>(false)
37
+ const [password, setPassword] = useState<string>(
38
+ typeof defaultValue === 'string' ? defaultValue : '',
39
+ )
40
+
41
+ const doChange = useCallback(
42
+ (event: ChangeEvent<HTMLInputElement>) => {
43
+ onChange?.(event)
44
+ setPassword(event.target.value)
45
+ },
46
+ [onChange],
47
+ )
48
+
49
+ return (
50
+ <InputText
51
+ placeholder={t`Password`}
52
+ aria-label={t`Password`}
53
+ title={t`Password`}
54
+ {...props}
55
+ ref={mergeRefs([ref, inputRef])}
56
+ dir={dir}
57
+ autoCapitalize={autoCapitalize}
58
+ autoCorrect={autoCorrect}
59
+ spellCheck={spellCheck}
60
+ icon={icon}
61
+ onBlur={
62
+ autoHide
63
+ ? (event) => {
64
+ onBlur?.(event)
65
+ if (!event.defaultPrevented) setVisible(false)
66
+ }
67
+ : onBlur
68
+ }
69
+ value={password}
70
+ onChange={doChange}
71
+ type={visible ? 'text' : 'password'}
72
+ autoComplete={autoComplete}
73
+ append={
74
+ <>
75
+ <ButtonToggleVisibility
76
+ className="m-1"
77
+ visible={visible}
78
+ toggleVisible={() => {
79
+ setVisible((prev) => !prev)
80
+ inputRef.current?.focus()
81
+ }}
82
+ />
83
+ {append}
84
+ </>
85
+ }
86
+ />
87
+ )
88
+ }
@@ -0,0 +1,76 @@
1
+ import { JSX, ReactNode, useContext, useRef } from 'react'
2
+ import { clsx } from '../../lib/clsx.ts'
3
+ import { mergeRefs } from '../../lib/ref.ts'
4
+ import { Override } from '../../lib/util.ts'
5
+ import { FieldsetContext } from './fieldset.tsx'
6
+ import { InputContainer } from './input-container.tsx'
7
+
8
+ export type InputTextProps = Override<
9
+ Omit<JSX.IntrinsicElements['input'], 'children'>,
10
+ {
11
+ icon?: ReactNode
12
+ append?: ReactNode
13
+ bellow?: ReactNode
14
+ className?: string
15
+ }
16
+ >
17
+
18
+ export function InputText({
19
+ icon,
20
+ append,
21
+ bellow,
22
+ className,
23
+
24
+ // input
25
+ onFocus,
26
+ onBlur,
27
+ ref,
28
+ disabled,
29
+ 'aria-labelledby': ariaLabelledBy,
30
+ ...props
31
+ }: InputTextProps) {
32
+ const ctx = useContext(FieldsetContext)
33
+
34
+ const inputRef = useRef<HTMLInputElement>(null)
35
+ const focusedRef = useRef(false) // ref instead of state to avoid re-renders
36
+
37
+ return (
38
+ <InputContainer
39
+ icon={icon}
40
+ append={append}
41
+ bellow={bellow}
42
+ className={clsx('cursor-text', className)}
43
+ tabIndex={-1}
44
+ onClick={(event) => {
45
+ if (inputRef.current !== event.target) {
46
+ event.preventDefault()
47
+ event.stopPropagation()
48
+ inputRef.current?.focus()
49
+ }
50
+ }}
51
+ onMouseDown={(event) => {
52
+ if (focusedRef.current && event.target !== inputRef.current) {
53
+ // Prevent "blur" event from firing when clicking outside the input
54
+ event.preventDefault()
55
+ event.stopPropagation()
56
+ }
57
+ }}
58
+ >
59
+ <input
60
+ {...props}
61
+ disabled={disabled ?? ctx.disabled}
62
+ aria-labelledby={ariaLabelledBy ?? ctx.labelId}
63
+ ref={mergeRefs([ref, inputRef])}
64
+ className="w-full bg-transparent bg-clip-padding text-base text-inherit outline-none dark:placeholder-gray-500 text-ellipsis"
65
+ onFocus={(event) => {
66
+ onFocus?.(event)
67
+ if (!event.defaultPrevented) focusedRef.current = true
68
+ }}
69
+ onBlur={(event) => {
70
+ onBlur?.(event)
71
+ if (!event.defaultPrevented) focusedRef.current = false
72
+ }}
73
+ />
74
+ </InputContainer>
75
+ )
76
+ }