@workos-inc/widgets 1.0.0 → 1.1.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 (397) hide show
  1. package/README.md +68 -1
  2. package/dist/cjs/card-list.d.ts +6 -0
  3. package/dist/cjs/card-list.d.ts.map +1 -0
  4. package/dist/cjs/card-list.js +13 -0
  5. package/dist/cjs/card-list.js.map +1 -0
  6. package/dist/cjs/index.d.ts +3 -0
  7. package/dist/cjs/index.d.ts.map +1 -1
  8. package/dist/cjs/index.js +7 -1
  9. package/dist/cjs/index.js.map +1 -1
  10. package/dist/cjs/lib/add-mfa-dialog.d.ts +9 -0
  11. package/dist/cjs/lib/add-mfa-dialog.d.ts.map +1 -0
  12. package/dist/cjs/lib/add-mfa-dialog.js +135 -0
  13. package/dist/cjs/lib/add-mfa-dialog.js.map +1 -0
  14. package/dist/cjs/lib/api/user.d.ts +13 -57
  15. package/dist/cjs/lib/api/user.d.ts.map +1 -1
  16. package/dist/cjs/lib/api/user.js +44 -314
  17. package/dist/cjs/lib/api/user.js.map +1 -1
  18. package/dist/cjs/lib/change-password-dialog.d.ts +8 -0
  19. package/dist/cjs/lib/change-password-dialog.d.ts.map +1 -0
  20. package/dist/cjs/lib/change-password-dialog.js +97 -0
  21. package/dist/cjs/lib/change-password-dialog.js.map +1 -0
  22. package/dist/cjs/lib/copy-button.d.ts +8 -0
  23. package/dist/cjs/lib/copy-button.d.ts.map +1 -0
  24. package/dist/cjs/lib/copy-button.js +63 -0
  25. package/dist/cjs/lib/copy-button.js.map +1 -0
  26. package/dist/cjs/lib/delete-user-dialog.d.ts +2 -2
  27. package/dist/cjs/lib/delete-user-dialog.d.ts.map +1 -1
  28. package/dist/cjs/lib/delete-user-dialog.js +1 -1
  29. package/dist/cjs/lib/delete-user-dialog.js.map +1 -1
  30. package/dist/cjs/lib/edit-user-profile-dialog.d.ts +10 -0
  31. package/dist/cjs/lib/edit-user-profile-dialog.d.ts.map +1 -0
  32. package/dist/cjs/lib/edit-user-profile-dialog.js +85 -0
  33. package/dist/cjs/lib/edit-user-profile-dialog.js.map +1 -0
  34. package/dist/cjs/lib/edit-user-role-dialog.d.ts +12 -0
  35. package/dist/cjs/lib/edit-user-role-dialog.d.ts.map +1 -0
  36. package/dist/cjs/lib/{edit-user-details-dialog.js → edit-user-role-dialog.js} +11 -7
  37. package/dist/cjs/lib/edit-user-role-dialog.js.map +1 -0
  38. package/dist/cjs/lib/elements.d.ts +1 -0
  39. package/dist/cjs/lib/elements.d.ts.map +1 -1
  40. package/dist/cjs/lib/elements.js +10 -4
  41. package/dist/cjs/lib/elements.js.map +1 -1
  42. package/dist/cjs/lib/elevated-access.d.ts +8 -0
  43. package/dist/cjs/lib/elevated-access.d.ts.map +1 -0
  44. package/dist/cjs/lib/elevated-access.js +130 -0
  45. package/dist/cjs/lib/elevated-access.js.map +1 -0
  46. package/dist/cjs/lib/generic-error.d.ts +4 -0
  47. package/dist/cjs/lib/generic-error.d.ts.map +1 -0
  48. package/dist/cjs/lib/generic-error.js +57 -0
  49. package/dist/cjs/lib/generic-error.js.map +1 -0
  50. package/dist/cjs/lib/icon-panel.d.ts +3 -0
  51. package/dist/cjs/lib/icon-panel.d.ts.map +1 -0
  52. package/dist/cjs/lib/icon-panel.js +16 -0
  53. package/dist/cjs/lib/icon-panel.js.map +1 -0
  54. package/dist/cjs/lib/icons.d.ts +3 -0
  55. package/dist/cjs/lib/icons.d.ts.map +1 -0
  56. package/dist/cjs/lib/icons.js +8 -0
  57. package/dist/cjs/lib/icons.js.map +1 -0
  58. package/dist/cjs/lib/invite-user-dialog.d.ts.map +1 -1
  59. package/dist/cjs/lib/invite-user-dialog.js +7 -5
  60. package/dist/cjs/lib/invite-user-dialog.js.map +1 -1
  61. package/dist/cjs/lib/logout-all-sessions-dialog.d.ts +9 -0
  62. package/dist/cjs/lib/logout-all-sessions-dialog.d.ts.map +1 -0
  63. package/dist/cjs/lib/logout-all-sessions-dialog.js +52 -0
  64. package/dist/cjs/lib/logout-all-sessions-dialog.js.map +1 -0
  65. package/dist/cjs/lib/logout-dialog.d.ts +10 -0
  66. package/dist/cjs/lib/logout-dialog.d.ts.map +1 -0
  67. package/dist/cjs/lib/logout-dialog.js +58 -0
  68. package/dist/cjs/lib/logout-dialog.js.map +1 -0
  69. package/dist/cjs/lib/marker.d.ts +14 -0
  70. package/dist/cjs/lib/marker.d.ts.map +1 -0
  71. package/dist/cjs/lib/marker.js +38 -0
  72. package/dist/cjs/lib/marker.js.map +1 -0
  73. package/dist/cjs/lib/oauth-icons.d.ts +4 -0
  74. package/dist/cjs/lib/oauth-icons.d.ts.map +1 -0
  75. package/dist/cjs/lib/oauth-icons.js +67 -0
  76. package/dist/cjs/lib/oauth-icons.js.map +1 -0
  77. package/dist/cjs/lib/organization-switcher.d.ts +24 -0
  78. package/dist/cjs/lib/organization-switcher.d.ts.map +1 -0
  79. package/dist/cjs/lib/organization-switcher.js +35 -0
  80. package/dist/cjs/lib/organization-switcher.js.map +1 -0
  81. package/dist/cjs/lib/otp-input.d.ts +20 -0
  82. package/dist/cjs/lib/otp-input.d.ts.map +1 -0
  83. package/dist/cjs/lib/otp-input.js +174 -0
  84. package/dist/cjs/lib/otp-input.js.map +1 -0
  85. package/dist/cjs/lib/resend-invite-dialog.d.ts +2 -2
  86. package/dist/cjs/lib/resend-invite-dialog.d.ts.map +1 -1
  87. package/dist/cjs/lib/resend-invite-dialog.js +1 -1
  88. package/dist/cjs/lib/resend-invite-dialog.js.map +1 -1
  89. package/dist/cjs/lib/reset-mfa-dialog.d.ts +9 -0
  90. package/dist/cjs/lib/reset-mfa-dialog.d.ts.map +1 -0
  91. package/dist/cjs/lib/reset-mfa-dialog.js +60 -0
  92. package/dist/cjs/lib/reset-mfa-dialog.js.map +1 -0
  93. package/dist/cjs/lib/revoke-invite-dialog.d.ts +2 -2
  94. package/dist/cjs/lib/revoke-invite-dialog.d.ts.map +1 -1
  95. package/dist/cjs/lib/revoke-invite-dialog.js +1 -1
  96. package/dist/cjs/lib/revoke-invite-dialog.js.map +1 -1
  97. package/dist/cjs/lib/save-button.d.ts +11 -0
  98. package/dist/cjs/lib/save-button.d.ts.map +1 -0
  99. package/dist/cjs/lib/save-button.js +47 -0
  100. package/dist/cjs/lib/save-button.js.map +1 -0
  101. package/dist/cjs/lib/set-password-dialog.d.ts +8 -0
  102. package/dist/cjs/lib/set-password-dialog.d.ts.map +1 -0
  103. package/dist/cjs/lib/set-password-dialog.js +80 -0
  104. package/dist/cjs/lib/set-password-dialog.js.map +1 -0
  105. package/dist/cjs/lib/use-dialog-close.d.ts +2 -0
  106. package/dist/cjs/lib/use-dialog-close.d.ts.map +1 -0
  107. package/dist/cjs/lib/use-dialog-close.js +43 -0
  108. package/dist/cjs/lib/use-dialog-close.js.map +1 -0
  109. package/dist/cjs/lib/use-security-settings.d.ts +11 -0
  110. package/dist/cjs/lib/use-security-settings.d.ts.map +1 -0
  111. package/dist/cjs/lib/use-security-settings.js +39 -0
  112. package/dist/cjs/lib/use-security-settings.js.map +1 -0
  113. package/dist/cjs/lib/user-actions-dropdown.d.ts +2 -2
  114. package/dist/cjs/lib/user-actions-dropdown.d.ts.map +1 -1
  115. package/dist/cjs/lib/user-actions-dropdown.js +8 -4
  116. package/dist/cjs/lib/user-actions-dropdown.js.map +1 -1
  117. package/dist/cjs/lib/user-profile.d.ts +11 -0
  118. package/dist/cjs/lib/user-profile.d.ts.map +1 -0
  119. package/dist/cjs/lib/user-profile.js +36 -0
  120. package/dist/cjs/lib/user-profile.js.map +1 -0
  121. package/dist/cjs/lib/user-security.d.ts +11 -0
  122. package/dist/cjs/lib/user-security.d.ts.map +1 -0
  123. package/dist/cjs/lib/user-security.js +64 -0
  124. package/dist/cjs/lib/user-security.js.map +1 -0
  125. package/dist/cjs/lib/user-sessions.d.ts +12 -0
  126. package/dist/cjs/lib/user-sessions.d.ts.map +1 -0
  127. package/dist/cjs/lib/user-sessions.js +72 -0
  128. package/dist/cjs/lib/user-sessions.js.map +1 -0
  129. package/dist/cjs/lib/users-filter.d.ts +2 -2
  130. package/dist/cjs/lib/users-filter.d.ts.map +1 -1
  131. package/dist/cjs/lib/users-filter.js.map +1 -1
  132. package/dist/cjs/lib/users-management-context.d.ts +0 -9
  133. package/dist/cjs/lib/users-management-context.d.ts.map +1 -1
  134. package/dist/cjs/lib/users-management-context.js +13 -26
  135. package/dist/cjs/lib/users-management-context.js.map +1 -1
  136. package/dist/cjs/lib/users-management-state.d.ts +3 -3
  137. package/dist/cjs/lib/users-management-state.d.ts.map +1 -1
  138. package/dist/cjs/lib/users-management-state.js.map +1 -1
  139. package/dist/cjs/lib/users-management.d.ts +3 -4
  140. package/dist/cjs/lib/users-management.d.ts.map +1 -1
  141. package/dist/cjs/lib/users-management.js +8 -26
  142. package/dist/cjs/lib/users-management.js.map +1 -1
  143. package/dist/cjs/lib/utils.d.ts +10 -2
  144. package/dist/cjs/lib/utils.d.ts.map +1 -1
  145. package/dist/cjs/lib/utils.js +18 -0
  146. package/dist/cjs/lib/utils.js.map +1 -1
  147. package/dist/cjs/organization-switcher.client.d.ts +8 -0
  148. package/dist/cjs/organization-switcher.client.d.ts.map +1 -0
  149. package/dist/cjs/organization-switcher.client.js +37 -0
  150. package/dist/cjs/organization-switcher.client.js.map +1 -0
  151. package/dist/cjs/user-profile.client.d.ts +7 -0
  152. package/dist/cjs/user-profile.client.d.ts.map +1 -0
  153. package/dist/cjs/user-profile.client.js +31 -0
  154. package/dist/cjs/user-profile.client.js.map +1 -0
  155. package/dist/cjs/user-security.client.d.ts +7 -0
  156. package/dist/cjs/user-security.client.d.ts.map +1 -0
  157. package/dist/cjs/user-security.client.js +27 -0
  158. package/dist/cjs/user-security.client.js.map +1 -0
  159. package/dist/cjs/user-sessions.client.d.ts +12 -0
  160. package/dist/cjs/user-sessions.client.d.ts.map +1 -0
  161. package/dist/cjs/user-sessions.client.js +48 -0
  162. package/dist/cjs/user-sessions.client.js.map +1 -0
  163. package/dist/cjs/users-management.client.d.ts +2 -1
  164. package/dist/cjs/users-management.client.d.ts.map +1 -1
  165. package/dist/cjs/users-management.client.js +12 -43
  166. package/dist/cjs/users-management.client.js.map +1 -1
  167. package/dist/esm/card-list.d.ts +6 -0
  168. package/dist/esm/card-list.d.ts.map +1 -0
  169. package/dist/esm/card-list.js +9 -0
  170. package/dist/esm/card-list.js.map +1 -0
  171. package/dist/esm/index.d.ts +3 -0
  172. package/dist/esm/index.d.ts.map +1 -1
  173. package/dist/esm/index.js +3 -0
  174. package/dist/esm/index.js.map +1 -1
  175. package/dist/esm/lib/add-mfa-dialog.d.ts +9 -0
  176. package/dist/esm/lib/add-mfa-dialog.d.ts.map +1 -0
  177. package/dist/esm/lib/add-mfa-dialog.js +109 -0
  178. package/dist/esm/lib/add-mfa-dialog.js.map +1 -0
  179. package/dist/esm/lib/api/user.d.ts +13 -57
  180. package/dist/esm/lib/api/user.d.ts.map +1 -1
  181. package/dist/esm/lib/api/user.js +44 -313
  182. package/dist/esm/lib/api/user.js.map +1 -1
  183. package/dist/esm/lib/change-password-dialog.d.ts +8 -0
  184. package/dist/esm/lib/change-password-dialog.d.ts.map +1 -0
  185. package/dist/esm/lib/change-password-dialog.js +71 -0
  186. package/dist/esm/lib/change-password-dialog.js.map +1 -0
  187. package/dist/esm/lib/copy-button.d.ts +8 -0
  188. package/dist/esm/lib/copy-button.d.ts.map +1 -0
  189. package/dist/esm/lib/copy-button.js +37 -0
  190. package/dist/esm/lib/copy-button.js.map +1 -0
  191. package/dist/esm/lib/delete-user-dialog.d.ts +2 -2
  192. package/dist/esm/lib/delete-user-dialog.d.ts.map +1 -1
  193. package/dist/esm/lib/delete-user-dialog.js +1 -1
  194. package/dist/esm/lib/delete-user-dialog.js.map +1 -1
  195. package/dist/esm/lib/edit-user-profile-dialog.d.ts +10 -0
  196. package/dist/esm/lib/edit-user-profile-dialog.d.ts.map +1 -0
  197. package/dist/esm/lib/edit-user-profile-dialog.js +59 -0
  198. package/dist/esm/lib/edit-user-profile-dialog.js.map +1 -0
  199. package/dist/esm/lib/edit-user-role-dialog.d.ts +12 -0
  200. package/dist/esm/lib/edit-user-role-dialog.d.ts.map +1 -0
  201. package/dist/esm/lib/{edit-user-details-dialog.js → edit-user-role-dialog.js} +10 -6
  202. package/dist/esm/lib/edit-user-role-dialog.js.map +1 -0
  203. package/dist/esm/lib/elements.d.ts +1 -0
  204. package/dist/esm/lib/elements.d.ts.map +1 -1
  205. package/dist/esm/lib/elements.js +9 -3
  206. package/dist/esm/lib/elements.js.map +1 -1
  207. package/dist/esm/lib/elevated-access.d.ts +8 -0
  208. package/dist/esm/lib/elevated-access.d.ts.map +1 -0
  209. package/dist/esm/lib/elevated-access.js +104 -0
  210. package/dist/esm/lib/elevated-access.js.map +1 -0
  211. package/dist/esm/lib/generic-error.d.ts +4 -0
  212. package/dist/esm/lib/generic-error.d.ts.map +1 -0
  213. package/dist/esm/lib/generic-error.js +31 -0
  214. package/dist/esm/lib/generic-error.js.map +1 -0
  215. package/dist/esm/lib/icon-panel.d.ts +3 -0
  216. package/dist/esm/lib/icon-panel.d.ts.map +1 -0
  217. package/dist/esm/lib/icon-panel.js +13 -0
  218. package/dist/esm/lib/icon-panel.js.map +1 -0
  219. package/dist/esm/lib/icons.d.ts +3 -0
  220. package/dist/esm/lib/icons.d.ts.map +1 -0
  221. package/dist/esm/lib/icons.js +5 -0
  222. package/dist/esm/lib/icons.js.map +1 -0
  223. package/dist/esm/lib/invite-user-dialog.d.ts.map +1 -1
  224. package/dist/esm/lib/invite-user-dialog.js +7 -5
  225. package/dist/esm/lib/invite-user-dialog.js.map +1 -1
  226. package/dist/esm/lib/logout-all-sessions-dialog.d.ts +9 -0
  227. package/dist/esm/lib/logout-all-sessions-dialog.d.ts.map +1 -0
  228. package/dist/esm/lib/logout-all-sessions-dialog.js +26 -0
  229. package/dist/esm/lib/logout-all-sessions-dialog.js.map +1 -0
  230. package/dist/esm/lib/logout-dialog.d.ts +10 -0
  231. package/dist/esm/lib/logout-dialog.d.ts.map +1 -0
  232. package/dist/esm/lib/logout-dialog.js +32 -0
  233. package/dist/esm/lib/logout-dialog.js.map +1 -0
  234. package/dist/esm/lib/marker.d.ts +14 -0
  235. package/dist/esm/lib/marker.d.ts.map +1 -0
  236. package/dist/esm/lib/marker.js +9 -0
  237. package/dist/esm/lib/marker.js.map +1 -0
  238. package/dist/esm/lib/oauth-icons.d.ts +4 -0
  239. package/dist/esm/lib/oauth-icons.d.ts.map +1 -0
  240. package/dist/esm/lib/oauth-icons.js +39 -0
  241. package/dist/esm/lib/oauth-icons.js.map +1 -0
  242. package/dist/esm/lib/organization-switcher.d.ts +24 -0
  243. package/dist/esm/lib/organization-switcher.d.ts.map +1 -0
  244. package/dist/esm/lib/organization-switcher.js +29 -0
  245. package/dist/esm/lib/organization-switcher.js.map +1 -0
  246. package/dist/esm/lib/otp-input.d.ts +20 -0
  247. package/dist/esm/lib/otp-input.d.ts.map +1 -0
  248. package/dist/esm/lib/otp-input.js +148 -0
  249. package/dist/esm/lib/otp-input.js.map +1 -0
  250. package/dist/esm/lib/resend-invite-dialog.d.ts +2 -2
  251. package/dist/esm/lib/resend-invite-dialog.d.ts.map +1 -1
  252. package/dist/esm/lib/resend-invite-dialog.js +1 -1
  253. package/dist/esm/lib/resend-invite-dialog.js.map +1 -1
  254. package/dist/esm/lib/reset-mfa-dialog.d.ts +9 -0
  255. package/dist/esm/lib/reset-mfa-dialog.d.ts.map +1 -0
  256. package/dist/esm/lib/reset-mfa-dialog.js +34 -0
  257. package/dist/esm/lib/reset-mfa-dialog.js.map +1 -0
  258. package/dist/esm/lib/revoke-invite-dialog.d.ts +2 -2
  259. package/dist/esm/lib/revoke-invite-dialog.d.ts.map +1 -1
  260. package/dist/esm/lib/revoke-invite-dialog.js +1 -1
  261. package/dist/esm/lib/revoke-invite-dialog.js.map +1 -1
  262. package/dist/esm/lib/save-button.d.ts +11 -0
  263. package/dist/esm/lib/save-button.d.ts.map +1 -0
  264. package/dist/esm/lib/save-button.js +44 -0
  265. package/dist/esm/lib/save-button.js.map +1 -0
  266. package/dist/esm/lib/set-password-dialog.d.ts +8 -0
  267. package/dist/esm/lib/set-password-dialog.d.ts.map +1 -0
  268. package/dist/esm/lib/set-password-dialog.js +54 -0
  269. package/dist/esm/lib/set-password-dialog.js.map +1 -0
  270. package/dist/esm/lib/use-dialog-close.d.ts +2 -0
  271. package/dist/esm/lib/use-dialog-close.d.ts.map +1 -0
  272. package/dist/esm/lib/use-dialog-close.js +17 -0
  273. package/dist/esm/lib/use-dialog-close.js.map +1 -0
  274. package/dist/esm/lib/use-security-settings.d.ts +11 -0
  275. package/dist/esm/lib/use-security-settings.d.ts.map +1 -0
  276. package/dist/esm/lib/use-security-settings.js +36 -0
  277. package/dist/esm/lib/use-security-settings.js.map +1 -0
  278. package/dist/esm/lib/user-actions-dropdown.d.ts +2 -2
  279. package/dist/esm/lib/user-actions-dropdown.d.ts.map +1 -1
  280. package/dist/esm/lib/user-actions-dropdown.js +8 -4
  281. package/dist/esm/lib/user-actions-dropdown.js.map +1 -1
  282. package/dist/esm/lib/user-profile.d.ts +11 -0
  283. package/dist/esm/lib/user-profile.d.ts.map +1 -0
  284. package/dist/esm/lib/user-profile.js +27 -0
  285. package/dist/esm/lib/user-profile.js.map +1 -0
  286. package/dist/esm/lib/user-security.d.ts +11 -0
  287. package/dist/esm/lib/user-security.d.ts.map +1 -0
  288. package/dist/esm/lib/user-security.js +32 -0
  289. package/dist/esm/lib/user-security.js.map +1 -0
  290. package/dist/esm/lib/user-sessions.d.ts +12 -0
  291. package/dist/esm/lib/user-sessions.d.ts.map +1 -0
  292. package/dist/esm/lib/user-sessions.js +40 -0
  293. package/dist/esm/lib/user-sessions.js.map +1 -0
  294. package/dist/esm/lib/users-filter.d.ts +2 -2
  295. package/dist/esm/lib/users-filter.d.ts.map +1 -1
  296. package/dist/esm/lib/users-filter.js.map +1 -1
  297. package/dist/esm/lib/users-management-context.d.ts +0 -9
  298. package/dist/esm/lib/users-management-context.d.ts.map +1 -1
  299. package/dist/esm/lib/users-management-context.js +13 -25
  300. package/dist/esm/lib/users-management-context.js.map +1 -1
  301. package/dist/esm/lib/users-management-state.d.ts +3 -3
  302. package/dist/esm/lib/users-management-state.d.ts.map +1 -1
  303. package/dist/esm/lib/users-management-state.js.map +1 -1
  304. package/dist/esm/lib/users-management.d.ts +3 -4
  305. package/dist/esm/lib/users-management.d.ts.map +1 -1
  306. package/dist/esm/lib/users-management.js +9 -27
  307. package/dist/esm/lib/users-management.js.map +1 -1
  308. package/dist/esm/lib/users-search.d.ts +1 -1
  309. package/dist/esm/lib/users-search.d.ts.map +1 -1
  310. package/dist/esm/lib/utils.d.ts +10 -2
  311. package/dist/esm/lib/utils.d.ts.map +1 -1
  312. package/dist/esm/lib/utils.js +16 -0
  313. package/dist/esm/lib/utils.js.map +1 -1
  314. package/dist/esm/organization-switcher.client.d.ts +8 -0
  315. package/dist/esm/organization-switcher.client.d.ts.map +1 -0
  316. package/dist/esm/organization-switcher.client.js +33 -0
  317. package/dist/esm/organization-switcher.client.js.map +1 -0
  318. package/dist/esm/user-profile.client.d.ts +7 -0
  319. package/dist/esm/user-profile.client.d.ts.map +1 -0
  320. package/dist/esm/user-profile.client.js +27 -0
  321. package/dist/esm/user-profile.client.js.map +1 -0
  322. package/dist/esm/user-security.client.d.ts +7 -0
  323. package/dist/esm/user-security.client.d.ts.map +1 -0
  324. package/dist/esm/user-security.client.js +23 -0
  325. package/dist/esm/user-security.client.js.map +1 -0
  326. package/dist/esm/user-sessions.client.d.ts +12 -0
  327. package/dist/esm/user-sessions.client.d.ts.map +1 -0
  328. package/dist/esm/user-sessions.client.js +44 -0
  329. package/dist/esm/user-sessions.client.js.map +1 -0
  330. package/dist/esm/users-management.client.d.ts +2 -1
  331. package/dist/esm/users-management.client.d.ts.map +1 -1
  332. package/dist/esm/users-management.client.js +12 -20
  333. package/dist/esm/users-management.client.js.map +1 -1
  334. package/dist/tsconfig.cjs.tsbuildinfo +1 -1
  335. package/dist/tsconfig.esm.tsbuildinfo +1 -1
  336. package/package.json +10 -4
  337. package/src/card-list.tsx +26 -0
  338. package/src/index.ts +3 -0
  339. package/src/lib/add-mfa-dialog.tsx +379 -0
  340. package/src/lib/api/user.ts +54 -458
  341. package/src/lib/card-list.css +3 -0
  342. package/src/lib/change-password-dialog.tsx +290 -0
  343. package/src/lib/copy-button.tsx +53 -0
  344. package/src/lib/delete-user-dialog.tsx +9 -6
  345. package/src/lib/edit-user-profile-dialog.tsx +181 -0
  346. package/src/lib/{edit-user-details-dialog.tsx → edit-user-role-dialog.tsx} +12 -9
  347. package/src/lib/elements.tsx +34 -1
  348. package/src/lib/elevated-access.tsx +261 -0
  349. package/src/lib/generic-error.tsx +70 -0
  350. package/src/lib/icon-panel.tsx +26 -0
  351. package/src/lib/icons.tsx +21 -0
  352. package/src/lib/invite-user-dialog.tsx +15 -10
  353. package/src/lib/logout-all-sessions-dialog.tsx +82 -0
  354. package/src/lib/logout-dialog.tsx +89 -0
  355. package/src/lib/marker.css +81 -0
  356. package/src/lib/marker.tsx +39 -0
  357. package/src/lib/oauth-icons.tsx +138 -0
  358. package/src/lib/organization-switcher.tsx +160 -0
  359. package/src/lib/otp-input.tsx +276 -0
  360. package/src/lib/resend-invite-dialog.tsx +9 -6
  361. package/src/lib/reset-mfa-dialog.tsx +104 -0
  362. package/src/lib/revoke-invite-dialog.tsx +9 -6
  363. package/src/lib/save-button.css +60 -0
  364. package/src/lib/save-button.tsx +113 -0
  365. package/src/lib/set-password-dialog.tsx +204 -0
  366. package/src/lib/use-dialog-close.tsx +19 -0
  367. package/src/lib/use-security-settings.tsx +49 -0
  368. package/src/lib/user-actions-dropdown.tsx +10 -6
  369. package/src/lib/user-profile.tsx +247 -0
  370. package/src/lib/user-security.tsx +187 -0
  371. package/src/lib/user-sessions.tsx +204 -0
  372. package/src/lib/users-filter.tsx +2 -2
  373. package/src/lib/users-management-context.tsx +21 -36
  374. package/src/lib/users-management-state.ts +3 -3
  375. package/src/lib/users-management.tsx +21 -77
  376. package/src/lib/utils.ts +30 -2
  377. package/src/organization-switcher.client.tsx +77 -0
  378. package/src/styles.css +44 -0
  379. package/src/user-profile.client.tsx +51 -0
  380. package/src/user-security.client.tsx +55 -0
  381. package/src/user-sessions.client.tsx +96 -0
  382. package/src/users-management.client.tsx +28 -39
  383. package/dist/cjs/lib/api/role.d.ts +0 -9
  384. package/dist/cjs/lib/api/role.d.ts.map +0 -1
  385. package/dist/cjs/lib/api/role.js +0 -115
  386. package/dist/cjs/lib/api/role.js.map +0 -1
  387. package/dist/cjs/lib/edit-user-details-dialog.d.ts +0 -12
  388. package/dist/cjs/lib/edit-user-details-dialog.d.ts.map +0 -1
  389. package/dist/cjs/lib/edit-user-details-dialog.js.map +0 -1
  390. package/dist/esm/lib/api/role.d.ts +0 -9
  391. package/dist/esm/lib/api/role.d.ts.map +0 -1
  392. package/dist/esm/lib/api/role.js +0 -110
  393. package/dist/esm/lib/api/role.js.map +0 -1
  394. package/dist/esm/lib/edit-user-details-dialog.d.ts +0 -12
  395. package/dist/esm/lib/edit-user-details-dialog.d.ts.map +0 -1
  396. package/dist/esm/lib/edit-user-details-dialog.js.map +0 -1
  397. package/src/lib/api/role.ts +0 -147
