payload-reserve 1.6.0 → 2.0.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 (89) hide show
  1. package/README.md +40 -3
  2. package/dist/collections/Reservations.js +19 -7
  3. package/dist/collections/Reservations.js.map +1 -1
  4. package/dist/collections/Resources.js +11 -8
  5. package/dist/collections/Resources.js.map +1 -1
  6. package/dist/collections/Schedules.js +12 -6
  7. package/dist/collections/Schedules.js.map +1 -1
  8. package/dist/collections/Services.js +19 -10
  9. package/dist/collections/Services.js.map +1 -1
  10. package/dist/components/AvailabilityOverview/index.js +43 -18
  11. package/dist/components/AvailabilityOverview/index.js.map +1 -1
  12. package/dist/components/CalendarView/CalendarView.module.css +9 -0
  13. package/dist/components/CalendarView/LaneTimelineView.d.ts +4 -1
  14. package/dist/components/CalendarView/LaneTimelineView.js +17 -12
  15. package/dist/components/CalendarView/LaneTimelineView.js.map +1 -1
  16. package/dist/components/CalendarView/index.js +134 -44
  17. package/dist/components/CalendarView/index.js.map +1 -1
  18. package/dist/components/CustomerField/index.js +8 -3
  19. package/dist/components/CustomerField/index.js.map +1 -1
  20. package/dist/components/DashboardWidget/DashboardWidgetServer.js +79 -17
  21. package/dist/components/DashboardWidget/DashboardWidgetServer.js.map +1 -1
  22. package/dist/defaults.js +42 -8
  23. package/dist/defaults.js.map +1 -1
  24. package/dist/endpoints/cancelBooking.js +1 -1
  25. package/dist/endpoints/cancelBooking.js.map +1 -1
  26. package/dist/endpoints/checkAvailability.js +56 -7
  27. package/dist/endpoints/checkAvailability.js.map +1 -1
  28. package/dist/endpoints/createBooking.js +19 -10
  29. package/dist/endpoints/createBooking.js.map +1 -1
  30. package/dist/endpoints/customerSearch.js +5 -2
  31. package/dist/endpoints/customerSearch.js.map +1 -1
  32. package/dist/endpoints/getSlots.js +56 -7
  33. package/dist/endpoints/getSlots.js.map +1 -1
  34. package/dist/endpoints/resourceAvailability.d.ts +2 -1
  35. package/dist/endpoints/resourceAvailability.js +85 -25
  36. package/dist/endpoints/resourceAvailability.js.map +1 -1
  37. package/dist/hooks/reservations/calculateEndTime.js +48 -20
  38. package/dist/hooks/reservations/calculateEndTime.js.map +1 -1
  39. package/dist/hooks/reservations/enforceCustomerOwnership.d.ts +11 -0
  40. package/dist/hooks/reservations/enforceCustomerOwnership.js +30 -0
  41. package/dist/hooks/reservations/enforceCustomerOwnership.js.map +1 -0
  42. package/dist/hooks/reservations/onStatusChange.js +10 -4
  43. package/dist/hooks/reservations/onStatusChange.js.map +1 -1
  44. package/dist/hooks/reservations/validateCancellation.js +3 -2
  45. package/dist/hooks/reservations/validateCancellation.js.map +1 -1
  46. package/dist/hooks/reservations/validateConflicts.js +23 -4
  47. package/dist/hooks/reservations/validateConflicts.js.map +1 -1
  48. package/dist/hooks/reservations/validateGuestBooking.js +3 -4
  49. package/dist/hooks/reservations/validateGuestBooking.js.map +1 -1
  50. package/dist/hooks/reservations/validateStatusTransition.js +2 -2
  51. package/dist/hooks/reservations/validateStatusTransition.js.map +1 -1
  52. package/dist/hooks/users/provisionStaffResource.js +5 -8
  53. package/dist/hooks/users/provisionStaffResource.js.map +1 -1
  54. package/dist/plugin.js +81 -13
  55. package/dist/plugin.js.map +1 -1
  56. package/dist/services/AvailabilityService.d.ts +54 -2
  57. package/dist/services/AvailabilityService.js +180 -46
  58. package/dist/services/AvailabilityService.js.map +1 -1
  59. package/dist/translations/ar.json +1 -0
  60. package/dist/translations/de.json +1 -0
  61. package/dist/translations/en.json +1 -0
  62. package/dist/translations/es.json +1 -0
  63. package/dist/translations/fa.json +1 -0
  64. package/dist/translations/fr.json +1 -0
  65. package/dist/translations/hi.json +1 -0
  66. package/dist/translations/id.json +1 -0
  67. package/dist/translations/pl.json +1 -0
  68. package/dist/translations/ru.json +1 -0
  69. package/dist/translations/tr.json +1 -0
  70. package/dist/translations/zh.json +1 -0
  71. package/dist/types.d.ts +39 -1
  72. package/dist/types.js +2 -0
  73. package/dist/types.js.map +1 -1
  74. package/dist/utilities/collectionOverrides.d.ts +14 -0
  75. package/dist/utilities/collectionOverrides.js +47 -0
  76. package/dist/utilities/collectionOverrides.js.map +1 -0
  77. package/dist/utilities/ownerAccess.d.ts +6 -0
  78. package/dist/utilities/ownerAccess.js +25 -12
  79. package/dist/utilities/ownerAccess.js.map +1 -1
  80. package/dist/utilities/reservationChanges.d.ts +17 -0
  81. package/dist/utilities/reservationChanges.js +88 -0
  82. package/dist/utilities/reservationChanges.js.map +1 -0
  83. package/dist/utilities/scheduleUtils.d.ts +14 -8
  84. package/dist/utilities/scheduleUtils.js +26 -19
  85. package/dist/utilities/scheduleUtils.js.map +1 -1
  86. package/dist/utilities/timezoneUtils.d.ts +39 -0
  87. package/dist/utilities/timezoneUtils.js +134 -0
  88. package/dist/utilities/timezoneUtils.js.map +1 -0
  89. package/package.json +1 -1
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/components/CalendarView/index.tsx"],"sourcesContent":["'use client'\nimport type { AdminViewServerProps } from 'payload'\n\nimport { useConfig, useDocumentDrawer, useTranslation } from '@payloadcms/ui'\nimport React, { Fragment, useCallback, useEffect, useMemo, useRef, useState } from 'react'\n\nimport type { PluginT } from '../../translations/index.js'\nimport type { SlotInfo } from '../../utilities/computeSlotStates.js'\n\nimport { computeSlotStates } from '../../utilities/computeSlotStates.js'\nimport { statusToI18nKey } from '../../utilities/i18nUtils.js'\nimport { localDayKey } from '../../utilities/slotUtils.js'\nimport { useTenantFilter } from '../../utilities/useTenantFilter.js'\nimport styles from './CalendarView.module.css'\nimport { LaneTimelineView } from './LaneTimelineView.js'\nimport { useResourceAvailability } from './useResourceAvailability.js'\n\ntype ViewMode = 'day' | 'lanes' | 'month' | 'pending' | 'week'\n\ntype ReservationItem = {\n endTime?: string\n guestCount?: number\n resource?: { id?: string; name?: string } | string\n service?: { name?: string } | string\n startTime?: string\n}\n\ntype Reservation = {\n customer?: { firstName?: string; lastName?: string; name?: string } | string\n endTime?: string\n id: string\n items?: ReservationItem[]\n resource?: { id?: string; name?: string } | string\n service?: { name?: string } | string\n startTime: string\n status: string\n}\n\ntype ResourceOption = {\n id: string\n name: string\n}\n\n// Built-in status → CSS class map (for known statuses; custom statuses use inline style)\nconst STATUS_CLASS_MAP: Record<string, string> = {\n cancelled: styles.statusCancelled,\n completed: styles.statusCompleted,\n confirmed: styles.statusConfirmed,\n 'no-show': styles.statusNoShow,\n pending: styles.statusPending,\n}\n\n// Built-in default colors for known statuses\nconst BUILTIN_STATUS_COLORS: Record<string, string> = {\n cancelled: '#e5e7eb',\n completed: '#d1fae5',\n confirmed: '#dbeafe',\n 'no-show': '#fee2e2',\n pending: '#fef3c7',\n}\n\n// Palette for auto-assigning colors to custom statuses\nconst CUSTOM_STATUS_PALETTE = ['#fde68a', '#c7d2fe', '#a7f3d0', '#fca5a5', '#fdba74']\n\nexport const CalendarView: React.FC<AdminViewServerProps> = () => {\n const { config } = useConfig()\n const { t: _t } = useTranslation()\n const t = _t as PluginT\n\n const slugs = config.admin?.custom?.reservationSlugs\n const reservationSlug = slugs?.reservations ?? 'reservations'\n const apiUrl = `${config.serverURL ?? ''}${config.routes.api}/${reservationSlug}`\n const apiBase = `${config.serverURL ?? ''}${config.routes.api}`\n const resourceSlug = slugs?.resources ?? 'resources'\n const reservationTenantParams = useTenantFilter(reservationSlug)\n const resourceTenantParams = useTenantFilter(resourceSlug)\n\n const statusMachine = config.admin?.custom?.reservationStatusMachine as\n | {\n blockingStatuses?: string[]\n defaultStatus?: string\n statuses?: string[]\n terminalStatuses?: string[]\n transitions?: Record<string, string[]>\n }\n | undefined\n\n // The initial/pending status (what \"pending\" view shows)\n const defaultStatus = statusMachine?.defaultStatus ?? 'pending'\n\n // Build STATUS_COLORS dynamically: built-ins first, then auto-assign palette for custom statuses\n const STATUS_COLORS = useMemo<Record<string, string>>(() => {\n const colors = { ...BUILTIN_STATUS_COLORS }\n const statuses = statusMachine?.statuses ?? []\n statuses.forEach((s, i) => {\n if (!colors[s]) {\n colors[s] = CUSTOM_STATUS_PALETTE[i % CUSTOM_STATUS_PALETTE.length]\n }\n })\n return colors\n }, [statusMachine])\n\n // Derive confirm/cancel target statuses from config transitions\n // \"confirm\" = first non-terminal transition from defaultStatus\n // \"cancel\" = first terminal transition from defaultStatus (or fallback: 'cancelled')\n const { cancelStatus, confirmStatus } = useMemo(() => {\n const terminalStatuses = statusMachine?.terminalStatuses ?? ['completed', 'cancelled', 'no-show']\n const transitions = statusMachine?.transitions ?? {}\n const defaultTransitions: string[] = transitions[defaultStatus] ?? []\n\n const nonTerminal = defaultTransitions.find((s) => !terminalStatuses.includes(s))\n const terminal = defaultTransitions.find((s) => terminalStatuses.includes(s))\n\n return {\n cancelStatus: terminal ?? 'cancelled',\n confirmStatus: nonTerminal ?? 'confirmed',\n }\n }, [statusMachine, defaultStatus])\n\n const STATUS_LABELS = useMemo<Record<string, string>>(() => {\n const statuses = statusMachine?.statuses ?? [\n 'pending',\n 'confirmed',\n 'completed',\n 'cancelled',\n 'no-show',\n ]\n const labels: Record<string, string> = {}\n for (const s of statuses) {\n // Attempt to look up a translation key, e.g. reservation:statusPending\n const key = statusToI18nKey(s)\n const translated = t(key)\n // If translation returns the key itself, it's missing — fall back to capitalized status name\n labels[s] = translated !== key ? translated : s.charAt(0).toUpperCase() + s.slice(1)\n }\n return labels\n }, [statusMachine, t])\n\n const [currentDate, setCurrentDate] = useState(() => new Date())\n const [viewMode, setViewMode] = useState<ViewMode>('month')\n const [reservations, setReservations] = useState<Reservation[]>([])\n const [loading, setLoading] = useState(true)\n const [drawerDocId, setDrawerDocId] = useState<null | string>(null)\n const [initialData, setInitialData] = useState<Record<string, unknown> | undefined>(undefined)\n\n // Resource filter state\n const [resources, setResources] = useState<ResourceOption[]>([])\n const [selectedResourceId, setSelectedResourceId] = useState<string>('')\n\n // Pending tab state\n const [pendingReservations, setPendingReservations] = useState<Reservation[]>([])\n const [pendingCount, setPendingCount] = useState(0)\n const [selectedIds, setSelectedIds] = useState<Set<string>>(() => new Set())\n const [confirmingIds, setConfirmingIds] = useState<Set<string>>(() => new Set())\n const [actionFeedback, setActionFeedback] = useState<{\n message: string\n type: 'error' | 'success'\n } | null>(null)\n\n const [DocumentDrawer, , { openDrawer }] = useDocumentDrawer({\n id: drawerDocId ?? undefined,\n collectionSlug: reservationSlug,\n })\n\n const pendingDrawerOpen = useRef(false)\n\n useEffect(() => {\n if (pendingDrawerOpen.current) {\n pendingDrawerOpen.current = false\n openDrawer()\n }\n })\n\n // Fetch active resources for filter dropdown\n useEffect(() => {\n const fetchResources = async () => {\n try {\n const params = new URLSearchParams({\n depth: '0',\n limit: '100',\n sort: 'name',\n 'where[active][equals]': 'true',\n ...resourceTenantParams,\n })\n const url = `${config.serverURL ?? ''}${config.routes.api}/${resourceSlug}?${params}`\n const response = await fetch(url)\n const result = await response.json()\n const docs: Array<{ id: string; name?: string }> = result.docs ?? []\n setResources(docs.map((d) => ({ id: d.id, name: d.name ?? '' })))\n } catch {\n setResources([])\n }\n }\n void fetchResources()\n }, [config.routes.api, config.serverURL, resourceSlug, resourceTenantParams])\n\n const { rangeEnd, rangeStart } = useMemo(() => {\n const start = new Date(currentDate)\n const end = new Date(currentDate)\n\n if (viewMode === 'month') {\n start.setDate(1)\n start.setDate(start.getDate() - start.getDay())\n end.setMonth(end.getMonth() + 1, 0)\n end.setDate(end.getDate() + (6 - end.getDay()))\n } else if (viewMode === 'week') {\n const dayOfWeek = start.getDay()\n start.setDate(start.getDate() - dayOfWeek)\n end.setDate(start.getDate() + 6)\n }\n start.setHours(0, 0, 0, 0)\n end.setHours(23, 59, 59, 999)\n\n return { rangeEnd: end, rangeStart: start }\n }, [currentDate, viewMode])\n\n // Availability data for the selected resource (null when no resource selected — grid unshaded)\n const { data: availability } = useResourceAvailability(\n apiBase,\n selectedResourceId || undefined,\n rangeStart,\n rangeEnd,\n )\n\n const fetchReservations = useCallback(async () => {\n setLoading(true)\n try {\n const params = new URLSearchParams({\n depth: '1',\n limit: '500',\n sort: 'startTime',\n 'where[startTime][greater_than_equal]': rangeStart.toISOString(),\n 'where[startTime][less_than_equal]': rangeEnd.toISOString(),\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n setReservations(result.docs ?? [])\n } catch {\n setReservations([])\n }\n setLoading(false)\n }, [rangeStart, rangeEnd, apiUrl, reservationTenantParams])\n\n useEffect(() => {\n void fetchReservations()\n }, [fetchReservations])\n\n // Fetch pending count (always, for badge) — uses defaultStatus from config\n const fetchPendingCount = useCallback(async () => {\n try {\n const params = new URLSearchParams({\n limit: '0',\n 'where[status][equals]': defaultStatus,\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n setPendingCount(result.totalDocs ?? 0)\n } catch {\n // silently ignore\n }\n }, [apiUrl, defaultStatus, reservationTenantParams])\n\n useEffect(() => {\n void fetchPendingCount()\n }, [fetchPendingCount])\n\n // Fetch pending reservations when tab is active — uses defaultStatus from config\n const fetchPendingReservations = useCallback(async () => {\n try {\n const params = new URLSearchParams({\n depth: '1',\n limit: '500',\n sort: 'startTime',\n 'where[status][equals]': defaultStatus,\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n setPendingReservations(result.docs ?? [])\n } catch {\n setPendingReservations([])\n }\n }, [apiUrl, defaultStatus, reservationTenantParams])\n\n useEffect(() => {\n if (viewMode === 'pending') {\n void fetchPendingReservations()\n }\n }, [viewMode, fetchPendingReservations])\n\n // Client-side resource filtering\n const matchesResourceFilter = useCallback(\n (r: Reservation): boolean => {\n if (!selectedResourceId) {return true}\n // Check top-level resource\n const topId = typeof r.resource === 'string' ? r.resource : r.resource?.id\n if (topId === selectedResourceId) {return true}\n // Check items array for multi-resource bookings\n if (r.items && r.items.length > 0) {\n return r.items.some((item) => {\n const itemId =\n typeof item.resource === 'string' ? item.resource : item.resource?.id\n return itemId === selectedResourceId\n })\n }\n return false\n },\n [selectedResourceId],\n )\n\n const filteredReservations = useMemo(\n () => reservations.filter(matchesResourceFilter),\n [reservations, matchesResourceFilter],\n )\n\n const filteredPendingReservations = useMemo(\n () => pendingReservations.filter(matchesResourceFilter),\n [pendingReservations, matchesResourceFilter],\n )\n\n // Clear selection when leaving pending view or changing resource filter\n useEffect(() => {\n if (viewMode !== 'pending') {\n setSelectedIds(new Set())\n setActionFeedback(null)\n }\n }, [viewMode])\n\n useEffect(() => {\n setSelectedIds(new Set())\n }, [selectedResourceId])\n\n // Auto-clear feedback toast\n useEffect(() => {\n if (!actionFeedback) {return}\n const timer = setTimeout(() => setActionFeedback(null), 4000)\n return () => clearTimeout(timer)\n }, [actionFeedback])\n\n const patchReservation = useCallback(\n async (id: string, data: Record<string, unknown>): Promise<boolean> => {\n try {\n const response = await fetch(`${apiUrl}/${id}`, {\n body: JSON.stringify(data),\n headers: { 'Content-Type': 'application/json' },\n method: 'PATCH',\n })\n return response.ok\n } catch {\n return false\n }\n },\n [apiUrl],\n )\n\n // Uses confirmStatus derived from config transitions\n const handleQuickConfirm = useCallback(\n async (id: string) => {\n setConfirmingIds((prev) => new Set(prev).add(id))\n const ok = await patchReservation(id, { status: confirmStatus })\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n setActionFeedback({\n type: ok ? 'success' : 'error',\n message: ok\n ? t('reservation:pendingConfirmSuccess')\n : t('reservation:pendingConfirmError'),\n })\n if (ok) {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n void fetchPendingReservations()\n void fetchPendingCount()\n }\n },\n [patchReservation, fetchPendingReservations, fetchPendingCount, t, confirmStatus],\n )\n\n // Uses cancelStatus derived from config transitions\n const handleQuickCancel = useCallback(\n async (id: string) => {\n setConfirmingIds((prev) => new Set(prev).add(id))\n const ok = await patchReservation(id, { status: cancelStatus })\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n setActionFeedback({\n type: ok ? 'success' : 'error',\n message: ok\n ? t('reservation:pendingCancelSuccess')\n : t('reservation:pendingCancelError'),\n })\n if (ok) {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n void fetchPendingReservations()\n void fetchPendingCount()\n }\n },\n [patchReservation, fetchPendingReservations, fetchPendingCount, t, cancelStatus],\n )\n\n const confirmSelected = useCallback(async () => {\n const ids = Array.from(selectedIds)\n if (ids.length === 0) {return}\n\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n for (const id of ids) {next.add(id)}\n return next\n })\n\n const results = await Promise.allSettled(\n ids.map((id) => patchReservation(id, { status: confirmStatus })),\n )\n\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n for (const id of ids) {next.delete(id)}\n return next\n })\n\n const succeeded = results.filter(\n (r) => r.status === 'fulfilled' && r.value,\n ).length\n const failed = ids.length - succeeded\n\n if (failed === 0) {\n setActionFeedback({\n type: 'success',\n message: `${succeeded} ${t('reservation:pendingConfirmSuccess').toLowerCase()}`,\n })\n } else {\n setActionFeedback({\n type: failed === ids.length ? 'error' : 'success',\n message: t('reservation:pendingBulkConfirmSuccess')\n .replace('{{succeeded}}', String(succeeded))\n .replace('{{failed}}', String(failed)),\n })\n }\n\n setSelectedIds(new Set())\n void fetchPendingReservations()\n void fetchPendingCount()\n }, [selectedIds, patchReservation, fetchPendingReservations, fetchPendingCount, t, confirmStatus])\n\n const handleEventClick = useCallback((e: React.MouseEvent, id: string) => {\n e.stopPropagation()\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const handleEventKeyDown = useCallback((e: React.KeyboardEvent, id: string) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n e.stopPropagation()\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }\n }, [])\n\n const handleCreateNew = useCallback(() => {\n setDrawerDocId(null)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const handleDateClick = useCallback((date: Date) => {\n setDrawerDocId(null)\n setInitialData({ startTime: date.toISOString() })\n pendingDrawerOpen.current = true\n }, [])\n\n // Click-to-book: open new-reservation drawer pre-filled with startTime + optional resource\n const handleSlotClick = useCallback(\n (startIso: string) => {\n setDrawerDocId(null)\n setInitialData({\n ...(selectedResourceId ? { resource: selectedResourceId } : {}),\n startTime: startIso,\n })\n pendingDrawerOpen.current = true\n },\n [selectedResourceId],\n )\n\n // Lane-specific book: pre-fills both the specific resource and startTime\n const handleLaneBook = useCallback((resourceId: string, startIso: string) => {\n setDrawerDocId(null)\n setInitialData({ resource: resourceId, startTime: startIso })\n pendingDrawerOpen.current = true\n }, [])\n\n const openDocDrawer = useCallback((id: string) => {\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const navigate = useCallback(\n (direction: -1 | 1) => {\n setCurrentDate((prev) => {\n const next = new Date(prev)\n if (viewMode === 'month') {\n next.setMonth(next.getMonth() + direction)\n } else if (viewMode === 'week') {\n next.setDate(next.getDate() + 7 * direction)\n } else {\n // day, lanes: step one day at a time\n next.setDate(next.getDate() + direction)\n }\n return next\n })\n },\n [viewMode],\n )\n\n const goToToday = useCallback(() => setCurrentDate(new Date()), [])\n\n const getResName = (field: { name?: string } | string | undefined): string => {\n if (!field) {return ''}\n if (typeof field === 'string') {return ''}\n return field.name ?? ''\n }\n\n const getCustomerName = (field: Reservation['customer']): string => {\n if (!field) {return ''}\n if (typeof field === 'string') {return ''}\n const parts = [field.firstName, field.lastName].filter(Boolean)\n return parts.length > 0 ? parts.join(' ') : (field.name ?? '')\n }\n\n const getEventLabel = (r: Reservation, compact: boolean) => {\n const time = new Date(r.startTime).toLocaleTimeString([], {\n hour: '2-digit',\n minute: '2-digit',\n })\n const serviceName = getResName(r.service)\n if (compact) {\n return `${time} ${serviceName}`.trim()\n }\n const customerName = getCustomerName(r.customer)\n const parts = [time, serviceName, customerName].filter(Boolean)\n return parts.join(' - ')\n }\n\n // Returns all resource names for a reservation — from items array if present, otherwise top-level resource\n const getResourceNames = (r: Reservation): string[] => {\n if (r.items && r.items.length > 0) {\n const names = r.items\n .map((item) => getResName(item.resource))\n .filter((name) => name.length > 0)\n if (names.length > 0) {return names}\n }\n const single = getResName(r.resource)\n return single ? [single] : []\n }\n\n const getEventTooltip = (r: Reservation): string => {\n const serviceName = getResName(r.service) || t('reservation:calendarUnknownService')\n const startStr = new Date(r.startTime).toLocaleTimeString([], {\n hour: '2-digit',\n minute: '2-digit',\n })\n const endStr = r.endTime\n ? new Date(r.endTime).toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' })\n : '?'\n const customerName = getCustomerName(r.customer) || t('reservation:calendarUnknownCustomer')\n const resourceNames = getResourceNames(r)\n const resourceStr =\n resourceNames.length > 0\n ? resourceNames.join(', ')\n : t('reservation:calendarUnknownResource')\n const status = STATUS_LABELS[r.status] ?? r.status\n return [\n serviceName,\n `${startStr} - ${endStr}`,\n `${t('reservation:tooltipCustomer')} ${customerName}`,\n `${t('reservation:tooltipResource')} ${resourceStr}`,\n `${t('reservation:tooltipStatus')} ${status}`,\n ].join('\\n')\n }\n\n const renderEventItem = (r: Reservation, compact: boolean) => {\n // Use CSS class for built-in statuses; also apply inline background for custom statuses\n const cssClass = STATUS_CLASS_MAP[r.status] ?? ''\n const color = STATUS_COLORS[r.status]\n // Only apply inline style when there's no CSS class (custom statuses) or as a supplement\n const inlineStyle = cssClass ? undefined : { background: color }\n const hasItems = Array.isArray(r.items) && r.items.length > 0\n return (\n <div\n className={`${styles.eventItem} ${cssClass} ${hasItems && !compact ? styles.eventItemExpanded : ''}`}\n key={r.id}\n onClick={(e) => handleEventClick(e, r.id)}\n onKeyDown={(e) => handleEventKeyDown(e, r.id)}\n role=\"button\"\n style={inlineStyle}\n tabIndex={0}\n title={getEventTooltip(r)}\n >\n {getEventLabel(r, compact)}\n {hasItems && (\n <div className={styles.itemBadges}>\n {r.items!.map((it, i) => {\n const name = typeof it.resource === 'object' ? it.resource?.name : it.resource\n return (\n <span className={styles.itemBadge} key={i}>\n {String(name ?? '')}\n </span>\n )\n })}\n </div>\n )}\n </div>\n )\n }\n\n // Dynamic legend: iterates all statuses from the status machine config\n const renderStatusLegend = () => {\n const statuses = statusMachine?.statuses ?? Object.keys(BUILTIN_STATUS_COLORS)\n return (\n <div className={styles.statusLegend}>\n {statuses.map((key) => (\n <div className={styles.legendItem} key={key}>\n <span className={styles.legendDot} style={{ background: STATUS_COLORS[key] }} />\n {STATUS_LABELS[key] ?? key}\n </div>\n ))}\n </div>\n )\n }\n\n const renderCurrentTimeLine = (cellDate: Date, cellHour: number) => {\n const now = new Date()\n if (\n now.getFullYear() !== cellDate.getFullYear() ||\n now.getMonth() !== cellDate.getMonth() ||\n now.getDate() !== cellDate.getDate() ||\n now.getHours() !== cellHour\n ) {\n return null\n }\n const topPercent = (now.getMinutes() / 60) * 100\n return <div className={styles.currentTimeLine} style={{ top: `${topPercent}%` }} />\n }\n\n const renderMonthView = () => {\n const firstDay = new Date(currentDate.getFullYear(), currentDate.getMonth(), 1)\n const startDay = new Date(firstDay)\n startDay.setDate(startDay.getDate() - startDay.getDay())\n\n const days: Date[] = []\n const d = new Date(startDay)\n for (let i = 0; i < 42; i++) {\n days.push(new Date(d))\n d.setDate(d.getDate() + 1)\n }\n\n const today = new Date()\n const todayStr = `${today.getFullYear()}-${today.getMonth()}-${today.getDate()}`\n\n return (\n <div className={styles.monthGrid}>\n {[\n t('reservation:dayShortSun'),\n t('reservation:dayShortMon'),\n t('reservation:dayShortTue'),\n t('reservation:dayShortWed'),\n t('reservation:dayShortThu'),\n t('reservation:dayShortFri'),\n t('reservation:dayShortSat'),\n ].map((d) => (\n <div className={styles.dayHeader} key={d}>\n {d}\n </div>\n ))}\n {days.map((day, i) => {\n const dayStr = `${day.getFullYear()}-${day.getMonth()}-${day.getDate()}`\n const isToday = dayStr === todayStr\n const isOtherMonth = day.getMonth() !== currentDate.getMonth()\n const dayReservations = filteredReservations.filter((r) => {\n const rDate = new Date(r.startTime)\n return (\n rDate.getFullYear() === day.getFullYear() &&\n rDate.getMonth() === day.getMonth() &&\n rDate.getDate() === day.getDate()\n )\n })\n\n const clickDate = new Date(day)\n clickDate.setHours(9, 0, 0, 0)\n\n return (\n <div\n className={`${styles.dayCell} ${isOtherMonth ? styles.dayCellOtherMonth : ''} ${isToday ? styles.dayCellToday : ''}`}\n key={i}\n onClick={() => handleDateClick(clickDate)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n handleDateClick(clickDate)\n }\n }}\n role=\"button\"\n tabIndex={0}\n >\n <div className={styles.dayNumber}>{day.getDate()}</div>\n {dayReservations.map((r) => renderEventItem(r, true))}\n </div>\n )\n })}\n </div>\n )\n }\n\n const renderWeekView = () => {\n const startOfWeek = new Date(currentDate)\n startOfWeek.setDate(startOfWeek.getDate() - startOfWeek.getDay())\n startOfWeek.setHours(0, 0, 0, 0)\n\n const weekDays: Date[] = []\n for (let i = 0; i < 7; i++) {\n const d = new Date(startOfWeek)\n d.setDate(d.getDate() + i)\n weekDays.push(d)\n }\n\n const hours = Array.from({ length: 12 }, (_, i) => i + 7)\n // Grid bounds: hour 7 to hour 19 (end of last slot), step 60 min\n const gridStartHour = 7\n const gridEndHour = gridStartHour + hours.length // 19\n const gridStep = 60\n\n // Build per-day slot-state maps when a resource is selected\n const daySlotMaps = availability\n ? new Map(\n weekDays.map((day) => {\n const isoDay = localDayKey(day)\n const dayAvail = availability.days.find((d) => d.date === isoDay)\n const dayStart = new Date(day)\n dayStart.setHours(gridStartHour, 0, 0, 0)\n const dayEnd = new Date(day)\n dayEnd.setHours(gridEndHour, 0, 0, 0)\n const slots = dayAvail\n ? computeSlotStates({\n busy: availability.busy,\n capacityMode: availability.capacityMode,\n dayEnd,\n dayStart,\n quantity: availability.quantity,\n requiredPools: availability.requiredPools,\n shiftWindows: dayAvail.shiftWindows,\n step: gridStep,\n timeOff: dayAvail.timeOff,\n })\n : []\n // Index by slot start ISO for fast lookup\n const slotByStart = new Map(slots.map((s) => [s.start.toISOString(), s]))\n return [isoDay, slotByStart] as const\n }),\n )\n : null\n\n return (\n <div className={styles.weekView}>\n <div className={styles.dayHeader} />\n {weekDays.map((d, i) => (\n <div className={styles.dayHeader} key={i}>\n {d.toLocaleDateString([], { day: 'numeric', month: 'numeric', weekday: 'short' })}\n </div>\n ))}\n {hours.map((hour) => (\n <Fragment key={`row-${hour}`}>\n <div className={styles.timeLabel}>\n {hour.toString().padStart(2, '0')}:00\n </div>\n {weekDays.map((day, di) => {\n const cellReservations = filteredReservations.filter((r) => {\n const rDate = new Date(r.startTime)\n return (\n rDate.getFullYear() === day.getFullYear() &&\n rDate.getMonth() === day.getMonth() &&\n rDate.getDate() === day.getDate() &&\n rDate.getHours() === hour\n )\n })\n const clickDate = new Date(day)\n clickDate.setHours(hour, 0, 0, 0)\n\n // Slot state (only when a resource is selected)\n const isoDay = localDayKey(day)\n const slotMap = daySlotMaps?.get(isoDay)\n const slotInfo = slotMap?.get(clickDate.toISOString()) ?? null\n\n // Derive cell CSS class and interactivity based on slot state\n let slotClass = ''\n let isNonInteractive = false\n if (slotInfo) {\n if (slotInfo.state === 'off-shift') {\n slotClass = styles.slotOffShift\n isNonInteractive = true\n } else if (slotInfo.state === 'time-off') {\n slotClass = styles.slotTimeOff\n isNonInteractive = true\n } else if (slotInfo.state === 'full') {\n slotClass = styles.slotFull\n isNonInteractive = true\n } else {\n slotClass = styles.slotFree\n }\n }\n\n // Time-off label: show type/reason from dayAvail when in time-off state\n const dayAvail = availability?.days.find((d) => d.date === isoDay)\n const timeOffEntry =\n slotInfo?.state === 'time-off'\n ? dayAvail?.timeOff.find(\n (to) =>\n new Date(to.start) <= clickDate && clickDate < new Date(to.end),\n )\n : undefined\n const timeOffLabel = timeOffEntry?.type ?? timeOffEntry?.reason ?? null\n\n const handleClick = isNonInteractive\n ? undefined\n : availability\n ? () => handleSlotClick(clickDate.toISOString())\n : () => handleDateClick(clickDate)\n const handleKeyDown = isNonInteractive\n ? undefined\n : (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n if (availability) {\n handleSlotClick(clickDate.toISOString())\n } else {\n handleDateClick(clickDate)\n }\n }\n }\n\n return (\n <div\n className={`${styles.weekCell} ${slotClass}`}\n key={`cell-${hour}-${di}`}\n onClick={handleClick}\n onKeyDown={handleKeyDown}\n role=\"button\"\n tabIndex={isNonInteractive ? -1 : 0}\n >\n {renderCurrentTimeLine(day, hour)}\n {timeOffLabel && (\n <span className={styles.timeOffLabel}>{timeOffLabel}</span>\n )}\n {slotInfo && availability && availability.quantity > 1 && (\n <span className={styles.capacityBadge}>\n {slotInfo.occupancy}/{availability.quantity}\n </span>\n )}\n {cellReservations.map((r) => renderEventItem(r, false))}\n </div>\n )\n })}\n </Fragment>\n ))}\n </div>\n )\n }\n\n const renderDayView = () => {\n const hours = Array.from({ length: 14 }, (_, i) => i + 7)\n // Grid bounds: hour 7 to hour 21 (end of last slot), step 60 min\n const gridStartHour = 7\n const gridEndHour = gridStartHour + hours.length // 21\n const gridStep = 60\n\n // Build slot-state map for the current day when a resource is selected\n let daySlotMap: Map<string, SlotInfo> | null = null\n if (availability) {\n const isoDay = localDayKey(currentDate)\n const dayAvail = availability.days.find((d) => d.date === isoDay)\n const dayStart = new Date(currentDate)\n dayStart.setHours(gridStartHour, 0, 0, 0)\n const dayEnd = new Date(currentDate)\n dayEnd.setHours(gridEndHour, 0, 0, 0)\n const slots = dayAvail\n ? computeSlotStates({\n busy: availability.busy,\n capacityMode: availability.capacityMode,\n dayEnd,\n dayStart,\n quantity: availability.quantity,\n requiredPools: availability.requiredPools,\n shiftWindows: dayAvail.shiftWindows,\n step: gridStep,\n timeOff: dayAvail.timeOff,\n })\n : []\n daySlotMap = new Map(slots.map((s) => [s.start.toISOString(), s]))\n }\n\n return (\n <div className={styles.dayView}>\n {hours.map((hour) => {\n const hourReservations = filteredReservations.filter((r) => {\n const rDate = new Date(r.startTime)\n return (\n rDate.getFullYear() === currentDate.getFullYear() &&\n rDate.getMonth() === currentDate.getMonth() &&\n rDate.getDate() === currentDate.getDate() &&\n rDate.getHours() === hour\n )\n })\n const clickDate = new Date(currentDate)\n clickDate.setHours(hour, 0, 0, 0)\n\n // Slot state (only when a resource is selected)\n const slotInfo = daySlotMap?.get(clickDate.toISOString()) ?? null\n\n // Derive cell CSS class and interactivity based on slot state\n let slotClass = ''\n let isNonInteractive = false\n if (slotInfo) {\n if (slotInfo.state === 'off-shift') {\n slotClass = styles.slotOffShift\n isNonInteractive = true\n } else if (slotInfo.state === 'time-off') {\n slotClass = styles.slotTimeOff\n isNonInteractive = true\n } else if (slotInfo.state === 'full') {\n slotClass = styles.slotFull\n isNonInteractive = true\n } else {\n slotClass = styles.slotFree\n }\n }\n\n // Time-off label\n const isoDay = localDayKey(currentDate)\n const dayAvail = availability?.days.find((d) => d.date === isoDay)\n const timeOffEntry =\n slotInfo?.state === 'time-off'\n ? dayAvail?.timeOff.find(\n (to) => new Date(to.start) <= clickDate && clickDate < new Date(to.end),\n )\n : undefined\n const timeOffLabel = timeOffEntry?.type ?? timeOffEntry?.reason ?? null\n\n const handleClick = isNonInteractive\n ? undefined\n : availability\n ? () => handleSlotClick(clickDate.toISOString())\n : () => handleDateClick(clickDate)\n const handleKeyDown = isNonInteractive\n ? undefined\n : (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n if (availability) {\n handleSlotClick(clickDate.toISOString())\n } else {\n handleDateClick(clickDate)\n }\n }\n }\n\n return (\n <Fragment key={`row-${hour}`}>\n <div className={styles.timeLabel}>\n {hour.toString().padStart(2, '0')}:00\n </div>\n <div\n className={`${styles.dayViewCell} ${slotClass}`}\n onClick={handleClick}\n onKeyDown={handleKeyDown}\n role=\"button\"\n tabIndex={isNonInteractive ? -1 : 0}\n >\n {renderCurrentTimeLine(currentDate, hour)}\n {timeOffLabel && (\n <span className={styles.timeOffLabel}>{timeOffLabel}</span>\n )}\n {slotInfo && availability && availability.quantity > 1 && (\n <span className={styles.capacityBadge}>\n {slotInfo.occupancy}/{availability.quantity}\n </span>\n )}\n {hourReservations.map((r) => renderEventItem(r, false))}\n </div>\n </Fragment>\n )\n })}\n </div>\n )\n }\n\n const renderPendingView = () => {\n if (filteredPendingReservations.length === 0) {\n return <div className={styles.pendingEmpty}>{t('reservation:pendingEmpty')}</div>\n }\n\n const allSelected =\n filteredPendingReservations.length > 0 &&\n filteredPendingReservations.every((r) => selectedIds.has(r.id))\n\n const toggleSelectAll = () => {\n if (allSelected) {\n setSelectedIds(new Set())\n } else {\n setSelectedIds(new Set(filteredPendingReservations.map((r) => r.id)))\n }\n }\n\n const toggleSelect = (id: string) => {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n if (next.has(id)) {\n next.delete(id)\n } else {\n next.add(id)\n }\n return next\n })\n }\n\n const formatDateTime = (iso: string) => {\n const d = new Date(iso)\n return d.toLocaleString([], {\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n month: 'short',\n year: 'numeric',\n })\n }\n\n return (\n <div className={styles.pendingView}>\n <div className={styles.pendingToolbar}>\n <label className={styles.selectAllLabel}>\n <input\n aria-label={t('reservation:pendingSelectAll')}\n checked={allSelected}\n onChange={toggleSelectAll}\n type=\"checkbox\"\n />\n {t('reservation:pendingSelectAll')}\n </label>\n {selectedIds.size > 0 && (\n <button\n className={styles.bulkConfirmButton}\n disabled={confirmingIds.size > 0}\n onClick={() => void confirmSelected()}\n type=\"button\"\n >\n {confirmingIds.size > 0\n ? t('reservation:pendingConfirming')\n : t('reservation:pendingConfirmSelected').replace(\n '{{count}}',\n String(selectedIds.size),\n )}\n </button>\n )}\n </div>\n {actionFeedback && (\n <div\n className={`${styles.feedbackToast} ${actionFeedback.type === 'success' ? styles.feedbackSuccess : styles.feedbackError}`}\n >\n {actionFeedback.message}\n </div>\n )}\n <table className={styles.pendingTable}>\n <thead>\n <tr>\n <th aria-label={t('reservation:pendingSelectAll')} className={styles.pendingTh} />\n <th className={styles.pendingTh}>{t('reservation:fieldCustomer')}</th>\n <th className={styles.pendingTh}>{t('reservation:fieldService')}</th>\n <th className={styles.pendingTh}>{t('reservation:fieldResource')}</th>\n <th className={styles.pendingTh}>{t('reservation:pendingDateTime')}</th>\n <th className={styles.pendingTh}>{t('reservation:pendingActions')}</th>\n </tr>\n </thead>\n <tbody>\n {filteredPendingReservations.map((r) => {\n const isConfirming = confirmingIds.has(r.id)\n // Show all resources from items array if present, else top-level resource\n const resourceDisplay =\n getResourceNames(r).join(', ') || t('reservation:calendarUnknownResource')\n return (\n <tr className={styles.pendingRow} key={r.id}>\n <td className={styles.pendingTd}>\n <input\n aria-label={getCustomerName(r.customer) || r.id}\n checked={selectedIds.has(r.id)}\n onChange={() => toggleSelect(r.id)}\n type=\"checkbox\"\n />\n </td>\n <td className={styles.pendingTd}>\n <span\n className={styles.pendingCustomerLink}\n onClick={() => openDocDrawer(r.id)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n openDocDrawer(r.id)\n }\n }}\n role=\"button\"\n tabIndex={0}\n >\n {getCustomerName(r.customer) || t('reservation:calendarUnknownCustomer')}\n </span>\n </td>\n <td className={styles.pendingTd}>\n {getResName(r.service) || t('reservation:calendarUnknownService')}\n </td>\n <td className={styles.pendingTd}>{resourceDisplay}</td>\n <td className={styles.pendingTd}>{formatDateTime(r.startTime)}</td>\n <td className={styles.pendingTd}>\n <button\n className={styles.confirmButton}\n disabled={isConfirming}\n onClick={() => void handleQuickConfirm(r.id)}\n title={t('reservation:pendingConfirm')}\n type=\"button\"\n >\n &#x2713;\n </button>\n <button\n className={styles.cancelButton}\n disabled={isConfirming}\n onClick={() => void handleQuickCancel(r.id)}\n title={t('reservation:pendingCancel')}\n type=\"button\"\n >\n &#x2717;\n </button>\n </td>\n </tr>\n )\n })}\n </tbody>\n </table>\n </div>\n )\n }\n\n const dateLabel = useMemo(() => {\n if (viewMode === 'month') {\n return currentDate.toLocaleDateString([], { month: 'long', year: 'numeric' })\n }\n if (viewMode === 'week') {\n const startOfWeek = new Date(currentDate)\n startOfWeek.setDate(startOfWeek.getDate() - startOfWeek.getDay())\n const endOfWeek = new Date(startOfWeek)\n endOfWeek.setDate(endOfWeek.getDate() + 6)\n return `${startOfWeek.toLocaleDateString([], { day: 'numeric', month: 'short' })} - ${endOfWeek.toLocaleDateString([], { day: 'numeric', month: 'short', year: 'numeric' })}`\n }\n return currentDate.toLocaleDateString([], {\n day: 'numeric',\n month: 'long',\n weekday: 'long',\n year: 'numeric',\n })\n }, [currentDate, viewMode])\n\n const handleDrawerSave = useCallback(() => {\n void fetchReservations()\n void fetchPendingCount()\n if (viewMode === 'pending') {\n void fetchPendingReservations()\n }\n }, [fetchReservations, fetchPendingCount, fetchPendingReservations, viewMode])\n\n return (\n <div className={styles.wrapper}>\n <div className={styles.header}>\n {viewMode !== 'pending' && (\n <div className={styles.navButtons}>\n <button className={styles.navButton} onClick={() => navigate(-1)} type=\"button\">\n &larr;\n </button>\n <button className={styles.navButton} onClick={goToToday} type=\"button\">\n {t('reservation:calendarToday')}\n </button>\n <button className={styles.navButton} onClick={() => navigate(1)} type=\"button\">\n &rarr;\n </button>\n <span className={styles.currentDate}>{dateLabel}</span>\n </div>\n )}\n {viewMode === 'pending' && <div />}\n <div className={styles.viewToggle}>\n <button className={styles.createButton} onClick={handleCreateNew} type=\"button\">\n {t('reservation:calendarCreateNew')}\n </button>\n {([\n { key: 'month' as ViewMode, label: t('reservation:calendarMonth') },\n { key: 'week' as ViewMode, label: t('reservation:calendarWeek') },\n { key: 'day' as ViewMode, label: t('reservation:calendarDay') },\n { key: 'lanes' as ViewMode, label: t('reservation:calendarLanes') },\n { key: 'pending' as ViewMode, label: t('reservation:calendarPending') },\n ]).map(({ key, label }) => (\n <button\n className={`${styles.viewToggleButton} ${viewMode === key ? styles.viewToggleButtonActive : ''}`}\n key={key}\n onClick={() => setViewMode(key)}\n type=\"button\"\n >\n {label}\n {key === 'pending' && pendingCount > 0 && (\n <span className={styles.pendingBadge}>\n {selectedResourceId ? filteredPendingReservations.length : pendingCount}\n </span>\n )}\n </button>\n ))}\n </div>\n </div>\n {viewMode !== 'pending' && renderStatusLegend()}\n {resources.length > 1 && (\n <div className={styles.filterBar}>\n <select\n aria-label={t('reservation:filterByResource')}\n className={styles.resourceFilter}\n onChange={(e) => setSelectedResourceId(e.target.value)}\n value={selectedResourceId}\n >\n <option value=\"\">{t('reservation:filterAllResources')}</option>\n {resources.map((r) => (\n <option key={r.id} value={r.id}>\n {r.name}\n </option>\n ))}\n </select>\n </div>\n )}\n {loading && viewMode !== 'pending' && viewMode !== 'lanes' ? (\n <div className={styles.loading}>{t('reservation:calendarLoading')}</div>\n ) : (\n <>\n {viewMode === 'month' && renderMonthView()}\n {viewMode === 'week' && renderWeekView()}\n {viewMode === 'day' && renderDayView()}\n {viewMode === 'lanes' && (\n <LaneTimelineView\n apiBase={apiBase}\n day={currentDate}\n onBook={handleLaneBook}\n resources={\n selectedResourceId\n ? resources.filter((r) => r.id === selectedResourceId)\n : resources\n }\n />\n )}\n </>\n )}\n {viewMode === 'pending' && renderPendingView()}\n <DocumentDrawer initialData={initialData} onSave={handleDrawerSave} />\n </div>\n )\n}\n"],"names":["useConfig","useDocumentDrawer","useTranslation","React","Fragment","useCallback","useEffect","useMemo","useRef","useState","computeSlotStates","statusToI18nKey","localDayKey","useTenantFilter","styles","LaneTimelineView","useResourceAvailability","STATUS_CLASS_MAP","cancelled","statusCancelled","completed","statusCompleted","confirmed","statusConfirmed","statusNoShow","pending","statusPending","BUILTIN_STATUS_COLORS","CUSTOM_STATUS_PALETTE","CalendarView","config","t","_t","slugs","admin","custom","reservationSlugs","reservationSlug","reservations","apiUrl","serverURL","routes","api","apiBase","resourceSlug","resources","reservationTenantParams","resourceTenantParams","statusMachine","reservationStatusMachine","defaultStatus","STATUS_COLORS","colors","statuses","forEach","s","i","length","cancelStatus","confirmStatus","terminalStatuses","transitions","defaultTransitions","nonTerminal","find","includes","terminal","STATUS_LABELS","labels","key","translated","charAt","toUpperCase","slice","currentDate","setCurrentDate","Date","viewMode","setViewMode","setReservations","loading","setLoading","drawerDocId","setDrawerDocId","initialData","setInitialData","undefined","setResources","selectedResourceId","setSelectedResourceId","pendingReservations","setPendingReservations","pendingCount","setPendingCount","selectedIds","setSelectedIds","Set","confirmingIds","setConfirmingIds","actionFeedback","setActionFeedback","DocumentDrawer","openDrawer","id","collectionSlug","pendingDrawerOpen","current","fetchResources","params","URLSearchParams","depth","limit","sort","url","response","fetch","result","json","docs","map","d","name","rangeEnd","rangeStart","start","end","setDate","getDate","getDay","setMonth","getMonth","dayOfWeek","setHours","data","availability","fetchReservations","toISOString","fetchPendingCount","totalDocs","fetchPendingReservations","matchesResourceFilter","r","topId","resource","items","some","item","itemId","filteredReservations","filter","filteredPendingReservations","timer","setTimeout","clearTimeout","patchReservation","body","JSON","stringify","headers","method","ok","handleQuickConfirm","prev","add","status","next","delete","type","message","handleQuickCancel","confirmSelected","ids","Array","from","results","Promise","allSettled","succeeded","value","failed","toLowerCase","replace","String","handleEventClick","e","stopPropagation","handleEventKeyDown","preventDefault","handleCreateNew","handleDateClick","date","startTime","handleSlotClick","startIso","handleLaneBook","resourceId","openDocDrawer","navigate","direction","goToToday","getResName","field","getCustomerName","parts","firstName","lastName","Boolean","join","getEventLabel","compact","time","toLocaleTimeString","hour","minute","serviceName","service","trim","customerName","customer","getResourceNames","names","single","getEventTooltip","startStr","endStr","endTime","resourceNames","resourceStr","renderEventItem","cssClass","color","inlineStyle","background","hasItems","isArray","div","className","eventItem","eventItemExpanded","onClick","onKeyDown","role","style","tabIndex","title","itemBadges","it","span","itemBadge","renderStatusLegend","Object","keys","statusLegend","legendItem","legendDot","renderCurrentTimeLine","cellDate","cellHour","now","getFullYear","getHours","topPercent","getMinutes","currentTimeLine","top","renderMonthView","firstDay","startDay","days","push","today","todayStr","monthGrid","dayHeader","day","dayStr","isToday","isOtherMonth","dayReservations","rDate","clickDate","dayCell","dayCellOtherMonth","dayCellToday","dayNumber","renderWeekView","startOfWeek","weekDays","hours","_","gridStartHour","gridEndHour","gridStep","daySlotMaps","Map","isoDay","dayAvail","dayStart","dayEnd","slots","busy","capacityMode","quantity","requiredPools","shiftWindows","step","timeOff","slotByStart","weekView","toLocaleDateString","month","weekday","timeLabel","toString","padStart","di","cellReservations","slotMap","get","slotInfo","slotClass","isNonInteractive","state","slotOffShift","slotTimeOff","slotFull","slotFree","timeOffEntry","to","timeOffLabel","reason","handleClick","handleKeyDown","weekCell","capacityBadge","occupancy","renderDayView","daySlotMap","dayView","hourReservations","dayViewCell","renderPendingView","pendingEmpty","allSelected","every","has","toggleSelectAll","toggleSelect","formatDateTime","iso","toLocaleString","year","pendingView","pendingToolbar","label","selectAllLabel","input","aria-label","checked","onChange","size","button","bulkConfirmButton","disabled","feedbackToast","feedbackSuccess","feedbackError","table","pendingTable","thead","tr","th","pendingTh","tbody","isConfirming","resourceDisplay","pendingRow","td","pendingTd","pendingCustomerLink","confirmButton","cancelButton","dateLabel","endOfWeek","handleDrawerSave","wrapper","header","navButtons","navButton","viewToggle","createButton","viewToggleButton","viewToggleButtonActive","pendingBadge","filterBar","select","resourceFilter","target","option","onBook","onSave"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/components/CalendarView/index.tsx"],"sourcesContent":["'use client'\nimport type { AdminViewServerProps } from 'payload'\n\nimport { useConfig, useDocumentDrawer, useTranslation } from '@payloadcms/ui'\nimport React, { Fragment, useCallback, useEffect, useMemo, useRef, useState } from 'react'\n\nimport type { PluginT } from '../../translations/index.js'\nimport type { SlotInfo } from '../../utilities/computeSlotStates.js'\n\nimport { computeSlotStates } from '../../utilities/computeSlotStates.js'\nimport { statusToI18nKey } from '../../utilities/i18nUtils.js'\nimport { getDayKeyInTimezone, getHourInTimezone } from '../../utilities/timezoneUtils.js'\nimport { useTenantFilter } from '../../utilities/useTenantFilter.js'\nimport styles from './CalendarView.module.css'\nimport { LaneTimelineView } from './LaneTimelineView.js'\nimport { useResourceAvailability } from './useResourceAvailability.js'\n\ntype ViewMode = 'day' | 'lanes' | 'month' | 'pending' | 'week'\n\ntype ReservationItem = {\n endTime?: string\n guestCount?: number\n resource?: { id?: string; name?: string } | string\n service?: { name?: string } | string\n startTime?: string\n}\n\ntype Reservation = {\n customer?: { firstName?: string; lastName?: string; name?: string } | string\n endTime?: string\n id: string\n items?: ReservationItem[]\n resource?: { id?: string; name?: string } | string\n service?: { name?: string } | string\n startTime: string\n status: string\n}\n\ntype ResourceOption = {\n id: string\n name: string\n}\n\n// Built-in status → CSS class map (for known statuses; custom statuses use inline style)\nconst STATUS_CLASS_MAP: Record<string, string> = {\n cancelled: styles.statusCancelled,\n completed: styles.statusCompleted,\n confirmed: styles.statusConfirmed,\n 'no-show': styles.statusNoShow,\n pending: styles.statusPending,\n}\n\n// Built-in default colors for known statuses\nconst BUILTIN_STATUS_COLORS: Record<string, string> = {\n cancelled: '#e5e7eb',\n completed: '#d1fae5',\n confirmed: '#dbeafe',\n 'no-show': '#fee2e2',\n pending: '#fef3c7',\n}\n\n// Palette for auto-assigning colors to custom statuses\nconst CUSTOM_STATUS_PALETTE = ['#fde68a', '#c7d2fe', '#a7f3d0', '#fca5a5', '#fdba74']\n\n// Safe ceiling for list fetches; when totalDocs exceeds this we surface a\n// \"showing N of M\" notice rather than silently truncating (review D9).\nconst MAX_LIST_LIMIT = 2000\n\n// Default visible-hour window for the week/day/lane grids; the actual window\n// expands to include any booking outside it so nothing is hidden (review D8).\nconst DEFAULT_HOUR_START = 7\nconst DEFAULT_HOUR_END = 20\n\n/**\n * Visible-hour window covering `reservations` (in `timeZone`), never narrower\n * than the default business window. Every booking's start hour gets a row, so a\n * booking outside 7–20 is shown rather than silently dropped, and all three\n * time views share one window.\n */\nfunction computeHourWindow(\n reservations: Reservation[],\n timeZone: string,\n): { endHour: number; startHour: number } {\n let startHour = DEFAULT_HOUR_START\n let endHour = DEFAULT_HOUR_END\n for (const r of reservations) {\n if (!r.startTime) {continue}\n const sh = getHourInTimezone(new Date(r.startTime), timeZone)\n startHour = Math.min(startHour, sh)\n endHour = Math.max(endHour, sh + 1)\n if (r.endTime) {\n // round the ending hour up so a slot that ends mid-hour still has a row\n endHour = Math.max(endHour, getHourInTimezone(new Date(r.endTime), timeZone) + 1)\n }\n }\n return { endHour: Math.min(endHour, 24), startHour: Math.max(startHour, 0) }\n}\n\nexport const CalendarView: React.FC<AdminViewServerProps> = () => {\n const { config } = useConfig()\n const { t: _t } = useTranslation()\n const t = _t as PluginT\n\n const slugs = config.admin?.custom?.reservationSlugs\n const reservationSlug = slugs?.reservations ?? 'reservations'\n const apiUrl = `${config.serverURL ?? ''}${config.routes.api}/${reservationSlug}`\n const apiBase = `${config.serverURL ?? ''}${config.routes.api}`\n const resourceSlug = slugs?.resources ?? 'resources'\n const reservationTenantParams = useTenantFilter(reservationSlug)\n const resourceTenantParams = useTenantFilter(resourceSlug)\n\n const reservationTimezone =\n ((config.admin?.custom as Record<string, unknown> | undefined)?.reservationTimezone as\n | string\n | undefined) ?? 'UTC'\n\n const statusMachine = config.admin?.custom?.reservationStatusMachine as\n | {\n blockingStatuses?: string[]\n defaultStatus?: string\n statuses?: string[]\n terminalStatuses?: string[]\n transitions?: Record<string, string[]>\n }\n | undefined\n\n // The initial/pending status (what \"pending\" view shows)\n const defaultStatus = statusMachine?.defaultStatus ?? 'pending'\n\n // Build STATUS_COLORS dynamically: built-ins first, then auto-assign palette for custom statuses\n const STATUS_COLORS = useMemo<Record<string, string>>(() => {\n const colors = { ...BUILTIN_STATUS_COLORS }\n const statuses = statusMachine?.statuses ?? []\n statuses.forEach((s, i) => {\n if (!colors[s]) {\n colors[s] = CUSTOM_STATUS_PALETTE[i % CUSTOM_STATUS_PALETTE.length]\n }\n })\n return colors\n }, [statusMachine])\n\n // Derive confirm/cancel target statuses from config transitions\n // \"confirm\" = first non-terminal transition from defaultStatus\n // \"cancel\" = first terminal transition from defaultStatus (or fallback: 'cancelled')\n const { cancelStatus, confirmStatus } = useMemo(() => {\n const terminalStatuses = statusMachine?.terminalStatuses ?? ['completed', 'cancelled', 'no-show']\n const transitions = statusMachine?.transitions ?? {}\n const defaultTransitions: string[] = transitions[defaultStatus] ?? []\n\n const nonTerminal = defaultTransitions.find((s) => !terminalStatuses.includes(s))\n const terminal = defaultTransitions.find((s) => terminalStatuses.includes(s))\n\n return {\n cancelStatus: terminal ?? 'cancelled',\n confirmStatus: nonTerminal ?? 'confirmed',\n }\n }, [statusMachine, defaultStatus])\n\n const STATUS_LABELS = useMemo<Record<string, string>>(() => {\n const statuses = statusMachine?.statuses ?? [\n 'pending',\n 'confirmed',\n 'completed',\n 'cancelled',\n 'no-show',\n ]\n const labels: Record<string, string> = {}\n for (const s of statuses) {\n // Attempt to look up a translation key, e.g. reservation:statusPending\n const key = statusToI18nKey(s)\n const translated = t(key)\n // If translation returns the key itself, it's missing — fall back to capitalized status name\n labels[s] = translated !== key ? translated : s.charAt(0).toUpperCase() + s.slice(1)\n }\n return labels\n }, [statusMachine, t])\n\n const [currentDate, setCurrentDate] = useState(() => new Date())\n const [viewMode, setViewMode] = useState<ViewMode>('month')\n const [reservations, setReservations] = useState<Reservation[]>([])\n const [loading, setLoading] = useState(true)\n // { shown, total } when a fetch hit its cap, else null — drives a non-silent notice (D9)\n const [truncation, setTruncation] = useState<{ shown: number; total: number } | null>(null)\n // Monotonic request counters so a slow earlier fetch can't overwrite a newer one (D5)\n const reservationsSeq = useRef(0)\n const pendingSeq = useRef(0)\n const [drawerDocId, setDrawerDocId] = useState<null | string>(null)\n const [initialData, setInitialData] = useState<Record<string, unknown> | undefined>(undefined)\n\n // Resource filter state\n const [resources, setResources] = useState<ResourceOption[]>([])\n const [selectedResourceId, setSelectedResourceId] = useState<string>('')\n\n // Pending tab state\n const [pendingReservations, setPendingReservations] = useState<Reservation[]>([])\n const [pendingCount, setPendingCount] = useState(0)\n const [selectedIds, setSelectedIds] = useState<Set<string>>(() => new Set())\n const [confirmingIds, setConfirmingIds] = useState<Set<string>>(() => new Set())\n const [actionFeedback, setActionFeedback] = useState<{\n message: string\n type: 'error' | 'success'\n } | null>(null)\n\n const [DocumentDrawer, , { openDrawer }] = useDocumentDrawer({\n id: drawerDocId ?? undefined,\n collectionSlug: reservationSlug,\n })\n\n const pendingDrawerOpen = useRef(false)\n\n useEffect(() => {\n if (pendingDrawerOpen.current) {\n pendingDrawerOpen.current = false\n openDrawer()\n }\n })\n\n // Fetch active resources for filter dropdown\n useEffect(() => {\n const fetchResources = async () => {\n try {\n const params = new URLSearchParams({\n depth: '0',\n limit: '100',\n sort: 'name',\n 'where[active][equals]': 'true',\n ...resourceTenantParams,\n })\n const url = `${config.serverURL ?? ''}${config.routes.api}/${resourceSlug}?${params}`\n const response = await fetch(url)\n const result = await response.json()\n const docs: Array<{ id: string; name?: string }> = result.docs ?? []\n setResources(docs.map((d) => ({ id: d.id, name: d.name ?? '' })))\n } catch {\n setResources([])\n }\n }\n void fetchResources()\n }, [config.routes.api, config.serverURL, resourceSlug, resourceTenantParams])\n\n const { rangeEnd, rangeStart } = useMemo(() => {\n const start = new Date(currentDate)\n const end = new Date(currentDate)\n\n if (viewMode === 'month') {\n start.setDate(1)\n start.setDate(start.getDate() - start.getDay())\n // The grid always renders 42 cells (6 weeks) from `start`; fetch the same\n // span so trailing weeks aren't silently empty (review D1).\n end.setTime(start.getTime())\n end.setDate(start.getDate() + 41)\n } else if (viewMode === 'week') {\n const dayOfWeek = start.getDay()\n start.setDate(start.getDate() - dayOfWeek)\n end.setDate(start.getDate() + 6)\n }\n start.setHours(0, 0, 0, 0)\n end.setHours(23, 59, 59, 999)\n\n return { rangeEnd: end, rangeStart: start }\n }, [currentDate, viewMode])\n\n // Availability data for the selected resource (null when no resource selected — grid unshaded)\n const { data: availability } = useResourceAvailability(\n apiBase,\n selectedResourceId || undefined,\n rangeStart,\n rangeEnd,\n )\n\n const fetchReservations = useCallback(async () => {\n const seq = ++reservationsSeq.current\n setLoading(true)\n try {\n const params = new URLSearchParams({\n depth: '1',\n limit: String(MAX_LIST_LIMIT),\n sort: 'startTime',\n 'where[startTime][greater_than_equal]': rangeStart.toISOString(),\n 'where[startTime][less_than_equal]': rangeEnd.toISOString(),\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n if (seq !== reservationsSeq.current) {return} // a newer fetch superseded this one\n const docs = result.docs ?? []\n setReservations(docs)\n const total = result.totalDocs ?? docs.length\n setTruncation(total > docs.length ? { shown: docs.length, total } : null)\n } catch {\n if (seq !== reservationsSeq.current) {return}\n setReservations([])\n }\n if (seq === reservationsSeq.current) {setLoading(false)}\n }, [rangeStart, rangeEnd, apiUrl, reservationTenantParams])\n\n useEffect(() => {\n void fetchReservations()\n }, [fetchReservations])\n\n // Fetch pending count (always, for badge) — uses defaultStatus from config\n const fetchPendingCount = useCallback(async () => {\n try {\n // limit:1 + depth:0 returns totalDocs (the full count) without downloading\n // every pending doc — limit:0 in Payload means \"no limit\" (review D9).\n const params = new URLSearchParams({\n depth: '0',\n limit: '1',\n 'where[status][equals]': defaultStatus,\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n setPendingCount(result.totalDocs ?? 0)\n } catch {\n // silently ignore\n }\n }, [apiUrl, defaultStatus, reservationTenantParams])\n\n useEffect(() => {\n void fetchPendingCount()\n }, [fetchPendingCount])\n\n // Fetch pending reservations when tab is active — uses defaultStatus from config\n const fetchPendingReservations = useCallback(async () => {\n const seq = ++pendingSeq.current\n try {\n const params = new URLSearchParams({\n depth: '1',\n limit: String(MAX_LIST_LIMIT),\n sort: 'startTime',\n 'where[status][equals]': defaultStatus,\n ...reservationTenantParams,\n })\n const response = await fetch(`${apiUrl}?${params}`)\n const result = await response.json()\n if (seq !== pendingSeq.current) {return}\n setPendingReservations(result.docs ?? [])\n } catch {\n if (seq !== pendingSeq.current) {return}\n setPendingReservations([])\n }\n }, [apiUrl, defaultStatus, reservationTenantParams])\n\n useEffect(() => {\n if (viewMode === 'pending') {\n void fetchPendingReservations()\n }\n }, [viewMode, fetchPendingReservations])\n\n // Client-side resource filtering\n const matchesResourceFilter = useCallback(\n (r: Reservation): boolean => {\n if (!selectedResourceId) {return true}\n // Check top-level resource\n const topId = typeof r.resource === 'string' ? r.resource : r.resource?.id\n if (topId === selectedResourceId) {return true}\n // Check items array for multi-resource bookings\n if (r.items && r.items.length > 0) {\n return r.items.some((item) => {\n const itemId =\n typeof item.resource === 'string' ? item.resource : item.resource?.id\n return itemId === selectedResourceId\n })\n }\n return false\n },\n [selectedResourceId],\n )\n\n const filteredReservations = useMemo(\n () => reservations.filter(matchesResourceFilter),\n [reservations, matchesResourceFilter],\n )\n\n const filteredPendingReservations = useMemo(\n () => pendingReservations.filter(matchesResourceFilter),\n [pendingReservations, matchesResourceFilter],\n )\n\n // Clear selection when leaving pending view or changing resource filter\n useEffect(() => {\n if (viewMode !== 'pending') {\n setSelectedIds(new Set())\n setActionFeedback(null)\n }\n }, [viewMode])\n\n useEffect(() => {\n setSelectedIds(new Set())\n }, [selectedResourceId])\n\n // Auto-clear feedback toast\n useEffect(() => {\n if (!actionFeedback) {return}\n const timer = setTimeout(() => setActionFeedback(null), 4000)\n return () => clearTimeout(timer)\n }, [actionFeedback])\n\n const patchReservation = useCallback(\n async (id: string, data: Record<string, unknown>): Promise<boolean> => {\n try {\n const response = await fetch(`${apiUrl}/${id}`, {\n body: JSON.stringify(data),\n headers: { 'Content-Type': 'application/json' },\n method: 'PATCH',\n })\n return response.ok\n } catch {\n return false\n }\n },\n [apiUrl],\n )\n\n // Uses confirmStatus derived from config transitions\n const handleQuickConfirm = useCallback(\n async (id: string) => {\n setConfirmingIds((prev) => new Set(prev).add(id))\n const ok = await patchReservation(id, { status: confirmStatus })\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n setActionFeedback({\n type: ok ? 'success' : 'error',\n message: ok\n ? t('reservation:pendingConfirmSuccess')\n : t('reservation:pendingConfirmError'),\n })\n if (ok) {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n void fetchPendingReservations()\n void fetchPendingCount()\n }\n },\n [patchReservation, fetchPendingReservations, fetchPendingCount, t, confirmStatus],\n )\n\n // Uses cancelStatus derived from config transitions\n const handleQuickCancel = useCallback(\n async (id: string) => {\n setConfirmingIds((prev) => new Set(prev).add(id))\n const ok = await patchReservation(id, { status: cancelStatus })\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n setActionFeedback({\n type: ok ? 'success' : 'error',\n message: ok\n ? t('reservation:pendingCancelSuccess')\n : t('reservation:pendingCancelError'),\n })\n if (ok) {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n next.delete(id)\n return next\n })\n void fetchPendingReservations()\n void fetchPendingCount()\n }\n },\n [patchReservation, fetchPendingReservations, fetchPendingCount, t, cancelStatus],\n )\n\n const confirmSelected = useCallback(async () => {\n const ids = Array.from(selectedIds)\n if (ids.length === 0) {return}\n\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n for (const id of ids) {next.add(id)}\n return next\n })\n\n const results = await Promise.allSettled(\n ids.map((id) => patchReservation(id, { status: confirmStatus })),\n )\n\n setConfirmingIds((prev) => {\n const next = new Set(prev)\n for (const id of ids) {next.delete(id)}\n return next\n })\n\n const succeeded = results.filter(\n (r) => r.status === 'fulfilled' && r.value,\n ).length\n const failed = ids.length - succeeded\n\n if (failed === 0) {\n setActionFeedback({\n type: 'success',\n message: `${succeeded} ${t('reservation:pendingConfirmSuccess').toLowerCase()}`,\n })\n } else {\n setActionFeedback({\n type: failed === ids.length ? 'error' : 'success',\n message: t('reservation:pendingBulkConfirmSuccess')\n .replace('{{succeeded}}', String(succeeded))\n .replace('{{failed}}', String(failed)),\n })\n }\n\n setSelectedIds(new Set())\n void fetchPendingReservations()\n void fetchPendingCount()\n }, [selectedIds, patchReservation, fetchPendingReservations, fetchPendingCount, t, confirmStatus])\n\n const handleEventClick = useCallback((e: React.MouseEvent, id: string) => {\n e.stopPropagation()\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const handleEventKeyDown = useCallback((e: React.KeyboardEvent, id: string) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n e.stopPropagation()\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }\n }, [])\n\n const handleCreateNew = useCallback(() => {\n setDrawerDocId(null)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const handleDateClick = useCallback((date: Date) => {\n setDrawerDocId(null)\n setInitialData({ startTime: date.toISOString() })\n pendingDrawerOpen.current = true\n }, [])\n\n // Click-to-book: open new-reservation drawer pre-filled with startTime + optional resource\n const handleSlotClick = useCallback(\n (startIso: string) => {\n setDrawerDocId(null)\n setInitialData({\n ...(selectedResourceId ? { resource: selectedResourceId } : {}),\n startTime: startIso,\n })\n pendingDrawerOpen.current = true\n },\n [selectedResourceId],\n )\n\n // Lane-specific book: pre-fills both the specific resource and startTime\n const handleLaneBook = useCallback((resourceId: string, startIso: string) => {\n setDrawerDocId(null)\n setInitialData({ resource: resourceId, startTime: startIso })\n pendingDrawerOpen.current = true\n }, [])\n\n const openDocDrawer = useCallback((id: string) => {\n setDrawerDocId(id)\n setInitialData(undefined)\n pendingDrawerOpen.current = true\n }, [])\n\n const navigate = useCallback(\n (direction: -1 | 1) => {\n setCurrentDate((prev) => {\n const next = new Date(prev)\n if (viewMode === 'month') {\n next.setMonth(next.getMonth() + direction)\n } else if (viewMode === 'week') {\n next.setDate(next.getDate() + 7 * direction)\n } else {\n // day, lanes: step one day at a time\n next.setDate(next.getDate() + direction)\n }\n return next\n })\n },\n [viewMode],\n )\n\n const goToToday = useCallback(() => setCurrentDate(new Date()), [])\n\n const getResName = (field: { name?: string } | string | undefined): string => {\n if (!field) {return ''}\n if (typeof field === 'string') {return ''}\n return field.name ?? ''\n }\n\n const getCustomerName = (field: Reservation['customer']): string => {\n if (!field) {return ''}\n if (typeof field === 'string') {return ''}\n const parts = [field.firstName, field.lastName].filter(Boolean)\n return parts.length > 0 ? parts.join(' ') : (field.name ?? '')\n }\n\n const getEventLabel = (r: Reservation, compact: boolean) => {\n const time = new Date(r.startTime).toLocaleTimeString([], {\n hour: '2-digit',\n minute: '2-digit',\n timeZone: reservationTimezone,\n })\n const serviceName = getResName(r.service)\n if (compact) {\n return `${time} ${serviceName}`.trim()\n }\n const customerName = getCustomerName(r.customer)\n const parts = [time, serviceName, customerName].filter(Boolean)\n return parts.join(' - ')\n }\n\n // Returns all resource names for a reservation — from items array if present, otherwise top-level resource\n const getResourceNames = (r: Reservation): string[] => {\n if (r.items && r.items.length > 0) {\n const names = r.items\n .map((item) => getResName(item.resource))\n .filter((name) => name.length > 0)\n if (names.length > 0) {return names}\n }\n const single = getResName(r.resource)\n return single ? [single] : []\n }\n\n const getEventTooltip = (r: Reservation): string => {\n const serviceName = getResName(r.service) || t('reservation:calendarUnknownService')\n const startStr = new Date(r.startTime).toLocaleTimeString([], {\n hour: '2-digit',\n minute: '2-digit',\n timeZone: reservationTimezone,\n })\n const endStr = r.endTime\n ? new Date(r.endTime).toLocaleTimeString([], {\n hour: '2-digit',\n minute: '2-digit',\n timeZone: reservationTimezone,\n })\n : '?'\n const customerName = getCustomerName(r.customer) || t('reservation:calendarUnknownCustomer')\n const resourceNames = getResourceNames(r)\n const resourceStr =\n resourceNames.length > 0\n ? resourceNames.join(', ')\n : t('reservation:calendarUnknownResource')\n const status = STATUS_LABELS[r.status] ?? r.status\n return [\n serviceName,\n `${startStr} - ${endStr}`,\n `${t('reservation:tooltipCustomer')} ${customerName}`,\n `${t('reservation:tooltipResource')} ${resourceStr}`,\n `${t('reservation:tooltipStatus')} ${status}`,\n ].join('\\n')\n }\n\n const renderEventItem = (r: Reservation, compact: boolean) => {\n // Use CSS class for built-in statuses; also apply inline background for custom statuses\n const cssClass = STATUS_CLASS_MAP[r.status] ?? ''\n const color = STATUS_COLORS[r.status]\n // Only apply inline style when there's no CSS class (custom statuses) or as a supplement\n const inlineStyle = cssClass ? undefined : { background: color }\n const hasItems = Array.isArray(r.items) && r.items.length > 0\n return (\n <div\n className={`${styles.eventItem} ${cssClass} ${hasItems && !compact ? styles.eventItemExpanded : ''}`}\n key={r.id}\n onClick={(e) => handleEventClick(e, r.id)}\n onKeyDown={(e) => handleEventKeyDown(e, r.id)}\n role=\"button\"\n style={inlineStyle}\n tabIndex={0}\n title={getEventTooltip(r)}\n >\n {getEventLabel(r, compact)}\n {hasItems && (\n <div className={styles.itemBadges}>\n {r.items!.map((it, i) => {\n const name = typeof it.resource === 'object' ? it.resource?.name : it.resource\n return (\n <span className={styles.itemBadge} key={i}>\n {String(name ?? '')}\n </span>\n )\n })}\n </div>\n )}\n </div>\n )\n }\n\n // Dynamic legend: iterates all statuses from the status machine config\n const renderStatusLegend = () => {\n const statuses = statusMachine?.statuses ?? Object.keys(BUILTIN_STATUS_COLORS)\n return (\n <div className={styles.statusLegend}>\n {statuses.map((key) => (\n <div className={styles.legendItem} key={key}>\n <span className={styles.legendDot} style={{ background: STATUS_COLORS[key] }} />\n {STATUS_LABELS[key] ?? key}\n </div>\n ))}\n </div>\n )\n }\n\n const renderCurrentTimeLine = (cellDate: Date, cellHour: number) => {\n const now = new Date()\n if (\n now.getFullYear() !== cellDate.getFullYear() ||\n now.getMonth() !== cellDate.getMonth() ||\n now.getDate() !== cellDate.getDate() ||\n now.getHours() !== cellHour\n ) {\n return null\n }\n const topPercent = (now.getMinutes() / 60) * 100\n return <div className={styles.currentTimeLine} style={{ top: `${topPercent}%` }} />\n }\n\n const renderMonthView = () => {\n const firstDay = new Date(currentDate.getFullYear(), currentDate.getMonth(), 1)\n const startDay = new Date(firstDay)\n startDay.setDate(startDay.getDate() - startDay.getDay())\n\n const days: Date[] = []\n const d = new Date(startDay)\n for (let i = 0; i < 42; i++) {\n days.push(new Date(d))\n d.setDate(d.getDate() + 1)\n }\n\n const today = new Date()\n const todayStr = getDayKeyInTimezone(today, reservationTimezone)\n\n return (\n <div className={styles.monthGrid}>\n {[\n t('reservation:dayShortSun'),\n t('reservation:dayShortMon'),\n t('reservation:dayShortTue'),\n t('reservation:dayShortWed'),\n t('reservation:dayShortThu'),\n t('reservation:dayShortFri'),\n t('reservation:dayShortSat'),\n ].map((d) => (\n <div className={styles.dayHeader} key={d}>\n {d}\n </div>\n ))}\n {days.map((day, i) => {\n const dayKey = getDayKeyInTimezone(day, reservationTimezone)\n const isToday = dayKey === todayStr\n const isOtherMonth =\n dayKey.slice(0, 7) !== getDayKeyInTimezone(currentDate, reservationTimezone).slice(0, 7)\n const dayReservations = filteredReservations.filter((r) => {\n const rKey = getDayKeyInTimezone(new Date(r.startTime), reservationTimezone)\n return rKey === dayKey\n })\n\n const clickDate = new Date(day)\n clickDate.setHours(9, 0, 0, 0)\n\n return (\n <div\n className={`${styles.dayCell} ${isOtherMonth ? styles.dayCellOtherMonth : ''} ${isToday ? styles.dayCellToday : ''}`}\n key={i}\n onClick={() => handleDateClick(clickDate)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n handleDateClick(clickDate)\n }\n }}\n role=\"button\"\n tabIndex={0}\n >\n <div className={styles.dayNumber}>{day.getDate()}</div>\n {dayReservations.map((r) => renderEventItem(r, true))}\n </div>\n )\n })}\n </div>\n )\n }\n\n const renderWeekView = () => {\n const startOfWeek = new Date(currentDate)\n startOfWeek.setDate(startOfWeek.getDate() - startOfWeek.getDay())\n startOfWeek.setHours(0, 0, 0, 0)\n\n const weekDays: Date[] = []\n for (let i = 0; i < 7; i++) {\n const d = new Date(startOfWeek)\n d.setDate(d.getDate() + i)\n weekDays.push(d)\n }\n\n // Visible-hour window derived from the week's bookings (review D8)\n const weekReservations = filteredReservations.filter((r) => {\n const k = getDayKeyInTimezone(new Date(r.startTime), reservationTimezone)\n return weekDays.some((d) => getDayKeyInTimezone(d, reservationTimezone) === k)\n })\n const { endHour: gridEndHour, startHour: gridStartHour } = computeHourWindow(\n weekReservations,\n reservationTimezone,\n )\n const hours = Array.from({ length: gridEndHour - gridStartHour }, (_, i) => i + gridStartHour)\n const gridStep = 60\n\n // Build per-day slot-state maps when a resource is selected\n const daySlotMaps = availability\n ? new Map(\n weekDays.map((day) => {\n const isoDay = getDayKeyInTimezone(day, reservationTimezone)\n const dayAvail = availability.days.find((d) => d.date === isoDay)\n const dayStart = new Date(day)\n dayStart.setHours(gridStartHour, 0, 0, 0)\n const dayEnd = new Date(day)\n dayEnd.setHours(gridEndHour, 0, 0, 0)\n const slots = dayAvail\n ? computeSlotStates({\n busy: availability.busy,\n capacityMode: availability.capacityMode,\n dayEnd,\n dayStart,\n quantity: availability.quantity,\n requiredPools: availability.requiredPools,\n shiftWindows: dayAvail.shiftWindows,\n step: gridStep,\n timeOff: dayAvail.timeOff,\n })\n : []\n // Index by slot start ISO for fast lookup\n const slotByStart = new Map(slots.map((s) => [s.start.toISOString(), s]))\n return [isoDay, slotByStart] as const\n }),\n )\n : null\n\n return (\n <div className={styles.weekView}>\n <div className={styles.dayHeader} />\n {weekDays.map((d, i) => (\n <div className={styles.dayHeader} key={i}>\n {d.toLocaleDateString([], {\n day: 'numeric',\n month: 'numeric',\n timeZone: reservationTimezone,\n weekday: 'short',\n })}\n </div>\n ))}\n {hours.map((hour) => (\n <Fragment key={`row-${hour}`}>\n <div className={styles.timeLabel}>\n {hour.toString().padStart(2, '0')}:00\n </div>\n {weekDays.map((day, di) => {\n const isoDay = getDayKeyInTimezone(day, reservationTimezone)\n const cellReservations = filteredReservations.filter((r) => {\n const rDate = new Date(r.startTime)\n return (\n getDayKeyInTimezone(rDate, reservationTimezone) === isoDay &&\n getHourInTimezone(rDate, reservationTimezone) === hour\n )\n })\n const clickDate = new Date(day)\n clickDate.setHours(hour, 0, 0, 0)\n\n // Slot state (only when a resource is selected)\n const slotMap = daySlotMaps?.get(isoDay)\n const slotInfo = slotMap?.get(clickDate.toISOString()) ?? null\n\n // Derive cell CSS class and interactivity based on slot state\n let slotClass = ''\n let isNonInteractive = false\n if (slotInfo) {\n if (slotInfo.state === 'off-shift') {\n slotClass = styles.slotOffShift\n isNonInteractive = true\n } else if (slotInfo.state === 'time-off') {\n slotClass = styles.slotTimeOff\n isNonInteractive = true\n } else if (slotInfo.state === 'full') {\n slotClass = styles.slotFull\n isNonInteractive = true\n } else {\n slotClass = styles.slotFree\n }\n }\n\n // Time-off label: show type/reason from dayAvail when in time-off state\n const dayAvail = availability?.days.find((d) => d.date === isoDay)\n const timeOffEntry =\n slotInfo?.state === 'time-off'\n ? dayAvail?.timeOff.find(\n (to) =>\n new Date(to.start) <= clickDate && clickDate < new Date(to.end),\n )\n : undefined\n const timeOffLabel = timeOffEntry?.type ?? timeOffEntry?.reason ?? null\n\n const handleClick = isNonInteractive\n ? undefined\n : availability\n ? () => handleSlotClick(clickDate.toISOString())\n : () => handleDateClick(clickDate)\n const handleKeyDown = isNonInteractive\n ? undefined\n : (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n if (availability) {\n handleSlotClick(clickDate.toISOString())\n } else {\n handleDateClick(clickDate)\n }\n }\n }\n\n return (\n <div\n className={`${styles.weekCell} ${slotClass}`}\n key={`cell-${hour}-${di}`}\n onClick={handleClick}\n onKeyDown={handleKeyDown}\n role=\"button\"\n tabIndex={isNonInteractive ? -1 : 0}\n >\n {renderCurrentTimeLine(day, hour)}\n {timeOffLabel && (\n <span className={styles.timeOffLabel}>{timeOffLabel}</span>\n )}\n {slotInfo && availability && availability.quantity > 1 && (\n <span className={styles.capacityBadge}>\n {slotInfo.occupancy}/{availability.quantity}\n </span>\n )}\n {cellReservations.map((r) => renderEventItem(r, false))}\n </div>\n )\n })}\n </Fragment>\n ))}\n </div>\n )\n }\n\n const renderDayView = () => {\n // Build slot-state map for the current day when a resource is selected\n const currentDayKey = getDayKeyInTimezone(currentDate, reservationTimezone)\n\n // Visible-hour window derived from this day's bookings (review D8)\n const dayReservations = filteredReservations.filter(\n (r) => getDayKeyInTimezone(new Date(r.startTime), reservationTimezone) === currentDayKey,\n )\n const { endHour: gridEndHour, startHour: gridStartHour } = computeHourWindow(\n dayReservations,\n reservationTimezone,\n )\n const hours = Array.from({ length: gridEndHour - gridStartHour }, (_, i) => i + gridStartHour)\n const gridStep = 60\n let daySlotMap: Map<string, SlotInfo> | null = null\n if (availability) {\n const isoDay = currentDayKey\n const dayAvail = availability.days.find((d) => d.date === isoDay)\n const dayStart = new Date(currentDate)\n dayStart.setHours(gridStartHour, 0, 0, 0)\n const dayEnd = new Date(currentDate)\n dayEnd.setHours(gridEndHour, 0, 0, 0)\n const slots = dayAvail\n ? computeSlotStates({\n busy: availability.busy,\n capacityMode: availability.capacityMode,\n dayEnd,\n dayStart,\n quantity: availability.quantity,\n requiredPools: availability.requiredPools,\n shiftWindows: dayAvail.shiftWindows,\n step: gridStep,\n timeOff: dayAvail.timeOff,\n })\n : []\n daySlotMap = new Map(slots.map((s) => [s.start.toISOString(), s]))\n }\n\n return (\n <div className={styles.dayView}>\n {hours.map((hour) => {\n const hourReservations = filteredReservations.filter((r) => {\n const rDate = new Date(r.startTime)\n return (\n getDayKeyInTimezone(rDate, reservationTimezone) === currentDayKey &&\n getHourInTimezone(rDate, reservationTimezone) === hour\n )\n })\n const clickDate = new Date(currentDate)\n clickDate.setHours(hour, 0, 0, 0)\n\n // Slot state (only when a resource is selected)\n const slotInfo = daySlotMap?.get(clickDate.toISOString()) ?? null\n\n // Derive cell CSS class and interactivity based on slot state\n let slotClass = ''\n let isNonInteractive = false\n if (slotInfo) {\n if (slotInfo.state === 'off-shift') {\n slotClass = styles.slotOffShift\n isNonInteractive = true\n } else if (slotInfo.state === 'time-off') {\n slotClass = styles.slotTimeOff\n isNonInteractive = true\n } else if (slotInfo.state === 'full') {\n slotClass = styles.slotFull\n isNonInteractive = true\n } else {\n slotClass = styles.slotFree\n }\n }\n\n // Time-off label\n const dayAvail = availability?.days.find((d) => d.date === currentDayKey)\n const timeOffEntry =\n slotInfo?.state === 'time-off'\n ? dayAvail?.timeOff.find(\n (to) => new Date(to.start) <= clickDate && clickDate < new Date(to.end),\n )\n : undefined\n const timeOffLabel = timeOffEntry?.type ?? timeOffEntry?.reason ?? null\n\n const handleClick = isNonInteractive\n ? undefined\n : availability\n ? () => handleSlotClick(clickDate.toISOString())\n : () => handleDateClick(clickDate)\n const handleKeyDown = isNonInteractive\n ? undefined\n : (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n if (availability) {\n handleSlotClick(clickDate.toISOString())\n } else {\n handleDateClick(clickDate)\n }\n }\n }\n\n return (\n <Fragment key={`row-${hour}`}>\n <div className={styles.timeLabel}>\n {hour.toString().padStart(2, '0')}:00\n </div>\n <div\n className={`${styles.dayViewCell} ${slotClass}`}\n onClick={handleClick}\n onKeyDown={handleKeyDown}\n role=\"button\"\n tabIndex={isNonInteractive ? -1 : 0}\n >\n {renderCurrentTimeLine(currentDate, hour)}\n {timeOffLabel && (\n <span className={styles.timeOffLabel}>{timeOffLabel}</span>\n )}\n {slotInfo && availability && availability.quantity > 1 && (\n <span className={styles.capacityBadge}>\n {slotInfo.occupancy}/{availability.quantity}\n </span>\n )}\n {hourReservations.map((r) => renderEventItem(r, false))}\n </div>\n </Fragment>\n )\n })}\n </div>\n )\n }\n\n const renderPendingView = () => {\n if (filteredPendingReservations.length === 0) {\n return <div className={styles.pendingEmpty}>{t('reservation:pendingEmpty')}</div>\n }\n\n const allSelected =\n filteredPendingReservations.length > 0 &&\n filteredPendingReservations.every((r) => selectedIds.has(r.id))\n\n const toggleSelectAll = () => {\n if (allSelected) {\n setSelectedIds(new Set())\n } else {\n setSelectedIds(new Set(filteredPendingReservations.map((r) => r.id)))\n }\n }\n\n const toggleSelect = (id: string) => {\n setSelectedIds((prev) => {\n const next = new Set(prev)\n if (next.has(id)) {\n next.delete(id)\n } else {\n next.add(id)\n }\n return next\n })\n }\n\n const formatDateTime = (iso: string) => {\n const d = new Date(iso)\n return d.toLocaleString([], {\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n month: 'short',\n timeZone: reservationTimezone,\n year: 'numeric',\n })\n }\n\n return (\n <div className={styles.pendingView}>\n <div className={styles.pendingToolbar}>\n <label className={styles.selectAllLabel}>\n <input\n aria-label={t('reservation:pendingSelectAll')}\n checked={allSelected}\n onChange={toggleSelectAll}\n type=\"checkbox\"\n />\n {t('reservation:pendingSelectAll')}\n </label>\n {selectedIds.size > 0 && (\n <button\n className={styles.bulkConfirmButton}\n disabled={confirmingIds.size > 0}\n onClick={() => void confirmSelected()}\n type=\"button\"\n >\n {confirmingIds.size > 0\n ? t('reservation:pendingConfirming')\n : t('reservation:pendingConfirmSelected').replace(\n '{{count}}',\n String(selectedIds.size),\n )}\n </button>\n )}\n </div>\n {actionFeedback && (\n <div\n className={`${styles.feedbackToast} ${actionFeedback.type === 'success' ? styles.feedbackSuccess : styles.feedbackError}`}\n >\n {actionFeedback.message}\n </div>\n )}\n <table className={styles.pendingTable}>\n <thead>\n <tr>\n <th aria-label={t('reservation:pendingSelectAll')} className={styles.pendingTh} />\n <th className={styles.pendingTh}>{t('reservation:fieldCustomer')}</th>\n <th className={styles.pendingTh}>{t('reservation:fieldService')}</th>\n <th className={styles.pendingTh}>{t('reservation:fieldResource')}</th>\n <th className={styles.pendingTh}>{t('reservation:pendingDateTime')}</th>\n <th className={styles.pendingTh}>{t('reservation:pendingActions')}</th>\n </tr>\n </thead>\n <tbody>\n {filteredPendingReservations.map((r) => {\n const isConfirming = confirmingIds.has(r.id)\n // Show all resources from items array if present, else top-level resource\n const resourceDisplay =\n getResourceNames(r).join(', ') || t('reservation:calendarUnknownResource')\n return (\n <tr className={styles.pendingRow} key={r.id}>\n <td className={styles.pendingTd}>\n <input\n aria-label={getCustomerName(r.customer) || r.id}\n checked={selectedIds.has(r.id)}\n onChange={() => toggleSelect(r.id)}\n type=\"checkbox\"\n />\n </td>\n <td className={styles.pendingTd}>\n <span\n className={styles.pendingCustomerLink}\n onClick={() => openDocDrawer(r.id)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n openDocDrawer(r.id)\n }\n }}\n role=\"button\"\n tabIndex={0}\n >\n {getCustomerName(r.customer) || t('reservation:calendarUnknownCustomer')}\n </span>\n </td>\n <td className={styles.pendingTd}>\n {getResName(r.service) || t('reservation:calendarUnknownService')}\n </td>\n <td className={styles.pendingTd}>{resourceDisplay}</td>\n <td className={styles.pendingTd}>{formatDateTime(r.startTime)}</td>\n <td className={styles.pendingTd}>\n <button\n className={styles.confirmButton}\n disabled={isConfirming}\n onClick={() => void handleQuickConfirm(r.id)}\n title={t('reservation:pendingConfirm')}\n type=\"button\"\n >\n &#x2713;\n </button>\n <button\n className={styles.cancelButton}\n disabled={isConfirming}\n onClick={() => void handleQuickCancel(r.id)}\n title={t('reservation:pendingCancel')}\n type=\"button\"\n >\n &#x2717;\n </button>\n </td>\n </tr>\n )\n })}\n </tbody>\n </table>\n </div>\n )\n }\n\n const dateLabel = useMemo(() => {\n if (viewMode === 'month') {\n return currentDate.toLocaleDateString([], { month: 'long', year: 'numeric' })\n }\n if (viewMode === 'week') {\n const startOfWeek = new Date(currentDate)\n startOfWeek.setDate(startOfWeek.getDate() - startOfWeek.getDay())\n const endOfWeek = new Date(startOfWeek)\n endOfWeek.setDate(endOfWeek.getDate() + 6)\n return `${startOfWeek.toLocaleDateString([], { day: 'numeric', month: 'short', timeZone: reservationTimezone })} - ${endOfWeek.toLocaleDateString([], { day: 'numeric', month: 'short', timeZone: reservationTimezone, year: 'numeric' })}`\n }\n return currentDate.toLocaleDateString([], {\n day: 'numeric',\n month: 'long',\n timeZone: reservationTimezone,\n weekday: 'long',\n year: 'numeric',\n })\n }, [currentDate, reservationTimezone, viewMode])\n\n const handleDrawerSave = useCallback(() => {\n void fetchReservations()\n void fetchPendingCount()\n if (viewMode === 'pending') {\n void fetchPendingReservations()\n }\n }, [fetchReservations, fetchPendingCount, fetchPendingReservations, viewMode])\n\n return (\n <div className={styles.wrapper}>\n <div className={styles.header}>\n {viewMode !== 'pending' && (\n <div className={styles.navButtons}>\n <button className={styles.navButton} onClick={() => navigate(-1)} type=\"button\">\n &larr;\n </button>\n <button className={styles.navButton} onClick={goToToday} type=\"button\">\n {t('reservation:calendarToday')}\n </button>\n <button className={styles.navButton} onClick={() => navigate(1)} type=\"button\">\n &rarr;\n </button>\n <span className={styles.currentDate}>{dateLabel}</span>\n </div>\n )}\n {viewMode === 'pending' && <div />}\n <div className={styles.viewToggle}>\n <button className={styles.createButton} onClick={handleCreateNew} type=\"button\">\n {t('reservation:calendarCreateNew')}\n </button>\n {([\n { key: 'month' as ViewMode, label: t('reservation:calendarMonth') },\n { key: 'week' as ViewMode, label: t('reservation:calendarWeek') },\n { key: 'day' as ViewMode, label: t('reservation:calendarDay') },\n { key: 'lanes' as ViewMode, label: t('reservation:calendarLanes') },\n { key: 'pending' as ViewMode, label: t('reservation:calendarPending') },\n ]).map(({ key, label }) => (\n <button\n className={`${styles.viewToggleButton} ${viewMode === key ? styles.viewToggleButtonActive : ''}`}\n key={key}\n onClick={() => setViewMode(key)}\n type=\"button\"\n >\n {label}\n {key === 'pending' && pendingCount > 0 && (\n <span className={styles.pendingBadge}>\n {selectedResourceId ? filteredPendingReservations.length : pendingCount}\n </span>\n )}\n </button>\n ))}\n </div>\n </div>\n {viewMode !== 'pending' && renderStatusLegend()}\n {viewMode !== 'pending' && truncation && (\n <div className={styles.truncationNotice} role=\"status\">\n {t('reservation:calendarShowingNofM', {\n shown: String(truncation.shown),\n total: String(truncation.total),\n })}\n </div>\n )}\n {resources.length > 1 && (\n <div className={styles.filterBar}>\n <select\n aria-label={t('reservation:filterByResource')}\n className={styles.resourceFilter}\n onChange={(e) => setSelectedResourceId(e.target.value)}\n value={selectedResourceId}\n >\n <option value=\"\">{t('reservation:filterAllResources')}</option>\n {resources.map((r) => (\n <option key={r.id} value={r.id}>\n {r.name}\n </option>\n ))}\n </select>\n </div>\n )}\n {loading && viewMode !== 'pending' && viewMode !== 'lanes' ? (\n <div className={styles.loading}>{t('reservation:calendarLoading')}</div>\n ) : (\n <>\n {viewMode === 'month' && renderMonthView()}\n {viewMode === 'week' && renderWeekView()}\n {viewMode === 'day' && renderDayView()}\n {viewMode === 'lanes' &&\n (() => {\n const laneDayKey = getDayKeyInTimezone(currentDate, reservationTimezone)\n const { endHour, startHour } = computeHourWindow(\n filteredReservations.filter(\n (r) =>\n getDayKeyInTimezone(new Date(r.startTime), reservationTimezone) === laneDayKey,\n ),\n reservationTimezone,\n )\n return (\n <LaneTimelineView\n apiBase={apiBase}\n day={currentDate}\n endHour={endHour}\n onBook={handleLaneBook}\n resources={\n selectedResourceId\n ? resources.filter((r) => r.id === selectedResourceId)\n : resources\n }\n startHour={startHour}\n timeZone={reservationTimezone}\n />\n )\n })()}\n </>\n )}\n {viewMode === 'pending' && renderPendingView()}\n <DocumentDrawer initialData={initialData} onSave={handleDrawerSave} />\n </div>\n )\n}\n"],"names":["useConfig","useDocumentDrawer","useTranslation","React","Fragment","useCallback","useEffect","useMemo","useRef","useState","computeSlotStates","statusToI18nKey","getDayKeyInTimezone","getHourInTimezone","useTenantFilter","styles","LaneTimelineView","useResourceAvailability","STATUS_CLASS_MAP","cancelled","statusCancelled","completed","statusCompleted","confirmed","statusConfirmed","statusNoShow","pending","statusPending","BUILTIN_STATUS_COLORS","CUSTOM_STATUS_PALETTE","MAX_LIST_LIMIT","DEFAULT_HOUR_START","DEFAULT_HOUR_END","computeHourWindow","reservations","timeZone","startHour","endHour","r","startTime","sh","Date","Math","min","max","endTime","CalendarView","config","t","_t","slugs","admin","custom","reservationSlugs","reservationSlug","apiUrl","serverURL","routes","api","apiBase","resourceSlug","resources","reservationTenantParams","resourceTenantParams","reservationTimezone","statusMachine","reservationStatusMachine","defaultStatus","STATUS_COLORS","colors","statuses","forEach","s","i","length","cancelStatus","confirmStatus","terminalStatuses","transitions","defaultTransitions","nonTerminal","find","includes","terminal","STATUS_LABELS","labels","key","translated","charAt","toUpperCase","slice","currentDate","setCurrentDate","viewMode","setViewMode","setReservations","loading","setLoading","truncation","setTruncation","reservationsSeq","pendingSeq","drawerDocId","setDrawerDocId","initialData","setInitialData","undefined","setResources","selectedResourceId","setSelectedResourceId","pendingReservations","setPendingReservations","pendingCount","setPendingCount","selectedIds","setSelectedIds","Set","confirmingIds","setConfirmingIds","actionFeedback","setActionFeedback","DocumentDrawer","openDrawer","id","collectionSlug","pendingDrawerOpen","current","fetchResources","params","URLSearchParams","depth","limit","sort","url","response","fetch","result","json","docs","map","d","name","rangeEnd","rangeStart","start","end","setDate","getDate","getDay","setTime","getTime","dayOfWeek","setHours","data","availability","fetchReservations","seq","String","toISOString","total","totalDocs","shown","fetchPendingCount","fetchPendingReservations","matchesResourceFilter","topId","resource","items","some","item","itemId","filteredReservations","filter","filteredPendingReservations","timer","setTimeout","clearTimeout","patchReservation","body","JSON","stringify","headers","method","ok","handleQuickConfirm","prev","add","status","next","delete","type","message","handleQuickCancel","confirmSelected","ids","Array","from","results","Promise","allSettled","succeeded","value","failed","toLowerCase","replace","handleEventClick","e","stopPropagation","handleEventKeyDown","preventDefault","handleCreateNew","handleDateClick","date","handleSlotClick","startIso","handleLaneBook","resourceId","openDocDrawer","navigate","direction","setMonth","getMonth","goToToday","getResName","field","getCustomerName","parts","firstName","lastName","Boolean","join","getEventLabel","compact","time","toLocaleTimeString","hour","minute","serviceName","service","trim","customerName","customer","getResourceNames","names","single","getEventTooltip","startStr","endStr","resourceNames","resourceStr","renderEventItem","cssClass","color","inlineStyle","background","hasItems","isArray","div","className","eventItem","eventItemExpanded","onClick","onKeyDown","role","style","tabIndex","title","itemBadges","it","span","itemBadge","renderStatusLegend","Object","keys","statusLegend","legendItem","legendDot","renderCurrentTimeLine","cellDate","cellHour","now","getFullYear","getHours","topPercent","getMinutes","currentTimeLine","top","renderMonthView","firstDay","startDay","days","push","today","todayStr","monthGrid","dayHeader","day","dayKey","isToday","isOtherMonth","dayReservations","rKey","clickDate","dayCell","dayCellOtherMonth","dayCellToday","dayNumber","renderWeekView","startOfWeek","weekDays","weekReservations","k","gridEndHour","gridStartHour","hours","_","gridStep","daySlotMaps","Map","isoDay","dayAvail","dayStart","dayEnd","slots","busy","capacityMode","quantity","requiredPools","shiftWindows","step","timeOff","slotByStart","weekView","toLocaleDateString","month","weekday","timeLabel","toString","padStart","di","cellReservations","rDate","slotMap","get","slotInfo","slotClass","isNonInteractive","state","slotOffShift","slotTimeOff","slotFull","slotFree","timeOffEntry","to","timeOffLabel","reason","handleClick","handleKeyDown","weekCell","capacityBadge","occupancy","renderDayView","currentDayKey","daySlotMap","dayView","hourReservations","dayViewCell","renderPendingView","pendingEmpty","allSelected","every","has","toggleSelectAll","toggleSelect","formatDateTime","iso","toLocaleString","year","pendingView","pendingToolbar","label","selectAllLabel","input","aria-label","checked","onChange","size","button","bulkConfirmButton","disabled","feedbackToast","feedbackSuccess","feedbackError","table","pendingTable","thead","tr","th","pendingTh","tbody","isConfirming","resourceDisplay","pendingRow","td","pendingTd","pendingCustomerLink","confirmButton","cancelButton","dateLabel","endOfWeek","handleDrawerSave","wrapper","header","navButtons","navButton","viewToggle","createButton","viewToggleButton","viewToggleButtonActive","pendingBadge","truncationNotice","filterBar","select","resourceFilter","target","option","laneDayKey","onBook","onSave"],"mappings":"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"}
@@ -23,6 +23,8 @@ export const CustomerField = ({ field, path: pathProp })=>{
23
23
  });
