@wakastellar/ui 2.3.4 → 2.4.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 (194) hide show
  1. package/dist/blocks/dashboard/index.d.ts +4 -1
  2. package/dist/blocks/empty-states/index.d.ts +4 -1
  3. package/dist/blocks/error-pages/index.d.ts +4 -1
  4. package/dist/blocks/index.d.ts +1 -1
  5. package/dist/blocks/landing/index.d.ts +4 -1
  6. package/dist/blocks/pricing/index.d.ts +5 -1
  7. package/dist/blocks/sidebar/index.d.ts +5 -1
  8. package/dist/index.cjs.js +130 -130
  9. package/dist/index.d.ts +1 -0
  10. package/dist/index.es.js +7905 -7856
  11. package/dist/stories/Button.d.ts +14 -0
  12. package/dist/stories/Button.stories.d.ts +8 -0
  13. package/dist/stories/Header.d.ts +11 -0
  14. package/dist/stories/Header.stories.d.ts +6 -0
  15. package/dist/stories/Page.d.ts +2 -0
  16. package/dist/stories/Page.stories.d.ts +6 -0
  17. package/dist/types/index.d.ts +1 -0
  18. package/dist/types/link.d.ts +23 -0
  19. package/package.json +11 -3
  20. package/src/blocks/activity-timeline/ActivityTimeline.stories.tsx +460 -0
  21. package/src/blocks/apm-overview/APMOverview.stories.tsx +435 -0
  22. package/src/blocks/auth-2fa/Auth2FA.stories.tsx +308 -0
  23. package/src/blocks/calendar-view/CalendarView.stories.tsx +398 -0
  24. package/src/blocks/chat/Chat.stories.tsx +466 -0
  25. package/src/blocks/chat-interface/ChatInterface.stories.tsx +412 -0
  26. package/src/blocks/checkout-flow/CheckoutFlow.stories.tsx +408 -0
  27. package/src/blocks/cicd-builder/CICDBuilder.stories.tsx +499 -0
  28. package/src/blocks/cloud-cost-dashboard/CloudCostDashboard.stories.tsx +356 -0
  29. package/src/blocks/container-orchestrator/ContainerOrchestrator.stories.tsx +461 -0
  30. package/src/blocks/dashboard/Dashboard.stories.tsx +559 -0
  31. package/src/blocks/dashboard/index.tsx +68 -27
  32. package/src/blocks/dashboard-kpi/DashboardKPI.stories.tsx +422 -0
  33. package/src/blocks/database-admin/DatabaseAdmin.stories.tsx +393 -0
  34. package/src/blocks/deployment-dashboard/DeploymentDashboard.stories.tsx +457 -0
  35. package/src/blocks/empty-states/EmptyStates.stories.tsx +467 -0
  36. package/src/blocks/empty-states/index.tsx +26 -8
  37. package/src/blocks/error-pages/ErrorPages.stories.tsx +401 -0
  38. package/src/blocks/error-pages/index.tsx +26 -8
  39. package/src/blocks/faq/FAQ.stories.tsx +416 -0
  40. package/src/blocks/file-manager/FileManager.stories.tsx +413 -0
  41. package/src/blocks/footer/Footer.stories.tsx +328 -0
  42. package/src/blocks/gitops-sync-status/GitOpsSyncStatus.stories.tsx +529 -0
  43. package/src/blocks/header/WakaHeader.stories.tsx +455 -0
  44. package/src/blocks/headtab/Headtab.stories.tsx +369 -0
  45. package/src/blocks/i18n-editor/I18nEditor.stories.tsx +451 -0
  46. package/src/blocks/incident-manager/IncidentManager.stories.tsx +464 -0
  47. package/src/blocks/index.ts +1 -1
  48. package/src/blocks/infrastructure-map/InfrastructureMap.stories.tsx +533 -0
  49. package/src/blocks/kanban-board/KanbanBoard.stories.tsx +494 -0
  50. package/src/blocks/landing/WakaLanding.stories.tsx +449 -0
  51. package/src/blocks/landing/index.tsx +125 -66
  52. package/src/blocks/language-selector/LanguageSelector.stories.tsx +345 -0
  53. package/src/blocks/layout/Layout.stories.tsx +373 -0
  54. package/src/blocks/login/Login.stories.tsx +443 -0
  55. package/src/blocks/on-call-schedule/OnCallSchedule.stories.tsx +381 -0
  56. package/src/blocks/player-profile/PlayerProfile.stories.tsx +316 -0
  57. package/src/blocks/pricing/WakaPricing.stories.tsx +530 -0
  58. package/src/blocks/pricing/index.tsx +38 -4
  59. package/src/blocks/profile/Profile.stories.tsx +371 -0
  60. package/src/blocks/release-notes/ReleaseNotes.stories.tsx +431 -0
  61. package/src/blocks/settings/Settings.stories.tsx +520 -0
  62. package/src/blocks/sidebar/WakaSidebar.stories.tsx +513 -0
  63. package/src/blocks/sidebar/index.tsx +49 -20
  64. package/src/blocks/theme-creator-block/ThemeCreatorBlock.stories.tsx +370 -0
  65. package/src/blocks/user-management/UserManagement.stories.tsx +411 -0
  66. package/src/blocks/wizard/Wizard.stories.tsx +683 -0
  67. package/src/components/accordion/Accordion.stories.tsx +93 -0
  68. package/src/components/alert/Alert.stories.tsx +95 -0
  69. package/src/components/alert-dialog/AlertDialog.stories.tsx +126 -0
  70. package/src/components/aspect-ratio/AspectRatio.stories.tsx +118 -0
  71. package/src/components/avatar/Avatar.stories.tsx +104 -0
  72. package/src/components/button/Button.stories.tsx +12 -1
  73. package/src/components/calendar/Calendar.stories.tsx +102 -0
  74. package/src/components/card/Card.stories.tsx +125 -0
  75. package/src/components/checkbox/Checkbox.stories.tsx +100 -0
  76. package/src/components/code/Code.stories.tsx +402 -0
  77. package/src/components/collapsible/Collapsible.stories.tsx +123 -0
  78. package/src/components/command/Command.stories.tsx +207 -0
  79. package/src/components/context-menu/ContextMenu.stories.tsx +220 -0
  80. package/src/components/dialog/Dialog.stories.tsx +157 -0
  81. package/src/components/dropdown-menu/DropdownMenu.stories.tsx +225 -0
  82. package/src/components/form/Form.stories.tsx +413 -0
  83. package/src/components/hover-card/HoverCard.stories.tsx +148 -0
  84. package/src/components/input-otp/InputOTP.stories.tsx +255 -0
  85. package/src/components/label/Label.stories.tsx +68 -0
  86. package/src/components/menubar/Menubar.stories.tsx +278 -0
  87. package/src/components/navigation-menu/NavigationMenu.stories.tsx +202 -0
  88. package/src/components/popover/Popover.stories.tsx +199 -0
  89. package/src/components/progress/Progress.stories.tsx +104 -0
  90. package/src/components/radio-group/RadioGroup.stories.tsx +138 -0
  91. package/src/components/scroll-area/ScrollArea.stories.tsx +153 -0
  92. package/src/components/select/Select.stories.tsx +146 -0
  93. package/src/components/separator/Separator.stories.tsx +117 -0
  94. package/src/components/sheet/Sheet.stories.tsx +195 -0
  95. package/src/components/skeleton/Skeleton.stories.tsx +114 -0
  96. package/src/components/slider/Slider.stories.tsx +157 -0
  97. package/src/components/switch/Switch.stories.tsx +114 -0
  98. package/src/components/table/Table.stories.tsx +153 -0
  99. package/src/components/tabs/Tabs.stories.tsx +155 -0
  100. package/src/components/textarea/Textarea.stories.tsx +116 -0
  101. package/src/components/toast/Toast.stories.tsx +160 -0
  102. package/src/components/toggle/Toggle.stories.tsx +125 -0
  103. package/src/components/tooltip/Tooltip.stories.tsx +133 -0
  104. package/src/components/typography/Typography.stories.tsx +207 -0
  105. package/src/components/waka-3d-pie-chart/Waka3DPieChart.stories.tsx +308 -0
  106. package/src/components/waka-achievement-unlock/WakaAchievementUnlock.stories.tsx +353 -0
  107. package/src/components/waka-artifact-list/WakaArtifactList.stories.tsx +258 -0
  108. package/src/components/waka-autocomplete/WakaAutocomplete.stories.tsx +224 -0
  109. package/src/components/waka-badge-showcase/WakaBadgeShowcase.stories.tsx +269 -0
  110. package/src/components/waka-barcode/WakaBarcode.stories.tsx +227 -0
  111. package/src/components/waka-bottom-sheet/WakaBottomSheet.stories.tsx +408 -0
  112. package/src/components/waka-breadcrumb/WakaBreadcrumb.stories.tsx +237 -0
  113. package/src/components/waka-build-matrix/WakaBuildMatrix.stories.tsx +328 -0
  114. package/src/components/waka-carousel/WakaCarousel.stories.tsx +296 -0
  115. package/src/components/waka-charts/WakaCharts.stories.tsx +519 -0
  116. package/src/components/waka-color-picker/WakaColorPicker.stories.tsx +200 -0
  117. package/src/components/waka-combobox/WakaCombobox.stories.tsx +250 -0
  118. package/src/components/waka-container-list/WakaContainerList.stories.tsx +315 -0
  119. package/src/components/waka-contribution-graph/WakaContributionGraph.stories.tsx +354 -0
  120. package/src/components/waka-cost-breakdown/WakaCostBreakdown.stories.tsx +348 -0
  121. package/src/components/waka-daily-reward/WakaDailyReward.stories.tsx +365 -0
  122. package/src/components/waka-database-card/WakaDatabaseCard.stories.tsx +306 -0
  123. package/src/components/waka-date-range-picker/WakaDateRangePicker.stories.tsx +339 -0
  124. package/src/components/waka-datetime-picker/WakaDateTimePicker.stories.tsx +317 -0
  125. package/src/components/waka-deployment-lane/WakaDeploymentLane.stories.tsx +292 -0
  126. package/src/components/waka-dock/WakaDock.stories.tsx +332 -0
  127. package/src/components/waka-drawer/WakaDrawer.stories.tsx +437 -0
  128. package/src/components/waka-env-var-editor/WakaEnvVarEditor.stories.tsx +263 -0
  129. package/src/components/waka-error-shake/WakaErrorShake.stories.tsx +410 -0
  130. package/src/components/waka-file-upload/WakaFileUpload.stories.tsx +239 -0
  131. package/src/components/waka-flow-diagram/WakaFlowDiagram.stories.tsx +365 -0
  132. package/src/components/waka-funnel-chart/WakaFunnelChart.stories.tsx +281 -0
  133. package/src/components/waka-glow-card/WakaGlowCard.stories.tsx +274 -0
  134. package/src/components/waka-haptic-button/WakaHapticButton.stories.tsx +349 -0
  135. package/src/components/waka-health-pulse/WakaHealthPulse.stories.tsx +293 -0
  136. package/src/components/waka-heatmap/WakaHeatmap.stories.tsx +376 -0
  137. package/src/components/waka-image/WakaImage.stories.tsx +255 -0
  138. package/src/components/waka-incident-timeline/WakaIncidentTimeline.stories.tsx +300 -0
  139. package/src/components/waka-kanban/WakaKanban.stories.tsx +399 -0
  140. package/src/components/waka-kubernetes-overview/WakaKubernetesOverview.stories.tsx +281 -0
  141. package/src/components/waka-leaderboard/WakaLeaderboard.stories.tsx +300 -0
  142. package/src/components/waka-level-progress/WakaLevelProgress.stories.tsx +313 -0
  143. package/src/components/waka-loading-orbit/WakaLoadingOrbit.stories.tsx +413 -0
  144. package/src/components/waka-log-viewer/WakaLogViewer.stories.tsx +312 -0
  145. package/src/components/waka-loot-box/WakaLootBox.stories.tsx +374 -0
  146. package/src/components/waka-metric-sparkline/WakaMetricSparkline.stories.tsx +312 -0
  147. package/src/components/waka-migration-list/WakaMigrationList.stories.tsx +289 -0
  148. package/src/components/waka-modal/WakaModal.stories.tsx +434 -0
  149. package/src/components/waka-morph-button/WakaMorphButton.stories.tsx +405 -0
  150. package/src/components/waka-network-topology/WakaNetworkTopology.stories.tsx +364 -0
  151. package/src/components/waka-notifications/WakaNotifications.stories.tsx +290 -0
  152. package/src/components/waka-number-input/WakaNumberInput.stories.tsx +282 -0
  153. package/src/components/waka-pagination/WakaPagination.stories.tsx +328 -0
  154. package/src/components/waka-password-strength/WakaPasswordStrength.stories.tsx +318 -0
  155. package/src/components/waka-pipeline-view/WakaPipelineView.stories.tsx +386 -0
  156. package/src/components/waka-player-card/WakaPlayerCard.stories.tsx +333 -0
  157. package/src/components/waka-pod-card/WakaPodCard.stories.tsx +435 -0
  158. package/src/components/waka-qrcode/WakaQRCode.stories.tsx +232 -0
  159. package/src/components/waka-query-explain/WakaQueryExplain.stories.tsx +407 -0
  160. package/src/components/waka-quest-card/WakaQuestCard.stories.tsx +394 -0
  161. package/src/components/waka-quota-bar/WakaQuotaBar.stories.tsx +435 -0
  162. package/src/components/waka-radar-score/WakaRadarScore.stories.tsx +372 -0
  163. package/src/components/waka-resource-gauge/WakaResourceGauge.stories.tsx +366 -0
  164. package/src/components/waka-rich-text-editor/WakaRichTextEditor.stories.tsx +238 -0
  165. package/src/components/waka-sankey-diagram/WakaSankeyDiagram.stories.tsx +389 -0
  166. package/src/components/waka-scratch-card/WakaScratchCard.stories.tsx +388 -0
  167. package/src/components/waka-secret-card/WakaSecretCard.stories.tsx +314 -0
  168. package/src/components/waka-segmented-control/WakaSegmentedControl.stories.tsx +309 -0
  169. package/src/components/waka-server-rack/WakaServerRack.stories.tsx +382 -0
  170. package/src/components/waka-service-graph/WakaServiceGraph.stories.tsx +262 -0
  171. package/src/components/waka-skeleton-wave/WakaSkeletonWave.stories.tsx +321 -0
  172. package/src/components/waka-skill-tree/WakaSkillTree.stories.tsx +308 -0
  173. package/src/components/waka-spin-wheel/WakaSpinWheel.stories.tsx +368 -0
  174. package/src/components/waka-spinner/WakaSpinner.stories.tsx +156 -0
  175. package/src/components/waka-stat/WakaStat.stories.tsx +334 -0
  176. package/src/components/waka-status-matrix/WakaStatusMatrix.stories.tsx +331 -0
  177. package/src/components/waka-stepper/WakaStepper.stories.tsx +468 -0
  178. package/src/components/waka-streak-counter/WakaStreakCounter.stories.tsx +235 -0
  179. package/src/components/waka-success-explosion/WakaSuccessExplosion.stories.tsx +389 -0
  180. package/src/components/waka-tabs-morph/WakaTabsMorph.stories.tsx +471 -0
  181. package/src/components/waka-terminal-output/WakaTerminalOutput.stories.tsx +351 -0
  182. package/src/components/waka-test-report/WakaTestReport.stories.tsx +322 -0
  183. package/src/components/waka-tilt-card/WakaTiltCard.stories.tsx +300 -0
  184. package/src/components/waka-time-picker/WakaTimePicker.stories.tsx +227 -0
  185. package/src/components/waka-timeline/WakaTimeline.stories.tsx +383 -0
  186. package/src/components/waka-tournament-bracket/WakaTournamentBracket.stories.tsx +375 -0
  187. package/src/components/waka-trace-viewer/WakaTraceViewer.stories.tsx +445 -0
  188. package/src/components/waka-tree/WakaTree.stories.tsx +359 -0
  189. package/src/components/waka-treemap-chart/WakaTreemapChart.stories.tsx +378 -0
  190. package/src/components/waka-typewriter/WakaTypewriter.stories.tsx +366 -0
  191. package/src/components/waka-versus-card/WakaVersusCard.stories.tsx +530 -0
  192. package/src/components/waka-video/WakaVideo.stories.tsx +203 -0
  193. package/src/components/waka-virtual-list/WakaVirtualList.stories.tsx +273 -0
  194. package/src/components/waka-xp-bar/WakaXPBar.stories.tsx +305 -0
