@solidxai/core-ui 0.1.8-beta.8 → 0.1.8-beta.9

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 (70) hide show
  1. package/dist/components/auth/AuthLayout.d.ts.map +1 -1
  2. package/dist/components/auth/AuthLayout.js +16 -32
  3. package/dist/components/auth/AuthLayout.js.map +1 -1
  4. package/dist/components/auth/AuthLayout.tsx +4 -20
  5. package/dist/components/auth/AuthTabs.d.ts.map +1 -1
  6. package/dist/components/auth/AuthTabs.js +9 -9
  7. package/dist/components/auth/AuthTabs.js.map +1 -1
  8. package/dist/components/auth/AuthTabs.tsx +14 -15
  9. package/dist/components/auth/SolidRegister.d.ts.map +1 -1
  10. package/dist/components/auth/SolidRegister.js +1 -1
  11. package/dist/components/auth/SolidRegister.js.map +1 -1
  12. package/dist/components/auth/SolidRegister.tsx +12 -5
  13. package/dist/components/common/GeneralSettings.d.ts.map +1 -1
  14. package/dist/components/common/GeneralSettings.js +41 -46
  15. package/dist/components/common/GeneralSettings.js.map +1 -1
  16. package/dist/components/common/GeneralSettings.tsx +0 -19
  17. package/dist/components/common/SolidThemeProvider.d.ts.map +1 -1
  18. package/dist/components/common/SolidThemeProvider.js +5 -21
  19. package/dist/components/common/SolidThemeProvider.js.map +1 -1
  20. package/dist/components/common/SolidThemeProvider.tsx +6 -24
  21. package/dist/components/core/form/SolidFormView.js +42 -41
  22. package/dist/components/core/form/SolidFormView.js.map +1 -1
  23. package/dist/components/core/form/SolidFormView.tsx +16 -16
  24. package/dist/components/core/form/fields/widgets/SolidS3FileViewerWidget.d.ts +1 -1
  25. package/dist/components/core/form/fields/widgets/SolidS3FileViewerWidget.js +1 -1
  26. package/dist/components/core/form/fields/widgets/SolidS3FileViewerWidget.js.map +1 -1
  27. package/dist/components/core/form/fields/widgets/SolidS3FileViewerWidget.tsx +1 -1
  28. package/dist/components/core/kanban/SolidKanbanView.d.ts.map +1 -1
  29. package/dist/components/core/kanban/SolidKanbanView.js +41 -44
  30. package/dist/components/core/kanban/SolidKanbanView.js.map +1 -1
  31. package/dist/components/core/kanban/SolidKanbanView.tsx +16 -25
  32. package/dist/components/core/list/SolidListView.d.ts.map +1 -1
  33. package/dist/components/core/list/SolidListView.js +3 -11
  34. package/dist/components/core/list/SolidListView.js.map +1 -1
  35. package/dist/components/core/list/SolidListView.tsx +27 -48
  36. package/dist/components/core/users/ApiKeysTab/ApiKeysTab.css +283 -9
  37. package/dist/components/core/users/ApiKeysTab/ApiKeysTab.d.ts.map +1 -1
  38. package/dist/components/core/users/ApiKeysTab/ApiKeysTab.js +35 -28
  39. package/dist/components/core/users/ApiKeysTab/ApiKeysTab.js.map +1 -1
  40. package/dist/components/core/users/ApiKeysTab/ApiKeysTab.tsx +64 -62
  41. package/dist/components/core/users/ApiKeysTab/GenerateApiKeyModal.d.ts +2 -1
  42. package/dist/components/core/users/ApiKeysTab/GenerateApiKeyModal.d.ts.map +1 -1
  43. package/dist/components/core/users/ApiKeysTab/GenerateApiKeyModal.js +4 -4
  44. package/dist/components/core/users/ApiKeysTab/GenerateApiKeyModal.js.map +1 -1
  45. package/dist/components/core/users/ApiKeysTab/GenerateApiKeyModal.tsx +17 -10
  46. package/dist/components/core/users/ApiKeysTab/RevealApiKeyModal.d.ts.map +1 -1
  47. package/dist/components/core/users/ApiKeysTab/RevealApiKeyModal.js +2 -19
  48. package/dist/components/core/users/ApiKeysTab/RevealApiKeyModal.js.map +1 -1
  49. package/dist/components/core/users/ApiKeysTab/RevealApiKeyModal.tsx +24 -43
  50. package/dist/components/core/users/CreateUser.css +114 -0
  51. package/dist/components/core/users/CreateUser.d.ts +1 -0
  52. package/dist/components/core/users/CreateUser.d.ts.map +1 -1
  53. package/dist/components/core/users/CreateUser.js +4 -3
  54. package/dist/components/core/users/CreateUser.js.map +1 -1
  55. package/dist/components/core/users/CreateUser.tsx +32 -26
  56. package/dist/components/layout/context/layoutcontext.js +2 -2
  57. package/dist/components/layout/context/layoutcontext.js.map +1 -1
  58. package/dist/components/layout/context/layoutcontext.tsx +2 -2
  59. package/dist/components/shad-cn-ui/SolidConfirmDialog.d.ts +6 -1
  60. package/dist/components/shad-cn-ui/SolidConfirmDialog.d.ts.map +1 -1
  61. package/dist/components/shad-cn-ui/SolidConfirmDialog.js +3 -3
  62. package/dist/components/shad-cn-ui/SolidConfirmDialog.js.map +1 -1
  63. package/dist/components/shad-cn-ui/SolidConfirmDialog.tsx +15 -3
  64. package/dist/components/shad-cn-ui/SolidTabs.d.ts.map +1 -1
  65. package/dist/components/shad-cn-ui/SolidTabs.js +2 -2
  66. package/dist/components/shad-cn-ui/SolidTabs.js.map +1 -1
  67. package/dist/components/shad-cn-ui/SolidTabs.tsx +0 -2
  68. package/dist/resources/globals.css +92 -77
  69. package/dist/resources/shadcn-base.css +62 -304
  70. package/package.json +1 -1
@@ -37,6 +37,7 @@ import { SolidListViewRowActionsMenu } from "./SolidListViewRowActionsMenu";
37
37
  import { SolidHeaderRequestStatus } from "../../common/SolidHeaderRequestStatus";
38
38
  import {
39
39
  SolidButton,
40
+ SolidConfirmDialog,
40
41
  SolidDialog,
41
42
  SolidDialogBody,
42
43
  SolidDialogClose,
@@ -1714,32 +1715,21 @@ export const SolidListView = forwardRef<SolidListViewHandle, SolidListViewParams
1714
1715
  )}
1715
1716
  </div>
1716
1717
  </div>
1717
- <SolidDialog
1718
+ <SolidConfirmDialog
1718
1719
  open={isDialogVisible}
1719
- onOpenChange={(open) => {
1720
- if (!open) {
1721
- onDeleteClose();
1722
- }
1723
- }}
1720
+ onCancel={onDeleteClose}
1721
+ onConfirm={deleteBulk}
1724
1722
  className="solid-shadcn-confirm-dialog solid-delete-confirm-dialog"