24
24
  const slugs = config.admin?.custom?.reservationSlugs;
25
25
  const customersCollection = slugs?.customers ?? 'customers';
26
+ // Respect a custom API route / base path instead of hardcoding '/api' (review D6)
27
+ const apiBase = `${config.serverURL ?? ''}${config.routes?.api ?? '/api'}`;
26
28
  const [search, setSearch] = useState('');
27
29
  const [results, setResults] = useState([]);
28
30
  const [selectedCustomer, setSelectedCustomer] = useState(null);
@@ -45,7 +47,7 @@ export const CustomerField = ({ field, path: pathProp })=>{
45
47
  }
46
48
  const fetchCustomer = async ()=>{
47
49
  try {
48
- const res = await fetch(`/api/${customersCollection}/${value}`);
50
+ const res = await fetch(`${apiBase}/${customersCollection}/${value}`);
49
51
  if (res.ok) {
50
52
  const doc = await res.json();
51
53
  setSelectedCustomer({
@@ -64,6 +66,7 @@ export const CustomerField = ({ field, path: pathProp })=>{
64
66
  void fetchCustomer();
65
67
  }, [
66
68
  value,
69
+ apiBase,
67
70
  customersCollection,
68
71
  selectedCustomer?.id
69
72
  ]);
@@ -75,7 +78,7 @@ export const CustomerField = ({ field, path: pathProp })=>{
75
78
  limit: '10',
76
79
  search: query
77
80
  });
78
- const res = await fetch(`/api/reservation-customer-search?${params.toString()}`);
81
+ const res = await fetch(`${apiBase}/reservation-customer-search?${params.toString()}`);
79
82
  if (res.ok) {
80
83
  const data = await res.json();
81
84
  setResults(data.docs);
@@ -85,7 +88,9 @@ export const CustomerField = ({ field, path: pathProp })=>{
85
88
  } finally{
86
89
  setLoading(false);
87
90
  }
88
- }, []);
91
+ }, [
92
+ apiBase
93
+ ]);
89
94
  const handleSearchChange = useCallback((e)=>{
90
95
  const val = e.target.value;
91
96
  setSearch(val);
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/components/CustomerField/index.tsx"],"sourcesContent":["'use client'\nimport type { RelationshipFieldClientComponent } from 'payload'\n\nimport { FieldLabel, useConfig, useDocumentDrawer, useField, useTranslation } from '@payloadcms/ui'\nimport React, { useCallback, useEffect, useRef, useState } from 'react'\n\nimport type { PluginT } from '../../translations/index.js'\n\nimport styles from './CustomerField.module.css'\n\ntype CustomerDoc = {\n email: string\n firstName?: string\n id: string\n lastName?: string\n name?: string\n phone?: string\n}\n\nconst getDisplayName = (customer: CustomerDoc): string => {\n if (customer.name) {return customer.name}\n const fullName = [customer.firstName, customer.lastName].filter(Boolean).join(' ')\n return fullName || customer.email\n}\n\nexport const CustomerField: RelationshipFieldClientComponent = ({ field, path: pathProp }) => {\n const fieldPath = pathProp ?? field?.name ?? 'customer'\n const { config } = useConfig()\n const { t: _t } = useTranslation()\n const t = _t as PluginT\n\n const { setValue, value } = useField<string>({ path: fieldPath })\n\n const slugs = config.admin?.custom?.reservationSlugs\n const customersCollection: string = slugs?.customers ?? 'customers'\n\n const [search, setSearch] = useState('')\n const [results, setResults] = useState<CustomerDoc[]>([])\n const [selectedCustomer, setSelectedCustomer] = useState<CustomerDoc | null>(null)\n const [isOpen, setIsOpen] = useState(false)\n const [loading, setLoading] = useState(false)\n\n const wrapperRef = useRef<HTMLDivElement>(null)\n const debounceRef = useRef<null | ReturnType<typeof setTimeout>>(null)\n\n const [DocumentDrawer, , { openDrawer }] = useDocumentDrawer({\n collectionSlug: customersCollection,\n })\n\n // Fetch selected customer details when value changes\n useEffect(() => {\n if (!value) {\n setSelectedCustomer(null)\n return\n }\n\n // If we already have the selected customer data, skip fetch\n if (selectedCustomer?.id === value) {return}\n\n const fetchCustomer = async () => {\n try {\n const res = await fetch(`/api/${customersCollection}/${value}`)\n if (res.ok) {\n const doc = await res.json()\n setSelectedCustomer({\n id: doc.id,\n name: doc.name ?? undefined,\n email: doc.email ?? '',\n firstName: doc.firstName ?? undefined,\n lastName: doc.lastName ?? undefined,\n phone: doc.phone ?? undefined,\n })\n }\n } catch {\n // Silently fail — the field will still show the ID\n }\n }\n void fetchCustomer()\n }, [value, customersCollection, selectedCustomer?.id])\n\n // Debounced search\n const doSearch = useCallback(\n async (query: string) => {\n setLoading(true)\n try {\n const params = new URLSearchParams({ limit: '10', search: query })\n const res = await fetch(`/api/reservation-customer-search?${params.toString()}`)\n if (res.ok) {\n const data = await res.json()\n setResults(data.docs)\n }\n } catch {\n setResults([])\n } finally {\n setLoading(false)\n }\n },\n [],\n )\n\n const handleSearchChange = useCallback(\n (e: React.ChangeEvent<HTMLInputElement>) => {\n const val = e.target.value\n setSearch(val)\n\n if (debounceRef.current) {\n clearTimeout(debounceRef.current)\n }\n\n debounceRef.current = setTimeout(() => {\n void doSearch(val)\n }, 300)\n },\n [doSearch],\n )\n\n // Click outside to close\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (wrapperRef.current && !wrapperRef.current.contains(e.target as Node)) {\n setIsOpen(false)\n }\n }\n document.addEventListener('mousedown', handleClickOutside)\n return () => document.removeEventListener('mousedown', handleClickOutside)\n }, [])\n\n const handleSelect = useCallback(\n (customer: CustomerDoc) => {\n setValue(customer.id)\n setSelectedCustomer(customer)\n setIsOpen(false)\n setSearch('')\n },\n [setValue],\n )\n\n const handleClear = useCallback(() => {\n setValue(null as unknown as string)\n setSelectedCustomer(null)\n setSearch('')\n }, [setValue])\n\n const handleFocus = useCallback(() => {\n setIsOpen(true)\n if (results.length === 0) {\n void doSearch('')\n }\n }, [doSearch, results.length])\n\n const handleCreate = useCallback(() => {\n setIsOpen(false)\n openDrawer()\n }, [openDrawer])\n\n const handleDrawerSave = useCallback(\n ({ doc }: { doc: Record<string, unknown> }) => {\n const customer: CustomerDoc = {\n id: doc.id as string,\n name: (doc.name as string) ?? undefined,\n email: (doc.email as string) ?? '',\n firstName: (doc.firstName as string) ?? undefined,\n lastName: (doc.lastName as string) ?? undefined,\n phone: (doc.phone as string) ?? undefined,\n }\n setValue(customer.id)\n setSelectedCustomer(customer)\n },\n [setValue],\n )\n\n return (\n <div className={styles.wrapper} ref={wrapperRef}>\n <FieldLabel label={field?.label ?? t('reservation:fieldCustomer')} path={fieldPath} required={field?.required} />\n\n {selectedCustomer ? (\n <div className={styles.selected}>\n <div className={styles.selectedInfo}>\n <span className={styles.selectedName}>{getDisplayName(selectedCustomer)}</span>\n <span className={styles.selectedMeta}>\n {[selectedCustomer.phone, selectedCustomer.email].filter(Boolean).join(' · ')}\n </span>\n </div>\n <button\n className={styles.clearButton}\n onClick={handleClear}\n type=\"button\"\n >\n {t('reservation:fieldCustomerClear')}\n </button>\n </div>\n ) : (\n <input\n aria-label={t('reservation:fieldCustomerSearch')}\n className={styles.searchInput}\n onChange={handleSearchChange}\n onFocus={handleFocus}\n placeholder={t('reservation:fieldCustomerSearch')}\n type=\"text\"\n value={search}\n />\n )}\n\n {isOpen && !selectedCustomer && (\n <div className={styles.dropdown}>\n {loading && results.length === 0 ? (\n <div className={styles.noResults}>...</div>\n ) : results.length === 0 && search ? (\n <div className={styles.noResults}>{t('reservation:fieldCustomerNoResults')}</div>\n ) : (\n results.map((customer) => (\n <div\n aria-selected={false}\n className={styles.option}\n key={customer.id}\n onClick={() => handleSelect(customer)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n handleSelect(customer)\n }\n }}\n role=\"option\"\n tabIndex={0}\n >\n <span className={styles.optionName}>{getDisplayName(customer)}</span>\n <span className={styles.optionMeta}>\n {[customer.phone, customer.email].filter(Boolean).join(' · ')}\n </span>\n </div>\n ))\n )}\n <button\n className={styles.createButton}\n onClick={handleCreate}\n type=\"button\"\n >\n + {t('reservation:fieldCustomerCreateNew')}\n </button>\n </div>\n )}\n\n <DocumentDrawer onSave={handleDrawerSave} />\n </div>\n )\n}\n"],"names":["FieldLabel","useConfig","useDocumentDrawer","useField","useTranslation","React","useCallback","useEffect","useRef","useState","styles","getDisplayName","customer","name","fullName","firstName","lastName","filter","Boolean","join","email","CustomerField","field","path","pathProp","fieldPath","config","t","_t","setValue","value","slugs","admin","custom","reservationSlugs","customersCollection","customers","search","setSearch","results","setResults","selectedCustomer","setSelectedCustomer","isOpen","setIsOpen","loading","setLoading","wrapperRef","debounceRef","DocumentDrawer","openDrawer","collectionSlug","id","fetchCustomer","res","fetch","ok","doc","json","undefined","phone","doSearch","query","params","URLSearchParams","limit","toString","data","docs","handleSearchChange","e","val","target","current","clearTimeout","setTimeout","handleClickOutside","contains","document","addEventListener","removeEventListener","handleSelect","handleClear","handleFocus","length","handleCreate","handleDrawerSave","div","className","wrapper","ref","label","required","selected","selectedInfo","span","selectedName","selectedMeta","button","clearButton","onClick","type","input","aria-label","searchInput","onChange","onFocus","placeholder","dropdown","noResults","map","aria-selected","option","onKeyDown","key","preventDefault","role","tabIndex","optionName","optionMeta","createButton","onSave"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/components/CustomerField/index.tsx"],"sourcesContent":["'use client'\nimport type { RelationshipFieldClientComponent } from 'payload'\n\nimport { FieldLabel, useConfig, useDocumentDrawer, useField, useTranslation } from '@payloadcms/ui'\nimport React, { useCallback, useEffect, useRef, useState } from 'react'\n\nimport type { PluginT } from '../../translations/index.js'\n\nimport styles from './CustomerField.module.css'\n\ntype CustomerDoc = {\n email: string\n firstName?: string\n id: string\n lastName?: string\n name?: string\n phone?: string\n}\n\nconst getDisplayName = (customer: CustomerDoc): string => {\n if (customer.name) {return customer.name}\n const fullName = [customer.firstName, customer.lastName].filter(Boolean).join(' ')\n return fullName || customer.email\n}\n\nexport const CustomerField: RelationshipFieldClientComponent = ({ field, path: pathProp }) => {\n const fieldPath = pathProp ?? field?.name ?? 'customer'\n const { config } = useConfig()\n const { t: _t } = useTranslation()\n const t = _t as PluginT\n\n const { setValue, value } = useField<string>({ path: fieldPath })\n\n const slugs = config.admin?.custom?.reservationSlugs\n const customersCollection: string = slugs?.customers ?? 'customers'\n // Respect a custom API route / base path instead of hardcoding '/api' (review D6)\n const apiBase = `${config.serverURL ?? ''}${config.routes?.api ?? '/api'}`\n\n const [search, setSearch] = useState('')\n const [results, setResults] = useState<CustomerDoc[]>([])\n const [selectedCustomer, setSelectedCustomer] = useState<CustomerDoc | null>(null)\n const [isOpen, setIsOpen] = useState(false)\n const [loading, setLoading] = useState(false)\n\n const wrapperRef = useRef<HTMLDivElement>(null)\n const debounceRef = useRef<null | ReturnType<typeof setTimeout>>(null)\n\n const [DocumentDrawer, , { openDrawer }] = useDocumentDrawer({\n collectionSlug: customersCollection,\n })\n\n // Fetch selected customer details when value changes\n useEffect(() => {\n if (!value) {\n setSelectedCustomer(null)\n return\n }\n\n // If we already have the selected customer data, skip fetch\n if (selectedCustomer?.id === value) {return}\n\n const fetchCustomer = async () => {\n try {\n const res = await fetch(`${apiBase}/${customersCollection}/${value}`)\n if (res.ok) {\n const doc = await res.json()\n setSelectedCustomer({\n id: doc.id,\n name: doc.name ?? undefined,\n email: doc.email ?? '',\n firstName: doc.firstName ?? undefined,\n lastName: doc.lastName ?? undefined,\n phone: doc.phone ?? undefined,\n })\n }\n } catch {\n // Silently fail — the field will still show the ID\n }\n }\n void fetchCustomer()\n }, [value, apiBase, customersCollection, selectedCustomer?.id])\n\n // Debounced search\n const doSearch = useCallback(\n async (query: string) => {\n setLoading(true)\n try {\n const params = new URLSearchParams({ limit: '10', search: query })\n const res = await fetch(`${apiBase}/reservation-customer-search?${params.toString()}`)\n if (res.ok) {\n const data = await res.json()\n setResults(data.docs)\n }\n } catch {\n setResults([])\n } finally {\n setLoading(false)\n }\n },\n [apiBase],\n )\n\n const handleSearchChange = useCallback(\n (e: React.ChangeEvent<HTMLInputElement>) => {\n const val = e.target.value\n setSearch(val)\n\n if (debounceRef.current) {\n clearTimeout(debounceRef.current)\n }\n\n debounceRef.current = setTimeout(() => {\n void doSearch(val)\n }, 300)\n },\n [doSearch],\n )\n\n // Click outside to close\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (wrapperRef.current && !wrapperRef.current.contains(e.target as Node)) {\n setIsOpen(false)\n }\n }\n document.addEventListener('mousedown', handleClickOutside)\n return () => document.removeEventListener('mousedown', handleClickOutside)\n }, [])\n\n const handleSelect = useCallback(\n (customer: CustomerDoc) => {\n setValue(customer.id)\n setSelectedCustomer(customer)\n setIsOpen(false)\n setSearch('')\n },\n [setValue],\n )\n\n const handleClear = useCallback(() => {\n setValue(null as unknown as string)\n setSelectedCustomer(null)\n setSearch('')\n }, [setValue])\n\n const handleFocus = useCallback(() => {\n setIsOpen(true)\n if (results.length === 0) {\n void doSearch('')\n }\n }, [doSearch, results.length])\n\n const handleCreate = useCallback(() => {\n setIsOpen(false)\n openDrawer()\n }, [openDrawer])\n\n const handleDrawerSave = useCallback(\n ({ doc }: { doc: Record<string, unknown> }) => {\n const customer: CustomerDoc = {\n id: doc.id as string,\n name: (doc.name as string) ?? undefined,\n email: (doc.email as string) ?? '',\n firstName: (doc.firstName as string) ?? undefined,\n lastName: (doc.lastName as string) ?? undefined,\n phone: (doc.phone as string) ?? undefined,\n }\n setValue(customer.id)\n setSelectedCustomer(customer)\n },\n [setValue],\n )\n\n return (\n <div className={styles.wrapper} ref={wrapperRef}>\n <FieldLabel label={field?.label ?? t('reservation:fieldCustomer')} path={fieldPath} required={field?.required} />\n\n {selectedCustomer ? (\n <div className={styles.selected}>\n <div className={styles.selectedInfo}>\n <span className={styles.selectedName}>{getDisplayName(selectedCustomer)}</span>\n <span className={styles.selectedMeta}>\n {[selectedCustomer.phone, selectedCustomer.email].filter(Boolean).join(' · ')}\n </span>\n </div>\n <button\n className={styles.clearButton}\n onClick={handleClear}\n type=\"button\"\n >\n {t('reservation:fieldCustomerClear')}\n </button>\n </div>\n ) : (\n <input\n aria-label={t('reservation:fieldCustomerSearch')}\n className={styles.searchInput}\n onChange={handleSearchChange}\n onFocus={handleFocus}\n placeholder={t('reservation:fieldCustomerSearch')}\n type=\"text\"\n value={search}\n />\n )}\n\n {isOpen && !selectedCustomer && (\n <div className={styles.dropdown}>\n {loading && results.length === 0 ? (\n <div className={styles.noResults}>...</div>\n ) : results.length === 0 && search ? (\n <div className={styles.noResults}>{t('reservation:fieldCustomerNoResults')}</div>\n ) : (\n results.map((customer) => (\n <div\n aria-selected={false}\n className={styles.option}\n key={customer.id}\n onClick={() => handleSelect(customer)}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n handleSelect(customer)\n }\n }}\n role=\"option\"\n tabIndex={0}\n >\n <span className={styles.optionName}>{getDisplayName(customer)}</span>\n <span className={styles.optionMeta}>\n {[customer.phone, customer.email].filter(Boolean).join(' · ')}\n </span>\n </div>\n ))\n )}\n <button\n className={styles.createButton}\n onClick={handleCreate}\n type=\"button\"\n >\n + {t('reservation:fieldCustomerCreateNew')}\n </button>\n </div>\n )}\n\n <DocumentDrawer onSave={handleDrawerSave} />\n </div>\n )\n}\n"],"names":["FieldLabel","useConfig","useDocumentDrawer","useField","useTranslation","React","useCallback","useEffect","useRef","useState","styles","getDisplayName","customer","name","fullName","firstName","lastName","filter","Boolean","join","email","CustomerField","field","path","pathProp","fieldPath","config","t","_t","setValue","value","slugs","admin","custom","reservationSlugs","customersCollection","customers","apiBase","serverURL","routes","api","search","setSearch","results","setResults","selectedCustomer","setSelectedCustomer","isOpen","setIsOpen","loading","setLoading","wrapperRef","debounceRef","DocumentDrawer","openDrawer","collectionSlug","id","fetchCustomer","res","fetch","ok","doc","json","undefined","phone","doSearch","query","params","URLSearchParams","limit","toString","data","docs","handleSearchChange","e","val","target","current","clearTimeout","setTimeout","handleClickOutside","contains","document","addEventListener","removeEventListener","handleSelect","handleClear","handleFocus","length","handleCreate","handleDrawerSave","div","className","wrapper","ref","label","required","selected","selectedInfo","span","selectedName","selectedMeta","button","clearButton","onClick","type","input","aria-label","searchInput","onChange","onFocus","placeholder","dropdown","noResults","map","aria-selected","option","onKeyDown","key","preventDefault","role","tabIndex","optionName","optionMeta","createButton","onSave"],"mappings":"AAAA;;AAGA,SAASA,UAAU,EAAEC,SAAS,EAAEC,iBAAiB,EAAEC,QAAQ,EAAEC,cAAc,QAAQ,iBAAgB;AACnG,OAAOC,SAASC,WAAW,EAAEC,SAAS,EAAEC,MAAM,EAAEC,QAAQ,QAAQ,QAAO;AAIvE,OAAOC,YAAY,6BAA4B;AAW/C,MAAMC,iBAAiB,CAACC;IACtB,IAAIA,SAASC,IAAI,EAAE;QAAC,OAAOD,SAASC,IAAI;IAAA;IACxC,MAAMC,WAAW;QAACF,SAASG,SAAS;QAAEH,SAASI,QAAQ;KAAC,CAACC,MAAM,CAACC,SAASC,IAAI,CAAC;IAC9E,OAAOL,YAAYF,SAASQ,KAAK;AACnC;AAEA,OAAO,MAAMC,gBAAkD,CAAC,EAAEC,KAAK,EAAEC,MAAMC,QAAQ,EAAE;IACvF,MAAMC,YAAYD,YAAYF,OAAOT,QAAQ;IAC7C,MAAM,EAAEa,MAAM,EAAE,GAAGzB;IACnB,MAAM,EAAE0B,GAAGC,EAAE,EAAE,GAAGxB;IAClB,MAAMuB,IAAIC;IAEV,MAAM,EAAEC,QAAQ,EAAEC,KAAK,EAAE,GAAG3B,SAAiB;QAAEoB,MAAME;IAAU;IAE/D,MAAMM,QAAQL,OAAOM,KAAK,EAAEC,QAAQC;IACpC,MAAMC,sBAA8BJ,OAAOK,aAAa;IACxD,kFAAkF;IAClF,MAAMC,UAAU,GAAGX,OAAOY,SAAS,IAAI,KAAKZ,OAAOa,MAAM,EAAEC,OAAO,QAAQ;IAE1E,MAAM,CAACC,QAAQC,UAAU,GAAGjC,SAAS;IACrC,MAAM,CAACkC,SAASC,WAAW,GAAGnC,SAAwB,EAAE;IACxD,MAAM,CAACoC,kBAAkBC,oBAAoB,GAAGrC,SAA6B;IAC7E,MAAM,CAACsC,QAAQC,UAAU,GAAGvC,SAAS;IACrC,MAAM,CAACwC,SAASC,WAAW,GAAGzC,SAAS;IAEvC,MAAM0C,aAAa3C,OAAuB;IAC1C,MAAM4C,cAAc5C,OAA6C;IAEjE,MAAM,CAAC6C,kBAAkB,EAAEC,UAAU,EAAE,CAAC,GAAGpD,kBAAkB;QAC3DqD,gBAAgBpB;IAClB;IAEA,qDAAqD;IACrD5B,UAAU;QACR,IAAI,CAACuB,OAAO;YACVgB,oBAAoB;YACpB;QACF;QAEA,4DAA4D;QAC5D,IAAID,kBAAkBW,OAAO1B,OAAO;YAAC;QAAM;QAE3C,MAAM2B,gBAAgB;YACpB,IAAI;gBACF,MAAMC,MAAM,MAAMC,MAAM,GAAGtB,QAAQ,CAAC,EAAEF,oBAAoB,CAAC,EAAEL,OAAO;gBACpE,IAAI4B,IAAIE,EAAE,EAAE;oBACV,MAAMC,MAAM,MAAMH,IAAII,IAAI;oBAC1BhB,oBAAoB;wBAClBU,IAAIK,IAAIL,EAAE;wBACV3C,MAAMgD,IAAIhD,IAAI,IAAIkD;wBAClB3C,OAAOyC,IAAIzC,KAAK,IAAI;wBACpBL,WAAW8C,IAAI9C,SAAS,IAAIgD;wBAC5B/C,UAAU6C,IAAI7C,QAAQ,IAAI+C;wBAC1BC,OAAOH,IAAIG,KAAK,IAAID;oBACtB;gBACF;YACF,EAAE,OAAM;YACN,mDAAmD;YACrD;QACF;QACA,KAAKN;IACP,GAAG;QAAC3B;QAAOO;QAASF;QAAqBU,kBAAkBW;KAAG;IAE9D,mBAAmB;IACnB,MAAMS,WAAW3D,YACf,OAAO4D;QACLhB,WAAW;QACX,IAAI;YACF,MAAMiB,SAAS,IAAIC,gBAAgB;gBAAEC,OAAO;gBAAM5B,QAAQyB;YAAM;YAChE,MAAMR,MAAM,MAAMC,MAAM,GAAGtB,QAAQ,6BAA6B,EAAE8B,OAAOG,QAAQ,IAAI;YACrF,IAAIZ,IAAIE,EAAE,EAAE;gBACV,MAAMW,OAAO,MAAMb,IAAII,IAAI;gBAC3BlB,WAAW2B,KAAKC,IAAI;YACtB;QACF,EAAE,OAAM;YACN5B,WAAW,EAAE;QACf,SAAU;YACRM,WAAW;QACb;IACF,GACA;QAACb;KAAQ;IAGX,MAAMoC,qBAAqBnE,YACzB,CAACoE;QACC,MAAMC,MAAMD,EAAEE,MAAM,CAAC9C,KAAK;QAC1BY,UAAUiC;QAEV,IAAIvB,YAAYyB,OAAO,EAAE;YACvBC,aAAa1B,YAAYyB,OAAO;QAClC;QAEAzB,YAAYyB,OAAO,GAAGE,WAAW;YAC/B,KAAKd,SAASU;QAChB,GAAG;IACL,GACA;QAACV;KAAS;IAGZ,yBAAyB;IACzB1D,UAAU;QACR,MAAMyE,qBAAqB,CAACN;YAC1B,IAAIvB,WAAW0B,OAAO,IAAI,CAAC1B,WAAW0B,OAAO,CAACI,QAAQ,CAACP,EAAEE,MAAM,GAAW;gBACxE5B,UAAU;YACZ;QACF;QACAkC,SAASC,gBAAgB,CAAC,aAAaH;QACvC,OAAO,IAAME,SAASE,mBAAmB,CAAC,aAAaJ;IACzD,GAAG,EAAE;IAEL,MAAMK,eAAe/E,YACnB,CAACM;QACCiB,SAASjB,SAAS4C,EAAE;QACpBV,oBAAoBlC;QACpBoC,UAAU;QACVN,UAAU;IACZ,GACA;QAACb;KAAS;IAGZ,MAAMyD,cAAchF,YAAY;QAC9BuB,SAAS;QACTiB,oBAAoB;QACpBJ,UAAU;IACZ,GAAG;QAACb;KAAS;IAEb,MAAM0D,cAAcjF,YAAY;QAC9B0C,UAAU;QACV,IAAIL,QAAQ6C,MAAM,KAAK,GAAG;YACxB,KAAKvB,SAAS;QAChB;IACF,GAAG;QAACA;QAAUtB,QAAQ6C,MAAM;KAAC;IAE7B,MAAMC,eAAenF,YAAY;QAC/B0C,UAAU;QACVM;IACF,GAAG;QAACA;KAAW;IAEf,MAAMoC,mBAAmBpF,YACvB,CAAC,EAAEuD,GAAG,EAAoC;QACxC,MAAMjD,WAAwB;YAC5B4C,IAAIK,IAAIL,EAAE;YACV3C,MAAM,AAACgD,IAAIhD,IAAI,IAAekD;YAC9B3C,OAAO,AAACyC,IAAIzC,KAAK,IAAe;YAChCL,WAAW,AAAC8C,IAAI9C,SAAS,IAAegD;YACxC/C,UAAU,AAAC6C,IAAI7C,QAAQ,IAAe+C;YACtCC,OAAO,AAACH,IAAIG,KAAK,IAAeD;QAClC;QACAlC,SAASjB,SAAS4C,EAAE;QACpBV,oBAAoBlC;IACtB,GACA;QAACiB;KAAS;IAGZ,qBACE,MAAC8D;QAAIC,WAAWlF,OAAOmF,OAAO;QAAEC,KAAK3C;;0BACnC,KAACnD;gBAAW+F,OAAOzE,OAAOyE,SAASpE,EAAE;gBAA8BJ,MAAME;gBAAWuE,UAAU1E,OAAO0E;;YAEpGnD,iCACC,MAAC8C;gBAAIC,WAAWlF,OAAOuF,QAAQ;;kCAC7B,MAACN;wBAAIC,WAAWlF,OAAOwF,YAAY;;0CACjC,KAACC;gCAAKP,WAAWlF,OAAO0F,YAAY;0CAAGzF,eAAekC;;0CACtD,KAACsD;gCAAKP,WAAWlF,OAAO2F,YAAY;0CACjC;oCAACxD,iBAAiBmB,KAAK;oCAAEnB,iBAAiBzB,KAAK;iCAAC,CAACH,MAAM,CAACC,SAASC,IAAI,CAAC;;;;kCAG3E,KAACmF;wBACCV,WAAWlF,OAAO6F,WAAW;wBAC7BC,SAASlB;wBACTmB,MAAK;kCAEJ9E,EAAE;;;+BAIP,KAAC+E;gBACCC,cAAYhF,EAAE;gBACdiE,WAAWlF,OAAOkG,WAAW;gBAC7BC,UAAUpC;gBACVqC,SAASvB;gBACTwB,aAAapF,EAAE;gBACf8E,MAAK;gBACL3E,OAAOW;;YAIVM,UAAU,CAACF,kCACV,MAAC8C;gBAAIC,WAAWlF,OAAOsG,QAAQ;;oBAC5B/D,WAAWN,QAAQ6C,MAAM,KAAK,kBAC7B,KAACG;wBAAIC,WAAWlF,OAAOuG,SAAS;kCAAE;yBAChCtE,QAAQ6C,MAAM,KAAK,KAAK/C,uBAC1B,KAACkD;wBAAIC,WAAWlF,OAAOuG,SAAS;kCAAGtF,EAAE;yBAErCgB,QAAQuE,GAAG,CAAC,CAACtG,yBACX,MAAC+E;4BACCwB,iBAAe;4BACfvB,WAAWlF,OAAO0G,MAAM;4BAExBZ,SAAS,IAAMnB,aAAazE;4BAC5ByG,WAAW,CAAC3C;gCACV,IAAIA,EAAE4C,GAAG,KAAK,WAAW5C,EAAE4C,GAAG,KAAK,KAAK;oCACtC5C,EAAE6C,cAAc;oCAChBlC,aAAazE;gCACf;4BACF;4BACA4G,MAAK;4BACLC,UAAU;;8CAEV,KAACtB;oCAAKP,WAAWlF,OAAOgH,UAAU;8CAAG/G,eAAeC;;8CACpD,KAACuF;oCAAKP,WAAWlF,OAAOiH,UAAU;8CAC/B;wCAAC/G,SAASoD,KAAK;wCAAEpD,SAASQ,KAAK;qCAAC,CAACH,MAAM,CAACC,SAASC,IAAI,CAAC;;;2BAbpDP,SAAS4C,EAAE;kCAkBtB,MAAC8C;wBACCV,WAAWlF,OAAOkH,YAAY;wBAC9BpB,SAASf;wBACTgB,MAAK;;4BACN;4BACI9E,EAAE;;;;;0BAKX,KAAC0B;gBAAewE,QAAQnC;;;;AAG9B,EAAC"}