@@ -0,0 +1,411 @@
1
+ import type { Meta, StoryObj } from '@storybook/react'
2
+ import { WakaUserManagement, defaultUsers, defaultPermissions } from './index'
3
+ import type { UserData, UserRole, UserStatus, UserFormData, UserPermission } from './index'
4
+ import * as React from 'react'
5
+
6
+ const meta: Meta<typeof WakaUserManagement> = {
7
+ title: 'Blocks/UserManagement',
8
+ component: WakaUserManagement,
9
+ parameters: {
10
+ layout: 'fullscreen',
11
+ docs: {
12
+ description: {
13
+ component:
14
+ 'A user management component with CRUD operations, role management, status changes, search, filters, bulk actions, and permissions.',
15
+ },
16
+ },
17
+ },
18
+ tags: ['autodocs'],
19
+ argTypes: {
20
+ showBulkActions: {
21
+ control: 'boolean',
22
+ },
23
+ showCreateButton: {
24
+ control: 'boolean',
25
+ },
26
+ loading: {
27
+ control: 'boolean',
28
+ },
29
+ },
30
+ }
31
+
32
+ export default meta
33
+ type Story = StoryObj<typeof WakaUserManagement>
34
+
35
+ export const Default: Story = {
36
+ render: () => (
37
+ <div className="p-6">
38
+ <WakaUserManagement
39
+ users={defaultUsers}
40
+ onSearchChange={(query) => console.log('Search:', query)}
41
+ onRoleFilterChange={(role) => console.log('Role filter:', role)}
42
+ onStatusFilterChange={(status) => console.log('Status filter:', status)}
43
+ onCreate={(user) => console.log('Create:', user)}
44
+ onEdit={(userId, data) => console.log('Edit:', userId, data)}
45
+ onDelete={(userId) => console.log('Delete:', userId)}
46
+ onRoleChange={(userId, role) => console.log('Role change:', userId, role)}
47
+ onStatusChange={(userId, status) => console.log('Status change:', userId, status)}
48
+ onResetPassword={(userId) => console.log('Reset password:', userId)}
49
+ onRefresh={() => console.log('Refresh')}
50
+ onExport={() => console.log('Export')}
51
+ />
52
+ </div>
53
+ ),
54
+ }
55
+
56
+ export const WithSelection: Story = {
57
+ render: () => {
58
+ const [selectedUsers, setSelectedUsers] = React.useState<string[]>([])
59
+
60
+ return (
61
+ <div className="p-6">
62
+ <WakaUserManagement
63
+ users={defaultUsers}
64
+ selectedUsers={selectedUsers}
65
+ onSelectionChange={setSelectedUsers}
66
+ onBulkDelete={(userIds) => {
67
+ console.log('Bulk delete:', userIds)
68
+ setSelectedUsers([])
69
+ }}
70
+ onSearchChange={(query) => console.log('Search:', query)}
71
+ onCreate={(user) => console.log('Create:', user)}
72
+ onDelete={(userId) => console.log('Delete:', userId)}
73
+ />
74
+ </div>
75
+ )
76
+ },
77
+ }
78
+
79
+ export const WithPermissions: Story = {
80
+ render: () => (
81
+ <div className="p-6">
82
+ <WakaUserManagement
83
+ users={defaultUsers}
84
+ permissions={defaultPermissions}
85
+ onCreate={(user) => console.log('Create:', user)}
86
+ onEdit={(userId, data) => console.log('Edit:', userId, data)}
87
+ onSearchChange={(query) => console.log('Search:', query)}
88
+ />
89
+ </div>
90
+ ),
91
+ }
92
+
93
+ export const ManyUsers: Story = {
94
+ render: () => {
95
+ const manyUsers: UserData[] = Array.from({ length: 50 }, (_, i) => ({
96
+ id: `user-${i + 1}`,
97
+ email: `user${i + 1}@example.com`,
98
+ displayName: `User ${i + 1}`,
99
+ firstName: `First${i + 1}`,
100
+ lastName: `Last${i + 1}`,
101
+ role: (['admin', 'moderator', 'user', 'guest'] as UserRole[])[i % 4],
102
+ status: (['active', 'inactive', 'pending', 'banned'] as UserStatus[])[i % 4],
103
+ createdAt: new Date(Date.now() - Math.random() * 1000 * 60 * 60 * 24 * 365),
104
+ lastLogin: Math.random() > 0.3 ? new Date(Date.now() - Math.random() * 1000 * 60 * 60 * 24 * 30) : undefined,
105
+ }))
106
+
107
+ return (
108
+ <div className="p-6">
109
+ <WakaUserManagement
110
+ users={manyUsers}
111
+ onSearchChange={(query) => console.log('Search:', query)}
112
+ onRoleFilterChange={(role) => console.log('Role filter:', role)}
113
+ onStatusFilterChange={(status) => console.log('Status filter:', status)}
114
+ />
115
+ </div>
116
+ )
117
+ },
118
+ }
119
+
120
+ export const Loading: Story = {
121
+ render: () => (
122
+ <div className="p-6">
123
+ <WakaUserManagement
124
+ users={[]}
125
+ loading
126
+ onSearchChange={(query) => console.log('Search:', query)}
127
+ />
128
+ </div>
129
+ ),
130
+ }
131
+
132
+ export const Empty: Story = {
133
+ render: () => (
134
+ <div className="p-6">
135
+ <WakaUserManagement
136
+ users={[]}
137
+ onCreate={(user) => console.log('Create:', user)}
138
+ onSearchChange={(query) => console.log('Search:', query)}
139
+ />
140
+ </div>
141
+ ),
142
+ }
143
+
144
+ export const ReadOnly: Story = {
145
+ render: () => (
146
+ <div className="p-6">
147
+ <WakaUserManagement
148
+ users={defaultUsers}
149
+ showCreateButton={false}
150
+ showBulkActions={false}
151
+ title="Utilisateurs"
152
+ description="Liste des utilisateurs (lecture seule)"
153
+ />
154
+ </div>
155
+ ),
156
+ }
157
+
158
+ export const CustomRoles: Story = {
159
+ render: () => {
160
+ const customRoles = [
161
+ { value: 'admin' as UserRole, label: 'Super Admin', color: 'destructive' },
162
+ { value: 'moderator' as UserRole, label: 'Content Manager', color: 'warning' },
163
+ { value: 'user' as UserRole, label: 'Standard User', color: 'default' },
164
+ { value: 'guest' as UserRole, label: 'Viewer', color: 'secondary' },
165
+ ]
166
+
167
+ return (
168
+ <div className="p-6">
169
+ <WakaUserManagement
170
+ users={defaultUsers}
171
+ roles={customRoles}
172
+ onRoleChange={(userId, role) => console.log('Role change:', userId, role)}
173
+ onSearchChange={(query) => console.log('Search:', query)}
174
+ />
175
+ </div>
176
+ )
177
+ },
178
+ }
179
+
180
+ export const Interactive: Story = {
181
+ render: () => {
182
+ const [users, setUsers] = React.useState<UserData[]>(defaultUsers)
183
+ const [selectedUsers, setSelectedUsers] = React.useState<string[]>([])
184
+ const [searchQuery, setSearchQuery] = React.useState('')
185
+ const [roleFilter, setRoleFilter] = React.useState<UserRole | 'all'>('all')
186
+ const [statusFilter, setStatusFilter] = React.useState<UserStatus | 'all'>('all')
187
+ const [loading, setLoading] = React.useState(false)
188
+ const [actionLoading, setActionLoading] = React.useState(false)
189
+
190
+ const handleCreate = async (data: UserFormData) => {
191
+ setActionLoading(true)
192
+ await new Promise((r) => setTimeout(r, 1000))
193
+ const newUser: UserData = {
194
+ id: `user-${Date.now()}`,
195
+ ...data,
196
+ createdAt: new Date(),
197
+ }
198
+ setUsers((prev) => [...prev, newUser])
199
+ setActionLoading(false)
200
+ }
201
+
202
+ const handleEdit = async (userId: string, data: Partial<UserFormData>) => {
203
+ setActionLoading(true)
204
+ await new Promise((r) => setTimeout(r, 1000))
205
+ setUsers((prev) =>
206
+ prev.map((u) => (u.id === userId ? { ...u, ...data } : u))
207
+ )
208
+ setActionLoading(false)
209
+ }
210
+
211
+ const handleDelete = async (userId: string) => {
212
+ setActionLoading(true)
213
+ await new Promise((r) => setTimeout(r, 500))
214
+ setUsers((prev) => prev.filter((u) => u.id !== userId))
215
+ setActionLoading(false)
216
+ }
217
+
218
+ const handleBulkDelete = async (userIds: string[]) => {
219
+ setActionLoading(true)
220
+ await new Promise((r) => setTimeout(r, 500))
221
+ setUsers((prev) => prev.filter((u) => !userIds.includes(u.id)))
222
+ setSelectedUsers([])
223
+ setActionLoading(false)
224
+ }
225
+
226
+ const handleRefresh = async () => {
227
+ setLoading(true)
228
+ await new Promise((r) => setTimeout(r, 1500))
229
+ setLoading(false)
230
+ }
231
+
232
+ return (
233
+ <div className="p-6">
234
+ <WakaUserManagement
235
+ users={users}
236
+ permissions={defaultPermissions}
237
+ selectedUsers={selectedUsers}
238
+ onSelectionChange={setSelectedUsers}
239
+ searchQuery={searchQuery}
240
+ onSearchChange={setSearchQuery}
241
+ roleFilter={roleFilter}
242
+ onRoleFilterChange={setRoleFilter}
243
+ statusFilter={statusFilter}
244
+ onStatusFilterChange={setStatusFilter}
245
+ onCreate={handleCreate}
246
+ onEdit={handleEdit}
247
+ onDelete={handleDelete}
248
+ onBulkDelete={handleBulkDelete}
249
+ onRoleChange={(userId, role) => handleEdit(userId, { role })}
250
+ onStatusChange={(userId, status) => handleEdit(userId, { status })}
251
+ onResetPassword={(userId) => alert(`Reset password for ${userId}`)}
252
+ onRefresh={handleRefresh}
253
+ onExport={() => alert('Exporting...')}
254
+ loading={loading}
255
+ actionLoading={actionLoading}
256
+ />
257
+ </div>
258
+ )
259
+ },
260
+ }
261
+
262
+ export const FilteredByRole: Story = {
263
+ render: () => {
264
+ const [roleFilter, setRoleFilter] = React.useState<UserRole | 'all'>('admin')
265
+
266
+ return (
267
+ <div className="p-6">
268
+ <WakaUserManagement
269
+ users={defaultUsers}
270
+ roleFilter={roleFilter}
271
+ onRoleFilterChange={setRoleFilter}
272
+ onSearchChange={(query) => console.log('Search:', query)}
273
+ title="Administrateurs"
274
+ description="Utilisateurs avec le rôle admin"
275
+ />
276
+ </div>
277
+ )
278
+ },
279
+ }
280
+
281
+ export const FilteredByStatus: Story = {
282
+ render: () => {
283
+ const [statusFilter, setStatusFilter] = React.useState<UserStatus | 'all'>('pending')
284
+
285
+ return (
286
+ <div className="p-6">
287
+ <WakaUserManagement
288
+ users={defaultUsers}
289
+ statusFilter={statusFilter}
290
+ onStatusFilterChange={setStatusFilter}
291
+ onSearchChange={(query) => console.log('Search:', query)}
292
+ title="Utilisateurs en attente"
293
+ description="Comptes nécessitant une activation"
294
+ />
295
+ </div>
296
+ )
297
+ },
298
+ }
299
+
300
+ export const AdminPanel: Story = {
301
+ render: () => {
302
+ const adminUsers: UserData[] = [
303
+ {
304
+ id: '1',
305
+ email: 'admin@company.com',
306
+ displayName: 'Admin Principal',
307
+ firstName: 'Admin',
308
+ lastName: 'Principal',
309
+ avatarUrl: 'https://api.dicebear.com/7.x/avataaars/svg?seed=Admin',
310
+ role: 'admin',
311
+ status: 'active',
312
+ createdAt: new Date('2022-01-01'),
313
+ lastLogin: new Date(),
314
+ },
315
+ {
316
+ id: '2',
317
+ email: 'sarah@company.com',
318
+ displayName: 'Sarah Manager',
319
+ firstName: 'Sarah',
320
+ lastName: 'Manager',
321
+ avatarUrl: 'https://api.dicebear.com/7.x/avataaars/svg?seed=Sarah',
322
+ role: 'moderator',
323
+ status: 'active',
324
+ createdAt: new Date('2023-03-15'),
325
+ lastLogin: new Date(Date.now() - 1000 * 60 * 60 * 2),
326
+ },
327
+ {
328
+ id: '3',
329
+ email: 'john@company.com',
330
+ displayName: 'John Developer',
331
+ firstName: 'John',
332
+ lastName: 'Developer',
333
+ avatarUrl: 'https://api.dicebear.com/7.x/avataaars/svg?seed=John',
334
+ role: 'user',
335
+ status: 'active',
336
+ createdAt: new Date('2023-06-01'),
337
+ lastLogin: new Date(Date.now() - 1000 * 60 * 60 * 24),
338
+ },
339
+ {
340
+ id: '4',
341
+ email: 'new.hire@company.com',
342
+ displayName: 'New Hire',
343
+ firstName: 'New',
344
+ lastName: 'Hire',
345
+ role: 'user',
346
+ status: 'pending',
347
+ createdAt: new Date(),
348
+ },
349
+ {
350
+ id: '5',
351
+ email: 'contractor@external.com',
352
+ displayName: 'External Contractor',
353
+ firstName: 'External',
354
+ lastName: 'Contractor',
355
+ role: 'guest',
356
+ status: 'active',
357
+ createdAt: new Date('2023-09-01'),
358
+ lastLogin: new Date(Date.now() - 1000 * 60 * 60 * 24 * 7),
359
+ },
360
+ ]
361
+
362
+ const extendedPermissions: UserPermission[] = [
363
+ { id: 'read', name: 'Lecture', description: 'Accès en lecture aux ressources', category: 'Base' },
364
+ { id: 'write', name: 'Écriture', description: 'Création et modification', category: 'Base' },
365
+ { id: 'delete', name: 'Suppression', description: 'Suppression de ressources', category: 'Base' },
366
+ { id: 'users:read', name: 'Voir utilisateurs', description: 'Liste des utilisateurs', category: 'Utilisateurs' },
367
+ { id: 'users:write', name: 'Gérer utilisateurs', description: 'Créer et modifier', category: 'Utilisateurs' },
368
+ { id: 'users:delete', name: 'Supprimer utilisateurs', description: 'Supprimer des comptes', category: 'Utilisateurs' },
369
+ { id: 'settings:read', name: 'Voir paramètres', description: 'Accès aux paramètres', category: 'Admin' },
370
+ { id: 'settings:write', name: 'Modifier paramètres', description: 'Changer les paramètres', category: 'Admin' },
371
+ { id: 'billing', name: 'Facturation', description: 'Gestion de la facturation', category: 'Admin' },
372
+ ]
373
+
374
+ return (
375
+ <div className="min-h-screen bg-muted/30">
376
+ <header className="bg-background border-b px-6 py-4">
377
+ <div className="flex items-center justify-between">
378
+ <div>
379
+ <h1 className="font-bold text-xl">Admin Panel</h1>
380
+ <p className="text-sm text-muted-foreground">Company Dashboard</p>
381
+ </div>
382
+ <nav className="flex gap-4 text-sm">
383
+ <a href="#" className="text-muted-foreground hover:text-foreground">Dashboard</a>
384
+ <a href="#" className="text-foreground font-medium">Users</a>
385
+ <a href="#" className="text-muted-foreground hover:text-foreground">Settings</a>
386
+ </nav>
387
+ </div>
388
+ </header>
389
+ <main className="p-6">
390
+ <WakaUserManagement
391
+ users={adminUsers}
392
+ permissions={extendedPermissions}
393
+ onSearchChange={(query) => console.log('Search:', query)}
394
+ onRoleFilterChange={(role) => console.log('Role filter:', role)}
395
+ onStatusFilterChange={(status) => console.log('Status filter:', status)}
396
+ onCreate={(user) => console.log('Create:', user)}
397
+ onEdit={(userId, data) => console.log('Edit:', userId, data)}
398
+ onDelete={(userId) => console.log('Delete:', userId)}
399
+ onRoleChange={(userId, role) => console.log('Role change:', userId, role)}
400
+ onStatusChange={(userId, status) => console.log('Status change:', userId, status)}
401
+ onResetPassword={(userId) => console.log('Reset password:', userId)}
402
+ onRefresh={() => console.log('Refresh')}
403
+ onExport={() => console.log('Export')}
404
+ title="Team Members"
405
+ description="Manage your organization's users and their permissions."
406
+ />
407
+ </main>
408
+ </div>
409
+ )
410
+ },
411
+ }