1725
- >
1726
- <SolidDialogHeader className="solid-shadcn-dialog-head">
1727
- <SolidDialogTitle>Confirm Delete</SolidDialogTitle>
1728
- <SolidDialogClose />
1729
- </SolidDialogHeader>
1730
- <SolidDialogSeparator className="solid-shadcn-dialog-sep" />
1731
- <SolidDialogBody className="solid-shadcn-dialog-body">
1732
- <p className="solid-shadcn-dialog-text">Are you sure you want to delete the selected records?</p>
1733
- </SolidDialogBody>
1734
- <SolidDialogFooter className="solid-shadcn-dialog-actions">
1735
- <SolidButton variant="destructive" size="sm" autoFocus onClick={deleteBulk}>
1736
- Delete
1737
- </SolidButton>
1738
- <SolidButton variant="outline" size="sm" onClick={onDeleteClose}>
1739
- Cancel
1740
- </SolidButton>
1741
- </SolidDialogFooter>
1742
- </SolidDialog>
1723
+ headerClassName="solid-shadcn-dialog-head"
1724
+ bodyClassName="solid-shadcn-dialog-body"
1725
+ footerClassName="solid-shadcn-dialog-actions"
1726
+ separatorClassName="solid-shadcn-dialog-sep"
1727
+ showSeparator
1728
+ title="Delete Records"
1729
+ message={<p className="solid-shadcn-dialog-text">Are you sure you want to delete the selected records?</p>}
1730
+ confirmLabel="Delete"
1731
+ cancelLabel="Cancel"
1732
+ />
1743
1733
  <SolidDialog
1744
1734
  open={isRecoverDialogVisible}
1745
1735
  onOpenChange={(open) => {
@@ -1790,32 +1780,21 @@ export const SolidListView = forwardRef<SolidListViewHandle, SolidListViewParams
1790
1780
  </SolidDialog>
1791
1781
  )
1792
1782
  }
1793
- <SolidDialog
1783
+ <SolidConfirmDialog
1794
1784
  open={deleteEntity}
1795
- onOpenChange={(open) => {
1796
- if (!open) {
1797
- setDeleteEntity(false);
1798
- }
1799
- }}
1785
+ onCancel={() => setDeleteEntity(false)}
1786
+ onConfirm={handleDeleteEntity}
1800
1787
  className="solid-shadcn-confirm-dialog solid-delete-confirm-dialog"