@@ -0,0 +1,39 @@
1
+ import * as React from "react";
2
+ import { Text } from "@radix-ui/themes";
3
+ import { MarginProps } from "@radix-ui/themes/props";
4
+ import clsx from "clsx";
5
+ import { namespaceClassNames } from "./utils";
6
+
7
+ type TextProps = React.ComponentPropsWithoutRef<typeof Text>;
8
+
9
+ type MarkerOwnProps = {
10
+ color?: "gray" | "purple" | "blue" | "green" | "yellow" | "red";
11
+ highContrast?: boolean;
12
+ size?: TextProps["size"];
13
+ };
14
+
15
+ interface MarkerProps
16
+ extends Omit<React.ComponentPropsWithRef<"span">, "color">,
17
+ MarkerOwnProps,
18
+ MarginProps {}
19
+
20
+ export const Marker = React.forwardRef<HTMLSpanElement, MarkerProps>(
21
+ function Marker(
22
+ { children, className, highContrast, ...props },
23
+ forwardedRef,
24
+ ) {
25
+ return (
26
+ <Text
27
+ ref={forwardedRef}
28
+ className={clsx(className, namespaceClassNames("marker"))}
29
+ {...props}
30
+ >
31
+ <span className={namespaceClassNames("marker-circle")}>
32
+ <span className={namespaceClassNames("marker-content")}>
33
+ {children}
34
+ </span>
35
+ </span>
36
+ </Text>
37
+ );
38
+ },
39
+ );
@@ -0,0 +1,138 @@
1
+ import * as React from "react";
2
+
3
+ const GitHub = React.forwardRef<
4
+ SVGSVGElement,
5
+ React.ComponentPropsWithoutRef<"svg">
6
+ >((props, forwardedRef) => (
7
+ <svg
8
+ ref={forwardedRef}
9
+ fill="none"
10
+ height="16"
11
+ viewBox="0 0 15 15"
12
+ width="16"
13
+ xmlns="http://www.w3.org/2000/svg"
14
+ {...props}
15
+ >
16
+ <path
17
+ clipRule="evenodd"
18
+ d="M7.49933 0.25C3.49635 0.25 0.25 3.49593 0.25 7.50024C0.25 10.703 2.32715 13.4206 5.2081 14.3797C5.57084 14.446 5.70302 14.2222 5.70302 14.0299C5.70302 13.8576 5.69679 13.4019 5.69323 12.797C3.67661 13.235 3.25112 11.825 3.25112 11.825C2.92132 10.9874 2.44599 10.7644 2.44599 10.7644C1.78773 10.3149 2.49584 10.3238 2.49584 10.3238C3.22353 10.375 3.60629 11.0711 3.60629 11.0711C4.25298 12.1788 5.30335 11.8588 5.71638 11.6732C5.78225 11.205 5.96962 10.8854 6.17658 10.7043C4.56675 10.5209 2.87415 9.89918 2.87415 7.12104C2.87415 6.32925 3.15677 5.68257 3.62053 5.17563C3.54576 4.99226 3.29697 4.25521 3.69174 3.25691C3.69174 3.25691 4.30015 3.06196 5.68522 3.99973C6.26337 3.83906 6.8838 3.75895 7.50022 3.75583C8.1162 3.75895 8.73619 3.83906 9.31523 3.99973C10.6994 3.06196 11.3069 3.25691 11.3069 3.25691C11.7026 4.25521 11.4538 4.99226 11.3795 5.17563C11.8441 5.68257 12.1245 6.32925 12.1245 7.12104C12.1245 9.9063 10.4292 10.5192 8.81452 10.6985C9.07444 10.9224 9.30633 11.3648 9.30633 12.0413C9.30633 13.0102 9.29742 13.7922 9.29742 14.0299C9.29742 14.2239 9.42828 14.4496 9.79591 14.3788C12.6746 13.4179 14.75 10.7025 14.75 7.50024C14.75 3.49593 11.5036 0.25 7.49933 0.25Z"
19
+ fill="currentColor"
20
+ fillRule="evenodd"
21
+ ></path>
22
+ </svg>
23
+ ));
24
+
25
+ GitHub.displayName = "GitHub";
26
+
27
+ const Google = React.forwardRef<
28
+ SVGSVGElement,
29
+ React.ComponentPropsWithoutRef<"svg">
30
+ >((props, forwardedRef) => (
31
+ <svg
32
+ ref={forwardedRef}
33
+ fill="none"
34
+ height="15"
35
+ viewBox="0 0 16 16"
36
+ width="15"
37
+ xmlns="http://www.w3.org/2000/svg"
38
+ {...props}
39
+ >
40
+ <g>
41
+ <path
42
+ d="M15.83 8.18C15.83 7.65333 15.7833 7.15333 15.7033 6.66667H8.17V9.67333H12.4833C12.29 10.66 11.7233 11.4933 10.8833 12.06V14.06H13.4567C14.9633 12.6667 15.83 10.6133 15.83 8.18Z"
43
+ fill="#4285F4"
44
+ />
45
+ <path
46
+ d="M8.17 16C10.33 16 12.1367 15.28 13.4567 14.06L10.8833 12.06C10.1633 12.54 9.25 12.8333 8.17 12.8333C6.08334 12.8333 4.31667 11.4267 3.68334 9.52667H1.03V11.5867C2.34334 14.2 5.04334 16 8.17 16Z"
47
+ fill="#34A853"
48
+ />
49
+ <path
50
+ d="M3.68334 9.52667C3.51667 9.04667 3.43 8.53333 3.43 8C3.43 7.46667 3.52334 6.95334 3.68334 6.47334V4.41334H1.03C0.483335 5.49334 0.170002 6.70667 0.170002 8C0.170002 9.29333 0.483335 10.5067 1.03 11.5867L3.68334 9.52667Z"
51
+ fill="#FBBC05"
52
+ />
53
+ <path
54
+ d="M8.17 3.16667C9.35 3.16667 10.4033 3.57334 11.2367 4.36667L13.5167 2.08667C12.1367 0.793334 10.33 0 8.17 0C5.04334 0 2.34334 1.8 1.03 4.41334L3.68334 6.47334C4.31667 4.57334 6.08334 3.16667 8.17 3.16667Z"
55
+ fill="#EA4335"
56
+ />
57
+ </g>
58
+ </svg>
59
+ ));
60
+
61
+ Google.displayName = "Google";
62
+
63
+ const Microsoft = React.forwardRef<
64
+ SVGSVGElement,
65
+ React.ComponentPropsWithoutRef<"svg">
66
+ >((props, forwardedRef) => (
67
+ <svg
68
+ ref={forwardedRef}
69
+ fill="none"
70
+ height="15"
71
+ viewBox="0 0 15 15"
72
+ width="15"
73
+ xmlns="http://www.w3.org/2000/svg"
74
+ {...props}
75
+ >
76
+ <g>
77
+ <path d="M0 0H7L7 7H-4.76837e-07L0 0Z" fill="#F35325" />
78
+ <path d="M8 0H15V7H8L8 0Z" fill="#81BC06" />
79
+ <path d="M0 8H7V15H0V8Z" fill="#05A6F0" />
80
+ <path d="M8 8L15 8V15L8 15V8Z" fill="#FFBA08" />
81
+ </g>
82
+ </svg>
83
+ ));
84
+
85
+ Microsoft.displayName = "Microsoft";
86
+
87
+ const Apple = React.forwardRef<
88
+ SVGSVGElement,
89
+ React.ComponentPropsWithoutRef<"svg">
90
+ >(({ style, ...props }, forwardedRef) => (
91
+ <svg
92
+ ref={forwardedRef}
93
+ fill="none"
94
+ height="15"
95
+ style={{ overflow: "visible", ...style }}
96
+ viewBox="0 0 15 15"
97
+ width="15"
98
+ xmlns="http://www.w3.org/2000/svg"
99
+ {...props}
100
+ >
101
+ <path
102
+ d="M14.219 3.33667C14.1169 3.41674 12.3137 4.44312 12.3137 6.72534C12.3137 9.3651 14.6082 10.299 14.6769 10.3221C14.6663 10.379 14.3124 11.601 13.4671 12.8462C12.7134 13.942 11.9263 15.0359 10.7288 15.0359C9.53134 15.0359 9.22317 14.3333 7.84081 14.3333C6.49366 14.3333 6.01469 15.0591 4.91935 15.0591C3.82401 15.0591 3.05978 14.0451 2.18104 12.8C1.1632 11.3378 0.34082 9.06625 0.34082 6.91034C0.34082 3.45232 2.56668 1.61835 4.75732 1.61835C5.92133 1.61835 6.89164 2.39036 7.62246 2.39036C8.31802 2.39036 9.40277 1.5721 10.7271 1.5721C11.2289 1.5721 13.0321 1.61835 14.219 3.33667ZM10.0984 0.108136C10.646 -0.548247 11.0334 -1.459 11.0334 -2.36975C11.0334 -2.49605 11.0229 -2.62412 11 -2.72729C10.1089 -2.6935 9.04883 -2.12783 8.40959 -1.37895C7.90772 -0.802617 7.43931 0.108136 7.43931 1.03134C7.43931 1.1701 7.46218 1.30883 7.47277 1.35331C7.52909 1.36398 7.62067 1.37643 7.71224 1.37643C8.51172 1.37643 9.51724 0.835672 10.0984 0.108136Z"
103
+ fill="currentColor"
104
+ />
105
+ </svg>
106
+ ));
107
+
108
+ Apple.displayName = "Apple";
109
+
110
+ export const getOAuthIcon = (account: string) => {
111
+ switch (account) {
112
+ case "GithubOAuth":
113
+ return GitHub;
114
+ case "GoogleOAuth":
115
+ return Google;
116
+ case "MicrosoftOAuth":
117
+ return Microsoft;
118
+ case "AppleOAuth":
119
+ return Apple;
120
+ default:
121
+ throw new Error(`Unknown OAuth account type: ${account}`);
122
+ }
123
+ };
124
+
125
+ export const getOAuthName = (account: string) => {
126
+ switch (account) {
127
+ case "GithubOAuth":
128
+ return "GitHub";
129
+ case "GoogleOAuth":
130
+ return "Google";
131
+ case "MicrosoftOAuth":
132
+ return "Microsoft";
133
+ case "AppleOAuth":
134
+ return "Apple";
135
+ default:
136
+ throw new Error(`Unknown OAuth account type: ${account}`);
137
+ }
138
+ };
@@ -0,0 +1,160 @@
1
+ "use client";
2
+
3
+ import { CheckIcon } from "@radix-ui/react-icons";
4
+ import {
5
+ Box,
6
+ Button,
7
+ ChevronDownIcon,
8
+ DropdownMenu,
9
+ Flex,
10
+ Skeleton,
11
+ Text,
12
+ VisuallyHidden,
13
+ } from "@radix-ui/themes";
14
+ import {
15
+ OrganizationInfo,
16
+ Organizations403,
17
+ Organizations404,
18
+ } from "@repo/api";
19
+
20
+ type OrganizationSwitcherVariant = "ghost" | "outline";
21
+
22
+ // Rename all uses of `org` to `organization`
23
+ export type OrganizationSwitcherPassthroughProps = {
24
+ switchToOrganization: ({
25
+ organizationId,
26
+ }: {
27
+ organizationId: string;
28
+ }) => void;
29
+ // Simple props to affect the overall style
30
+ variant?: OrganizationSwitcherVariant;
31
+ organizationLabel?: string | null;
32
+ children?: React.ReactNode;
33
+ };
34
+
35
+ export interface OrganizationSwitcherProps
36
+ extends OrganizationSwitcherPassthroughProps {
37
+ organizations: OrganizationInfo[];
38
+ }
39
+
40
+ export const OrganizationSwitcher = ({
41
+ organizations,
42
+ switchToOrganization,
43
+ variant = "outline",
44
+ organizationLabel = "Organizations",
45
+ children,
46
+ }: OrganizationSwitcherProps) => {
47
+ const currentOrganization = organizations.find(
48
+ (organization) => organization.current,
49
+ );
50
+
51
+ // Possible if the user has no organizations - we should figure out what to do in this case
52
+ if (!currentOrganization) {
53
+ return null;
54
+ }
55
+
56
+ return (
57
+ <DropdownMenu.Root>
58
+ <DropdownMenu.Trigger>
59
+ <Button
60
+ color="gray"
61
+ variant={variant}
62
+ className="OrganizationSwitcherTrigger"
63
+ >
64
+ <Flex align="center" justify="between" gap="2" flexGrow="1">
65
+ <Text>{currentOrganization.name}</Text>
66
+ <DropdownMenu.TriggerIcon />
67
+ </Flex>
68
+ </Button>
69
+ </DropdownMenu.Trigger>
70
+ <DropdownMenu.Content>
71
+ <DropdownMenu.Group>
72
+ {organizationLabel ? (
73
+ <DropdownMenu.Label>
74
+ <Text>{organizationLabel}</Text>
75
+ </DropdownMenu.Label>
76
+ ) : null}
77
+ {organizations.map((organization) => (
78
+ <Flex
79
+ key={organization.id}
80
+ asChild
81
+ pr="2"
82
+ maxWidth="280px"
83
+ minWidth="180px"
84
+ >
85
+ <DropdownMenu.Item
86
+ onClick={() => {
87
+ if (organization.id !== currentOrganization.id) {
88
+ switchToOrganization({ organizationId: organization.id });
89
+ }
90
+ }}
91
+ >
92
+ <Flex
93
+ justify="between"
94
+ align="center"
95
+ gap="4"
96
+ flexGrow="1"
97
+ overflow="hidden"
98
+ >
99
+ <Text truncate>
100
+ {organization.name}
101
+ {organization.current && (
102
+ <VisuallyHidden> (current)</VisuallyHidden>
103
+ )}
104
+ </Text>
105
+ <Flex
106
+ aria-hidden
107
+ align="center"
108
+ justify="center"
109
+ flexShrink="0"
110
+ >
111
+ {organization.current ? (
112
+ <CheckIcon width="18px" height="18px" />
113
+ ) : (
114
+ // make the extra space for
115
+ <Box width="18px" height="18px" />
116
+ )}
117
+ </Flex>
118
+ </Flex>
119
+ </DropdownMenu.Item>
120
+ </Flex>
121
+ ))}
122
+ </DropdownMenu.Group>
123
+ {children}
124
+ </DropdownMenu.Content>
125
+ </DropdownMenu.Root>
126
+ );
127
+ };
128
+
129
+ export function OrganizationSwitcherLoading({
130
+ variant = "outline",
131
+ organizations,
132
+ }: {
133
+ variant?: OrganizationSwitcherVariant;
134
+ organizations: OrganizationInfo[];
135
+ }) {
136
+ const currentOrganization = organizations.find(
137
+ (organization) => organization.current,
138
+ );
139
+
140
+ return (
141
+ // Always need DropdownMenu.Root to wrap children than may include
142
+ <Button color="gray" variant={variant} disabled>
143
+ <Flex align="center" gap="2">
144
+ <Skeleton loading={!currentOrganization}>
145
+ <Text>{currentOrganization?.name ?? "Loading..."}</Text>
146
+ </Skeleton>
147
+ <ChevronDownIcon />
148
+ </Flex>
149
+ </Button>
150
+ );
151
+ }
152
+
153
+ interface OrganizationSwitcherErrorProps {
154
+ error: Organizations403 | Organizations404;
155
+ }
156
+
157
+ export function OrganizationSwitcherError(_: OrganizationSwitcherErrorProps) {
158
+ // TODO: consider other error state options
159
+ return null;
160
+ }
@@ -0,0 +1,276 @@
1
+ "use client";
2
+
3
+ import { composeRefs, useComposedRefs } from "@radix-ui/react-compose-refs";
4
+ import { useControllableState } from "@radix-ui/react-use-controllable-state";
5
+ import * as Form from "@radix-ui/react-form";
6
+ import { Grid } from "@radix-ui/themes";
7
+ import * as React from "react";
8
+ import { TextField } from "./elements";
9
+
10
+ interface OptContextType {
11
+ value: string[];
12
+ readOnly?: boolean;
13
+ state?: "valid" | "invalid";
14
+ onEnterPressed: () => void;
15
+ onChildAdd: (input: HTMLInputElement) => void;
16
+ onCharChange: (char: string, index: number) => void;
17
+ allChildrenAdded: boolean;
18
+ }
19
+
20
+ const OtpContext = React.createContext<OptContextType | undefined>(undefined);
21
+
22
+ type OtpRootProps = React.ComponentPropsWithoutRef<typeof Grid> & {
23
+ onValueChange?: (value: string) => void;
24
+ id?: string;
25
+ name?: string;
26
+ readOnly?: boolean;
27
+ state?: "valid" | "invalid";
28
+ value?: string;
29
+ defaultValue?: string;
30
+ autoSubmit?: boolean;
31
+ };
32
+
33
+ export const Root = React.forwardRef<HTMLInputElement, OtpRootProps>(
34
+ function Root(
35
+ {
36
+ name,
37
+ id,
38
+ defaultValue,
39
+ value: valueProp,
40
+ onValueChange,
41
+ autoSubmit,
42
+ children,
43
+ readOnly,
44
+ state,
45
+ ...gridProps
46
+ },
47
+ forwardedRef,
48
+ ) {
49
+ const [lastCharIndex, setLastCharIndex] = React.useState<number>(0);
50
+ const [allChildrenAdded, setAllChildrenAdded] =
51
+ React.useState<boolean>(false);
52
+ const childCount = React.Children.count(children);
53
+
54
+ const [value, setValue] = useControllableState({
55
+ prop: getValueAsArray(valueProp, childCount),
56
+ defaultProp: getValueAsArray(defaultValue, childCount),
57
+ onChange: (value) => onValueChange?.(value.join("")),
58
+ });
59
+
60
+ const hiddenInputRef = React.useRef<HTMLInputElement>(null);
61
+ const childrenRefs = React.useRef<HTMLInputElement[]>([]);
62
+
63
+ const attemptAutoSubmit = React.useCallback(
64
+ (enterPressed = false) => {
65
+ if (
66
+ autoSubmit &&
67
+ value &&
68
+ value.every((char) => char !== "") &&
69
+ (enterPressed || lastCharIndex + 1 === childCount)
70
+ ) {
71
+ hiddenInputRef.current?.form?.requestSubmit();
72
+ }
73
+ },
74
+ [value, childCount, lastCharIndex, autoSubmit],
75
+ );
76
+
77
+ const handleEnterPressed = React.useCallback(
78
+ () => attemptAutoSubmit(true),
79
+ [attemptAutoSubmit],
80
+ );
81
+
82
+ const handleChildAdd = React.useCallback(
83
+ (input: HTMLInputElement) => {
84
+ if (input) {
85
+ input.dataset.index = `${childrenRefs.current.length}`;
86
+ childrenRefs.current.push(input);
87
+ } else {
88
+ childrenRefs.current.pop();
89
+ }
90
+
91
+ if (childrenRefs.current.length === childCount) {
92
+ setAllChildrenAdded(true);
93
+ }
94
+ },
95
+ [childCount],
96
+ );
97
+
98
+ const handleCharChange = React.useCallback(
99
+ (char: string, index: number) => {
100
+ setValue((previousValue) => {
101
+ const arrayToCopy = previousValue ?? createEmptyArray(childCount);
102
+ const newValue = [...arrayToCopy];
103
+ newValue[index] = char;
104
+ return newValue;
105
+ });
106
+ setLastCharIndex(index);
107
+ },
108
+ [childCount, setValue],
109
+ );
110
+
111
+ const otpContext = React.useMemo(
112
+ () => ({
113
+ value: value ?? createEmptyArray(childCount),
114
+ readOnly,
115
+ state,
116
+ allChildrenAdded,
117
+ onEnterPressed: handleEnterPressed,
118
+ onChildAdd: handleChildAdd,
119
+ onCharChange: handleCharChange,
120
+ }),
121
+ [
122
+ value,
123
+ allChildrenAdded,
124
+ readOnly,
125
+ state,
126
+ childCount,
127
+ handleEnterPressed,
128
+ handleChildAdd,
129
+ handleCharChange,
130
+ ],
131
+ );
132
+
133
+ React.useEffect(attemptAutoSubmit, [attemptAutoSubmit]);
134
+
135
+ return (
136
+ <OtpContext.Provider value={otpContext}>
137
+ <Grid
138
+ columns={`repeat(${childCount}, 1fr)`}
139
+ {...gridProps}
140
+ onPaste={(event: React.ClipboardEvent<HTMLDivElement>) => {
141
+ event.preventDefault();
142
+ const pastedValue = event.clipboardData.getData("Text");
143
+ const sanitizedValue = pastedValue
144
+ .replace(/[^\d]/g, "")
145
+ .slice(0, childCount);
146
+ const value = sanitizedValue
147
+ .padEnd(childCount, "#")
148
+ .split("")
149
+ .map((char) => (char === "#" ? "" : char));
150
+
151
+ setValue(value);
152
+ setLastCharIndex(sanitizedValue.length - 1);
153
+
154
+ const index = Math.min(sanitizedValue.length, childCount - 1);
155
+ childrenRefs.current?.[index]?.focus();
156
+ }}
157
+ >
158
+ {children}
159
+ <input
160
+ ref={composeRefs(forwardedRef, hiddenInputRef)}
161
+ defaultValue={value?.join("")}
162
+ minLength={childCount}
163
+ name={name}
164
+ type="hidden"
165
+ />
166
+ </Grid>
167
+ </OtpContext.Provider>
168
+ );
169
+ },
170
+ );
171
+
172
+ interface InputProps extends React.ComponentProps<typeof TextField> {
173
+ autoComplete?: "one-time-code" | "off";
174
+ }
175
+
176
+ export const Input = React.forwardRef<HTMLInputElement, InputProps>(
177
+ function Input(
178
+ { style, readOnly, autoComplete = "off", ...props },
179
+ forwardedRef,
180
+ ) {
181
+ const otpContext = useOptContext();
182
+ const inputRef = React.useRef<HTMLInputElement>(null);
183
+ const composedInputRef = useComposedRefs(
184
+ forwardedRef,
185
+ inputRef,
186
+ otpContext.onChildAdd,
187
+ );
188
+
189
+ const index = Number(inputRef.current?.dataset.index ?? -1);
190
+ const char = otpContext.value[index] ?? "";
191
+
192
+ return (
193
+ <Form.Field name={`otp-${index}`} asChild>
194
+ <Form.Control asChild>
195
+ <TextField
196
+ ref={composedInputRef}
197
+ autoComplete={index === 0 ? autoComplete : "off"}
198
+ color={otpContext.state === "invalid" ? "red" : undefined}
199
+ inputMode="numeric"
200
+ maxLength={1}
201
+ pattern="\d{1}"
202
+ readOnly={readOnly ?? otpContext.readOnly}
203
+ size="3"
204
+ value={char}
205
+ variant={otpContext.state === "invalid" ? "soft" : undefined}
206
+ style={{
207
+ ...style,
208
+ height: "auto",
209
+ "--text-field-padding": 0,
210
+ textAlign: "center",
211
+ }}
212
+ onChange={(event) => {
213
+ // Only update the value if it matches the input pattern (number only)
214
+ if (event.target.validity.valid) {
215
+ const char = event.target.value;
216
+ const index = Number(event.target.dataset.index ?? -1);
217
+ otpContext.onCharChange(char, index);
218
+ if (char !== "") {
219
+ focusSibling(event.currentTarget, { back: char === "" });
220
+ }
221
+ }
222
+ }}
223
+ onKeyDown={(event) => {
224
+ if (event.key === "ArrowLeft") {
225
+ focusSibling(event.currentTarget, { back: true });
226
+ event.preventDefault();
227
+ } else if (event.key === "ArrowRight") {
228
+ focusSibling(event.currentTarget);
229
+ event.preventDefault();
230
+ } else if (event.key === "Backspace" && char === "") {
231
+ focusSibling(event.currentTarget, { back: true });
232
+ } else if (event.key === "Enter" && char !== "") {
233
+ otpContext.onEnterPressed();
234
+ }
235
+ }}
236
+ {...props}
237
+ />
238
+ </Form.Control>
239
+ </Form.Field>
240
+ );
241
+ },
242
+ );
243
+
244
+ const useOptContext = () => {
245
+ const optContext = React.useContext(OtpContext);
246
+
247
+ if (!optContext) {
248
+ throw new Error(
249
+ "OtpInput compound components cannot be rendered outside the OtpRoot component",
250
+ );
251
+ }
252
+
253
+ return optContext;
254
+ };
255
+
256
+ function focusSibling(input: HTMLInputElement, { back = false } = {}) {
257
+ const sibling = back
258
+ ? input.parentElement?.previousSibling
259
+ : input.parentElement?.nextSibling;
260
+ const siblingInput = sibling?.firstChild;
261
+ if (siblingInput && siblingInput instanceof HTMLInputElement) {
262
+ siblingInput?.focus();
263
+ siblingInput?.select();
264
+ }
265
+ }
266
+
267
+ const getValueAsArray = (value: string | undefined, length: number) => {
268
+ if (!value) {
269
+ return undefined;
270
+ }
271
+
272
+ return createEmptyArray(length).map((_, index) => value?.[index] ?? "");
273
+ };
274
+
275
+ const createEmptyArray = (length: number): string[] =>
276
+ Array.from<string>({ length }).fill("");
@@ -10,7 +10,6 @@ import {
10
10
  } from "@radix-ui/themes";
