* feat(PBI-49): add debugProps helper + Vitest test
Adds lib/debug.ts with debugProps(id, component, file) that returns
data-debug-id and data-debug-label attrs in dev mode, empty object in
production. Adds __tests__/lib/debug.test.ts covering both modes.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* docs(PBI-49): add debug-id pattern doc + CLAUDE.md reference
Adds docs/patterns/debug-id.md documenting the named-component boundary
rule (6 punten), helper-voorbeeld, skip-criteria en motivatie voor
handmatige pad-argumenten. Voegt verwijzing toe aan CLAUDE.md
patterns-tabel.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* refactor(PBI-49): migrate 17 shared/ components to debugProps helper
Replace hardcoded data-debug-id + data-debug-label attribute pairs with
{...debugProps(id, component, file)} spread in all 17 components/shared/
files. Existing debug-ids preserved unchanged.
* feat(PBI-49): add debugProps to backlog/, sprint/, solo/ components
* feat(PBI-49): add debugProps to jobs/ + ideas/ components
* feat(PBI-49): add debugProps to products/ + settings/ + notifications/ components
* feat(PBI-49): add debugProps to admin/ + dashboard/ + dialogs/ + mobile/ + split-pane/
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* fix(PBI-49): use attr(data-debug-id) for debug tooltip in globals.css
* refactor(PBI-49): remove data-debug-label from debugProps helper + test
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* refactor(PBI-49): strip unused component/file args from debugProps in shared/
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to StatusBar, NavBar, PanelNavBar
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to components/sprint/*
- new-sprint-dialog: __submit on submit button
- sprint-backlog: __list on SprintBacklogLeft + SprintBacklogRight scroll areas
- sprint-board-client: root wrapper div (display:contents) + __drag-overlay
- sprint-header: __title on goal button, __dates on dates button, __actions on action cluster
- sprint-run-controls: root on controls div, __start/__cancel on action buttons; __blockers-dialog on dialog content
- start-sprint-button: root on trigger button, __dialog on dialog content, __submit on submit button
- sync-active-sprint-cookie: no debug-id (returns null, side-effect only), comment added
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to components/backlog/*
* feat(PBI-49): add BEM sub-element data-debug-id to components/ideas/*
* feat(PBI-49): add BEM sub-element data-debug-id to components/dashboard/* + components/markdown.tsx
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to new-product-button
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to components/solo/*
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-elements to nav-status-indicators
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to components/jobs/*
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to components/products/*
* feat(PBI-49): add BEM sub-element data-debug-id to components/notifications/*
- answer-modal: __content (scroll area), __submit (footer)
- notifications-bridge: skip comment (bridge, non-rendering wrapper)
- notifications-realtime-mount: skip comment (returns null)
- notifications-sheet: __header, __items (questions list)
- push-toggle: __switch (button), __label (button text) on subscribed/unsubscribed states
* feat(PBI-49): add BEM sub-element data-debug-id to components/settings/*
- leave-product-button: root only (single-button component)
- min-quota-editor: __input (number input), __save (save button)
- profile-editor: __username (bio/short-description input), __save (submit)
- role-manager: __roles (checkbox list), __add (save button)
- token-manager: __tokens (active tokens list), __generate (create button)
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
* feat(PBI-49): add BEM sub-element data-debug-id to admin, auth, dialogs, entity-dialog, mobile, split-pane
* docs(PBI-49): add debug-labels BEM pattern doc + CLAUDE.md entry
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
---------
Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
240 lines
7.3 KiB
TypeScript
240 lines
7.3 KiB
TypeScript
'use client'
|
|
|
|
import { useState, useTransition } from 'react'
|
|
import { Role } from '@prisma/client'
|
|
import { Badge } from '@/components/ui/badge'
|
|
import { Button } from '@/components/ui/button'
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
DialogTrigger,
|
|
DialogClose,
|
|
} from '@/components/ui/dialog'
|
|
import {
|
|
Table,
|
|
TableBody,
|
|
TableCell,
|
|
TableHead,
|
|
TableHeader,
|
|
TableRow,
|
|
} from '@/components/ui/table'
|
|
import {
|
|
deleteUserAction,
|
|
updateUserRolesAction,
|
|
setMustResetPasswordAction,
|
|
} from '@/actions/admin/users'
|
|
import { debugProps } from '@/lib/debug'
|
|
|
|
type UserWithRoles = {
|
|
id: string
|
|
username: string
|
|
email: string | null
|
|
must_reset_password: boolean
|
|
created_at: Date
|
|
roles: { role: Role }[]
|
|
}
|
|
|
|
const ALL_ROLES: Role[] = [Role.PRODUCT_OWNER, Role.SCRUM_MASTER, Role.DEVELOPER, Role.ADMIN]
|
|
|
|
const ROLE_LABEL: Record<Role, string> = {
|
|
PRODUCT_OWNER: 'Product Owner',
|
|
SCRUM_MASTER: 'Scrum Master',
|
|
DEVELOPER: 'Developer',
|
|
ADMIN: 'Admin',
|
|
}
|
|
|
|
function RoleBadge({ role }: { role: Role }) {
|
|
const cls =
|
|
role === Role.ADMIN
|
|
? 'bg-status-done text-white border-transparent'
|
|
: role === Role.PRODUCT_OWNER
|
|
? 'bg-status-in-progress text-white border-transparent'
|
|
: role === Role.SCRUM_MASTER
|
|
? 'bg-priority-medium text-white border-transparent'
|
|
: 'bg-secondary text-secondary-foreground'
|
|
return <Badge className={cls}>{ROLE_LABEL[role]}</Badge>
|
|
}
|
|
|
|
function RolesDialog({ user, currentUserId }: { user: UserWithRoles; currentUserId: string }) {
|
|
const [open, setOpen] = useState(false)
|
|
const [selected, setSelected] = useState<Set<Role>>(new Set(user.roles.map(r => r.role)))
|
|
const [pending, startTransition] = useTransition()
|
|
const isSelf = user.id === currentUserId
|
|
|
|
function toggle(role: Role) {
|
|
setSelected(prev => {
|
|
const next = new Set(prev)
|
|
if (next.has(role)) next.delete(role)
|
|
else next.add(role)
|
|
return next
|
|
})
|
|
}
|
|
|
|
function handleSave() {
|
|
startTransition(async () => {
|
|
await updateUserRolesAction(user.id, Array.from(selected))
|
|
setOpen(false)
|
|
})
|
|
}
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={setOpen}>
|
|
<DialogTrigger render={<Button variant="outline" size="sm" />}>
|
|
Rollen
|
|
</DialogTrigger>
|
|
<DialogContent>
|
|
<DialogHeader>
|
|
<DialogTitle>Rollen voor {user.username}</DialogTitle>
|
|
</DialogHeader>
|
|
<div className="flex flex-col gap-2 py-2">
|
|
{ALL_ROLES.map(role => {
|
|
const isDisabled = isSelf && role === Role.ADMIN && selected.has(role)
|
|
return (
|
|
<label key={role} className="flex items-center gap-2 cursor-pointer">
|
|
<input
|
|
type="checkbox"
|
|
checked={selected.has(role)}
|
|
disabled={isDisabled}
|
|
onChange={() => toggle(role)}
|
|
className="rounded border-border"
|
|
/>
|
|
<span className="text-sm">{ROLE_LABEL[role]}</span>
|
|
{isDisabled && <span className="text-xs text-muted-foreground">(eigen rol)</span>}
|
|
</label>
|
|
)
|
|
})}
|
|
</div>
|
|
<DialogFooter showCloseButton>
|
|
<Button onClick={handleSave} disabled={pending}>
|
|
{pending ? 'Opslaan…' : 'Opslaan'}
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|
|
|
|
function DeleteDialog({ user, currentUserId }: { user: UserWithRoles; currentUserId: string }) {
|
|
const [open, setOpen] = useState(false)
|
|
const [pending, startTransition] = useTransition()
|
|
const isSelf = user.id === currentUserId
|
|
|
|
function handleDelete() {
|
|
startTransition(async () => {
|
|
await deleteUserAction(user.id)
|
|
setOpen(false)
|
|
})
|
|
}
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={setOpen}>
|
|
<DialogTrigger
|
|
render={
|
|
<Button
|
|
variant="destructive"
|
|
size="sm"
|
|
disabled={isSelf}
|
|
title={isSelf ? 'Zelfverwijdering niet toegestaan' : undefined}
|
|
/>
|
|
}
|
|
>
|
|
Verwijder
|
|
</DialogTrigger>
|
|
<DialogContent>
|
|
<DialogHeader>
|
|
<DialogTitle>Gebruiker verwijderen</DialogTitle>
|
|
</DialogHeader>
|
|
<p className="text-sm text-muted-foreground">
|
|
Weet je zeker dat je <strong>{user.username}</strong> wilt verwijderen? Dit kan niet ongedaan worden gemaakt.
|
|
</p>
|
|
<DialogFooter>
|
|
<DialogClose render={<Button variant="outline" />}>Annuleer</DialogClose>
|
|
<Button variant="destructive" onClick={handleDelete} disabled={pending}>
|
|
{pending ? 'Verwijderen…' : 'Verwijderen'}
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|
|
|
|
function ResetToggle({ user }: { user: UserWithRoles }) {
|
|
const [pending, startTransition] = useTransition()
|
|
|
|
function handleToggle() {
|
|
startTransition(async () => {
|
|
await setMustResetPasswordAction(user.id, !user.must_reset_password)
|
|
})
|
|
}
|
|
|
|
return (
|
|
<Button
|
|
variant={user.must_reset_password ? 'default' : 'outline'}
|
|
size="sm"
|
|
onClick={handleToggle}
|
|
disabled={pending}
|
|
title="Forceer wachtwoord-reset bij volgende login"
|
|
>
|
|
{user.must_reset_password ? 'Reset gepland' : 'Reset pw'}
|
|
</Button>
|
|
)
|
|
}
|
|
|
|
export function UsersTable({
|
|
users,
|
|
currentUserId,
|
|
}: {
|
|
users: UserWithRoles[]
|
|
currentUserId: string
|
|
}) {
|
|
return (
|
|
<Table {...debugProps('admin-users-table', 'UsersTable', 'components/admin/users-table.tsx')}>
|
|
<TableHeader data-debug-id="admin-users-table__header">
|
|
<TableRow>
|
|
<TableHead>Gebruiker</TableHead>
|
|
<TableHead>Email</TableHead>
|
|
<TableHead>Rollen</TableHead>
|
|
<TableHead>Reset pw</TableHead>
|
|
<TableHead>Aangemaakt</TableHead>
|
|
<TableHead className="text-right">Acties</TableHead>
|
|
</TableRow>
|
|
</TableHeader>
|
|
<TableBody data-debug-id="admin-users-table__table">
|
|
{users.map(user => (
|
|
<TableRow key={user.id}>
|
|
<TableCell className="font-medium">{user.username}</TableCell>
|
|
<TableCell className="text-muted-foreground">{user.email ?? '—'}</TableCell>
|
|
<TableCell>
|
|
<div className="flex flex-wrap gap-1">
|
|
{user.roles.map(r => (
|
|
<RoleBadge key={r.role} role={r.role} />
|
|
))}
|
|
{user.roles.length === 0 && <span className="text-muted-foreground text-xs">Geen</span>}
|
|
</div>
|
|
</TableCell>
|
|
<TableCell>
|
|
{user.must_reset_password ? (
|
|
<Badge className="bg-priority-high text-white border-transparent">Ja</Badge>
|
|
) : (
|
|
<span className="text-muted-foreground text-xs">—</span>
|
|
)}
|
|
</TableCell>
|
|
<TableCell className="text-muted-foreground text-xs">
|
|
{new Date(user.created_at).toLocaleDateString('nl-NL')}
|
|
</TableCell>
|
|
<TableCell>
|
|
<div className="flex gap-2 justify-end">
|
|
<ResetToggle user={user} />
|
|
<RolesDialog user={user} currentUserId={currentUserId} />
|
|
<DeleteDialog user={user} currentUserId={currentUserId} />
|
|
</div>
|
|
</TableCell>
|
|
</TableRow>
|
|
))}
|
|
</TableBody>
|
|
</Table>
|
|
)
|
|
}
|