* 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>
161 lines
5.3 KiB
TypeScript
161 lines
5.3 KiB
TypeScript
'use client'
|
|
|
|
import { usePathname, useRouter } from 'next/navigation'
|
|
import { useState, useTransition } from 'react'
|
|
import { Check, ChevronDown } from 'lucide-react'
|
|
import { toast } from 'sonner'
|
|
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip'
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuSeparator,
|
|
DropdownMenuTrigger,
|
|
} from '@/components/ui/dropdown-menu'
|
|
import { cn } from '@/lib/utils'
|
|
import { setActiveSprintAction } from '@/actions/active-sprint'
|
|
import type { SprintStatusApi } from '@/lib/task-status'
|
|
import { debugProps } from '@/lib/debug'
|
|
|
|
type SprintItem = { id: string; code: string; sprint_goal: string; status: SprintStatusApi }
|
|
|
|
interface SprintSwitcherProps {
|
|
productId: string
|
|
sprints: SprintItem[]
|
|
activeSprint: SprintItem | null
|
|
buildingSprintIds: string[]
|
|
}
|
|
|
|
const SPRINT_STATUS_LABEL: Record<SprintStatusApi, string> = {
|
|
open: 'Open',
|
|
closed: 'Gesloten',
|
|
archived: 'Gearchiveerd',
|
|
failed: 'Mislukt',
|
|
}
|
|
|
|
export function SprintSwitcher({
|
|
productId,
|
|
sprints,
|
|
activeSprint,
|
|
buildingSprintIds,
|
|
}: SprintSwitcherProps) {
|
|
const pathname = usePathname()
|
|
const router = useRouter()
|
|
const [isPending, startTransition] = useTransition()
|
|
const [showClosed, setShowClosed] = useState(false)
|
|
const buildingSet = new Set(buildingSprintIds)
|
|
|
|
const visibleSprints = sprints.filter(s => {
|
|
if (showClosed) return true
|
|
if (s.id === activeSprint?.id) return true
|
|
return s.status === 'open'
|
|
})
|
|
|
|
function handleSwitchSprint(sprintId: string) {
|
|
if (sprintId === activeSprint?.id) return
|
|
startTransition(async () => {
|
|
const result = await setActiveSprintAction(productId, sprintId)
|
|
if (result?.error) {
|
|
toast.error(typeof result.error === 'string' ? result.error : 'Wisselen mislukt')
|
|
return
|
|
}
|
|
if (pathname.includes('/sprint')) {
|
|
router.push(`/products/${productId}/sprint/${sprintId}`)
|
|
} else {
|
|
router.refresh()
|
|
}
|
|
})
|
|
}
|
|
|
|
if (sprints.length === 0) {
|
|
return (
|
|
<span {...debugProps('sprint-switcher')}>
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger
|
|
className="text-xs text-muted-foreground/50 px-2 cursor-not-allowed select-none"
|
|
aria-disabled="true"
|
|
>
|
|
Geen sprints
|
|
</TooltipTrigger>
|
|
<TooltipContent>Maak een sprint aan vanuit de Product Backlog</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</span>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<span {...debugProps('sprint-switcher')}>
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger
|
|
disabled={isPending}
|
|
className="flex items-center gap-1.5 text-xs text-muted-foreground hover:text-foreground transition-colors px-2 py-1 rounded-md hover:bg-surface-container focus:outline-none"
|
|
>
|
|
<span className="truncate max-w-[160px]">
|
|
{activeSprint ? activeSprint.code : 'Selecteer sprint'}
|
|
</span>
|
|
{activeSprint && (
|
|
<span
|
|
className={cn(
|
|
'text-[10px]',
|
|
buildingSet.has(activeSprint.id) ? 'text-warning' : 'text-muted-foreground',
|
|
)}
|
|
>
|
|
{buildingSet.has(activeSprint.id) ? 'BUILDING' : SPRINT_STATUS_LABEL[activeSprint.status]}
|
|
</span>
|
|
)}
|
|
<ChevronDown className="w-3 h-3 shrink-0 text-muted-foreground" />
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent align="center" className="w-80">
|
|
<button
|
|
type="button"
|
|
onClick={(e) => {
|
|
e.preventDefault()
|
|
setShowClosed(v => !v)
|
|
}}
|
|
className="flex items-center gap-2 w-full px-2 py-1.5 text-xs text-muted-foreground hover:bg-surface-container rounded-md"
|
|
>
|
|
<span
|
|
className={cn(
|
|
'inline-flex items-center justify-center w-3.5 h-3.5 rounded border',
|
|
showClosed ? 'bg-primary border-primary text-primary-foreground' : 'border-border',
|
|
)}
|
|
>
|
|
{showClosed && <Check className="w-3 h-3" />}
|
|
</span>
|
|
Toon afgeronde sprints
|
|
</button>
|
|
<DropdownMenuSeparator />
|
|
{visibleSprints.length === 0 ? (
|
|
<div className="px-2 py-2 text-xs text-muted-foreground/70 italic">
|
|
Geen open sprints
|
|
</div>
|
|
) : (
|
|
visibleSprints.map(s => (
|
|
<DropdownMenuItem
|
|
key={s.id}
|
|
onClick={() => handleSwitchSprint(s.id)}
|
|
className={cn(
|
|
'flex items-center gap-2',
|
|
s.id === activeSprint?.id && 'bg-primary-container text-primary-container-foreground font-medium',
|
|
)}
|
|
>
|
|
<span className="text-xs font-medium shrink-0">{s.code}</span>
|
|
<span className="text-xs truncate flex-1">{s.sprint_goal}</span>
|
|
<span
|
|
className={cn(
|
|
'text-[10px] shrink-0',
|
|
buildingSet.has(s.id) ? 'text-warning' : 'text-muted-foreground',
|
|
)}
|
|
>
|
|
{buildingSet.has(s.id) ? 'BUILDING' : SPRINT_STATUS_LABEL[s.status]}
|
|
</span>
|
|
</DropdownMenuItem>
|
|
))
|
|
)}
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
</span>
|
|
)
|
|
}
|