11
11
  import * as React from "react";
12
12
  import { useResendUserInvite } from "./api/user";
13
- import type { User } from "./api/user";
14
13
  import {
15
14
  AlertDialogContent,
16
15
  DestructiveButton,
@@ -18,9 +17,10 @@ import {
18
17
  PrimaryButton,
19
18
  SecondaryButton,
20
19
  } from "./elements";
20
+ import { Member } from "@repo/api";
21
21
 
22
22
  interface ResendInviteDialogProps extends AlertDialog.RootProps {
23
- user: User;
23
+ user: Member;
24
24
  children?: React.ReactNode;
25
25
  }
26
26
 
@@ -35,11 +35,14 @@ export function ResendInviteDialog({
35
35
  const [successDialogIsOpen, setSuccessDialogIsOpen] = React.useState(false);
36
36
 
37
37
  const onSubmitForm = () => {
38
- resendInvite.mutate(user.id, {
39
- onSuccess: () => {
40
- setSuccessDialogIsOpen(true);
38
+ resendInvite.mutate(
39
+ { userId: user.id },
40
+ {
41
+ onSuccess: () => {
42
+ setSuccessDialogIsOpen(true);
43
+ },
41
44
  },
42
- });
45
+ );
43
46
  };
44
47
 
45
48
  return (