1801
- >
1802
- <SolidDialogHeader className="solid-shadcn-dialog-head">
1803
- <SolidDialogTitle>{`Delete ${entityDisplayName}`}</SolidDialogTitle>
1804
- <SolidDialogClose />
1805
- </SolidDialogHeader>
1806
- <SolidDialogSeparator className="solid-shadcn-dialog-sep" />
1807
- <SolidDialogBody className="solid-shadcn-dialog-body">
1808
- <p className="solid-shadcn-dialog-text">{`Are you sure you want to delete this ${entityDisplayName}?`}</p>
1809
- </SolidDialogBody>
1810
- <SolidDialogFooter className="solid-shadcn-dialog-actions">
1811
- <SolidButton variant="destructive" size="sm" onClick={handleDeleteEntity}>
1812
- Delete
1813
- </SolidButton>
1814
- <SolidButton variant="outline" size="sm" onClick={() => setDeleteEntity(false)}>
1815
- Cancel
1816
- </SolidButton>
1817
- </SolidDialogFooter>
1818
- </SolidDialog>
1788
+ headerClassName="solid-shadcn-dialog-head"
1789
+ bodyClassName="solid-shadcn-dialog-body"
1790
+ footerClassName="solid-shadcn-dialog-actions"
1791
+ separatorClassName="solid-shadcn-dialog-sep"
1792
+ showSeparator
1793
+ title={`Delete ${entityDisplayName}`}
1794
+ message={<p className="solid-shadcn-dialog-text">{`Are you sure you want to delete this ${entityDisplayName}?`}</p>}
1795
+ confirmLabel="Delete"
1796
+ cancelLabel="Cancel"
1797
+ />
1819
1798
  {openLightbox && (
1820
1799
  <SolidLightbox
1821
1800
  open={openLightbox}
@@ -1,38 +1,312 @@
1
1
  .solid-api-keys-tab {
2
- padding: 4px 0;
2
+ padding: 0;
3
+ width: 100%;
4
+ }
5
+
6
+ .solid-api-keys-hero {
7
+ display: flex;
8
+ align-items: flex-start;
9
+ justify-content: space-between;
10
+ gap: 16px;
11
+ margin-bottom: 18px;
12
+ }
13
+
14
+ .solid-api-keys-copy {
15
+ display: flex;
16
+ flex-direction: column;
17
+ gap: 4px;
18
+ }
19
+
20
+ .solid-api-keys-title {
21
+ font-size: 18px;
22
+ font-weight: 700;
23
+ color: var(--foreground);
24
+ }
25
+
26
+ .solid-api-keys-subtitle {
27
+ max-width: 700px;
28
+ font-size: 13px;
29
+ line-height: 1.5;
30
+ color: var(--muted-foreground);
31
+ }
32
+
33
+ .solid-api-keys-actions {
34
+ display: flex;
35
+ align-items: center;
36
+ gap: 10px;
37
+ }
38
+
39
+ .solid-api-keys-count {
40
+ display: inline-flex;
41
+ align-items: center;
42
+ min-height: 32px;
43
+ padding: 0 10px;
44
+ border-radius: 999px;
45
+ border: 1px solid color-mix(in srgb, var(--border) 90%, transparent);
46
+ background: color-mix(in srgb, var(--accent) 30%, transparent);
47
+ font-size: 12px;
48
+ font-weight: 600;
49
+ color: var(--muted-foreground);
50
+ }
51
+
52
+ .solid-api-keys-state {
53
+ min-height: 220px;
54
+ display: grid;
55
+ place-items: center;
56
+ }
57
+
58
+ .solid-api-keys-error {
59
+ min-height: 220px;
60
+ display: grid;
61
+ place-items: center;
62
+ text-align: center;
63
+ gap: 6px;
64
+ color: var(--muted-foreground);
65
+ font-size: 13px;
66
+ }
67
+
68
+ .solid-api-keys-empty {
69
+ width: 100%;
70
+ min-height: 320px;
71
+ border: 1px dashed color-mix(in srgb, var(--border) 86%, transparent);
72
+ border-radius: 16px;
73
+ background: linear-gradient(180deg, color-mix(in srgb, var(--accent) 32%, transparent), transparent);
74
+ display: flex;
75
+ flex-direction: column;
76
+ align-items: center;
77
+ justify-content: center;
78
+ gap: 14px;
79
+ padding: 28px;
80
+ }
81
+
82
+ .solid-api-keys-empty-icon {
83
+ width: 56px;
84
+ height: 56px;
85
+ border-radius: 999px;
86
+ display: grid;
87
+ place-items: center;
88
+ background: color-mix(in srgb, var(--primary) 10%, var(--card));
89
+ color: color-mix(in srgb, var(--foreground) 72%, transparent);
90
+ }
91
+
92
+ .solid-api-keys-empty-title {
93
+ font-size: 16px;
94
+ font-weight: 700;
95
+ color: var(--foreground);
96
+ }
97
+
98
+ .solid-api-keys-empty-copy {
99
+ max-width: 360px;
100
+ margin-top: 6px;
101
+ font-size: 13px;
102
+ line-height: 1.5;
103
+ color: var(--muted-foreground);
104
+ }
105
+
106
+ .solid-api-keys-table-shell {
107
+ width: 100%;
108
+ border: 1px solid color-mix(in srgb, var(--border) 88%, transparent);
109
+ border-radius: 16px;
110
+ background: color-mix(in srgb, var(--card) 98%, white 2%);
111
+ overflow: hidden;
112
+ }
113
+
114
+ .solid-api-keys-table-wrap {
115
+ overflow-x: auto;
116
+ }
117
+
118
+ .solid-api-keys-table {
119
+ width: 100%;
120
+ border-collapse: collapse;
121
+ table-layout: fixed;
3
122
  }
4
123
 
5
124
  .solid-api-keys-th {
6
- padding: 8px 12px;
125
+ padding: 13px 16px;
7
126
  text-align: left;
8
127
  font-size: 12px;
9
128
  font-weight: 600;
10
- color: var(--text-color-secondary, #6b7280);
129
+ color: var(--muted-foreground);
11
130
  letter-spacing: 0.02em;
12
- border-bottom: 1px solid var(--surface-border, #dfe7ef);
131
+ border-bottom: 1px solid color-mix(in srgb, var(--border) 88%, transparent);
132
+ background: color-mix(in srgb, var(--accent) 32%, transparent);
13
133
  white-space: nowrap;
14
134
  }
15
135
 
136
+ .solid-api-keys-th--right {
137
+ text-align: right;
138
+ }
139
+
16
140
  .solid-api-keys-td {
17
- padding: 12px 12px;
141
+ padding: 16px;
18
142
  font-size: 13px;
19
- color: var(--text-color, #4b5563);
20
- border-bottom: 1px solid var(--surface-border, #dfe7ef);
143
+ color: var(--foreground);
144
+ border-bottom: 1px solid color-mix(in srgb, var(--border) 88%, transparent);
21
145
  vertical-align: middle;
22
146
  }
23
147
 
148
+ .solid-api-keys-td--right {
149
+ text-align: right;
150
+ }
151
+
152
+ .solid-api-keys-name {
153
+ font-weight: 600;
154
+ color: var(--foreground);
155
+ }
156
+
157
+ .solid-api-keys-code {
158
+ display: inline-flex;
159
+ max-width: 100%;
160
+ font-family: monospace;
161
+ font-size: 12.5px;
162
+ background: color-mix(in srgb, var(--accent) 34%, transparent);
163
+ padding: 4px 8px;
164
+ border-radius: 999px;
165
+ color: color-mix(in srgb, var(--foreground) 90%, transparent);
166
+ overflow-wrap: anywhere;
167
+ }
168
+
169
+ .solid-api-keys-muted {
170
+ color: var(--muted-foreground);
171
+ }
172
+
173
+ .solid-api-keys-date--expired {
174
+ color: var(--destructive);
175
+ }
176
+
177
+ .solid-api-keys-date--warning {
178
+ color: #b7791f;
179
+ }
180
+
24
181
  .solid-api-keys-row--expired td {
25
182
  opacity: 0.6;
26
183
  }
27
184
 
28
185
  .solid-api-keys-row--expired:hover td {
29
- background: var(--surface-hover, #f6f9fc);
186
+ background: color-mix(in srgb, var(--accent) 42%, transparent);
30
187
  }
31
188
 
32
189
  tbody tr:hover .solid-api-keys-td {
33
- background: var(--surface-hover, #f6f9fc);
190
+ background: color-mix(in srgb, var(--accent) 42%, transparent);
34
191
  }
35
192
 
36
193
  tbody tr:last-child .solid-api-keys-td {
37
194
  border-bottom: none;
38
195
  }
196
+
197
+ .solid-api-key-dialog-header {
198
+ padding-bottom: 14px;
199
+ }
200
+
201
+ .solid-api-key-dialog-body {
202
+ padding-top: 18px;
203
+ }
204
+
205
+ .solid-api-key-dialog-fields {
206
+ display: flex;
207
+ flex-direction: column;
208
+ gap: 16px;
209
+ }
210
+
211
+ .solid-api-key-dialog-field {
212
+ display: flex;
213
+ flex-direction: column;
214
+ gap: 8px;
215
+ }
216
+
217
+ .solid-api-key-dialog-hint {
218
+ font-size: 12px;
219
+ line-height: 1.45;
220
+ color: var(--muted-foreground);
221
+ }
222
+
223
+ .solid-api-key-dialog-footer {
224
+ padding-top: 14px;
225
+ }
226
+
227
+ .solid-api-key-reveal-warning {
228
+ display: flex;
229
+ align-items: flex-start;
230
+ gap: 10px;
231
+ padding: 12px 14px;
232
+ margin-bottom: 18px;
233
+ border-radius: 12px;
234
+ border: 1px solid color-mix(in srgb, #f59e0b 28%, var(--border));
235
+ background: color-mix(in srgb, #f59e0b 10%, transparent);
236
+ font-size: 13px;
237
+ line-height: 1.5;
238
+ color: var(--foreground);
239
+ }
240
+
241
+ .solid-api-key-reveal-warning-icon {
242
+ flex: 0 0 auto;
243
+ margin-top: 1px;
244
+ }
245
+
246
+ .solid-api-key-reveal-box {
247
+ display: flex;
248
+ align-items: center;
249
+ gap: 10px;
250
+ padding: 14px;
251
+ border-radius: 12px;
252
+ border: 1px solid color-mix(in srgb, var(--border) 90%, transparent);
253
+ background: color-mix(in srgb, var(--accent) 28%, transparent);
254
+ }
255
+
256
+ .solid-api-key-reveal-code {
257
+ flex: 1 1 auto;
258
+ margin: 0;
259
+ font-family: monospace;
260
+ font-size: 13px;
261
+ line-height: 1.45;
262
+ color: var(--foreground);
263
+ word-break: break-all;
264
+ user-select: all;
265
+ }
266
+
267
+ .solid-api-key-reveal-copy {
268
+ width: 34px;
269
+ height: 34px;
270
+ border: 1px solid color-mix(in srgb, var(--border) 90%, transparent);
271
+ border-radius: 10px;
272
+ background: var(--card);
273
+ color: var(--muted-foreground);
274
+ display: inline-flex;
275
+ align-items: center;
276
+ justify-content: center;
277
+ cursor: pointer;
278
+ flex-shrink: 0;
279
+ transition: background-color 0.15s ease, border-color 0.15s ease, color 0.15s ease;
280
+ }
281
+
282
+ .solid-api-key-reveal-copy:hover {
283
+ background: var(--accent);
284
+ color: var(--foreground);
285
+ }
286
+
287
+ .solid-api-key-reveal-copy-success,
288
+ .solid-api-key-reveal-copied {
289
+ color: #16a34a;
290
+ }
291
+
292
+ .solid-api-key-reveal-copied {
293
+ font-size: 12px;
294
+ }
295
+
296
+ @media (max-width: 900px) {
297
+ .solid-api-keys-hero {
298
+ flex-direction: column;
299
+ align-items: stretch;
300
+ }
301
+
302
+ .solid-api-keys-actions {
303
+ justify-content: space-between;
304
+ }
305
+ }
306
+
307
+ @media (max-width: 640px) {
308
+ .solid-api-keys-actions {
309
+ flex-direction: column;
310
+ align-items: stretch;
311
+ }
312
+ }
@@ -1 +1 @@
1
- {"version":3,"file":"ApiKeysTab.d.ts","sourceRoot":"","sources":["../../../../../src/components/core/users/ApiKeysTab/ApiKeysTab.tsx"],"names":[],"mappings":"AAGA,OAAO,kBAAkB,CAAC;AA4J1B,UAAU,eAAe;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,wBAAgB,UAAU,CAAC,EAAE,MAAM,EAAE,SAAiB,EAAE,EAAE,eAAe,2CA2FxE"}
1
+ {"version":3,"file":"ApiKeysTab.d.ts","sourceRoot":"","sources":["../../../../../src/components/core/users/ApiKeysTab/ApiKeysTab.tsx"],"names":[],"mappings":"AAGA,OAAO,kBAAkB,CAAC;AAmJ1B,UAAU,eAAe;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,wBAAgB,UAAU,CAAC,EAAE,MAAM,EAAE,SAAiB,EAAE,EAAE,eAAe,2CAsGxE"}
@@ -34,8 +34,17 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
34
34
  if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
35
  }
36
36
  };
37
+ var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
38
+ if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
39
+ if (ar || !(i in from)) {
40
+ if (!ar) ar = Array.prototype.slice.call(from, 0, i);
41
+ ar[i] = from[i];
42
+ }
43
+ }
44
+ return to.concat(ar || Array.prototype.slice.call(from));
45
+ };
37
46
  import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
38
- import { useState } from "react";
47
+ import { useMemo, useState } from "react";
39
48
  import { useDispatch } from "react-redux";
40
49
  import { KeyRound, Plus } from "lucide-react";
41
50
  import "./ApiKeysTab.css";
@@ -64,39 +73,35 @@ function getExpiryStatus(expiresAt) {
64
73
  return "ok";
65
74
  }
66
75
  function ApiKeysTable(_a) {
67
- var keys = _a.keys, onToggleActive = _a.onToggleActive, isTogglingId = _a.isTogglingId;
76
+ var keys = _a.keys, onToggleActive = _a.onToggleActive, isTogglingId = _a.isTogglingId, onGenerateFirstKey = _a.onGenerateFirstKey, canCreate = _a.canCreate;
68
77
  if (keys.length === 0) {
69
- return (_jsxs("div", { className: "flex flex-column align-items-center justify-content-center gap-3 py-6", style: { color: "var(--solid-text-secondary, #888)" }, children: [_jsx(KeyRound, { size: 32, strokeWidth: 1.5 }), _jsxs("div", { className: "text-center", children: [_jsx("p", { className: "m-0", style: { fontSize: 14, fontWeight: 500 }, children: "No API keys" }), _jsx("p", { className: "m-0 mt-1", style: { fontSize: 12 }, children: "Generate a key to enable programmatic access." })] })] }));
78
+ return (_jsxs("div", { className: "solid-api-keys-empty", children: [_jsx("div", { className: "solid-api-keys-empty-icon", children: _jsx(KeyRound, { size: 30, strokeWidth: 1.6 }) }), _jsxs("div", { className: "text-center", children: [_jsx("p", { className: "solid-api-keys-empty-title m-0", children: "No API keys yet" }), _jsx("p", { className: "solid-api-keys-empty-copy m-0", children: canCreate
79
+ ? "Generate a key to enable secure programmatic access for this user."
80
+ : "API key generation is disabled for this user account." })] }), canCreate ? (_jsxs(SolidButton, { type: "button", onClick: onGenerateFirstKey, children: [_jsx(Plus, { size: 14 }), "Generate First Key"] })) : null] }));
70
81
  }
71
- return (_jsx("div", { style: { overflowX: "auto" }, children: _jsxs("table", { style: { width: "100%", borderCollapse: "collapse", tableLayout: "fixed" }, children: [_jsxs("colgroup", { children: [_jsx("col", { style: { width: "18%" } }), _jsx("col", { style: { width: "22%" } }), _jsx("col", { style: { width: "14%" } }), _jsx("col", { style: { width: "16%" } }), _jsx("col", { style: { width: "18%" } }), _jsx("col", { style: { width: "12%" } })] }), _jsx("thead", { children: _jsxs("tr", { children: [_jsx("th", { className: "solid-api-keys-th", children: "Name" }), _jsx("th", { className: "solid-api-keys-th", children: "Key" }), _jsx("th", { className: "solid-api-keys-th", children: "Status" }), _jsx("th", { className: "solid-api-keys-th", children: "Expires" }), _jsx("th", { className: "solid-api-keys-th", children: "Last Used" }), _jsx("th", { className: "solid-api-keys-th", style: { textAlign: "right" }, children: "Active" })] }) }), _jsx("tbody", { children: keys.map(function (key) {
72
- var expiryStatus = getExpiryStatus(key.expiresAt);
73
- var isExpired = expiryStatus === "expired";
74
- return (_jsxs("tr", { className: isExpired ? "solid-api-keys-row--expired" : undefined, children: [_jsx("td", { className: "solid-api-keys-td", children: _jsx("span", { style: { fontWeight: 500 }, children: key.name }) }), _jsx("td", { className: "solid-api-keys-td", children: _jsx("code", { style: {
75
- fontFamily: "monospace",
76
- fontSize: 13,
77
- background: "var(--solid-surface-secondary, #f5f5f5)",
78
- padding: "2px 6px",
79
- borderRadius: 4,
80
- }, children: key.maskedKey }) }), _jsx("td", { className: "solid-api-keys-td", children: isExpired ? (_jsx(SolidTag, { tone: "danger", children: "Expired" })) : !key.isActive ? (_jsx(SolidTag, { tone: "warn", children: "Inactive" })) : expiryStatus === "expiring-soon" ? (_jsx(SolidTag, { tone: "warn", children: "Expiring soon" })) : (_jsx(SolidTag, { tone: "success", children: "Active" })) }), _jsx("td", { className: "solid-api-keys-td", style: {
81
- color: expiryStatus === "expired"
82
- ? "var(--solid-danger-color, #ef4444)"
83
- : expiryStatus === "expiring-soon"
84
- ? "var(--solid-warn-color, #f59e0b)"
85
- : undefined,
86
- }, children: expiryStatus === "never" ? (_jsx("span", { style: { color: "var(--solid-text-secondary, #888)" }, children: "Never" })) : (formatDate(key.expiresAt)) }), _jsx("td", { className: "solid-api-keys-td", style: { color: "var(--solid-text-secondary, #888)" }, children: formatDate(key.lastUsedAt) }), _jsx("td", { className: "solid-api-keys-td", style: { textAlign: "right" }, children: isTogglingId === key.id ? (_jsx(SolidSpinner, {})) : (_jsx(SolidSwitch, { checked: key.isActive, disabled: isExpired, onChange: function () { return onToggleActive(key); } })) })] }, key.id));
87
- }) })] }) }));
82
+ return (_jsx("div", { className: "solid-api-keys-table-shell", children: _jsx("div", { className: "solid-api-keys-table-wrap", children: _jsxs("table", { className: "solid-api-keys-table", children: [_jsxs("colgroup", { children: [_jsx("col", { style: { width: "18%" } }), _jsx("col", { style: { width: "22%" } }), _jsx("col", { style: { width: "14%" } }), _jsx("col", { style: { width: "16%" } }), _jsx("col", { style: { width: "18%" } }), _jsx("col", { style: { width: "12%" } })] }), _jsx("thead", { children: _jsxs("tr", { children: [_jsx("th", { className: "solid-api-keys-th", children: "Name" }), _jsx("th", { className: "solid-api-keys-th", children: "Key" }), _jsx("th", { className: "solid-api-keys-th", children: "Status" }), _jsx("th", { className: "solid-api-keys-th", children: "Expires" }), _jsx("th", { className: "solid-api-keys-th", children: "Last Used" }), _jsx("th", { className: "solid-api-keys-th solid-api-keys-th--right", children: "Active" })] }) }), _jsx("tbody", { children: keys.map(function (key) {
83
+ var expiryStatus = getExpiryStatus(key.expiresAt);
84
+ var isExpired = expiryStatus === "expired";
85
+ return (_jsxs("tr", { className: isExpired ? "solid-api-keys-row--expired" : undefined, children: [_jsx("td", { className: "solid-api-keys-td", children: _jsx("span", { className: "solid-api-keys-name", children: key.name }) }), _jsx("td", { className: "solid-api-keys-td", children: _jsx("code", { className: "solid-api-keys-code", children: key.maskedKey }) }), _jsx("td", { className: "solid-api-keys-td", children: isExpired ? (_jsx(SolidTag, { tone: "danger", children: "Expired" })) : !key.isActive ? (_jsx(SolidTag, { tone: "warn", children: "Inactive" })) : expiryStatus === "expiring-soon" ? (_jsx(SolidTag, { tone: "warn", children: "Expiring soon" })) : (_jsx(SolidTag, { tone: "success", children: "Active" })) }), _jsx("td", { className: "solid-api-keys-td ".concat(expiryStatus === "expired" ? "solid-api-keys-date--expired" : "", " ").concat(expiryStatus === "expiring-soon" ? "solid-api-keys-date--warning" : ""), children: expiryStatus === "never" ? (_jsx("span", { className: "solid-api-keys-muted", children: "Never" })) : (formatDate(key.expiresAt)) }), _jsx("td", { className: "solid-api-keys-td solid-api-keys-muted", children: formatDate(key.lastUsedAt) }), _jsx("td", { className: "solid-api-keys-td solid-api-keys-td--right", children: isTogglingId === key.id ? (_jsx(SolidSpinner, {})) : (_jsx(SolidSwitch, { checked: key.isActive, disabled: isExpired, onChange: function () { return onToggleActive(key); } })) })] }, key.id));
86
+ }) })] }) }) }));
88
87
  }
89
88
  export function ApiKeysTab(_a) {
90
89
  var _this = this;
91
- var _b, _c;
92
- var userId = _a.userId, _d = _a.canCreate, canCreate = _d === void 0 ? false : _d;
90
+ var _b;
91
+ var userId = _a.userId, _c = _a.canCreate, canCreate = _c === void 0 ? false : _c;
93
92
  var dispatch = useDispatch();
94
- var _e = useGetUserApiKeysQuery(userId), data = _e.data, isLoading = _e.isLoading, isError = _e.isError;
93
+ var _d = useGetUserApiKeysQuery(userId), data = _d.data, isLoading = _d.isLoading, isError = _d.isError, refetch = _d.refetch;
95
94
  var updateApiKey = useUpdateApiKeyMutation()[0];
96
- var _f = useState(null), togglingId = _f[0], setTogglingId = _f[1];
97
- var _g = useState(false), showGenerate = _g[0], setShowGenerate = _g[1];
98
- var _h = useState(null), revealKey = _h[0], setRevealKey = _h[1];
99
- var keys = (_c = (_b = data === null || data === void 0 ? void 0 : data.data) === null || _b === void 0 ? void 0 : _b.apiKeys) !== null && _c !== void 0 ? _c : [];
95
+ var _e = useState(null), togglingId = _e[0], setTogglingId = _e[1];
96
+ var _f = useState(false), showGenerate = _f[0], setShowGenerate = _f[1];
97
+ var _g = useState(null), revealKey = _g[0], setRevealKey = _g[1];
98
+ var _h = useState([]), optimisticKeys = _h[0], setOptimisticKeys = _h[1];
99
+ var keys = useMemo(function () {
100
+ var _a, _b;
101
+ var serverKeys = (_b = (_a = data === null || data === void 0 ? void 0 : data.data) === null || _a === void 0 ? void 0 : _a.apiKeys) !== null && _b !== void 0 ? _b : [];
102
+ var merged = __spreadArray(__spreadArray([], optimisticKeys.filter(function (key) { return !serverKeys.some(function (serverKey) { return serverKey.id === key.id; }); }), true), serverKeys, true);
103
+ return merged.sort(function (a, b) { return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime(); });
104
+ }, [(_b = data === null || data === void 0 ? void 0 : data.data) === null || _b === void 0 ? void 0 : _b.apiKeys, optimisticKeys]);
100
105
  var handleToggle = function (key) { return __awaiter(_this, void 0, void 0, function () {
101
106
  var err_1;
102
107
  var _a;
@@ -131,8 +136,10 @@ export function ApiKeysTab(_a) {
131
136
  }
132
137
  });
133
138
  }); };
134
- return (_jsxs(_Fragment, { children: [_jsxs("div", { className: "solid-api-keys-tab", children: [_jsxs("div", { className: "flex align-items-center justify-content-between mb-4", children: [_jsxs("div", { children: [_jsx("p", { className: "m-0", style: { fontWeight: 600, fontSize: 14 }, children: "API Keys" }), _jsx("p", { className: "m-0 mt-1", style: { fontSize: 12, color: "var(--solid-text-secondary, #888)" }, children: "Keys grant programmatic access. Store them securely \u2014 they are shown only once." })] }), canCreate && (_jsxs(SolidButton, { size: "small", type: "button", onClick: function () { return setShowGenerate(true); }, children: [_jsx(Plus, { size: 14 }), "Generate Key"] }))] }), isLoading ? (_jsx("div", { className: "flex justify-content-center py-5", children: _jsx(SolidSpinner, {}) })) : isError && keys.length === 0 ? (_jsxs("div", { className: "flex flex-column align-items-center justify-content-center gap-2 py-5", style: { color: "var(--solid-text-secondary, #888)", fontSize: 13 }, children: [_jsx("p", { className: "m-0", children: "Something went wrong while loading API keys." }), _jsx("p", { className: "m-0", children: "Please refresh the page and try again." })] })) : (_jsx(ApiKeysTable, { keys: keys, onToggleActive: handleToggle, isTogglingId: togglingId }))] }), _jsx(GenerateApiKeyModal, { open: showGenerate, userId: Number(userId), onClose: function () { return setShowGenerate(false); }, onCreated: function (apiKey, keyName) {
139
+ return (_jsxs(_Fragment, { children: [_jsxs("div", { className: "solid-api-keys-tab", children: [_jsxs("div", { className: "solid-api-keys-hero", children: [_jsxs("div", { className: "solid-api-keys-copy", children: [_jsx("p", { className: "solid-api-keys-title m-0", children: "API Keys" }), _jsx("p", { className: "solid-api-keys-subtitle m-0", children: "Keys grant programmatic access. Store them securely \u2014 they are shown only once." })] }), _jsxs("div", { className: "solid-api-keys-actions", children: [_jsxs("span", { className: "solid-api-keys-count", children: [keys.length, " key", keys.length === 1 ? "" : "s"] }), canCreate ? (_jsxs(SolidButton, { size: "small", type: "button", onClick: function () { return setShowGenerate(true); }, children: [_jsx(Plus, { size: 14 }), "Generate Key"] })) : null] })] }), isLoading ? (_jsx("div", { className: "solid-api-keys-state", children: _jsx(SolidSpinner, {}) })) : isError && keys.length === 0 ? (_jsxs("div", { className: "solid-api-keys-error", children: [_jsx("p", { className: "m-0", children: "Something went wrong while loading API keys." }), _jsx("p", { className: "m-0", children: "Please refresh the page and try again." })] })) : (_jsx(ApiKeysTable, { keys: keys, onToggleActive: handleToggle, isTogglingId: togglingId, onGenerateFirstKey: function () { return setShowGenerate(true); }, canCreate: canCreate }))] }), _jsx(GenerateApiKeyModal, { open: showGenerate, userId: Number(userId), onClose: function () { return setShowGenerate(false); }, onCreated: function (apiKey, keyName, record) {
135
140
  setShowGenerate(false);
141
+ setOptimisticKeys(function (previous) { return __spreadArray([record], previous.filter(function (existing) { return existing.id !== record.id; }), true); });
142
+ void refetch();
136
143
  setRevealKey({ apiKey: apiKey, keyName: keyName });
137
144
  } }), revealKey && (_jsx(RevealApiKeyModal, { open: true, apiKey: revealKey.apiKey, keyName: revealKey.keyName, onClose: function () { return setRevealKey(null); } }))] }));
138
145
  }
@@ -1 +1 @@
1
- {"version":3,"file":"ApiKeysTab.js","sourceRoot":"","sources":["../../../../../src/components/core/users/ApiKeysTab/ApiKeysTab.tsx"],"names":[],"mappings":"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","sourcesContent":["import { useState } from \"react\";\nimport { useDispatch } from \"react-redux\";\nimport { KeyRound, Plus } from \"lucide-react\";\nimport \"./ApiKeysTab.css\";\nimport { SolidButton, SolidSpinner, SolidSwitch, SolidTag } from \"../../../shad-cn-ui\";\nimport { showToast } from \"../../../../redux/features/toastSlice\";\nimport {\n useGetUserApiKeysQuery,\n useUpdateApiKeyMutation,\n type ApiKeyRecord,\n} from \"../../../../redux/api/apiKeyApi\";\nimport { GenerateApiKeyModal } from \"./GenerateApiKeyModal\";\nimport { RevealApiKeyModal } from \"./RevealApiKeyModal\";\n\nfunction formatDate(iso: string | null): string {\n if (!iso) return \"—\";\n return new Date(iso).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n });\n}\n\nfunction getExpiryStatus(expiresAt: string | null): \"expired\" | \"expiring-soon\" | \"ok\" | \"never\" {\n if (!expiresAt) return \"never\";\n const diff = new Date(expiresAt).getTime() - Date.now();\n if (diff < 0) return \"expired\";\n if (diff < 7 * 24 * 60 * 60 * 1000) return \"expiring-soon\";\n return \"ok\";\n}\n\nfunction ApiKeysTable({\n keys,\n onToggleActive,\n isTogglingId,\n}: {\n keys: ApiKeyRecord[];\n onToggleActive: (key: ApiKeyRecord) => void;\n isTogglingId: string | null;\n}) {\n if (keys.length === 0) {\n return (\n <div\n className=\"flex flex-column align-items-center justify-content-center gap-3 py-6\"\n style={{ color: \"var(--solid-text-secondary, #888)\" }}\n >\n <KeyRound size={32} strokeWidth={1.5} />\n <div className=\"text-center\">\n <p className=\"m-0\" style={{ fontSize: 14, fontWeight: 500 }}>\n No API keys\n </p>\n <p className=\"m-0 mt-1\" style={{ fontSize: 12 }}>\n Generate a key to enable programmatic access.\n </p>\n </div>\n </div>\n );\n }\n\n return (\n <div style={{ overflowX: \"auto\" }}>\n <table style={{ width: \"100%\", borderCollapse: \"collapse\", tableLayout: \"fixed\" }}>\n <colgroup>\n <col style={{ width: \"18%\" }} />\n <col style={{ width: \"22%\" }} />\n <col style={{ width: \"14%\" }} />\n <col style={{ width: \"16%\" }} />\n <col style={{ width: \"18%\" }} />\n <col style={{ width: \"12%\" }} />\n </colgroup>\n <thead>\n <tr>\n <th className=\"solid-api-keys-th\">Name</th>\n <th className=\"solid-api-keys-th\">Key</th>\n <th className=\"solid-api-keys-th\">Status</th>\n <th className=\"solid-api-keys-th\">Expires</th>\n <th className=\"solid-api-keys-th\">Last Used</th>\n <th className=\"solid-api-keys-th\" style={{ textAlign: \"right\" }}>Active</th>\n </tr>\n </thead>\n <tbody>\n {keys.map((key) => {\n const expiryStatus = getExpiryStatus(key.expiresAt);\n const isExpired = expiryStatus === \"expired\";\n\n return (\n <tr key={key.id} className={isExpired ? \"solid-api-keys-row--expired\" : undefined}>\n <td className=\"solid-api-keys-td\">\n <span style={{ fontWeight: 500 }}>{key.name}</span>\n </td>\n\n <td className=\"solid-api-keys-td\">\n <code\n style={{\n fontFamily: \"monospace\",\n fontSize: 13,\n background: \"var(--solid-surface-secondary, #f5f5f5)\",\n padding: \"2px 6px\",\n borderRadius: 4,\n }}\n >\n {key.maskedKey}\n </code>\n </td>\n\n <td className=\"solid-api-keys-td\">\n {isExpired ? (\n <SolidTag tone=\"danger\">Expired</SolidTag>\n ) : !key.isActive ? (\n <SolidTag tone=\"warn\">Inactive</SolidTag>\n ) : expiryStatus === \"expiring-soon\" ? (\n <SolidTag tone=\"warn\">Expiring soon</SolidTag>\n ) : (\n <SolidTag tone=\"success\">Active</SolidTag>\n )}\n </td>\n\n <td\n className=\"solid-api-keys-td\"\n style={{\n color:\n expiryStatus === \"expired\"\n ? \"var(--solid-danger-color, #ef4444)\"\n : expiryStatus === \"expiring-soon\"\n ? \"var(--solid-warn-color, #f59e0b)\"\n : undefined,\n }}\n >\n {expiryStatus === \"never\" ? (\n <span style={{ color: \"var(--solid-text-secondary, #888)\" }}>Never</span>\n ) : (\n formatDate(key.expiresAt)\n )}\n </td>\n\n <td className=\"solid-api-keys-td\" style={{ color: \"var(--solid-text-secondary, #888)\" }}>\n {formatDate(key.lastUsedAt)}\n </td>\n\n <td className=\"solid-api-keys-td\" style={{ textAlign: \"right\" }}>\n {isTogglingId === key.id ? (\n <SolidSpinner />\n ) : (\n <SolidSwitch\n checked={key.isActive}\n disabled={isExpired}\n onChange={() => onToggleActive(key)}\n />\n )}\n </td>\n </tr>\n );\n })}\n </tbody>\n </table>\n </div>\n );\n}\n\ninterface ApiKeysTabProps {\n userId: string;\n canCreate?: boolean;\n}\n\nexport function ApiKeysTab({ userId, canCreate = false }: ApiKeysTabProps) {\n const dispatch = useDispatch();\n const { data, isLoading, isError } = useGetUserApiKeysQuery(userId);\n const [updateApiKey] = useUpdateApiKeyMutation();\n const [togglingId, setTogglingId] = useState<string | null>(null);\n const [showGenerate, setShowGenerate] = useState(false);\n const [revealKey, setRevealKey] = useState<{ apiKey: string; keyName: string } | null>(null);\n\n const keys: ApiKeyRecord[] = data?.data?.apiKeys ?? [];\n\n const handleToggle = async (key: ApiKeyRecord) => {\n setTogglingId(key.id);\n try {\n await updateApiKey({ id: key.id, isActive: !key.isActive }).unwrap();\n dispatch(\n showToast({\n severity: \"success\",\n summary: \"Updated\",\n detail: `API key \"${key.name}\" ${!key.isActive ? \"activated\" : \"deactivated\"}.`,\n })\n );\n } catch (err: any) {\n dispatch(\n showToast({\n severity: \"error\",\n summary: \"Error\",\n detail: err?.data?.message || \"Failed to update API key.\",\n })\n );\n } finally {\n setTogglingId(null);\n }\n };\n\n return (\n <>\n <div className=\"solid-api-keys-tab\">\n <div className=\"flex align-items-center justify-content-between mb-4\">\n <div>\n <p className=\"m-0\" style={{ fontWeight: 600, fontSize: 14 }}>\n API Keys\n </p>\n <p className=\"m-0 mt-1\" style={{ fontSize: 12, color: \"var(--solid-text-secondary, #888)\" }}>\n Keys grant programmatic access. Store them securely — they are shown only once.\n </p>\n </div>\n {canCreate && (\n <SolidButton size=\"small\" type=\"button\" onClick={() => setShowGenerate(true)}>\n <Plus size={14} />\n Generate Key\n </SolidButton>\n )}\n </div>\n\n {isLoading ? (\n <div className=\"flex justify-content-center py-5\">\n <SolidSpinner />\n </div>\n ) : isError && keys.length === 0 ? (\n <div\n className=\"flex flex-column align-items-center justify-content-center gap-2 py-5\"\n style={{ color: \"var(--solid-text-secondary, #888)\", fontSize: 13 }}\n >\n <p className=\"m-0\">Something went wrong while loading API keys.</p>\n <p className=\"m-0\">Please refresh the page and try again.</p>\n </div>\n ) : (\n <ApiKeysTable keys={keys} onToggleActive={handleToggle} isTogglingId={togglingId} />\n )}\n </div>\n\n <GenerateApiKeyModal\n open={showGenerate}\n userId={Number(userId)}\n onClose={() => setShowGenerate(false)}\n onCreated={(apiKey, keyName) => {\n setShowGenerate(false);\n setRevealKey({ apiKey, keyName });\n }}\n />\n\n {revealKey && (\n <RevealApiKeyModal\n open={true}\n apiKey={revealKey.apiKey}\n keyName={revealKey.keyName}\n onClose={() => setRevealKey(null)}\n />\n )}\n </>\n );\n}\n"]}
1
+ {"version":3,"file":"ApiKeysTab.js","sourceRoot":"","sources":["../../../../../src/components/core/users/ApiKeysTab/ApiKeysTab.tsx"],"names":[],"mappings":"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","sourcesContent":["import { useMemo, useState } from \"react\";\nimport { useDispatch } from \"react-redux\";\nimport { KeyRound, Plus } from \"lucide-react\";\nimport \"./ApiKeysTab.css\";\nimport { SolidButton, SolidSpinner, SolidSwitch, SolidTag } from \"../../../shad-cn-ui\";\nimport { showToast } from \"../../../../redux/features/toastSlice\";\nimport {\n useGetUserApiKeysQuery,\n useUpdateApiKeyMutation,\n type ApiKeyRecord,\n} from \"../../../../redux/api/apiKeyApi\";\nimport { GenerateApiKeyModal } from \"./GenerateApiKeyModal\";\nimport { RevealApiKeyModal } from \"./RevealApiKeyModal\";\n\nfunction formatDate(iso: string | null): string {\n if (!iso) return \"—\";\n return new Date(iso).toLocaleDateString(undefined, {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n });\n}\n\nfunction getExpiryStatus(expiresAt: string | null): \"expired\" | \"expiring-soon\" | \"ok\" | \"never\" {\n if (!expiresAt) return \"never\";\n const diff = new Date(expiresAt).getTime() - Date.now();\n if (diff < 0) return \"expired\";\n if (diff < 7 * 24 * 60 * 60 * 1000) return \"expiring-soon\";\n return \"ok\";\n}\n\nfunction ApiKeysTable({\n keys,\n onToggleActive,\n isTogglingId,\n onGenerateFirstKey,\n canCreate,\n}: {\n keys: ApiKeyRecord[];\n onToggleActive: (key: ApiKeyRecord) => void;\n isTogglingId: string | null;\n onGenerateFirstKey: () => void;\n canCreate: boolean;\n}) {\n if (keys.length === 0) {\n return (\n <div className=\"solid-api-keys-empty\">\n <div className=\"solid-api-keys-empty-icon\">\n <KeyRound size={30} strokeWidth={1.6} />\n </div>\n <div className=\"text-center\">\n <p className=\"solid-api-keys-empty-title m-0\">No API keys yet</p>\n <p className=\"solid-api-keys-empty-copy m-0\">\n {canCreate\n ? \"Generate a key to enable secure programmatic access for this user.\"\n : \"API key generation is disabled for this user account.\"}\n </p>\n </div>\n {canCreate ? (\n <SolidButton type=\"button\" onClick={onGenerateFirstKey}>\n <Plus size={14} />\n Generate First Key\n </SolidButton>\n ) : null}\n </div>\n );\n }\n\n return (\n <div className=\"solid-api-keys-table-shell\">\n <div className=\"solid-api-keys-table-wrap\">\n <table className=\"solid-api-keys-table\">\n <colgroup>\n <col style={{ width: \"18%\" }} />\n <col style={{ width: \"22%\" }} />\n <col style={{ width: \"14%\" }} />\n <col style={{ width: \"16%\" }} />\n <col style={{ width: \"18%\" }} />\n <col style={{ width: \"12%\" }} />\n </colgroup>\n <thead>\n <tr>\n <th className=\"solid-api-keys-th\">Name</th>\n <th className=\"solid-api-keys-th\">Key</th>\n <th className=\"solid-api-keys-th\">Status</th>\n <th className=\"solid-api-keys-th\">Expires</th>\n <th className=\"solid-api-keys-th\">Last Used</th>\n <th className=\"solid-api-keys-th solid-api-keys-th--right\">Active</th>\n </tr>\n </thead>\n <tbody>\n {keys.map((key) => {\n const expiryStatus = getExpiryStatus(key.expiresAt);\n const isExpired = expiryStatus === \"expired\";\n\n return (\n <tr key={key.id} className={isExpired ? \"solid-api-keys-row--expired\" : undefined}>\n <td className=\"solid-api-keys-td\">\n <span className=\"solid-api-keys-name\">{key.name}</span>\n </td>\n\n <td className=\"solid-api-keys-td\">\n <code className=\"solid-api-keys-code\">{key.maskedKey}</code>\n </td>\n\n <td className=\"solid-api-keys-td\">\n {isExpired ? (\n <SolidTag tone=\"danger\">Expired</SolidTag>\n ) : !key.isActive ? (\n <SolidTag tone=\"warn\">Inactive</SolidTag>\n ) : expiryStatus === \"expiring-soon\" ? (\n <SolidTag tone=\"warn\">Expiring soon</SolidTag>\n ) : (\n <SolidTag tone=\"success\">Active</SolidTag>\n )}\n </td>\n\n <td className={`solid-api-keys-td ${expiryStatus === \"expired\" ? \"solid-api-keys-date--expired\" : \"\"} ${expiryStatus === \"expiring-soon\" ? \"solid-api-keys-date--warning\" : \"\"}`}>\n {expiryStatus === \"never\" ? (\n <span className=\"solid-api-keys-muted\">Never</span>\n ) : (\n formatDate(key.expiresAt)\n )}\n </td>\n\n <td className=\"solid-api-keys-td solid-api-keys-muted\">\n {formatDate(key.lastUsedAt)}\n </td>\n\n <td className=\"solid-api-keys-td solid-api-keys-td--right\">\n {isTogglingId === key.id ? (\n <SolidSpinner />\n ) : (\n <SolidSwitch\n checked={key.isActive}\n disabled={isExpired}\n onChange={() => onToggleActive(key)}\n />\n )}\n </td>\n </tr>\n );\n })}\n </tbody>\n </table>\n </div>\n </div>\n );\n}\n\ninterface ApiKeysTabProps {\n userId: string;\n canCreate?: boolean;\n}\n\nexport function ApiKeysTab({ userId, canCreate = false }: ApiKeysTabProps) {\n const dispatch = useDispatch();\n const { data, isLoading, isError, refetch } = useGetUserApiKeysQuery(userId);\n const [updateApiKey] = useUpdateApiKeyMutation();\n const [togglingId, setTogglingId] = useState<string | null>(null);\n const [showGenerate, setShowGenerate] = useState(false);\n const [revealKey, setRevealKey] = useState<{ apiKey: string; keyName: string } | null>(null);\n const [optimisticKeys, setOptimisticKeys] = useState<ApiKeyRecord[]>([]);\n\n const keys = useMemo(() => {\n const serverKeys = data?.data?.apiKeys ?? [];\n const merged = [...optimisticKeys.filter((key) => !serverKeys.some((serverKey) => serverKey.id === key.id)), ...serverKeys];\n return merged.sort((a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime());\n }, [data?.data?.apiKeys, optimisticKeys]);\n\n const handleToggle = async (key: ApiKeyRecord) => {\n setTogglingId(key.id);\n try {\n await updateApiKey({ id: key.id, isActive: !key.isActive }).unwrap();\n dispatch(\n showToast({\n severity: \"success\",\n summary: \"Updated\",\n detail: `API key \"${key.name}\" ${!key.isActive ? \"activated\" : \"deactivated\"}.`,\n })\n );\n } catch (err: any) {\n dispatch(\n showToast({\n severity: \"error\",\n summary: \"Error\",\n detail: err?.data?.message || \"Failed to update API key.\",\n })\n );\n } finally {\n setTogglingId(null);\n }\n };\n\n return (\n <>\n <div className=\"solid-api-keys-tab\">\n <div className=\"solid-api-keys-hero\">\n <div className=\"solid-api-keys-copy\">\n <p className=\"solid-api-keys-title m-0\">API Keys</p>\n <p className=\"solid-api-keys-subtitle m-0\">\n Keys grant programmatic access. Store them securely — they are shown only once.\n </p>\n </div>\n <div className=\"solid-api-keys-actions\">\n <span className=\"solid-api-keys-count\">{keys.length} key{keys.length === 1 ? \"\" : \"s\"}</span>\n {canCreate ? (\n <SolidButton size=\"small\" type=\"button\" onClick={() => setShowGenerate(true)}>\n <Plus size={14} />\n Generate Key\n </SolidButton>\n ) : null}\n </div>\n </div>\n\n {isLoading ? (\n <div className=\"solid-api-keys-state\">\n <SolidSpinner />\n </div>\n ) : isError && keys.length === 0 ? (\n <div className=\"solid-api-keys-error\">\n <p className=\"m-0\">Something went wrong while loading API keys.</p>\n <p className=\"m-0\">Please refresh the page and try again.</p>\n </div>\n ) : (\n <ApiKeysTable\n keys={keys}\n onToggleActive={handleToggle}\n isTogglingId={togglingId}\n onGenerateFirstKey={() => setShowGenerate(true)}\n canCreate={canCreate}\n />\n )}\n </div>\n\n <GenerateApiKeyModal\n open={showGenerate}\n userId={Number(userId)}\n onClose={() => setShowGenerate(false)}\n onCreated={(apiKey, keyName, record) => {\n setShowGenerate(false);\n setOptimisticKeys((previous) => [record, ...previous.filter((existing) => existing.id !== record.id)]);\n void refetch();\n setRevealKey({ apiKey, keyName });\n }}\n />\n\n {revealKey && (\n <RevealApiKeyModal\n open={true}\n apiKey={revealKey.apiKey}\n keyName={revealKey.keyName}\n onClose={() => setRevealKey(null)}\n />\n )}\n </>\n );\n}\n"]}