@innosolutions/inno-calendar 1.0.63 → 1.0.65
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.
- package/README.md +5 -5
- package/dist/{agenda-widget-DbqQHhyM.js → agenda-widget-CD_UoYWC.js} +245 -242
- package/dist/agenda-widget-CD_UoYWC.js.map +1 -0
- package/dist/agenda-widget-COxgmpv4.cjs +2 -0
- package/dist/agenda-widget-COxgmpv4.cjs.map +1 -0
- package/dist/components/index.cjs +1 -1
- package/dist/components/index.mjs +2 -2
- package/dist/components/inno-calendar.d.ts +9 -7
- package/dist/components/inno-calendar.d.ts.map +1 -1
- package/dist/core/context/inno-calendar-provider.d.ts.map +1 -1
- package/dist/core/index.cjs +1 -1
- package/dist/core/index.mjs +165 -154
- package/dist/core/utils/index.d.ts +1 -0
- package/dist/core/utils/index.d.ts.map +1 -1
- package/dist/core/utils/view-storage.d.ts +64 -0
- package/dist/core/utils/view-storage.d.ts.map +1 -0
- package/dist/index.cjs +1 -1
- package/dist/index.mjs +210 -199
- package/dist/position-utils-Do2ciBXl.cjs +2 -0
- package/dist/position-utils-Do2ciBXl.cjs.map +1 -0
- package/dist/{position-utils-DMVQFywD.js → position-utils-DpaKqYhO.js} +430 -346
- package/dist/position-utils-DpaKqYhO.js.map +1 -0
- package/dist/presets/index.cjs +1 -1
- package/dist/presets/index.mjs +1 -1
- package/dist/slot-selection-context-BlrL9b5P.cjs +2 -0
- package/dist/slot-selection-context-BlrL9b5P.cjs.map +1 -0
- package/dist/{slot-selection-context-CRstOosL.js → slot-selection-context-D3MW4WJ7.js} +159 -159
- package/dist/slot-selection-context-D3MW4WJ7.js.map +1 -0
- package/dist/{tailwind-calendar-CNhXkxzW.js → tailwind-calendar-CzbuXRmn.js} +65 -65
- package/dist/{tailwind-calendar-CNhXkxzW.js.map → tailwind-calendar-CzbuXRmn.js.map} +1 -1
- package/dist/{tailwind-calendar-VRvPJQwa.cjs → tailwind-calendar-smlkVdSq.cjs} +2 -2
- package/dist/{tailwind-calendar-VRvPJQwa.cjs.map → tailwind-calendar-smlkVdSq.cjs.map} +1 -1
- package/dist/{use-calendar-DkT1_V3j.cjs → use-calendar-Bd-d8oGj.cjs} +2 -2
- package/dist/{use-calendar-DkT1_V3j.cjs.map → use-calendar-Bd-d8oGj.cjs.map} +1 -1
- package/dist/{use-calendar-D8XaVe44.js → use-calendar-mU1joTwp.js} +53 -53
- package/dist/{use-calendar-D8XaVe44.js.map → use-calendar-mU1joTwp.js.map} +1 -1
- package/dist/{use-slot-selection-Dj_tWg1O.cjs → use-slot-selection-CCOSsRSh.cjs} +2 -2
- package/dist/{use-slot-selection-Dj_tWg1O.cjs.map → use-slot-selection-CCOSsRSh.cjs.map} +1 -1
- package/dist/{use-slot-selection-CWRk_17s.js → use-slot-selection-CauKdLYv.js} +3 -3
- package/dist/{use-slot-selection-CWRk_17s.js.map → use-slot-selection-CauKdLYv.js.map} +1 -1
- package/dist/utils.cjs +1 -1
- package/dist/utils.mjs +120 -109
- package/dist/{week-view-DY167Wok.js → week-view-BQXIaSUZ.js} +3 -3
- package/dist/{week-view-DY167Wok.js.map → week-view-BQXIaSUZ.js.map} +1 -1
- package/dist/{week-view-C1Vu2ErD.cjs → week-view-xrKlAJos.cjs} +2 -2
- package/dist/{week-view-C1Vu2ErD.cjs.map → week-view-xrKlAJos.cjs.map} +1 -1
- package/package.json +1 -1
- package/dist/agenda-widget-BiZIpa3f.cjs +0 -2
- package/dist/agenda-widget-BiZIpa3f.cjs.map +0 -1
- package/dist/agenda-widget-DbqQHhyM.js.map +0 -1
- package/dist/position-utils-BQpbtF6N.cjs +0 -2
- package/dist/position-utils-BQpbtF6N.cjs.map +0 -1
- package/dist/position-utils-DMVQFywD.js.map +0 -1
- package/dist/slot-selection-context-CRstOosL.js.map +0 -1
- package/dist/slot-selection-context-DBCZI2Dn.cjs +0 -2
- package/dist/slot-selection-context-DBCZI2Dn.cjs.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"position-utils-DpaKqYhO.js","sources":["../src/core/utils/react-node-utils.ts","../src/core/utils/view-storage.ts","../src/core/constants.ts","../src/core/utils/date-utils.ts","../src/core/utils/event-utils.ts","../src/core/utils/grid-utils.ts","../src/core/utils/position-utils.ts"],"sourcesContent":["/**\n * Utility to extract plain text from a ReactNode.\n *\n * Useful when a prop accepts ReactNode for flexible rendering\n * but the underlying logic needs a string (search, aria-label, join, etc.).\n */\n\nimport { type ReactNode, isValidElement } from 'react';\n\n/**\n * Recursively extracts plain text from a ReactNode tree.\n *\n * @example\n * ```ts\n * reactNodeToText(\"hello\") // \"hello\"\n * reactNodeToText(<span>hello <b>world</b></span>) // \"hello world\"\n * reactNodeToText(null) // \"\"\n * reactNodeToText(42) // \"42\"\n * ```\n */\nexport function reactNodeToText(node: ReactNode): string {\n\tif (node == null || typeof node === 'boolean') return '';\n\tif (typeof node === 'string') return node;\n\tif (typeof node === 'number') return String(node);\n\tif (Array.isArray(node)) return node.map(reactNodeToText).join('');\n\tif (isValidElement(node)) {\n\t\tconst props = node.props as { children?: ReactNode };\n\t\treturn reactNodeToText(props.children);\n\t}\n\treturn '';\n}\n","/**\n * View Storage Utilities\n *\n * Persist the user's last selected calendar view and resource view to\n * localStorage so the choice survives page reloads.\n *\n * The two view families live in separate header dropdowns:\n * - Calendar views: 'day' | 'week' | 'month' | 'year' | 'agenda'\n * - Resource views: 'timeline-day' | 'timeline-3day' | 'timeline-week'\n * | 'resource-day' | 'resource-week'\n *\n * Each family is stored under its own key so switching between dropdowns\n * can restore the last value for each independently.\n */\n\nimport type { TCalendarView } from '../types';\n\n// ============================================================================\n// CONSTANTS\n// ============================================================================\n\n/** localStorage key for the last selected calendar (grid) view */\nexport const CALENDAR_VIEW_STORAGE_KEY = 'inno-calendar-view';\n\n/** localStorage key for the last selected resource/timeline view */\nexport const RESOURCE_VIEW_STORAGE_KEY = 'inno-calendar-resource-view';\n\n/**\n * Calendar view values (non-resource, non-timeline).\n * Matches the \"Calendar View\" dropdown in the built-in header.\n */\nexport type TCalendarGridView = 'day' | 'week' | 'month' | 'year' | 'agenda';\n\n/**\n * Resource view values (timeline + resource).\n * Matches the \"Resource View\" dropdown in the built-in header.\n */\nexport type TResourceView =\n\t| 'timeline-day'\n\t| 'timeline-3day'\n\t| 'timeline-week'\n\t| 'resource-day'\n\t| 'resource-week';\n\nconst CALENDAR_GRID_VIEWS: readonly TCalendarGridView[] = [\n\t'day',\n\t'week',\n\t'month',\n\t'year',\n\t'agenda',\n] as const;\n\nconst RESOURCE_VIEWS: readonly TResourceView[] = [\n\t'timeline-day',\n\t'timeline-3day',\n\t'timeline-week',\n\t'resource-day',\n\t'resource-week',\n] as const;\n\n// ============================================================================\n// TYPE GUARDS\n// ============================================================================\n\n/** Returns true if `view` belongs to the calendar (grid) view family */\nexport function isCalendarGridView(view: unknown): view is TCalendarGridView {\n\treturn typeof view === 'string' && (CALENDAR_GRID_VIEWS as readonly string[]).includes(view);\n}\n\n/** Returns true if `view` belongs to the resource/timeline view family */\nexport function isResourceView(view: unknown): view is TResourceView {\n\treturn typeof view === 'string' && (RESOURCE_VIEWS as readonly string[]).includes(view);\n}\n\n// ============================================================================\n// INTERNAL HELPERS\n// ============================================================================\n\nfunction isBrowser(): boolean {\n\treturn typeof window !== 'undefined' && typeof window.localStorage !== 'undefined';\n}\n\nfunction readKey(key: string): string | null {\n\tif (!isBrowser()) return null;\n\ttry {\n\t\treturn window.localStorage.getItem(key);\n\t} catch {\n\t\tconsole.warn(`[InnoCalendar] Failed to read \"${key}\" from localStorage`);\n\t\treturn null;\n\t}\n}\n\nfunction writeKey(key: string, value: string): void {\n\tif (!isBrowser()) return;\n\ttry {\n\t\twindow.localStorage.setItem(key, value);\n\t} catch {\n\t\tconsole.warn(`[InnoCalendar] Failed to write \"${key}\" to localStorage`);\n\t}\n}\n\nfunction removeKey(key: string): void {\n\tif (!isBrowser()) return;\n\ttry {\n\t\twindow.localStorage.removeItem(key);\n\t} catch {\n\t\tconsole.warn(`[InnoCalendar] Failed to remove \"${key}\" from localStorage`);\n\t}\n}\n\n// ============================================================================\n// CALENDAR VIEW\n// ============================================================================\n\n/**\n * Save the last selected calendar (grid) view.\n *\n * @param view - One of: `'day' | 'week' | 'month' | 'year' | 'agenda'`\n * @param storageKey - Optional override for the localStorage key\n */\nexport function saveCalendarView(\n\tview: TCalendarGridView,\n\tstorageKey: string = CALENDAR_VIEW_STORAGE_KEY\n): void {\n\tif (!isCalendarGridView(view)) return;\n\twriteKey(storageKey, view);\n}\n\n/**\n * Read the last saved calendar (grid) view from localStorage.\n * Returns `null` if no value is stored, the stored value is invalid,\n * or localStorage is not available (SSR).\n *\n * @param storageKey - Optional override for the localStorage key\n */\nexport function readCalendarView(\n\tstorageKey: string = CALENDAR_VIEW_STORAGE_KEY\n): TCalendarGridView | null {\n\tconst value = readKey(storageKey);\n\treturn isCalendarGridView(value) ? value : null;\n}\n\n/**\n * Remove the saved calendar (grid) view from localStorage.\n */\nexport function clearCalendarView(storageKey: string = CALENDAR_VIEW_STORAGE_KEY): void {\n\tremoveKey(storageKey);\n}\n\n// ============================================================================\n// RESOURCE VIEW\n// ============================================================================\n\n/**\n * Save the last selected resource/timeline view.\n *\n * @param view - One of: `'timeline-day' | 'timeline-3day' | 'timeline-week'\n * | 'resource-day' | 'resource-week'`\n * @param storageKey - Optional override for the localStorage key\n */\nexport function saveResourceView(\n\tview: TResourceView,\n\tstorageKey: string = RESOURCE_VIEW_STORAGE_KEY\n): void {\n\tif (!isResourceView(view)) return;\n\twriteKey(storageKey, view);\n}\n\n/**\n * Read the last saved resource/timeline view from localStorage.\n * Returns `null` if no value is stored, the stored value is invalid,\n * or localStorage is not available (SSR).\n *\n * @param storageKey - Optional override for the localStorage key\n */\nexport function readResourceView(\n\tstorageKey: string = RESOURCE_VIEW_STORAGE_KEY\n): TResourceView | null {\n\tconst value = readKey(storageKey);\n\treturn isResourceView(value) ? value : null;\n}\n\n/**\n * Remove the saved resource/timeline view from localStorage.\n */\nexport function clearResourceView(storageKey: string = RESOURCE_VIEW_STORAGE_KEY): void {\n\tremoveKey(storageKey);\n}\n\n// ============================================================================\n// CONVENIENCE\n// ============================================================================\n\n/**\n * Save a view to the matching slot (calendar vs resource) automatically.\n * No-op for views that do not belong to either family.\n */\nexport function saveViewByFamily(view: TCalendarView): void {\n\tif (isCalendarGridView(view)) {\n\t\tsaveCalendarView(view);\n\t\treturn;\n\t}\n\tif (isResourceView(view)) {\n\t\tsaveResourceView(view);\n\t}\n}\n","/**\n * Constants for @inno/calendar\n *\n * Default values, view configurations, and color mappings.\n */\n\nimport type {\n\tICalendarPreferences,\n\tIViewConfig,\n\tIVisibleHours,\n\tTCalendarView,\n\tTEventColor,\n} from './types';\n\n// ============================================================================\n// CALENDAR VIEWS\n// ============================================================================\n\n/**\n * All available calendar views\n */\nexport const CALENDAR_VIEWS: TCalendarView[] = [\n\t'day',\n\t'week',\n\t'month',\n\t'year',\n\t'agenda',\n\t'resource-day',\n\t'resource-week',\n\t'timeline-day',\n\t'timeline-week',\n];\n\n// ============================================================================\n// DEFAULT PREFERENCES\n// ============================================================================\n\n/**\n * Default calendar preferences\n */\nexport const DEFAULT_PREFERENCES: ICalendarPreferences = {\n\tstartHour: 8,\n\tendHour: 18,\n\tfirstDayOfWeek: 1, // Monday\n\tslotDuration: 30,\n\tshowWeekends: true,\n\ttimeFormat: '24h',\n\tbadgeVariant: 'colored',\n\tshowCanceledEvents: false,\n};\n\n/**\n * Default visible hours\n */\nexport const DEFAULT_VISIBLE_HOURS: IVisibleHours = {\n\tstartHour: 0,\n\tendHour: 24,\n};\n\n/**\n * Default business hours range (used as a fallback for visible hours)\n */\nexport const DEFAULT_BUSINESS_HOURS: IVisibleHours = {\n\tstartHour: 8,\n\tendHour: 18,\n};\n\n// ============================================================================\n// VIEW CONFIGURATIONS\n// ============================================================================\n\n/**\n * Default configuration for each view type\n */\nexport const DEFAULT_VIEW_CONFIGS: Record<TCalendarView, IViewConfig> = {\n\tday: {\n\t\ttype: 'day',\n\t\tslotMinTime: '00:00',\n\t\tslotMaxTime: '24:00',\n\t\tslotDuration: '00:30:00',\n\t\tscrollTime: '08:00:00',\n\t\tallDaySlot: true,\n\t},\n\tweek: {\n\t\ttype: 'week',\n\t\tslotMinTime: '00:00',\n\t\tslotMaxTime: '24:00',\n\t\tslotDuration: '00:30:00',\n\t\tscrollTime: '08:00:00',\n\t\tdayMaxEventRows: 3,\n\t\tallDaySlot: true,\n\t},\n\tmonth: {\n\t\ttype: 'month',\n\t\tdayMaxEventRows: 4,\n\t\teventMaxStack: 3,\n\t\tallDaySlot: false,\n\t},\n\tyear: {\n\t\ttype: 'year',\n\t\tdayMaxEventRows: 2,\n\t\tallDaySlot: false,\n\t},\n\tagenda: {\n\t\ttype: 'agenda',\n\t\tallDaySlot: false,\n\t},\n\t'resource-day': {\n\t\ttype: 'resource-day',\n\t\tslotMinTime: '00:00',\n\t\tslotMaxTime: '24:00',\n\t\tslotDuration: '00:30:00',\n\t\tscrollTime: '08:00:00',\n\t\tallDaySlot: true,\n\t},\n\t'resource-week': {\n\t\ttype: 'resource-week',\n\t\tslotMinTime: '00:00',\n\t\tslotMaxTime: '24:00',\n\t\tslotDuration: '00:30:00',\n\t\tscrollTime: '08:00:00',\n\t\tallDaySlot: true,\n\t},\n\t'timeline-day': {\n\t\ttype: 'timeline-day',\n\t\tslotDuration: '01:00:00',\n\t},\n\t'timeline-3day': {\n\t\ttype: 'timeline-3day',\n\t\tslotDuration: '01:00:00',\n\t},\n\t'timeline-week': {\n\t\ttype: 'timeline-week',\n\t\tslotDuration: '01:00:00',\n\t},\n};\n\n// ============================================================================\n// EVENT COLORS\n// ============================================================================\n\n/**\n * Event color palette with hex values\n */\nexport const EVENT_COLORS: Record<TEventColor, string> = {\n\tblue: '#3b82f6',\n\tgreen: '#22c55e',\n\tred: '#ef4444',\n\tyellow: '#eab308',\n\tpurple: '#a855f7',\n\torange: '#f97316',\n\tpink: '#ec4899',\n\tteal: '#14b8a6',\n\tgray: '#6b7280',\n\tindigo: '#6366f1',\n};\n\n/**\n * TailwindCSS class mappings for event colors\n */\nexport const EVENT_COLOR_CLASSES: Record<\n\tTEventColor,\n\t{ bg: string; text: string; border: string }\n> = {\n\tblue: { bg: 'bg-blue-100', text: 'text-blue-800', border: 'border-blue-300' },\n\tgreen: { bg: 'bg-green-100', text: 'text-green-800', border: 'border-green-300' },\n\tred: { bg: 'bg-red-100', text: 'text-red-800', border: 'border-red-300' },\n\tyellow: { bg: 'bg-yellow-100', text: 'text-yellow-800', border: 'border-yellow-300' },\n\tpurple: { bg: 'bg-purple-100', text: 'text-purple-800', border: 'border-purple-300' },\n\torange: { bg: 'bg-orange-100', text: 'text-orange-800', border: 'border-orange-300' },\n\tpink: { bg: 'bg-pink-100', text: 'text-pink-800', border: 'border-pink-300' },\n\tteal: { bg: 'bg-teal-100', text: 'text-teal-800', border: 'border-teal-300' },\n\tgray: { bg: 'bg-gray-100', text: 'text-gray-800', border: 'border-gray-300' },\n\tindigo: { bg: 'bg-indigo-100', text: 'text-indigo-800', border: 'border-indigo-300' },\n};\n\n/**\n * Map hex colors to event color names\n */\nexport const HEX_TO_EVENT_COLOR: Record<string, TEventColor> = {\n\t'#3b82f6': 'blue',\n\t'#22c55e': 'green',\n\t'#ef4444': 'red',\n\t'#eab308': 'yellow',\n\t'#a855f7': 'purple',\n\t'#f97316': 'orange',\n\t'#ec4899': 'pink',\n\t'#14b8a6': 'teal',\n\t'#6b7280': 'gray',\n\t'#6366f1': 'indigo',\n};\n\n// ============================================================================\n// TIME CONSTANTS\n// ============================================================================\n\n/**\n * Hours in a day\n */\nexport const HOURS_IN_DAY = 24;\n\n/**\n * Minutes in an hour\n */\nexport const MINUTES_IN_HOUR = 60;\n\n/**\n * Milliseconds in a minute\n */\nexport const MS_PER_MINUTE = 60 * 1000;\n\n/**\n * Milliseconds in an hour\n */\nexport const MS_PER_HOUR = 60 * MS_PER_MINUTE;\n\n/**\n * Milliseconds in a day\n */\nexport const MS_PER_DAY = 24 * MS_PER_HOUR;\n\n// ============================================================================\n// GRID CONSTANTS\n// ============================================================================\n\n/**\n * Default hour height in pixels for time grid views\n */\nexport const DEFAULT_HOUR_HEIGHT = 64;\n\n/**\n * Default slot height in pixels\n */\nexport const DEFAULT_SLOT_HEIGHT = 32;\n\n/**\n * Default header height in pixels\n */\nexport const DEFAULT_HEADER_HEIGHT = 56;\n\n/**\n * Default resource column width in pixels\n */\nexport const DEFAULT_RESOURCE_WIDTH = 200;\n\n/**\n * Minimum event height in pixels\n */\nexport const MIN_EVENT_HEIGHT = 20;\n","/**\n * Date Utilities for @inno/calendar\n *\n * Pure functions for date manipulation using native Date API.\n * No external dependencies.\n */\n\nimport type { TWorkingHours } from '../types';\n\n// Re-export TWorkingHours for convenience\nexport type { TWorkingHours };\n\n// ============================================================================\n// DAY OPERATIONS\n// ============================================================================\n\n/**\n * Get start of day (00:00:00.000)\n */\nexport function startOfDay(date: Date): Date {\n\tconst result = new Date(date);\n\tresult.setHours(0, 0, 0, 0);\n\treturn result;\n}\n\n/**\n * Get end of day (23:59:59.999)\n */\nexport function endOfDay(date: Date): Date {\n\tconst result = new Date(date);\n\tresult.setHours(23, 59, 59, 999);\n\treturn result;\n}\n\n/**\n * Add days to a date\n */\nexport function addDays(date: Date, days: number): Date {\n\tconst result = new Date(date);\n\tresult.setDate(result.getDate() + days);\n\treturn result;\n}\n\n/**\n * Subtract days from a date\n */\nexport function subDays(date: Date, days: number): Date {\n\treturn addDays(date, -days);\n}\n\n/**\n * Check if two dates are the same day\n */\nexport function isSameDay(date1: Date, date2: Date): boolean {\n\treturn (\n\t\tdate1.getFullYear() === date2.getFullYear() &&\n\t\tdate1.getMonth() === date2.getMonth() &&\n\t\tdate1.getDate() === date2.getDate()\n\t);\n}\n\n/**\n * Check if date is today\n */\nexport function isToday(date: Date): boolean {\n\treturn isSameDay(date, new Date());\n}\n\n/**\n * Get day of week (0-6, where 0 is Sunday)\n */\nexport function getDayOfWeek(date: Date): number {\n\treturn date.getDay();\n}\n\n/**\n * Check if date is a weekend\n */\nexport function isWeekend(date: Date): boolean {\n\tconst day = date.getDay();\n\treturn day === 0 || day === 6;\n}\n\n// ============================================================================\n// WEEK OPERATIONS\n// ============================================================================\n\n/**\n * Get start of week\n */\nexport function startOfWeek(date: Date, weekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1): Date {\n\tconst result = new Date(date);\n\tconst day = result.getDay();\n\tconst diff = (day < weekStartsOn ? 7 : 0) + day - weekStartsOn;\n\tresult.setDate(result.getDate() - diff);\n\tresult.setHours(0, 0, 0, 0);\n\treturn result;\n}\n\n/**\n * Get end of week\n */\nexport function endOfWeek(date: Date, weekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1): Date {\n\tconst result = startOfWeek(date, weekStartsOn);\n\tresult.setDate(result.getDate() + 6);\n\tresult.setHours(23, 59, 59, 999);\n\treturn result;\n}\n\n/**\n * Add weeks to a date\n */\nexport function addWeeks(date: Date, weeks: number): Date {\n\treturn addDays(date, weeks * 7);\n}\n\n/**\n * Subtract weeks from a date\n */\nexport function subWeeks(date: Date, weeks: number): Date {\n\treturn addDays(date, -weeks * 7);\n}\n\n/**\n * Get week number in year (ISO 8601)\n */\nexport function getWeekNumber(date: Date): number {\n\tconst d = new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()));\n\tconst dayNum = d.getUTCDay() || 7;\n\td.setUTCDate(d.getUTCDate() + 4 - dayNum);\n\tconst yearStart = new Date(Date.UTC(d.getUTCFullYear(), 0, 1));\n\treturn Math.ceil(((d.getTime() - yearStart.getTime()) / 86400000 + 1) / 7);\n}\n\n/**\n * Check if two dates are in the same week\n */\nexport function isSameWeek(\n\tdate1: Date,\n\tdate2: Date,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): boolean {\n\tconst start1 = startOfWeek(date1, weekStartsOn);\n\tconst start2 = startOfWeek(date2, weekStartsOn);\n\treturn isSameDay(start1, start2);\n}\n\n// ============================================================================\n// MONTH OPERATIONS\n// ============================================================================\n\n/**\n * Get start of month\n */\nexport function startOfMonth(date: Date): Date {\n\tconst result = new Date(date);\n\tresult.setDate(1);\n\tresult.setHours(0, 0, 0, 0);\n\treturn result;\n}\n\n/**\n * Get end of month\n */\nexport function endOfMonth(date: Date): Date {\n\tconst result = new Date(date.getFullYear(), date.getMonth() + 1, 0);\n\tresult.setHours(23, 59, 59, 999);\n\treturn result;\n}\n\n/**\n * Add months to a date\n */\nexport function addMonths(date: Date, months: number): Date {\n\tconst result = new Date(date);\n\tresult.setMonth(result.getMonth() + months);\n\treturn result;\n}\n\n/**\n * Subtract months from a date\n */\nexport function subMonths(date: Date, months: number): Date {\n\treturn addMonths(date, -months);\n}\n\n/**\n * Get number of days in a month\n */\nexport function getDaysInMonth(date: Date): number {\n\treturn new Date(date.getFullYear(), date.getMonth() + 1, 0).getDate();\n}\n\n/**\n * Check if two dates are in the same month\n */\nexport function isSameMonth(date1: Date, date2: Date): boolean {\n\treturn date1.getFullYear() === date2.getFullYear() && date1.getMonth() === date2.getMonth();\n}\n\n// ============================================================================\n// YEAR OPERATIONS\n// ============================================================================\n\n/**\n * Get start of year\n */\nexport function startOfYear(date: Date): Date {\n\treturn new Date(date.getFullYear(), 0, 1, 0, 0, 0, 0);\n}\n\n/**\n * Get end of year\n */\nexport function endOfYear(date: Date): Date {\n\treturn new Date(date.getFullYear(), 11, 31, 23, 59, 59, 999);\n}\n\n/**\n * Add years to a date\n */\nexport function addYears(date: Date, years: number): Date {\n\tconst result = new Date(date);\n\tresult.setFullYear(result.getFullYear() + years);\n\treturn result;\n}\n\n/**\n * Subtract years from a date\n */\nexport function subYears(date: Date, years: number): Date {\n\treturn addYears(date, -years);\n}\n\n/**\n * Check if two dates are in the same year\n */\nexport function isSameYear(date1: Date, date2: Date): boolean {\n\treturn date1.getFullYear() === date2.getFullYear();\n}\n\n// ============================================================================\n// TIME OPERATIONS\n// ============================================================================\n\n/**\n * Add hours to a date\n */\nexport function addHours(date: Date, hours: number): Date {\n\tconst result = new Date(date);\n\tresult.setHours(result.getHours() + hours);\n\treturn result;\n}\n\n/**\n * Add minutes to a date\n */\nexport function addMinutes(date: Date, minutes: number): Date {\n\tconst result = new Date(date);\n\tresult.setMinutes(result.getMinutes() + minutes);\n\treturn result;\n}\n\n/**\n * Set time on a date\n */\nexport function setTime(date: Date, hours: number, minutes = 0, seconds = 0): Date {\n\tconst result = new Date(date);\n\tresult.setHours(hours, minutes, seconds, 0);\n\treturn result;\n}\n\n/**\n * Get time as decimal hours (e.g., 14:30 = 14.5)\n */\nexport function getDecimalHours(date: Date): number {\n\treturn date.getHours() + date.getMinutes() / 60;\n}\n\n// ============================================================================\n// COMPARISON OPERATIONS\n// ============================================================================\n\n/**\n * Check if date is before another date\n */\nexport function isBefore(date: Date, dateToCompare: Date): boolean {\n\treturn date.getTime() < dateToCompare.getTime();\n}\n\n/**\n * Check if date is after another date\n */\nexport function isAfter(date: Date, dateToCompare: Date): boolean {\n\treturn date.getTime() > dateToCompare.getTime();\n}\n\n/**\n * Check if date is between two dates (inclusive)\n */\nexport function isBetween(date: Date, start: Date, end: Date): boolean {\n\tconst time = date.getTime();\n\treturn time >= start.getTime() && time <= end.getTime();\n}\n\n/**\n * Get minimum date from array\n */\nexport function minDate(dates: Date[]): Date | undefined {\n\tif (dates.length === 0) return undefined;\n\treturn dates.reduce((min, date) => (date < min ? date : min));\n}\n\n/**\n * Get maximum date from array\n */\nexport function maxDate(dates: Date[]): Date | undefined {\n\tif (dates.length === 0) return undefined;\n\treturn dates.reduce((max, date) => (date > max ? date : max));\n}\n\n// ============================================================================\n// DIFFERENCE OPERATIONS\n// ============================================================================\n\n/**\n * Get difference in milliseconds\n */\nexport function differenceInMilliseconds(date1: Date, date2: Date): number {\n\treturn date1.getTime() - date2.getTime();\n}\n\n/**\n * Get difference in minutes\n */\nexport function differenceInMinutes(date1: Date, date2: Date): number {\n\treturn Math.floor(differenceInMilliseconds(date1, date2) / 60000);\n}\n\n/**\n * Get difference in hours\n */\nexport function differenceInHours(date1: Date, date2: Date): number {\n\treturn Math.floor(differenceInMilliseconds(date1, date2) / 3600000);\n}\n\n/**\n * Get difference in days\n */\nexport function differenceInDays(date1: Date, date2: Date): number {\n\tconst start = startOfDay(date1);\n\tconst end = startOfDay(date2);\n\treturn Math.floor(differenceInMilliseconds(start, end) / 86400000);\n}\n\n// ============================================================================\n// FORMATTING\n// ============================================================================\n\n/**\n * Format time as HH:MM\n */\nexport function formatTime(date: Date, format: '12h' | '24h' = '24h'): string {\n\tconst hours = date.getHours();\n\tconst minutes = date.getMinutes();\n\n\tif (format === '12h') {\n\t\tconst period = hours >= 12 ? 'PM' : 'AM';\n\t\tconst hour12 = hours % 12 || 12;\n\t\treturn `${hour12}:${minutes.toString().padStart(2, '0')} ${period}`;\n\t}\n\n\treturn `${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}`;\n}\n\n/**\n * Format date as YYYY-MM-DD\n */\nexport function formatDateISO(date: Date): string {\n\tconst year = date.getFullYear();\n\tconst month = (date.getMonth() + 1).toString().padStart(2, '0');\n\tconst day = date.getDate().toString().padStart(2, '0');\n\treturn `${year}-${month}-${day}`;\n}\n\n/**\n * Parse ISO date string to Date\n */\nexport function parseISO(dateString: string): Date {\n\treturn new Date(dateString);\n}\n\n// ============================================================================\n// ARRAY GENERATION\n// ============================================================================\n\n/**\n * Generate array of dates between start and end (inclusive)\n */\nexport function eachDayOfInterval(start: Date, end: Date): Date[] {\n\tconst dates: Date[] = [];\n\tlet current = startOfDay(start);\n\tconst endDay = startOfDay(end);\n\n\twhile (current <= endDay) {\n\t\tdates.push(new Date(current));\n\t\tcurrent = addDays(current, 1);\n\t}\n\n\treturn dates;\n}\n\n/**\n * Generate array of hours between start and end\n */\nexport function eachHourOfInterval(start: Date, end: Date): Date[] {\n\tconst hours: Date[] = [];\n\tlet current = new Date(start);\n\n\twhile (current <= end) {\n\t\thours.push(new Date(current));\n\t\tcurrent = addHours(current, 1);\n\t}\n\n\treturn hours;\n}\n\n/**\n * Get weekday names\n */\nexport function getWeekdayNames(\n\tlocale = 'en-US',\n\tformat: 'long' | 'short' | 'narrow' = 'short'\n): string[] {\n\tconst formatter = new Intl.DateTimeFormat(locale, { weekday: format });\n\tconst days: string[] = [];\n\n\t// Use a known Sunday to start\n\tconst baseDate = new Date(2024, 0, 7); // Sunday, Jan 7, 2024\n\n\tfor (let i = 0; i < 7; i++) {\n\t\tdays.push(formatter.format(addDays(baseDate, i)));\n\t}\n\n\treturn days;\n}\n\n/**\n * Get month names\n */\nexport function getMonthNames(\n\tlocale = 'en-US',\n\tformat: 'long' | 'short' | 'narrow' = 'long'\n): string[] {\n\tconst formatter = new Intl.DateTimeFormat(locale, { month: format });\n\tconst months: string[] = [];\n\n\tfor (let i = 0; i < 12; i++) {\n\t\tmonths.push(formatter.format(new Date(2024, i, 1)));\n\t}\n\n\treturn months;\n}\n\n// ============================================================================\n// VIEW HELPERS\n// ============================================================================\n\n/**\n * Get array of visible hours based on configuration\n */\nexport function getVisibleHoursArray(visibleHours: {\n\tstartHour: number;\n\tendHour: number;\n}): number[] {\n\tconst hours: number[] = [];\n\tfor (let h = visibleHours.startHour; h <= visibleHours.endHour; h++) {\n\t\thours.push(h);\n\t}\n\treturn hours;\n}\n\n/**\n * Format hour for display (e.g., 8 -> \"8:00\", 14 -> \"14:00\")\n */\nexport function formatHourLabel(hour: number, format: '12h' | '24h' = '24h'): string {\n\tif (format === '12h') {\n\t\tconst period = hour >= 12 ? 'PM' : 'AM';\n\t\tconst hour12 = hour % 12 || 12;\n\t\treturn `${hour12} ${period}`;\n\t}\n\t// 24h format with leading zero (e.g., \"09:00\", \"14:00\")\n\treturn `${String(hour).padStart(2, '0')}:00`;\n}\n\n/**\n * Get an array of 7 days for a week starting from the given date\n */\nexport function getWeekDays(date: Date, weekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1): Date[] {\n\tconst weekStart = startOfWeek(date, weekStartsOn);\n\tconst days: Date[] = [];\n\n\tfor (let i = 0; i < 7; i++) {\n\t\tdays.push(addDays(weekStart, i));\n\t}\n\n\treturn days;\n}\n\n/**\n * Generate month grid cells (typically 6 weeks x 7 days = 42 days)\n * Returns enriched cell objects with date, isCurrentMonth, and isWeekend\n */\nexport function generateMonthGrid(\n\tdate: Date,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): Array<{ date: Date; isCurrentMonth: boolean; isWeekend: boolean }> {\n\tconst monthStart = startOfMonth(date);\n\tconst monthEnd = endOfMonth(date);\n\tconst gridStart = startOfWeek(monthStart, weekStartsOn);\n\tconst gridEnd = endOfWeek(monthEnd, weekStartsOn);\n\tconst currentMonth = date.getMonth();\n\n\tconst dates = eachDayOfInterval(gridStart, gridEnd);\n\n\treturn dates.map((d) => ({\n\t\tdate: d,\n\t\tisCurrentMonth: d.getMonth() === currentMonth,\n\t\tisWeekend: d.getDay() === 0 || d.getDay() === 6,\n\t}));\n}\n\n// ============================================================================\n// ALL-DAY EVENT DETECTION\n// ============================================================================\n\n/**\n * Detect whether an event should be treated as an \"all-day\" event.\n *\n * An event is all-day if:\n * 1. Its `isAllDay` flag is explicitly `true`, OR\n * 2. It starts at 00:00 and ends at 23:59 or 00:00 (next day boundary)\n *\n * All-day events should **not** display time ranges — showing \"00:00 – 23:59\"\n * is confusing for users. Instead, display \"All day\" or a date range.\n *\n * @example\n * ```ts\n * detectAllDayEvent({ startDate: new Date('2026-02-16T00:00'), endDate: new Date('2026-02-16T23:59'), isAllDay: false })\n * // => true (midnight boundaries)\n *\n * detectAllDayEvent({ startDate: new Date('2026-02-16T09:00'), endDate: new Date('2026-02-16T17:00') })\n * // => false (normal timed event)\n * ```\n */\nexport function detectAllDayEvent(event: {\n\tstartDate: Date;\n\tendDate: Date;\n\tisAllDay?: boolean;\n}): boolean {\n\t// Explicit flag takes priority — respect both true AND false\n\tif (event.isAllDay === true) return true;\n\tif (event.isAllDay === false) return false;\n\n\t// Check if times sit on day boundaries (midnight start, midnight or end-of-day end)\n\tconst startHour = event.startDate.getHours();\n\tconst startMinute = event.startDate.getMinutes();\n\tconst endHour = event.endDate.getHours();\n\tconst endMinute = event.endDate.getMinutes();\n\n\tconst startsAtMidnight = startHour === 0 && startMinute === 0;\n\tconst endsAtDayBoundary =\n\t\t(endHour === 0 && endMinute === 0) || // midnight next day\n\t\t(endHour === 23 && endMinute >= 59); // end of day (23:59 or 23:59:59)\n\n\tif (startsAtMidnight && endsAtDayBoundary) return true;\n\n\treturn false;\n}\n\n/**\n * Separate events into single-day timed events and banner events\n * (multi-day + all-day).\n *\n * All-day events (detected via {@link detectAllDayEvent}) are placed in\n * the `multiDay` bucket even when they span a single calendar day, because\n * they belong in the banner strip — not in the hourly time grid.\n */\nexport function separateEventsByDuration<\n\tTEvent extends { id: string; startDate: Date; endDate: Date; isAllDay?: boolean },\n>(\n\tevents: TEvent[]\n): {\n\tsingleDay: TEvent[];\n\tmultiDay: TEvent[];\n} {\n\tconst singleDay: TEvent[] = [];\n\tconst multiDay: TEvent[] = [];\n\n\tfor (const event of events) {\n\t\t// All-day events always go to the banner, even single-day ones\n\t\tif (detectAllDayEvent(event)) {\n\t\t\tmultiDay.push(event);\n\t\t} else if (isSameDay(event.startDate, event.endDate)) {\n\t\t\tsingleDay.push(event);\n\t\t} else {\n\t\t\tmultiDay.push(event);\n\t\t}\n\t}\n\n\treturn { singleDay, multiDay };\n}\n\n/**\n * Get events within a date range\n */\nexport function getEventsInRange<TEvent extends { startDate: Date; endDate: Date }>(\n\tevents: TEvent[],\n\trangeStart: Date,\n\trangeEnd: Date\n): TEvent[] {\n\treturn events.filter((event) => {\n\t\treturn event.startDate <= rangeEnd && event.endDate >= rangeStart;\n\t});\n}\n\n// ============================================================================\n// WORKING HOURS UTILITIES\n// ============================================================================\n\n/**\n * Check if a given hour is within working hours for a specific day\n * Handles both TWorkingHours (with `enabled`) and TWorkingHoursConfig (without `enabled`)\n */\nexport function isWorkingHour(date: Date, hour: number, workingHours?: TWorkingHours): boolean {\n\tif (!workingHours) return true; // If no working hours defined, all hours are valid\n\n\tconst dayOfWeek = date.getDay();\n\tconst dayHours = workingHours[dayOfWeek];\n\n\tif (!dayHours) return false;\n\n\t// Treat as enabled if `enabled` is undefined (backward compat with TWorkingHoursConfig)\n\tconst isEnabled = dayHours.enabled !== false;\n\tif (!isEnabled || dayHours.from === dayHours.to) {\n\t\treturn false;\n\t}\n\n\treturn hour >= dayHours.from && hour < dayHours.to;\n}\n\n/**\n * Get working hours for a specific day\n * Returns null if the day is disabled or has no working hours\n * Handles both TWorkingHours (with `enabled`) and TWorkingHoursConfig (without `enabled`)\n */\nexport function getWorkingHoursForDay(\n\tdate: Date,\n\tworkingHours: TWorkingHours\n): { from: number; to: number } | null {\n\tconst dayOfWeek = date.getDay();\n\tconst dayHours = workingHours[dayOfWeek];\n\n\tif (!dayHours) return null;\n\n\t// Treat as enabled if `enabled` is undefined (backward compat with TWorkingHoursConfig)\n\tconst isEnabled = dayHours.enabled !== false;\n\tif (!isEnabled || dayHours.from === dayHours.to) {\n\t\treturn null;\n\t}\n\n\treturn { from: dayHours.from, to: dayHours.to };\n}\n\n/**\n * Get all months for a year (for year view)\n */\nexport function getYearMonths(year: number): Date[] {\n\tconst months: Date[] = [];\n\tfor (let month = 0; month < 12; month++) {\n\t\tmonths.push(new Date(year, month, 1));\n\t}\n\treturn months;\n}\n","/**\n * Event Utilities for @inno/calendar\n *\n * Functions for filtering, sorting, and grouping events.\n * All functions are generic to work with any event data type.\n */\n\nimport type { CalendarEvent, TEventColor } from '../types';\nimport { detectAllDayEvent, endOfDay, formatTime, isSameDay, startOfDay } from './date-utils';\nimport { reactNodeToText } from './react-node-utils';\n\n// ============================================================================\n// FILTERING\n// ============================================================================\n\n/**\n * Filter events by date range\n */\nexport function filterEventsByDateRange<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tstartDate: Date,\n\tendDate: Date\n): CalendarEvent<TData>[] {\n\tconst rangeStart = startOfDay(startDate);\n\tconst rangeEnd = endOfDay(endDate);\n\n\treturn events.filter((event) => {\n\t\tconst eventStart = event.startDate;\n\t\tconst eventEnd = event.endDate;\n\n\t\t// Event overlaps with range if:\n\t\t// - Event starts before range ends AND\n\t\t// - Event ends after range starts\n\t\treturn eventStart <= rangeEnd && eventEnd >= rangeStart;\n\t});\n}\n\n/**\n * Filter events by schedule type IDs\n */\nexport function filterEventsByScheduleType<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tscheduleTypeIds: number[]\n): CalendarEvent<TData>[] {\n\tif (scheduleTypeIds.length === 0) return events;\n\treturn events.filter(\n\t\t(event) => event.scheduleTypeId !== undefined && scheduleTypeIds.includes(event.scheduleTypeId)\n\t);\n}\n\n/**\n * Filter events by resource IDs\n */\nexport function filterEventsByResource<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tresourceIds: string[]\n): CalendarEvent<TData>[] {\n\tif (resourceIds.length === 0) return events;\n\treturn events.filter(\n\t\t(event) => event.resourceId !== undefined && resourceIds.includes(event.resourceId)\n\t);\n}\n\n/**\n * Filter events by search term\n */\nexport function filterEventsBySearch<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tsearch: string\n): CalendarEvent<TData>[] {\n\tif (!search.trim()) return events;\n\n\tconst searchLower = search.toLowerCase();\n\treturn events.filter((event) => {\n\t\tconst titleMatch = reactNodeToText(event.title).toLowerCase().includes(searchLower);\n\t\tconst descriptionMatch = reactNodeToText(event.description).toLowerCase().includes(searchLower);\n\t\treturn titleMatch || descriptionMatch;\n\t});\n}\n\n/**\n * Filter out canceled events\n */\nexport function filterOutCanceled<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn events.filter((event) => !event.isCanceled);\n}\n\n/**\n * Apply all filters at once\n */\nexport function applyEventFilters<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\toptions: {\n\t\tdateRange?: { startDate: Date; endDate: Date };\n\t\tscheduleTypeIds?: number[];\n\t\tresourceIds?: string[];\n\t\tsearch?: string;\n\t\tshowCanceled?: boolean;\n\t}\n): CalendarEvent<TData>[] {\n\tlet filtered = [...events];\n\n\tif (options.dateRange) {\n\t\tfiltered = filterEventsByDateRange(\n\t\t\tfiltered,\n\t\t\toptions.dateRange.startDate,\n\t\t\toptions.dateRange.endDate\n\t\t);\n\t}\n\n\tif (options.scheduleTypeIds && options.scheduleTypeIds.length > 0) {\n\t\tfiltered = filterEventsByScheduleType(filtered, options.scheduleTypeIds);\n\t}\n\n\tif (options.resourceIds && options.resourceIds.length > 0) {\n\t\tfiltered = filterEventsByResource(filtered, options.resourceIds);\n\t}\n\n\tif (options.search) {\n\t\tfiltered = filterEventsBySearch(filtered, options.search);\n\t}\n\n\tif (!options.showCanceled) {\n\t\tfiltered = filterOutCanceled(filtered);\n\t}\n\n\treturn filtered;\n}\n\n// ============================================================================\n// SORTING\n// ============================================================================\n\n/**\n * Sort events by start date (ascending)\n */\nexport function sortEventsByStart<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn [...events].sort((a, b) => a.startDate.getTime() - b.startDate.getTime());\n}\n\n/**\n * Sort events by end date (ascending)\n */\nexport function sortEventsByEnd<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn [...events].sort((a, b) => a.endDate.getTime() - b.endDate.getTime());\n}\n\n/**\n * Sort events by duration (longest first)\n */\nexport function sortEventsByDuration<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn [...events].sort((a, b) => {\n\t\tconst durationA = a.endDate.getTime() - a.startDate.getTime();\n\t\tconst durationB = b.endDate.getTime() - b.startDate.getTime();\n\t\treturn durationB - durationA;\n\t});\n}\n\n// ============================================================================\n// GROUPING\n// ============================================================================\n\n/**\n * Group events by date\n */\nexport function groupEventsByDate<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): Map<string, CalendarEvent<TData>[]> {\n\tconst groups = new Map<string, CalendarEvent<TData>[]>();\n\n\tfor (const event of events) {\n\t\tconst dateKey = startOfDay(event.startDate).toISOString();\n\t\tconst existing = groups.get(dateKey) ?? [];\n\t\texisting.push(event);\n\t\tgroups.set(dateKey, existing);\n\t}\n\n\treturn groups;\n}\n\n/**\n * Group events by schedule type\n */\nexport function groupEventsByScheduleType<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): Map<number | undefined, CalendarEvent<TData>[]> {\n\tconst groups = new Map<number | undefined, CalendarEvent<TData>[]>();\n\n\tfor (const event of events) {\n\t\tconst existing = groups.get(event.scheduleTypeId) ?? [];\n\t\texisting.push(event);\n\t\tgroups.set(event.scheduleTypeId, existing);\n\t}\n\n\treturn groups;\n}\n\n/**\n * Group events by resource\n */\nexport function groupEventsByResource<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): Map<string | undefined, CalendarEvent<TData>[]> {\n\tconst groups = new Map<string | undefined, CalendarEvent<TData>[]>();\n\n\tfor (const event of events) {\n\t\tconst existing = groups.get(event.resourceId) ?? [];\n\t\texisting.push(event);\n\t\tgroups.set(event.resourceId, existing);\n\t}\n\n\treturn groups;\n}\n\n// ============================================================================\n// QUERIES\n// ============================================================================\n\n/**\n * Get events for a specific day\n */\nexport function getEventsForDay<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tdate: Date\n): CalendarEvent<TData>[] {\n\treturn events.filter((event) => {\n\t\t// Check if event spans this day\n\t\tconst dayStart = startOfDay(date);\n\t\tconst dayEnd = endOfDay(date);\n\t\treturn event.startDate <= dayEnd && event.endDate >= dayStart;\n\t});\n}\n\n/**\n * Get all-day events\n */\nexport function getAllDayEvents<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn events.filter((event) => event.isAllDay);\n}\n\n/**\n * Get timed events (not all-day)\n */\nexport function getTimedEvents<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn events.filter((event) => !event.isAllDay);\n}\n\n/**\n * Get multi-day events\n */\nexport function getMultiDayEvents<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[] {\n\treturn events.filter((event) => {\n\t\tif (event.isMultiDay) return true;\n\t\treturn !isSameDay(event.startDate, event.endDate);\n\t});\n}\n\n/**\n * Check if event spans multiple days\n */\nexport function isMultiDayEvent<TData = Record<string, unknown>>(\n\tevent: CalendarEvent<TData>\n): boolean {\n\tif (event.isMultiDay) return true;\n\treturn !isSameDay(event.startDate, event.endDate);\n}\n\n/**\n * Get event duration in minutes\n */\nexport function getEventDurationMinutes<TData = Record<string, unknown>>(\n\tevent: CalendarEvent<TData>\n): number {\n\treturn Math.floor((event.endDate.getTime() - event.startDate.getTime()) / 60000);\n}\n\n// ============================================================================\n// COLOR UTILITIES\n// ============================================================================\n\n/**\n * Get event color with fallback\n */\nexport function getEventColor<TData = Record<string, unknown>>(\n\tevent: CalendarEvent<TData>,\n\tdefaultColor: TEventColor = 'blue'\n): TEventColor {\n\treturn event.color ?? defaultColor;\n}\n\n/**\n * Resolve color from schedule type or event\n */\nexport function resolveEventColor<TData = Record<string, unknown>>(\n\tevent: CalendarEvent<TData>,\n\tscheduleTypeColors?: Map<number, TEventColor | string>,\n\tdefaultColor: TEventColor = 'blue'\n): TEventColor | string {\n\t// Event color takes precedence\n\tif (event.color) return event.color;\n\n\t// Then schedule type color\n\tif (event.scheduleTypeId !== undefined && scheduleTypeColors) {\n\t\tconst typeColor = scheduleTypeColors.get(event.scheduleTypeId);\n\t\tif (typeColor) return typeColor;\n\t}\n\n\treturn defaultColor;\n}\n\n// ============================================================================\n// ALL-DAY / TIME DISPLAY\n// ============================================================================\n\n/**\n * Format event time display intelligently based on event type.\n *\n * - **All-day events** → `\"All day\"` (no confusing \"00:00 – 23:59\")\n * - **Multi-day all-day** → `\"All day · Feb 16 – Feb 18\"`\n * - **Multi-day timed** → `\"Feb 16, 09:00 – Feb 18, 17:00\"`\n * - **Regular events** → `\"09:00 – 17:00\"`\n *\n * @example\n * ```ts\n * formatEventTimeDisplay({ startDate: allDayStart, endDate: allDayEnd, isAllDay: true });\n * // => \"All day\"\n *\n * formatEventTimeDisplay({ startDate: normalStart, endDate: normalEnd });\n * // => \"09:00 - 17:00\"\n * ```\n */\nexport function formatEventTimeDisplay(event: {\n\tstartDate: Date;\n\tendDate: Date;\n\tisAllDay?: boolean;\n\tisMultiDay?: boolean;\n}): string {\n\tconst isAllDay = detectAllDayEvent(event);\n\n\tif (isAllDay) {\n\t\tif (isSameDay(event.startDate, event.endDate)) {\n\t\t\treturn 'All day';\n\t\t}\n\t\t// Multi-day all-day: \"All day · Feb 16 – Feb 18\"\n\t\tconst startStr = event.startDate.toLocaleDateString(undefined, {\n\t\t\tmonth: 'short',\n\t\t\tday: 'numeric',\n\t\t});\n\t\tconst endStr = event.endDate.toLocaleDateString(undefined, {\n\t\t\tmonth: 'short',\n\t\t\tday: 'numeric',\n\t\t});\n\t\treturn `All day · ${startStr} – ${endStr}`;\n\t}\n\n\t// Multi-day timed (not all-day): \"Feb 16, 09:00 – Feb 18, 17:00\"\n\tif (!isSameDay(event.startDate, event.endDate)) {\n\t\tconst startDateStr = event.startDate.toLocaleDateString(undefined, {\n\t\t\tmonth: 'short',\n\t\t\tday: 'numeric',\n\t\t});\n\t\tconst endDateStr = event.endDate.toLocaleDateString(undefined, {\n\t\t\tmonth: 'short',\n\t\t\tday: 'numeric',\n\t\t});\n\t\treturn `${startDateStr}, ${formatTime(event.startDate)} – ${endDateStr}, ${formatTime(event.endDate)}`;\n\t}\n\n\t// Regular single-day timed event: \"09:00 - 17:00\"\n\treturn `${formatTime(event.startDate)} - ${formatTime(event.endDate)}`;\n}\n\n/**\n * Check whether an event should display its time range.\n *\n * Returns `false` for all-day events (where times like \"00:00 – 23:59\"\n * would be meaningless and confusing to users).\n *\n * @example\n * ```ts\n * shouldShowEventTime({ isAllDay: true, startDate, endDate }); // false\n * shouldShowEventTime({ startDate: new Date('2026-02-16T09:00'), endDate: new Date('2026-02-16T17:00') }); // true\n * ```\n */\nexport function shouldShowEventTime(event: {\n\tstartDate: Date;\n\tendDate: Date;\n\tisAllDay?: boolean;\n}): boolean {\n\treturn !detectAllDayEvent(event);\n}\n","/**\n * Grid Utilities for @inno/calendar\n *\n * Functions for view navigation, date ranges, and cell generation.\n */\n\nimport type { ICalendarCell, IDateRange, TCalendarView } from '../types';\nimport {\n\taddDays,\n\taddMonths,\n\taddWeeks,\n\taddYears,\n\tendOfDay,\n\tendOfMonth,\n\tendOfWeek,\n\tendOfYear,\n\tisSameMonth,\n\tisToday,\n\tisWeekend,\n\tstartOfDay,\n\tstartOfMonth,\n\tstartOfWeek,\n\tstartOfYear,\n} from './date-utils';\n\n// ============================================================================\n// VIEW DATE RANGES\n// ============================================================================\n\n/**\n * Get date range for a specific view\n */\nexport function getViewDateRange(\n\tdate: Date,\n\tview: TCalendarView,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): IDateRange {\n\tswitch (view) {\n\t\tcase 'day':\n\t\tcase 'resource-day':\n\t\tcase 'timeline-day':\n\t\t\treturn {\n\t\t\t\tstartDate: startOfDay(date),\n\t\t\t\tendDate: endOfDay(date),\n\t\t\t};\n\n\t\tcase 'week':\n\t\tcase 'resource-week':\n\t\tcase 'timeline-week':\n\t\t\treturn {\n\t\t\t\tstartDate: startOfWeek(date, weekStartsOn),\n\t\t\t\tendDate: endOfWeek(date, weekStartsOn),\n\t\t\t};\n\n\t\tcase 'month': {\n\t\t\t// Include partial weeks at start and end\n\t\t\tconst monthStart = startOfMonth(date);\n\t\t\tconst monthEnd = endOfMonth(date);\n\t\t\treturn {\n\t\t\t\tstartDate: startOfWeek(monthStart, weekStartsOn),\n\t\t\t\tendDate: endOfWeek(monthEnd, weekStartsOn),\n\t\t\t};\n\t\t}\n\n\t\tcase 'timeline-3day':\n\t\t\treturn {\n\t\t\t\tstartDate: startOfDay(date),\n\t\t\t\tendDate: endOfDay(addDays(date, 2)),\n\t\t\t};\n\n\t\tcase 'year':\n\t\t\treturn {\n\t\t\t\tstartDate: startOfYear(date),\n\t\t\t\tendDate: endOfYear(date),\n\t\t\t};\n\n\t\tcase 'agenda':\n\t\t\t// Agenda typically shows next 30 days\n\t\t\treturn {\n\t\t\t\tstartDate: startOfDay(date),\n\t\t\t\tendDate: endOfDay(addDays(date, 29)),\n\t\t\t};\n\n\t\tdefault:\n\t\t\treturn {\n\t\t\t\tstartDate: startOfDay(date),\n\t\t\t\tendDate: endOfDay(date),\n\t\t\t};\n\t}\n}\n\n// ============================================================================\n// NAVIGATION\n// ============================================================================\n\n/**\n * Navigate to next period based on view\n */\nexport function navigateNext(date: Date, view: TCalendarView): Date {\n\tswitch (view) {\n\t\tcase 'day':\n\t\tcase 'resource-day':\n\t\tcase 'timeline-day':\n\t\t\treturn addDays(date, 1);\n\n\t\tcase 'timeline-3day':\n\t\t\treturn addDays(date, 3);\n\n\t\tcase 'week':\n\t\tcase 'resource-week':\n\t\tcase 'timeline-week':\n\t\t\treturn addWeeks(date, 1);\n\n\t\tcase 'month':\n\t\t\treturn addMonths(date, 1);\n\n\t\tcase 'year':\n\t\t\treturn addYears(date, 1);\n\n\t\tcase 'agenda':\n\t\t\treturn addWeeks(date, 1);\n\n\t\tdefault:\n\t\t\treturn addDays(date, 1);\n\t}\n}\n\n/**\n * Navigate to previous period based on view\n */\nexport function navigatePrev(date: Date, view: TCalendarView): Date {\n\tswitch (view) {\n\t\tcase 'day':\n\t\tcase 'resource-day':\n\t\tcase 'timeline-day':\n\t\t\treturn addDays(date, -1);\n\n\t\tcase 'timeline-3day':\n\t\t\treturn addDays(date, -3);\n\n\t\tcase 'week':\n\t\tcase 'resource-week':\n\t\tcase 'timeline-week':\n\t\t\treturn addWeeks(date, -1);\n\n\t\tcase 'month':\n\t\t\treturn addMonths(date, -1);\n\n\t\tcase 'year':\n\t\t\treturn addYears(date, -1);\n\n\t\tcase 'agenda':\n\t\t\treturn addWeeks(date, -1);\n\n\t\tdefault:\n\t\t\treturn addDays(date, -1);\n\t}\n}\n\n/**\n * Navigate to today\n */\nexport function navigateToday(): Date {\n\treturn startOfDay(new Date());\n}\n\n// ============================================================================\n// CELL GENERATION\n// ============================================================================\n\n/**\n * Generate cells for month view\n */\nexport function generateMonthCells(\n\tdate: Date,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): ICalendarCell[] {\n\tconst monthStart = startOfMonth(date);\n\tconst monthEnd = endOfMonth(date);\n\tconst viewStart = startOfWeek(monthStart, weekStartsOn);\n\tconst viewEnd = endOfWeek(monthEnd, weekStartsOn);\n\n\tconst cells: ICalendarCell[] = [];\n\tlet current = viewStart;\n\n\twhile (current <= viewEnd) {\n\t\tcells.push({\n\t\t\tdate: new Date(current),\n\t\t\tisToday: isToday(current),\n\t\t\tisCurrentMonth: isSameMonth(current, date),\n\t\t\tisWeekend: isWeekend(current),\n\t\t\tdayOfWeek: current.getDay(),\n\t\t});\n\t\tcurrent = addDays(current, 1);\n\t}\n\n\treturn cells;\n}\n\n/**\n * Generate cells for week view\n */\nexport function generateWeekCells(\n\tdate: Date,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): ICalendarCell[] {\n\tconst weekStart = startOfWeek(date, weekStartsOn);\n\tconst cells: ICalendarCell[] = [];\n\n\tfor (let i = 0; i < 7; i++) {\n\t\tconst current = addDays(weekStart, i);\n\t\tcells.push({\n\t\t\tdate: current,\n\t\t\tisToday: isToday(current),\n\t\t\tisCurrentMonth: isSameMonth(current, date),\n\t\t\tisWeekend: isWeekend(current),\n\t\t\tdayOfWeek: current.getDay(),\n\t\t});\n\t}\n\n\treturn cells;\n}\n\n/**\n * Generate cells for year view (months)\n */\nexport function generateYearCells(date: Date): ICalendarCell[] {\n\tconst yearStart = startOfYear(date);\n\tconst cells: ICalendarCell[] = [];\n\n\tfor (let i = 0; i < 12; i++) {\n\t\tconst current = addMonths(yearStart, i);\n\t\tcells.push({\n\t\t\tdate: current,\n\t\t\tisToday: isSameMonth(current, new Date()),\n\t\t\tisCurrentMonth: true,\n\t\t\tisWeekend: false,\n\t\t\tdayOfWeek: current.getDay(),\n\t\t});\n\t}\n\n\treturn cells;\n}\n\n// ============================================================================\n// TIME SLOTS\n// ============================================================================\n\n/**\n * Generate time slots for a day\n */\nexport function generateTimeSlots(\n\tstartHour: number,\n\tendHour: number,\n\tslotDuration: number = 30\n): { hour: number; minute: number; label: string }[] {\n\tconst slots: { hour: number; minute: number; label: string }[] = [];\n\tconst slotsPerHour = 60 / slotDuration;\n\n\tfor (let hour = startHour; hour < endHour; hour++) {\n\t\tfor (let slot = 0; slot < slotsPerHour; slot++) {\n\t\t\tconst minute = slot * slotDuration;\n\t\t\tslots.push({\n\t\t\t\thour,\n\t\t\t\tminute,\n\t\t\t\tlabel: `${hour.toString().padStart(2, '0')}:${minute.toString().padStart(2, '0')}`,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn slots;\n}\n\n/**\n * Generate hour labels\n */\nexport function generateHourLabels(\n\tstartHour: number,\n\tendHour: number,\n\tformat: '12h' | '24h' = '24h'\n): { hour: number; label: string }[] {\n\tconst labels: { hour: number; label: string }[] = [];\n\n\tfor (let hour = startHour; hour <= endHour; hour++) {\n\t\tlet label: string;\n\t\tif (format === '12h') {\n\t\t\tconst period = hour >= 12 ? 'PM' : 'AM';\n\t\t\tconst hour12 = hour % 12 || 12;\n\t\t\tlabel = `${hour12} ${period}`;\n\t\t} else {\n\t\t\tlabel = `${hour.toString().padStart(2, '0')}:00`;\n\t\t}\n\t\tlabels.push({ hour, label });\n\t}\n\n\treturn labels;\n}\n\n// ============================================================================\n// VIEW TITLE\n// ============================================================================\n\n/**\n * Get range text for date navigator (e.g., \"Mon, Jan 1 - Sun, Jan 7\")\n * Shows a more compact range description below the month/year title\n */\nexport function getRangeText(date: Date, view: TCalendarView, locale = 'en-US'): string {\n\tconst dayMonthFormat = new Intl.DateTimeFormat(locale, {\n\t\tweekday: 'short',\n\t\tmonth: 'short',\n\t\tday: 'numeric',\n\t});\n\tconst shortFormat = new Intl.DateTimeFormat(locale, { month: 'short', day: 'numeric' });\n\n\tswitch (view) {\n\t\tcase 'day':\n\t\tcase 'resource-day':\n\t\tcase 'timeline-day':\n\t\t\treturn dayMonthFormat.format(date);\n\n\t\tcase 'timeline-3day': {\n\t\t\tconst start3 = startOfDay(date);\n\t\t\tconst end3 = addDays(date, 2);\n\t\t\treturn `${shortFormat.format(start3)} - ${shortFormat.format(end3)}`;\n\t\t}\n\n\t\tcase 'week':\n\t\tcase 'resource-week':\n\t\tcase 'timeline-week': {\n\t\t\tconst weekStart = startOfWeek(date, 1);\n\t\t\tconst weekEnd = endOfWeek(date, 1);\n\t\t\treturn `${shortFormat.format(weekStart)} - ${shortFormat.format(weekEnd)}`;\n\t\t}\n\n\t\tcase 'month': {\n\t\t\tconst monthStart = startOfMonth(date);\n\t\t\tconst monthEnd = endOfMonth(date);\n\t\t\treturn `${shortFormat.format(monthStart)} - ${shortFormat.format(monthEnd)}`;\n\t\t}\n\n\t\tcase 'year':\n\t\t\treturn `Jan 1 - Dec 31, ${date.getFullYear()}`;\n\n\t\tcase 'agenda': {\n\t\t\tconst monthStart = startOfMonth(date);\n\t\t\tconst monthEnd = endOfMonth(date);\n\t\t\treturn `${shortFormat.format(monthStart)} - ${shortFormat.format(monthEnd)}`;\n\t\t}\n\n\t\tdefault:\n\t\t\treturn '';\n\t}\n}\n\n/**\n * Count events in the current view's date range\n * Useful for showing an event count badge in the navigation\n */\nexport function getEventsCountInView(\n\tevents: { startDate: Date; endDate: Date }[],\n\tdate: Date,\n\tview: TCalendarView,\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1\n): number {\n\tconst range = getViewDateRange(date, view, weekStartsOn);\n\n\treturn events.filter((event) => {\n\t\t// Event overlaps with period if it starts before period ends AND ends after period starts\n\t\treturn event.startDate <= range.endDate && event.endDate >= range.startDate;\n\t}).length;\n}\n\n/**\n * Get formatted title for current view\n */\nexport function getViewTitle(date: Date, view: TCalendarView, locale = 'en-US'): string {\n\tconst monthFormat = new Intl.DateTimeFormat(locale, { month: 'long', year: 'numeric' });\n\tconst dateFormat = new Intl.DateTimeFormat(locale, {\n\t\tweekday: 'long',\n\t\tmonth: 'long',\n\t\tday: 'numeric',\n\t\tyear: 'numeric',\n\t});\n\n\tswitch (view) {\n\t\tcase 'day':\n\t\tcase 'resource-day':\n\t\tcase 'timeline-day':\n\t\t\treturn dateFormat.format(date);\n\n\t\tcase 'week':\n\t\tcase 'resource-week':\n\t\tcase 'timeline-week': {\n\t\t\tconst weekStart = startOfWeek(date, 1);\n\t\t\tconst weekEnd = endOfWeek(date, 1);\n\t\t\tconst startMonth = new Intl.DateTimeFormat(locale, { month: 'short' }).format(weekStart);\n\t\t\tconst endMonth = new Intl.DateTimeFormat(locale, { month: 'short' }).format(weekEnd);\n\t\t\tconst year = weekStart.getFullYear();\n\n\t\t\tif (startMonth === endMonth) {\n\t\t\t\treturn `${startMonth} ${weekStart.getDate()} - ${weekEnd.getDate()}, ${year}`;\n\t\t\t}\n\t\t\treturn `${startMonth} ${weekStart.getDate()} - ${endMonth} ${weekEnd.getDate()}, ${year}`;\n\t\t}\n\n\t\tcase 'month':\n\t\t\treturn monthFormat.format(date);\n\n\t\tcase 'year':\n\t\t\treturn date.getFullYear().toString();\n\n\t\tcase 'agenda':\n\t\t\treturn `Agenda - ${monthFormat.format(date)}`;\n\n\t\tdefault:\n\t\t\treturn monthFormat.format(date);\n\t}\n}\n\n// ============================================================================\n// WEEK DAYS\n// ============================================================================\n\n/**\n * Get weekday headers for week/month views\n */\nexport function getWeekdayHeaders(\n\tweekStartsOn: 0 | 1 | 2 | 3 | 4 | 5 | 6 = 1,\n\tformat: 'long' | 'short' | 'narrow' = 'short',\n\tlocale = 'en-US'\n): { dayOfWeek: number; label: string }[] {\n\tconst formatter = new Intl.DateTimeFormat(locale, { weekday: format });\n\tconst headers: { dayOfWeek: number; label: string }[] = [];\n\n\t// Start from a known date (Jan 7, 2024 is a Sunday)\n\tconst baseDate = new Date(2024, 0, 7);\n\n\tfor (let i = 0; i < 7; i++) {\n\t\tconst dayIndex = (weekStartsOn + i) % 7;\n\t\tconst date = addDays(baseDate, dayIndex);\n\t\theaders.push({\n\t\t\tdayOfWeek: dayIndex,\n\t\t\tlabel: formatter.format(date),\n\t\t});\n\t}\n\n\treturn headers;\n}\n","/**\n * Position Utilities for @inno/calendar\n *\n * Functions for calculating event positions in time grid views.\n */\n\nimport { DEFAULT_HOUR_HEIGHT, MIN_EVENT_HEIGHT } from '../constants';\nimport type { CalendarEvent, IEventPosition, IPositionedEvent, IVisibleHours } from '../types';\n\n// ============================================================================\n// POSITION CALCULATION\n// ============================================================================\n\n/**\n * Calculate event position in time grid\n */\nexport function calculateEventPosition<TData = Record<string, unknown>>(\n\tevent: CalendarEvent<TData>,\n\t_dayStart: Date,\n\tvisibleHours: IVisibleHours,\n\thourHeight: number = DEFAULT_HOUR_HEIGHT\n): IEventPosition {\n\tconst { startHour, endHour } = visibleHours;\n\tconst totalVisibleHours = endHour - startHour;\n\tconst totalHeight = totalVisibleHours * hourHeight;\n\n\t// Get event times as decimal hours\n\tconst eventStartHour = event.startDate.getHours() + event.startDate.getMinutes() / 60;\n\tconst eventEndHour = event.endDate.getHours() + event.endDate.getMinutes() / 60;\n\n\t// Clamp to visible range\n\tconst clampedStart = Math.max(eventStartHour, startHour);\n\tconst clampedEnd = Math.min(eventEndHour, endHour);\n\n\t// Calculate position relative to visible area\n\tconst top = ((clampedStart - startHour) / totalVisibleHours) * totalHeight;\n\tconst bottom = ((clampedEnd - startHour) / totalVisibleHours) * totalHeight;\n\tconst height = Math.max(bottom - top, MIN_EVENT_HEIGHT);\n\n\treturn {\n\t\ttop,\n\t\theight,\n\t\tleft: 0,\n\t\twidth: 100,\n\t\tzIndex: 1,\n\t};\n}\n\n/**\n * Calculate positions for overlapping events using group-based columns.\n *\n * Events are first grouped into overlap clusters. Within each cluster,\n * columns are assigned independently so that non-overlapping events can\n * expand to full width instead of being constrained by unrelated events.\n */\nexport function calculateOverlappingPositions<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[],\n\tdayStart: Date,\n\tvisibleHours: IVisibleHours,\n\thourHeight: number = DEFAULT_HOUR_HEIGHT\n): IPositionedEvent<TData>[] {\n\tif (events.length === 0) return [];\n\n\t// Sort by start time, then by duration (longer events first)\n\tconst sortedEvents = [...events].sort((a, b) => {\n\t\tconst startDiff = a.startDate.getTime() - b.startDate.getTime();\n\t\tif (startDiff !== 0) return startDiff;\n\t\tconst durationA = a.endDate.getTime() - a.startDate.getTime();\n\t\tconst durationB = b.endDate.getTime() - b.startDate.getTime();\n\t\treturn durationB - durationA;\n\t});\n\n\t// --- Step 1: Group events into overlap clusters ---\n\tconst groups: CalendarEvent<TData>[][] = [];\n\tlet currentGroup: CalendarEvent<TData>[] = [];\n\tlet groupEnd = 0;\n\n\tfor (const event of sortedEvents) {\n\t\tif (currentGroup.length === 0 || event.startDate.getTime() < groupEnd) {\n\t\t\tcurrentGroup.push(event);\n\t\t\tgroupEnd = Math.max(groupEnd, event.endDate.getTime());\n\t\t} else {\n\t\t\tgroups.push(currentGroup);\n\t\t\tcurrentGroup = [event];\n\t\t\tgroupEnd = event.endDate.getTime();\n\t\t}\n\t}\n\tif (currentGroup.length > 0) {\n\t\tgroups.push(currentGroup);\n\t}\n\n\t// --- Step 2: Assign columns within each group independently ---\n\tconst positionedEvents: IPositionedEvent<TData>[] = [];\n\n\tfor (const group of groups) {\n\t\tconst columns: { event: CalendarEvent<TData>; endTime: number }[][] = [];\n\n\t\tfor (const event of group) {\n\t\t\tconst eventStart = event.startDate.getTime();\n\t\t\tconst eventEnd = event.endDate.getTime();\n\n\t\t\t// Find first available column within this group\n\t\t\tlet columnIndex = -1;\n\t\t\tfor (let i = 0; i < columns.length; i++) {\n\t\t\t\tconst column = columns[i];\n\t\t\t\tif (!column) continue;\n\t\t\t\tconst lastInColumn = column[column.length - 1];\n\t\t\t\tif (lastInColumn && lastInColumn.endTime <= eventStart) {\n\t\t\t\t\tcolumnIndex = i;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (columnIndex === -1) {\n\t\t\t\tcolumnIndex = columns.length;\n\t\t\t\tcolumns.push([]);\n\t\t\t}\n\n\t\t\tconst column = columns[columnIndex];\n\t\t\tif (column) {\n\t\t\t\tcolumn.push({ event, endTime: eventEnd });\n\t\t\t}\n\n\t\t\tconst basePosition = calculateEventPosition<TData>(event, dayStart, visibleHours, hourHeight);\n\n\t\t\tpositionedEvents.push({\n\t\t\t\tevent,\n\t\t\t\tposition: basePosition,\n\t\t\t\tcolumn: columnIndex,\n\t\t\t\tcolumnSpan: 1,\n\t\t\t});\n\t\t}\n\n\t\t// Calculate width and left using this group's column count\n\t\tconst totalColumns = columns.length;\n\t\tconst widthPerColumn = 100 / totalColumns;\n\n\t\t// Only update positions for events in this group\n\t\tconst groupIds = new Set(group.map((e) => e.id));\n\t\tfor (const positioned of positionedEvents) {\n\t\t\tif (!groupIds.has(positioned.event.id)) continue;\n\t\t\tpositioned.position.left = positioned.column * widthPerColumn;\n\t\t\tpositioned.position.width = widthPerColumn * positioned.columnSpan;\n\t\t\tpositioned.position.zIndex = positioned.column + 1;\n\t\t}\n\t}\n\n\treturn positionedEvents;\n}\n\n// ============================================================================\n// TIME TO POSITION CONVERSION\n// ============================================================================\n\n/**\n * Convert Y position to time\n */\nexport function yToTime(\n\ty: number,\n\tdayStart: Date,\n\tvisibleHours: IVisibleHours,\n\thourHeight: number = DEFAULT_HOUR_HEIGHT,\n\tslotDuration: number = 30\n): Date {\n\tconst { startHour, endHour } = visibleHours;\n\tconst totalVisibleHours = endHour - startHour;\n\tconst totalHeight = totalVisibleHours * hourHeight;\n\n\t// Calculate hours from top\n\tconst hoursFromTop = (y / totalHeight) * totalVisibleHours;\n\tconst absoluteHours = startHour + hoursFromTop;\n\n\t// Snap to slot duration\n\tconst totalMinutes = absoluteHours * 60;\n\tconst snappedMinutes = Math.round(totalMinutes / slotDuration) * slotDuration;\n\n\t// Create date with snapped time\n\tconst result = new Date(dayStart);\n\tresult.setHours(0, 0, 0, 0);\n\tresult.setMinutes(snappedMinutes);\n\n\treturn result;\n}\n\n/**\n * Convert time to Y position\n */\nexport function timeToY(\n\ttime: Date,\n\tvisibleHours: IVisibleHours,\n\thourHeight: number = DEFAULT_HOUR_HEIGHT\n): number {\n\tconst { startHour, endHour } = visibleHours;\n\tconst totalVisibleHours = endHour - startHour;\n\tconst totalHeight = totalVisibleHours * hourHeight;\n\n\tconst hours = time.getHours() + time.getMinutes() / 60;\n\tconst clampedHours = Math.max(startHour, Math.min(hours, endHour));\n\n\treturn ((clampedHours - startHour) / totalVisibleHours) * totalHeight;\n}\n\n// ============================================================================\n// OVERLAP DETECTION\n// ============================================================================\n\n/**\n * Check if two events overlap in time\n */\nexport function eventsOverlap(event1: CalendarEvent, event2: CalendarEvent): boolean {\n\treturn event1.startDate < event2.endDate && event1.endDate > event2.startDate;\n}\n\n/**\n * Get overlapping event groups\n */\nexport function getOverlappingGroups<TData = Record<string, unknown>>(\n\tevents: CalendarEvent<TData>[]\n): CalendarEvent<TData>[][] {\n\tif (events.length === 0) return [];\n\n\tconst sorted = [...events].sort((a, b) => a.startDate.getTime() - b.startDate.getTime());\n\tconst groups: CalendarEvent<TData>[][] = [];\n\tlet currentGroup: CalendarEvent<TData>[] = [];\n\tlet groupEnd = 0;\n\n\tfor (const event of sorted) {\n\t\tif (currentGroup.length === 0 || event.startDate.getTime() < groupEnd) {\n\t\t\t// Event overlaps with current group\n\t\t\tcurrentGroup.push(event);\n\t\t\tgroupEnd = Math.max(groupEnd, event.endDate.getTime());\n\t\t} else {\n\t\t\t// Start new group\n\t\t\tif (currentGroup.length > 0) {\n\t\t\t\tgroups.push(currentGroup);\n\t\t\t}\n\t\t\tcurrentGroup = [event];\n\t\t\tgroupEnd = event.endDate.getTime();\n\t\t}\n\t}\n\n\tif (currentGroup.length > 0) {\n\t\tgroups.push(currentGroup);\n\t}\n\n\treturn groups;\n}\n\n// ============================================================================\n// SELECTION OVERLAY\n// ============================================================================\n\n/**\n * Calculate selection overlay dimensions\n */\nexport function calculateSelectionOverlay(\n\tstartY: number,\n\tcurrentY: number,\n\tcontainerHeight: number\n): { top: number; height: number } {\n\tconst minY = Math.max(0, Math.min(startY, currentY));\n\tconst maxY = Math.min(containerHeight, Math.max(startY, currentY));\n\n\treturn {\n\t\ttop: minY,\n\t\theight: maxY - minY,\n\t};\n}\n"],"names":["reactNodeToText","node","isValidElement","props","CALENDAR_VIEW_STORAGE_KEY","RESOURCE_VIEW_STORAGE_KEY","CALENDAR_GRID_VIEWS","RESOURCE_VIEWS","isCalendarGridView","view","isResourceView","isBrowser","readKey","key","writeKey","value","removeKey","saveCalendarView","storageKey","readCalendarView","clearCalendarView","saveResourceView","readResourceView","clearResourceView","saveViewByFamily","CALENDAR_VIEWS","DEFAULT_PREFERENCES","DEFAULT_VISIBLE_HOURS","DEFAULT_BUSINESS_HOURS","DEFAULT_VIEW_CONFIGS","EVENT_COLORS","EVENT_COLOR_CLASSES","HEX_TO_EVENT_COLOR","HOURS_IN_DAY","MINUTES_IN_HOUR","MS_PER_MINUTE","MS_PER_HOUR","MS_PER_DAY","DEFAULT_HOUR_HEIGHT","DEFAULT_SLOT_HEIGHT","DEFAULT_HEADER_HEIGHT","DEFAULT_RESOURCE_WIDTH","MIN_EVENT_HEIGHT","startOfDay","date","result","endOfDay","addDays","days","subDays","isSameDay","date1","date2","isToday","getDayOfWeek","isWeekend","day","startOfWeek","weekStartsOn","diff","endOfWeek","addWeeks","weeks","subWeeks","getWeekNumber","d","dayNum","yearStart","isSameWeek","start1","start2","startOfMonth","endOfMonth","addMonths","months","subMonths","getDaysInMonth","isSameMonth","startOfYear","endOfYear","addYears","years","subYears","isSameYear","addHours","hours","addMinutes","minutes","setTime","seconds","getDecimalHours","isBefore","dateToCompare","isAfter","isBetween","start","end","time","minDate","dates","min","maxDate","max","differenceInMilliseconds","differenceInMinutes","differenceInHours","differenceInDays","formatTime","format","period","formatDateISO","year","month","parseISO","dateString","eachDayOfInterval","current","endDay","eachHourOfInterval","getWeekdayNames","locale","formatter","baseDate","i","getMonthNames","getVisibleHoursArray","visibleHours","h","formatHourLabel","hour","getWeekDays","weekStart","generateMonthGrid","monthStart","monthEnd","gridStart","gridEnd","currentMonth","detectAllDayEvent","event","startHour","startMinute","endHour","endMinute","startsAtMidnight","endsAtDayBoundary","separateEventsByDuration","events","singleDay","multiDay","getEventsInRange","rangeStart","rangeEnd","isWorkingHour","workingHours","dayOfWeek","dayHours","getWorkingHoursForDay","getYearMonths","filterEventsByDateRange","startDate","endDate","eventStart","eventEnd","filterEventsByScheduleType","scheduleTypeIds","filterEventsByResource","resourceIds","filterEventsBySearch","search","searchLower","titleMatch","descriptionMatch","filterOutCanceled","applyEventFilters","options","filtered","sortEventsByStart","a","b","sortEventsByEnd","sortEventsByDuration","durationA","groupEventsByDate","groups","dateKey","existing","groupEventsByScheduleType","groupEventsByResource","getEventsForDay","dayStart","dayEnd","getAllDayEvents","getTimedEvents","getMultiDayEvents","isMultiDayEvent","getEventDurationMinutes","getEventColor","defaultColor","resolveEventColor","scheduleTypeColors","typeColor","formatEventTimeDisplay","startStr","endStr","startDateStr","endDateStr","shouldShowEventTime","getViewDateRange","navigateNext","navigatePrev","navigateToday","generateMonthCells","viewStart","viewEnd","cells","generateWeekCells","generateYearCells","generateTimeSlots","slotDuration","slots","slotsPerHour","slot","minute","generateHourLabels","labels","label","getRangeText","dayMonthFormat","shortFormat","start3","end3","weekEnd","getEventsCountInView","range","getViewTitle","monthFormat","dateFormat","startMonth","endMonth","getWeekdayHeaders","headers","dayIndex","calculateEventPosition","_dayStart","hourHeight","totalVisibleHours","totalHeight","eventStartHour","eventEndHour","clampedStart","clampedEnd","top","bottom","height","calculateOverlappingPositions","sortedEvents","startDiff","currentGroup","groupEnd","positionedEvents","group","columns","columnIndex","column","lastInColumn","basePosition","widthPerColumn","groupIds","e","positioned","yToTime","y","hoursFromTop","totalMinutes","snappedMinutes","timeToY","eventsOverlap","event1","event2","getOverlappingGroups","sorted","calculateSelectionOverlay","startY","currentY","containerHeight","minY","maxY"],"mappings":"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"}
|
package/dist/presets/index.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const a=require("../tailwind-calendar-
|
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const a=require("../tailwind-calendar-smlkVdSq.cjs");exports.DefaultCalendar=a.DefaultCalendar;exports.TailwindCalendar=a.TailwindCalendar;
|
|
2
2
|
//# sourceMappingURL=index.cjs.map
|
package/dist/presets/index.mjs
CHANGED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
"use strict";const F=require("react/jsx-runtime"),e=require("react"),P=require("./position-utils-Do2ciBXl.cjs"),M=e.createContext(void 0);function we({children:s,onEventDrop:n}){const[t,r]=e.useState(null),g=e.useCallback(a=>{const f=typeof a.startDate=="string"?new Date(a.startDate):a.startDate,D=typeof a.endDate=="string"?new Date(a.endDate):a.endDate;r({event:a,originalStartDate:f,originalEndDate:D})},[]),i=e.useCallback((a,f,D,l)=>{r(m=>m?{...m,previewDate:a,previewHour:f,previewMinute:D,...l!==void 0&&{targetResourceId:l}}:null)},[]),S=e.useCallback(()=>{if(!t)return null;const{event:a,originalStartDate:f,originalEndDate:D,previewDate:l,previewHour:m,previewMinute:h,targetResourceId:I}=t,H=D.getTime()-f.getTime();let w;l?(w=new Date(l),typeof m=="number"?w.setHours(m,h??0,0,0):w.setHours(f.getHours(),f.getMinutes(),f.getSeconds(),0)):w=f;const b=new Date(w.getTime()+H),y={event:a,newStartDate:w,newEndDate:b,...I!==void 0&&{newResourceId:I}};return r(null),n?.(y),y},[t,n]),c=e.useCallback(()=>{r(null)},[]);return F.jsx(M.Provider,{value:{dragState:t,isDragging:t!==null,startDrag:g,updateDragPreview:i,endDrag:S,cancelDrag:c},children:s})}function be(){const s=e.useContext(M);if(!s)throw new Error("useDragDrop must be used within a DragDropProvider");return s}function ke(){return e.useContext(M)??null}const oe="inno-calendar-preferences",ae={from:0,to:24},ue={0:{enabled:!1,from:8,to:18},1:{enabled:!0,from:8,to:18},2:{enabled:!0,from:8,to:18},3:{enabled:!0,from:8,to:18},4:{enabled:!0,from:8,to:18},5:{enabled:!0,from:8,to:18},6:{enabled:!1,from:8,to:18}},le=30,Ce={view:"month",badgeVariant:"colored",slotDuration:le,visibleHours:ae,workingHours:ue,showWorkingHoursOnly:!1,showWeekends:!0,firstDayOfWeek:1};function pe(s){if(typeof window>"u")return{};try{const n=localStorage.getItem(s);if(!n)return{};const t=JSON.parse(n),r={};return t.view&&typeof t.view=="string"&&(r.view=t.view),t.badgeVariant&&typeof t.badgeVariant=="string"&&(r.badgeVariant=t.badgeVariant),t.slotDuration&&typeof t.slotDuration=="number"&&(r.slotDuration=t.slotDuration),t.visibleHours&&typeof t.visibleHours=="object"&&(r.visibleHours=t.visibleHours),t.workingHours&&typeof t.workingHours=="object"&&(r.workingHours=t.workingHours),typeof t.showWorkingHoursOnly=="boolean"&&(r.showWorkingHoursOnly=t.showWorkingHoursOnly),typeof t.showWeekends=="boolean"&&(r.showWeekends=t.showWeekends),typeof t.firstDayOfWeek=="number"&&(r.firstDayOfWeek=t.firstDayOfWeek),r}catch{return console.warn("[InnoCalendar] Failed to load preferences from localStorage"),{}}}function me(s,n){if(!(typeof window>"u"))try{localStorage.setItem(s,JSON.stringify(n))}catch{console.warn("[InnoCalendar] Failed to save preferences to localStorage")}}function ie(s={}){const{modes:n={},locked:t={},defaults:r={},storageKey:g=oe,disableStorage:i=!1}=s,S=e.useMemo(()=>({...Ce,...r}),[r]),[c,a]=e.useState(()=>i?{}:pe(g)),[f,D]=e.useState({});e.useEffect(()=>{i||me(g,c)},[c,g,i]);const l=e.useCallback(o=>n[o]??"user",[n]),m=e.useCallback(o=>l(o)==="locked",[l]),h=e.useCallback(o=>l(o)==="user"&&!i,[l,i]),I=e.useMemo(()=>{const o={...S};for(const d of Object.keys(o)){const p=l(d);p==="locked"&&d in t?o[d]=t[d]:p==="session"&&d in f?o[d]=f[d]:p==="user"&&d in c&&(o[d]=c[d])}return o},[S,c,f,t,l]),H=e.useCallback((o,d)=>{const p=l(o);if(p==="locked"){console.warn(`[InnoCalendar] Preference "${o}" is locked and cannot be modified`);return}p==="session"?D(k=>({...k,[o]:d})):a(k=>({...k,[o]:d}))},[l]),w=e.useCallback(o=>{const d={},p={};for(const k of Object.keys(o)){const v=l(k),O=o[k];if(O!==void 0){if(v==="locked"){console.warn(`[InnoCalendar] Preference "${k}" is locked and cannot be modified`);continue}v==="session"?d[k]=O:p[k]=O}}Object.keys(d).length>0&&D(k=>({...k,...d})),Object.keys(p).length>0&&a(k=>({...k,...p}))},[l]),b=e.useCallback(()=>{a({}),D({}),i||localStorage.removeItem(g)},[g,i]),y=e.useCallback(o=>{const d=l(o);if(d==="locked"){console.warn(`[InnoCalendar] Preference "${o}" is locked and cannot be reset`);return}d==="session"?D(p=>{const{[o]:k,...v}=p;return v}):a(p=>{const{[o]:k,...v}=p;return v})},[l]);return{preferences:I,setPreference:H,setPreferences:w,resetPreferences:b,resetPreference:y,isLocked:m,isPersisted:h,getMode:l}}const N=e.createContext(void 0),se={0:{enabled:!1,from:8,to:17},1:{enabled:!0,from:8,to:17},2:{enabled:!0,from:8,to:17},3:{enabled:!0,from:8,to:17},4:{enabled:!0,from:8,to:17},5:{enabled:!0,from:8,to:17},6:{enabled:!0,from:8,to:12}},He={start:0,end:24};function ye({children:s,initialEvents:n=[],initialUsers:t=[],initialScheduleTypes:r=[],initialView:g,initialDate:i,initialSelectedUserId:S="all",initialScheduleTypeIds:c=[],initialParticipantIds:a=[],initialWorkingHoursView:f="default",initialSearchQuery:D="",preferencesConfig:l,initialFocusEventId:m=null,onDateChange:h,onViewChange:I}){const{preferences:H,setPreference:w,isLocked:b}=ie(l),[y,o]=e.useState(g??H.view??P.readCalendarView()??P.readResourceView()??"week"),[d,p]=e.useState(i??new Date),[k,v]=e.useState(S),O=H.badgeVariant??"colored",j=e.useCallback(u=>{b("badgeVariant")||w("badgeVariant",u)},[w,b]),K=H.workingHours??se,B=e.useCallback(u=>{if(!b("workingHours")){const C=H.workingHours??se,T=typeof u=="function"?u(C):u;w("workingHours",T)}},[w,b,H.workingHours]),E=H.visibleHours,U=e.useMemo(()=>E?{start:E.start??E.from??0,end:E.end??E.to??24}:He,[E]),G=e.useCallback(u=>{if(!b("visibleHours")){const C=typeof u=="function"?u(U):u;w("visibleHours",{from:C.start,to:C.end})}},[w,b,U]),[ce,de]=e.useState(f==="enabled"?!0:f==="disabled"?!1:H.showWorkingHoursOnly??!1),Y=ce,$=e.useCallback(u=>{de(u),b("showWorkingHoursOnly")||w("showWorkingHoursOnly",u)},[w,b]),q=H.slotDuration??30,z=e.useCallback(u=>{b("slotDuration")||w("slotDuration",u)},[w,b]),[L,Q]=e.useState(n);e.useEffect(()=>{Q(n)},[n]);const[R,fe]=e.useState(c),[V,Ue]=e.useState(a),[x,ge]=e.useState(D),[J,X]=e.useState(m??null);e.useEffect(()=>{m&&X(m)},[m]);const Z=e.useCallback(()=>X(null),[]),ee=e.useCallback(u=>{o(u),b("view")||w("view",u),P.saveViewByFamily(u),I?.(u)},[I,w,b]),te=e.useCallback((u,C)=>{p(u),h?.(u,C??y)},[h,y]),ne=e.useMemo(()=>{let u=L;if(R.length>0&&(u=u.filter(C=>C.scheduleTypeId!==void 0&&R.includes(C.scheduleTypeId))),V.length>0&&(u=u.filter(C=>C.participants?.length?C.participants.some(T=>V.includes(T.id)):!1)),k!=="all"&&(u=u.filter(C=>C.participants?.some(T=>T.id===k))),x.trim()){const C=x.toLowerCase();u=u.filter(T=>P.reactNodeToText(T.title).toLowerCase().includes(C)||P.reactNodeToText(T.description).toLowerCase().includes(C)||P.reactNodeToText(T.scheduleTypeName).toLowerCase().includes(C)||T.participants?.some(De=>De.name.toLowerCase().includes(C)))}return u},[L,R,V,k,x]),Se=e.useMemo(()=>({view:y,setView:ee,selectedDate:d,setSelectedDate:te,selectedUserId:k,setSelectedUserId:v,badgeVariant:O,setBadgeVariant:j,workingHours:K,setWorkingHours:B,visibleHours:U,setVisibleHours:G,showWorkingHoursOnly:Y,setShowWorkingHoursOnly:$,slotDuration:q,setSlotDuration:z,isPreferenceLocked:b,events:L,setEvents:Q,users:t,scheduleTypes:r,selectedScheduleTypeIds:R,setSelectedScheduleTypeIds:fe,searchQuery:x,setSearchQuery:ge,filteredEvents:ne,focusEventId:J,clearFocusEventId:Z}),[y,ee,d,te,k,O,j,K,B,U,G,Y,$,q,z,b,L,t,r,R,x,ne,J,Z]);return F.jsx(N.Provider,{value:Se,children:s})}function _(){const s=e.useContext(N);if(!s)throw new Error("useInnoCalendar must be used within an InnoCalendarProvider");return s}function Ie(){return e.useContext(N)}function Te(){const{view:s,setView:n,selectedDate:t,setSelectedDate:r,slotDuration:g}=_();return{view:s,setView:n,selectedDate:t,setSelectedDate:r,slotDuration:g}}function he(){const{events:s,setEvents:n,filteredEvents:t}=_();return{events:s,setEvents:n,filteredEvents:t}}function ve(){const{selectedScheduleTypeIds:s,setSelectedScheduleTypeIds:n,selectedUserId:t,setSelectedUserId:r,searchQuery:g,setSearchQuery:i}=_();return{selectedScheduleTypeIds:s,setSelectedScheduleTypeIds:n,selectedUserId:t,setSelectedUserId:r,searchQuery:g,setSearchQuery:i}}function Oe(){const{workingHours:s,setWorkingHours:n,visibleHours:t,setVisibleHours:r,showWorkingHoursOnly:g,setShowWorkingHoursOnly:i}=_();return{workingHours:s,setWorkingHours:n,visibleHours:t,setVisibleHours:r,showWorkingHoursOnly:g,setShowWorkingHoursOnly:i}}const A=e.createContext(void 0);function W(s){const n=new Date(s.date);return typeof s.hour=="number"&&n.setHours(s.hour,s.minute??0,0,0),n}function Ee(s){const n=new Date(s),t=n.getMinutes(),r=Math.floor(t/30)*30;return n.setMinutes(r,0,0),n}function Pe(s,n){return s.getFullYear()===n.getFullYear()&&s.getMonth()===n.getMonth()&&s.getDate()===n.getDate()}function re(s,n,t,r,g=30){const i=W(s),S=W(n);let c=i,a=S;if(S<i&&(c=S,a=i),t==="day")if(Pe(c,a)){const D=Ee(new Date);c=new Date(c),c.setHours(D.getHours(),D.getMinutes(),0,0),a=new Date(c.getTime()+r*60*1e3)}else c.setHours(0,0,0,0),a.setHours(23,59,59,999);else{const f=g*60*1e3;a=new Date(a.getTime()+f);const D=a.getTime()-c.getTime(),l=r*60*1e3;D<l&&(a=new Date(c.getTime()+l))}return{startDate:c,endDate:a}}function Re(s,n,t,r){const g=W(s),i=W(n),S=W(t),c=i<=S?i:S,a=i<=S?S:i;if(r==="day"){const f=new Date(g);f.setHours(0,0,0,0);const D=new Date(c);D.setHours(0,0,0,0);const l=new Date(a);return l.setHours(0,0,0,0),f>=D&&f<=l}return g>=c&&g<=a}function xe({children:s,mode:n="time",onSelect:t,minDurationMinutes:r=30,slotDurationMinutes:g=30,enabled:i=!0}){const[S,c]=e.useState(null),a=e.useRef(!1),f=e.useRef(!1),D=e.useRef(null),l=e.useCallback(()=>{f.current=!0,D.current&&clearTimeout(D.current),D.current=setTimeout(()=>{f.current=!1},300)},[]),m=e.useCallback(o=>{!i||f.current||(a.current=!0,c({start:o,end:o,isSelecting:!0}))},[i]),h=e.useCallback(o=>{i&&a.current&&c(d=>d?{...d,end:o}:null)},[i]),I=e.useCallback(()=>{a.current&&(a.current=!1,c(o=>{if(!o)return null;const d=re(o.start,o.end,n,r,g);return t?.(d),{...o,isSelecting:!1}}),setTimeout(()=>{c(null)},100))},[n,r,g,t]),H=e.useCallback(()=>{a.current=!1,c(null)},[]),w=e.useCallback(o=>S?Re(o,S.start,S.end,n):!1,[S,n]),b=e.useCallback(()=>S?re(S.start,S.end,n,r,g):null,[S,n,r,g]),y=e.useMemo(()=>({isSelecting:S?.isSelecting??!1,startSelection:m,updateSelection:h,endSelection:I,cancelSelection:H,isSlotSelected:w,getSelectionResult:b,mode:n,suppressSelection:l}),[S?.isSelecting,m,h,I,H,w,b,n,l]);return F.jsx(A.Provider,{value:y,children:s})}function We(){const s=e.useContext(A);if(!s)throw new Error("useSlotSelectionContext must be used within a SlotSelectionProvider");return s}function _e(){return e.useContext(A)}exports.DEFAULT_SLOT_DURATION=le;exports.DEFAULT_VISIBLE_HOURS=ae;exports.DEFAULT_WORKING_HOURS=ue;exports.DragDropProvider=we;exports.InnoCalendarProvider=ye;exports.PREFERENCES_STORAGE_KEY=oe;exports.SlotSelectionProvider=xe;exports.useCalendarPreferences=ie;exports.useDragDrop=be;exports.useInnoCalendar=_;exports.useInnoCalendarEvents=he;exports.useInnoCalendarFilters=ve;exports.useInnoCalendarTimeConfig=Oe;exports.useInnoCalendarView=Te;exports.useOptionalDragDrop=ke;exports.useOptionalInnoCalendar=Ie;exports.useOptionalSlotSelection=_e;exports.useSlotSelectionContext=We;
|
|
2
|
+
//# sourceMappingURL=slot-selection-context-BlrL9b5P.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"slot-selection-context-BlrL9b5P.cjs","sources":["../src/core/context/drag-drop-context.tsx","../src/core/preferences/types.ts","../src/core/preferences/use-preferences.ts","../src/core/context/inno-calendar-provider.tsx","../src/core/context/slot-selection-context.tsx"],"sourcesContent":["/**\n * DragDropContext - Drag & Drop State Management for Calendar Events\n *\n * Provides drag & drop state management for calendar events.\n * Enables dragging events between time slots and days.\n *\n * Features:\n * - Generic event type support via CalendarEvent<TData>\n * - Preview position tracking during drag\n * - Duration preservation when dropping\n * - Optional provider pattern (components can work without drag-drop)\n */\n\nimport { createContext, type ReactNode, useCallback, useContext, useState } from 'react';\nimport type { CalendarEvent } from '../types';\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport interface IDragState<TData = Record<string, unknown>> {\n\t/** The event being dragged */\n\tevent: CalendarEvent<TData>;\n\t/** Original start date of the event */\n\toriginalStartDate: Date;\n\t/** Original end date of the event */\n\toriginalEndDate: Date;\n\t/** Current preview position (during drag) */\n\tpreviewDate?: Date | undefined;\n\t/** Current preview hour (for time-based views) */\n\tpreviewHour?: number | undefined;\n\t/** Current preview minute */\n\tpreviewMinute?: number | undefined;\n\t/** Original resource/row ID (for resource/timeline views) */\n\toriginalResourceId?: string | undefined;\n\t/** Current target resource/row ID during drag (for resource/timeline views) */\n\ttargetResourceId?: string | undefined;\n}\n\nexport interface IDropResult<TData = Record<string, unknown>> {\n\t/** The event that was dropped */\n\tevent: CalendarEvent<TData>;\n\t/** New start date after drop */\n\tnewStartDate: Date;\n\t/** New end date after drop (maintains original duration) */\n\tnewEndDate: Date;\n\t/** New resource/row ID after drop (for resource/timeline views) */\n\tnewResourceId?: string | undefined;\n}\n\nexport interface IDragDropContext<TData = Record<string, unknown>> {\n\t/** Current drag state, null when not dragging */\n\tdragState: IDragState<TData> | null;\n\t/** Whether a drag operation is in progress */\n\tisDragging: boolean;\n\t/** Start dragging an event */\n\t// biome-ignore lint/suspicious/noExplicitAny: Accept any event type for flexibility\n\tstartDrag: (event: CalendarEvent<any>) => void;\n\t/** Update drag preview position */\n\tupdateDragPreview: (date: Date, hour?: number, minute?: number, resourceId?: string) => void;\n\t/** End drag operation and return drop result */\n\tendDrag: () => IDropResult<TData> | null;\n\t/** Cancel drag operation */\n\tcancelDrag: () => void;\n}\n\n// ============================================================================\n// CONTEXT\n// ============================================================================\n\n// biome-ignore lint/suspicious/noExplicitAny: Context needs to be flexible for different event types\nconst DragDropContext = createContext<IDragDropContext<any> | undefined>(undefined);\n\n// ============================================================================\n// PROVIDER PROPS\n// ============================================================================\n\nexport interface DragDropProviderProps<TData = Record<string, unknown>> {\n\tchildren: ReactNode;\n\t/** Callback when an event is dropped at a new location */\n\tonEventDrop?: (result: IDropResult<TData>) => void;\n}\n\n// ============================================================================\n// PROVIDER\n// ============================================================================\n\nexport function DragDropProvider<TData = Record<string, unknown>>({\n\tchildren,\n\tonEventDrop,\n}: DragDropProviderProps<TData>) {\n\tconst [dragState, setDragState] = useState<IDragState<TData> | null>(null);\n\n\t// biome-ignore lint/suspicious/noExplicitAny: Accept any event type for flexibility\n\tconst startDrag = useCallback((event: CalendarEvent<any>) => {\n\t\tconst startDate =\n\t\t\ttypeof event.startDate === 'string' ? new Date(event.startDate) : event.startDate;\n\t\tconst endDate = typeof event.endDate === 'string' ? new Date(event.endDate) : event.endDate;\n\n\t\tsetDragState({\n\t\t\tevent: event as CalendarEvent<TData>,\n\t\t\toriginalStartDate: startDate,\n\t\t\toriginalEndDate: endDate,\n\t\t});\n\t}, []);\n\n\tconst updateDragPreview = useCallback((date: Date, hour?: number, minute?: number, resourceId?: string) => {\n\t\tsetDragState((prev) => {\n\t\t\tif (!prev) return null;\n\t\t\treturn {\n\t\t\t\t...prev,\n\t\t\t\tpreviewDate: date,\n\t\t\t\tpreviewHour: hour,\n\t\t\t\tpreviewMinute: minute,\n\t\t\t\t...(resourceId !== undefined && { targetResourceId: resourceId }),\n\t\t\t};\n\t\t});\n\t}, []);\n\n\tconst endDrag = useCallback(() => {\n\t\tif (!dragState) return null;\n\n\t\tconst { event, originalStartDate, originalEndDate, previewDate, previewHour, previewMinute, targetResourceId } =\n\t\t\tdragState;\n\n\t\t// Calculate duration of original event\n\t\tconst durationMs = originalEndDate.getTime() - originalStartDate.getTime();\n\n\t\t// Calculate new start date\n\t\tlet newStartDate: Date;\n\n\t\tif (previewDate) {\n\t\t\tnewStartDate = new Date(previewDate);\n\n\t\t\tif (typeof previewHour === 'number') {\n\t\t\t\t// Time-based drop (day/week view)\n\t\t\t\tnewStartDate.setHours(previewHour, previewMinute ?? 0, 0, 0);\n\t\t\t} else {\n\t\t\t\t// Day-based drop (month view) - keep original time\n\t\t\t\tnewStartDate.setHours(\n\t\t\t\t\toriginalStartDate.getHours(),\n\t\t\t\t\toriginalStartDate.getMinutes(),\n\t\t\t\t\toriginalStartDate.getSeconds(),\n\t\t\t\t\t0\n\t\t\t\t);\n\t\t\t}\n\t\t} else {\n\t\t\t// No preview, return to original position\n\t\t\tnewStartDate = originalStartDate;\n\t\t}\n\n\t\t// Calculate new end date maintaining duration\n\t\tconst newEndDate = new Date(newStartDate.getTime() + durationMs);\n\n\t\tconst result: IDropResult<TData> = {\n\t\t\tevent,\n\t\t\tnewStartDate,\n\t\t\tnewEndDate,\n\t\t\t...(targetResourceId !== undefined && { newResourceId: targetResourceId }),\n\t\t};\n\n\t\t// Clear drag state\n\t\tsetDragState(null);\n\n\t\t// Notify parent\n\t\tonEventDrop?.(result);\n\n\t\treturn result;\n\t}, [dragState, onEventDrop]);\n\n\tconst cancelDrag = useCallback(() => {\n\t\tsetDragState(null);\n\t}, []);\n\n\treturn (\n\t\t<DragDropContext.Provider\n\t\t\tvalue={{\n\t\t\t\tdragState,\n\t\t\t\tisDragging: dragState !== null,\n\t\t\t\tstartDrag,\n\t\t\t\tupdateDragPreview,\n\t\t\t\tendDrag,\n\t\t\t\tcancelDrag,\n\t\t\t}}\n\t\t>\n\t\t\t{children}\n\t\t</DragDropContext.Provider>\n\t);\n}\n\n// ============================================================================\n// HOOKS\n// ============================================================================\n\n/**\n * Access drag & drop context (required)\n *\n * @throws Error if used outside of DragDropProvider\n */\nexport function useDragDrop<TData = Record<string, unknown>>(): IDragDropContext<TData> {\n\tconst context = useContext(DragDropContext);\n\tif (!context) {\n\t\tthrow new Error('useDragDrop must be used within a DragDropProvider');\n\t}\n\treturn context as IDragDropContext<TData>;\n}\n\n/**\n * Access drag & drop context (optional)\n *\n * Use this for components that may be rendered outside of DragDropProvider.\n * Returns null when drag-drop is not available.\n */\nexport function useOptionalDragDrop<\n\tTData = Record<string, unknown>,\n>(): IDragDropContext<TData> | null {\n\tconst context = useContext(DragDropContext);\n\treturn (context as IDragDropContext<TData>) ?? null;\n}\n","/**\n * Calendar Preferences Types\n *\n * Type definitions for the calendar preferences system.\n * This module enables:\n * - Persistent user preferences via localStorage\n * - Developer-controlled defaults and overrides\n * - Locking preferences to prevent user modifications\n * - Complete customization for different deployment scenarios\n */\n\nimport type { TBadgeVariant, TCalendarView, TSlotDuration } from '../types';\n\n// ============================================================================\n// PREFERENCE KEYS\n// ============================================================================\n\n/**\n * All available preference keys\n */\nexport type TPreferenceKey =\n\t| 'view'\n\t| 'badgeVariant'\n\t| 'slotDuration'\n\t| 'visibleHours'\n\t| 'workingHours'\n\t| 'showWorkingHoursOnly'\n\t| 'showWeekends'\n\t| 'firstDayOfWeek';\n\n/**\n * Storage key for localStorage\n */\nexport const PREFERENCES_STORAGE_KEY = 'inno-calendar-preferences';\n\n// ============================================================================\n// PREFERENCE VALUE TYPES\n// ============================================================================\n\n/**\n * Visible hours configuration\n */\nexport interface IVisibleHoursConfig {\n\tfrom: number;\n\tto: number;\n}\n\n/**\n * Working hours configuration per day of week\n * Key: 0 = Sunday, 1 = Monday, ..., 6 = Saturday\n */\nexport type TWorkingHoursConfig = {\n\t[dayIndex: number]: { from: number; to: number };\n};\n\n// ============================================================================\n// PREFERENCE VALUES\n// ============================================================================\n\n/**\n * Complete preferences object structure\n */\nexport interface IPreferences {\n\t/** Default calendar view */\n\tview: TCalendarView;\n\t/** Event badge display style */\n\tbadgeVariant: TBadgeVariant;\n\t/** Time slot duration in minutes (15, 30, 60) */\n\tslotDuration: TSlotDuration;\n\t/** Visible hours range for day/week views */\n\tvisibleHours: IVisibleHoursConfig;\n\t/** Working hours configuration per day */\n\tworkingHours: TWorkingHoursConfig;\n\t/** Show only working hours in day/week views */\n\tshowWorkingHoursOnly: boolean;\n\t/** Show weekend days */\n\tshowWeekends: boolean;\n\t/** First day of week (0 = Sunday, 1 = Monday, etc.) */\n\tfirstDayOfWeek: 0 | 1 | 2 | 3 | 4 | 5 | 6;\n}\n\n/**\n * Partial preferences for updates\n */\nexport type TPartialPreferences = Partial<IPreferences>;\n\n// ============================================================================\n// PREFERENCE CONTROL\n// ============================================================================\n\n/**\n * Control mode for each preference\n * - 'user': User can modify, persisted to localStorage\n * - 'locked': Developer-controlled, cannot be modified by user\n * - 'session': User can modify during session, not persisted\n */\nexport type TPreferenceMode = 'user' | 'locked' | 'session';\n\n/**\n * Configuration for preference control\n * Allows developers to lock certain preferences to specific values\n */\nexport type TPreferenceModes = {\n\t[K in TPreferenceKey]?: TPreferenceMode;\n};\n\n/**\n * Locked values for preferences\n * When a preference mode is 'locked', this value is used\n */\nexport type TLockedPreferences = TPartialPreferences;\n\n// ============================================================================\n// PREFERENCES CONFIG\n// ============================================================================\n\n/**\n * Complete preferences configuration for developers\n *\n * @example\n * // Allow all preferences to be user-controlled (default)\n * const config: IPreferencesConfig = {};\n *\n * @example\n * // Lock slot duration to 30 minutes, let users control the rest\n * const config: IPreferencesConfig = {\n * modes: { slotDuration: 'locked' },\n * locked: { slotDuration: 30 }\n * };\n *\n * @example\n * // Completely override all preferences (no user control)\n * const config: IPreferencesConfig = {\n * modes: {\n * view: 'locked',\n * badgeVariant: 'locked',\n * slotDuration: 'locked',\n * visibleHours: 'locked',\n * workingHours: 'locked',\n * showWorkingHoursOnly: 'locked'\n * },\n * locked: {\n * view: 'week',\n * badgeVariant: 'colored',\n * slotDuration: 30,\n * visibleHours: { from: 8, to: 18 },\n * workingHours: { ... },\n * showWorkingHoursOnly: true\n * }\n * };\n */\nexport interface IPreferencesConfig {\n\t/**\n\t * Control mode for each preference\n\t * Defaults to 'user' for all preferences\n\t */\n\tmodes?: TPreferenceModes;\n\n\t/**\n\t * Values to use when preference mode is 'locked'\n\t */\n\tlocked?: TLockedPreferences;\n\n\t/**\n\t * Custom default values (used when no stored preference exists)\n\t * These override the system defaults\n\t */\n\tdefaults?: TPartialPreferences;\n\n\t/**\n\t * Custom localStorage key (default: 'inno-calendar-preferences')\n\t * Useful when multiple calendar instances need separate storage\n\t */\n\tstorageKey?: string;\n\n\t/**\n\t * Disable localStorage entirely\n\t * All preferences will reset on page reload\n\t */\n\tdisableStorage?: boolean;\n}\n\n// ============================================================================\n// HOOK RETURN TYPE\n// ============================================================================\n\n/**\n * Return type for useCalendarPreferences hook\n */\nexport interface IUsePreferencesReturn {\n\t/** Current preference values */\n\tpreferences: IPreferences;\n\n\t/** Update a single preference */\n\tsetPreference: <K extends TPreferenceKey>(key: K, value: IPreferences[K]) => void;\n\n\t/** Update multiple preferences at once */\n\tsetPreferences: (updates: TPartialPreferences) => void;\n\n\t/** Reset preferences to defaults */\n\tresetPreferences: () => void;\n\n\t/** Reset a single preference to default */\n\tresetPreference: (key: TPreferenceKey) => void;\n\n\t/** Check if a preference is locked */\n\tisLocked: (key: TPreferenceKey) => boolean;\n\n\t/** Check if a preference is persisted */\n\tisPersisted: (key: TPreferenceKey) => boolean;\n\n\t/** Get the mode for a preference */\n\tgetMode: (key: TPreferenceKey) => TPreferenceMode;\n}\n","/**\n * Calendar Preferences Hook\n *\n * Manages calendar preferences with localStorage persistence and developer control.\n *\n * Features:\n * - Automatic localStorage persistence\n * - Developer-controlled defaults and overrides\n * - Preference locking for controlled deployments\n * - Session-only preferences that don't persist\n * - Type-safe preference access and updates\n *\n * @example Basic usage\n * ```tsx\n * const { preferences, setPreference } = useCalendarPreferences();\n * setPreference('slotDuration', 30);\n * ```\n *\n * @example With developer configuration\n * ```tsx\n * const { preferences } = useCalendarPreferences({\n * modes: { slotDuration: 'locked' },\n * locked: { slotDuration: 60 },\n * defaults: { view: 'week' }\n * });\n * ```\n */\n\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport type { TWorkingHours } from '../types';\nimport {\n\ttype IPreferences,\n\ttype IPreferencesConfig,\n\ttype IUsePreferencesReturn,\n\tPREFERENCES_STORAGE_KEY,\n\ttype TPartialPreferences,\n\ttype TPreferenceKey,\n\ttype TPreferenceMode,\n} from './types';\n\n// ============================================================================\n// SYSTEM DEFAULTS\n// ============================================================================\n\n/**\n * Default visible hours\n */\nexport const DEFAULT_VISIBLE_HOURS = {\n\tfrom: 0,\n\tto: 24,\n};\n\n/**\n * Default working hours (Monday-Friday, 8am-6pm)\n */\nexport const DEFAULT_WORKING_HOURS: TWorkingHours = {\n\t0: { enabled: false, from: 8, to: 18 }, // Sunday - disabled\n\t1: { enabled: true, from: 8, to: 18 }, // Monday\n\t2: { enabled: true, from: 8, to: 18 }, // Tuesday\n\t3: { enabled: true, from: 8, to: 18 }, // Wednesday\n\t4: { enabled: true, from: 8, to: 18 }, // Thursday\n\t5: { enabled: true, from: 8, to: 18 }, // Friday\n\t6: { enabled: false, from: 8, to: 18 }, // Saturday - disabled\n};\n\n/**\n * Default slot duration in minutes\n */\nexport const DEFAULT_SLOT_DURATION = 30;\n\n/**\n * System default preferences\n * These are used when no stored preference or custom default exists\n */\nconst SYSTEM_DEFAULTS: IPreferences = {\n\tview: 'month',\n\tbadgeVariant: 'colored',\n\tslotDuration: DEFAULT_SLOT_DURATION,\n\tvisibleHours: DEFAULT_VISIBLE_HOURS,\n\tworkingHours: DEFAULT_WORKING_HOURS,\n\tshowWorkingHoursOnly: false,\n\tshowWeekends: true,\n\tfirstDayOfWeek: 1,\n};\n\n// ============================================================================\n// STORAGE HELPERS\n// ============================================================================\n\n/**\n * Load preferences from localStorage\n */\nfunction loadFromStorage(key: string): TPartialPreferences {\n\tif (typeof window === 'undefined') return {};\n\n\ttry {\n\t\tconst stored = localStorage.getItem(key);\n\t\tif (!stored) return {};\n\n\t\tconst parsed = JSON.parse(stored);\n\n\t\t// Validate and sanitize stored values\n\t\tconst sanitized: TPartialPreferences = {};\n\n\t\tif (parsed.view && typeof parsed.view === 'string') {\n\t\t\tsanitized.view = parsed.view;\n\t\t}\n\n\t\tif (parsed.badgeVariant && typeof parsed.badgeVariant === 'string') {\n\t\t\tsanitized.badgeVariant = parsed.badgeVariant;\n\t\t}\n\n\t\tif (parsed.slotDuration && typeof parsed.slotDuration === 'number') {\n\t\t\tsanitized.slotDuration = parsed.slotDuration;\n\t\t}\n\n\t\tif (parsed.visibleHours && typeof parsed.visibleHours === 'object') {\n\t\t\tsanitized.visibleHours = parsed.visibleHours;\n\t\t}\n\n\t\tif (parsed.workingHours && typeof parsed.workingHours === 'object') {\n\t\t\tsanitized.workingHours = parsed.workingHours;\n\t\t}\n\n\t\tif (typeof parsed.showWorkingHoursOnly === 'boolean') {\n\t\t\tsanitized.showWorkingHoursOnly = parsed.showWorkingHoursOnly;\n\t\t}\n\n\t\tif (typeof parsed.showWeekends === 'boolean') {\n\t\t\tsanitized.showWeekends = parsed.showWeekends;\n\t\t}\n\n\t\tif (typeof parsed.firstDayOfWeek === 'number') {\n\t\t\tsanitized.firstDayOfWeek = parsed.firstDayOfWeek;\n\t\t}\n\n\t\treturn sanitized;\n\t} catch {\n\t\tconsole.warn('[InnoCalendar] Failed to load preferences from localStorage');\n\t\treturn {};\n\t}\n}\n\n/**\n * Save preferences to localStorage\n */\nfunction saveToStorage(key: string, preferences: TPartialPreferences): void {\n\tif (typeof window === 'undefined') return;\n\n\ttry {\n\t\tlocalStorage.setItem(key, JSON.stringify(preferences));\n\t} catch {\n\t\tconsole.warn('[InnoCalendar] Failed to save preferences to localStorage');\n\t}\n}\n\n// ============================================================================\n// HOOK\n// ============================================================================\n\n/**\n * Calendar preferences hook with localStorage persistence\n *\n * @param config - Optional configuration for preferences behavior\n * @returns Preferences state and control functions\n */\nexport function useCalendarPreferences(config: IPreferencesConfig = {}): IUsePreferencesReturn {\n\tconst {\n\t\tmodes = {},\n\t\tlocked = {},\n\t\tdefaults = {},\n\t\tstorageKey = PREFERENCES_STORAGE_KEY,\n\t\tdisableStorage = false,\n\t} = config;\n\n\t// Compute effective defaults (system + custom)\n\tconst effectiveDefaults = useMemo<IPreferences>(\n\t\t() => ({\n\t\t\t...SYSTEM_DEFAULTS,\n\t\t\t...defaults,\n\t\t}),\n\t\t[defaults]\n\t);\n\n\t// Initialize state with stored + defaults\n\tconst [storedPreferences, setStoredPreferences] = useState<TPartialPreferences>(() => {\n\t\tif (disableStorage) return {};\n\t\treturn loadFromStorage(storageKey);\n\t});\n\n\t// Session-only preferences (not persisted)\n\tconst [sessionPreferences, setSessionPreferences] = useState<TPartialPreferences>({});\n\n\t// Save to localStorage when stored preferences change\n\tuseEffect(() => {\n\t\tif (disableStorage) return;\n\t\tsaveToStorage(storageKey, storedPreferences);\n\t}, [storedPreferences, storageKey, disableStorage]);\n\n\t// Get mode for a preference\n\tconst getMode = useCallback(\n\t\t(key: TPreferenceKey): TPreferenceMode => {\n\t\t\treturn modes[key] ?? 'user';\n\t\t},\n\t\t[modes]\n\t);\n\n\t// Check if preference is locked\n\tconst isLocked = useCallback(\n\t\t(key: TPreferenceKey): boolean => {\n\t\t\treturn getMode(key) === 'locked';\n\t\t},\n\t\t[getMode]\n\t);\n\n\t// Check if preference is persisted\n\tconst isPersisted = useCallback(\n\t\t(key: TPreferenceKey): boolean => {\n\t\t\tconst mode = getMode(key);\n\t\t\treturn mode === 'user' && !disableStorage;\n\t\t},\n\t\t[getMode, disableStorage]\n\t);\n\n\t// Compute final preferences (locked > session > stored > defaults)\n\tconst preferences = useMemo<IPreferences>(() => {\n\t\tconst result = { ...effectiveDefaults };\n\n\t\t// Apply stored preferences (for 'user' mode)\n\t\tfor (const key of Object.keys(result) as TPreferenceKey[]) {\n\t\t\tconst mode = getMode(key);\n\n\t\t\tif (mode === 'locked' && key in locked) {\n\t\t\t\t// Use locked value\n\t\t\t\t(result as Record<string, unknown>)[key] = locked[key];\n\t\t\t} else if (mode === 'session' && key in sessionPreferences) {\n\t\t\t\t// Use session value\n\t\t\t\t(result as Record<string, unknown>)[key] = sessionPreferences[key];\n\t\t\t} else if (mode === 'user' && key in storedPreferences) {\n\t\t\t\t// Use stored value\n\t\t\t\t(result as Record<string, unknown>)[key] = storedPreferences[key];\n\t\t\t}\n\t\t\t// Otherwise, keep default\n\t\t}\n\n\t\treturn result;\n\t}, [effectiveDefaults, storedPreferences, sessionPreferences, locked, getMode]);\n\n\t// Set a single preference\n\tconst setPreference = useCallback(\n\t\t<K extends TPreferenceKey>(key: K, value: IPreferences[K]) => {\n\t\t\tconst mode = getMode(key);\n\n\t\t\t// Can't modify locked preferences\n\t\t\tif (mode === 'locked') {\n\t\t\t\tconsole.warn(`[InnoCalendar] Preference \"${key}\" is locked and cannot be modified`);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (mode === 'session') {\n\t\t\t\tsetSessionPreferences((prev) => ({ ...prev, [key]: value }));\n\t\t\t} else {\n\t\t\t\tsetStoredPreferences((prev) => ({ ...prev, [key]: value }));\n\t\t\t}\n\t\t},\n\t\t[getMode]\n\t);\n\n\t// Set multiple preferences at once\n\tconst setPreferences = useCallback(\n\t\t(updates: TPartialPreferences) => {\n\t\t\tconst sessionUpdates: TPartialPreferences = {};\n\t\t\tconst storedUpdates: TPartialPreferences = {};\n\n\t\t\tfor (const key of Object.keys(updates) as TPreferenceKey[]) {\n\t\t\t\tconst mode = getMode(key);\n\t\t\t\tconst value = updates[key];\n\n\t\t\t\tif (value === undefined) continue;\n\n\t\t\t\tif (mode === 'locked') {\n\t\t\t\t\tconsole.warn(`[InnoCalendar] Preference \"${key}\" is locked and cannot be modified`);\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tif (mode === 'session') {\n\t\t\t\t\t(sessionUpdates as Record<string, unknown>)[key] = value;\n\t\t\t\t} else {\n\t\t\t\t\t(storedUpdates as Record<string, unknown>)[key] = value;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (Object.keys(sessionUpdates).length > 0) {\n\t\t\t\tsetSessionPreferences((prev) => ({ ...prev, ...sessionUpdates }));\n\t\t\t}\n\n\t\t\tif (Object.keys(storedUpdates).length > 0) {\n\t\t\t\tsetStoredPreferences((prev) => ({ ...prev, ...storedUpdates }));\n\t\t\t}\n\t\t},\n\t\t[getMode]\n\t);\n\n\t// Reset all preferences to defaults\n\tconst resetPreferences = useCallback(() => {\n\t\tsetStoredPreferences({});\n\t\tsetSessionPreferences({});\n\n\t\tif (!disableStorage) {\n\t\t\tlocalStorage.removeItem(storageKey);\n\t\t}\n\t}, [storageKey, disableStorage]);\n\n\t// Reset a single preference to default\n\tconst resetPreference = useCallback(\n\t\t(key: TPreferenceKey) => {\n\t\t\tconst mode = getMode(key);\n\n\t\t\tif (mode === 'locked') {\n\t\t\t\tconsole.warn(`[InnoCalendar] Preference \"${key}\" is locked and cannot be reset`);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif (mode === 'session') {\n\t\t\t\tsetSessionPreferences((prev) => {\n\t\t\t\t\tconst { [key]: _, ...rest } = prev;\n\t\t\t\t\treturn rest;\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\tsetStoredPreferences((prev) => {\n\t\t\t\t\tconst { [key]: _, ...rest } = prev;\n\t\t\t\t\treturn rest;\n\t\t\t\t});\n\t\t\t}\n\t\t},\n\t\t[getMode]\n\t);\n\n\treturn {\n\t\tpreferences,\n\t\tsetPreference,\n\t\tsetPreferences,\n\t\tresetPreferences,\n\t\tresetPreference,\n\t\tisLocked,\n\t\tisPersisted,\n\t\tgetMode,\n\t};\n}\n\nexport default useCalendarPreferences;\n","/**\n * InnoCalendar Provider\n *\n * A CalendarProvider that matches agenda-v2's API exactly.\n * This is an all-in-one provider that manages state internally\n * rather than requiring consumers to use useCalendar separately.\n *\n * Features:\n * - Automatic preference persistence to localStorage\n * - Developer-controlled preference locking\n * - Same prop interface as agenda-v2's CalendarProvider\n * - Same context shape for component compatibility\n */\n\nimport {\n createContext,\n type Dispatch,\n type ReactNode,\n type SetStateAction,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useState,\n} from \"react\";\nimport {\n type IPreferencesConfig,\n type TPreferenceKey,\n useCalendarPreferences,\n} from \"../preferences\";\nimport type {\n CalendarEvent,\n ICalendarUser,\n IScheduleType,\n TBadgeVariant,\n TCalendarView,\n TSlotDuration,\n TWorkingHours,\n} from \"../types\";\nimport { reactNodeToText } from \"../utils/react-node-utils\";\nimport {\n readCalendarView,\n readResourceView,\n saveViewByFamily,\n} from \"../utils/view-storage\";\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\n// Re-export types from core/types for convenience\nexport type { IWorkingHoursDay, TSlotDuration, TWorkingHours } from \"../types\";\n\n/**\n * Visible hours configuration\n * Using start/end to match agenda-v2 naming\n */\nexport interface TVisibleHours {\n start: number;\n end: number;\n}\n\n// ============================================================================\n// CONTEXT INTERFACE - Matches agenda-v2 exactly\n// ============================================================================\n\nexport interface IInnoCalendarContext<TEventData = Record<string, unknown>> {\n // View & Navigation\n view: TCalendarView;\n setView: (view: TCalendarView) => void;\n selectedDate: Date;\n /** Set selected date. Pass optional forView to use that view for date range calculation. */\n setSelectedDate: (date: Date, forView?: TCalendarView) => void;\n\n // User/Participant Filtering\n selectedUserId: string | \"all\";\n setSelectedUserId: (userId: string | \"all\") => void;\n\n // Visual Customization\n badgeVariant: TBadgeVariant;\n setBadgeVariant: (variant: TBadgeVariant) => void;\n\n // Time Configuration\n workingHours: TWorkingHours;\n setWorkingHours: Dispatch<SetStateAction<TWorkingHours>>;\n visibleHours: TVisibleHours;\n setVisibleHours: Dispatch<SetStateAction<TVisibleHours>>;\n showWorkingHoursOnly: boolean;\n setShowWorkingHoursOnly: (show: boolean) => void;\n slotDuration: TSlotDuration;\n setSlotDuration: (duration: TSlotDuration) => void;\n\n // Preferences\n /**\n * Check if a specific preference is locked by the developer.\n * Useful for conditionally disabling UI controls.\n */\n isPreferenceLocked: (key: TPreferenceKey) => boolean;\n\n // Data\n events: CalendarEvent<TEventData>[];\n setEvents: Dispatch<SetStateAction<CalendarEvent<TEventData>[]>>;\n users: ICalendarUser[];\n scheduleTypes: IScheduleType[];\n\n // Filter state (schedule types)\n selectedScheduleTypeIds: number[];\n setSelectedScheduleTypeIds: Dispatch<SetStateAction<number[]>>;\n\n // Search\n searchQuery: string;\n setSearchQuery: (query: string) => void;\n\n // Computed / Filtered Events\n filteredEvents: CalendarEvent<TEventData>[];\n\n // Focus Event (auto-open popover for a specific event)\n /**\n * When set, the EventCard / EventBlock whose `event.id` matches this value\n * will automatically open its popover after rendering.\n * Cleared automatically after the popover opens.\n */\n focusEventId: string | null;\n /** Clear the focused event (called internally after popover opens). */\n clearFocusEventId: () => void;\n}\n\n// ============================================================================\n// CONTEXT\n// ============================================================================\n\n// biome-ignore lint/suspicious/noExplicitAny: Context needs flexibility for different event types\nconst InnoCalendarContext = createContext<\n IInnoCalendarContext<any> | undefined\n>(undefined);\n\n// ============================================================================\n// DEFAULT WORKING HOURS\n// ============================================================================\n\nconst DEFAULT_WORKING_HOURS: TWorkingHours = {\n 0: { enabled: false, from: 8, to: 17 }, // Sunday\n 1: { enabled: true, from: 8, to: 17 }, // Monday\n 2: { enabled: true, from: 8, to: 17 }, // Tuesday\n 3: { enabled: true, from: 8, to: 17 }, // Wednesday\n 4: { enabled: true, from: 8, to: 17 }, // Thursday\n 5: { enabled: true, from: 8, to: 17 }, // Friday\n 6: { enabled: true, from: 8, to: 12 }, // Saturday\n};\n\nconst DEFAULT_VISIBLE_HOURS: TVisibleHours = { start: 0, end: 24 };\n\n// ============================================================================\n// PROVIDER PROPS - Matches agenda-v2 exactly\n// ============================================================================\n\nexport interface InnoCalendarProviderProps<\n TEventData = Record<string, unknown>,\n> {\n children: ReactNode;\n\n // Initial data from server/loader\n initialEvents?: CalendarEvent<TEventData>[];\n initialUsers?: ICalendarUser[];\n initialScheduleTypes?: IScheduleType[];\n\n // Initial filter state (from URL)\n initialView?: TCalendarView;\n initialDate?: Date;\n initialSelectedUserId?: string | \"all\";\n initialScheduleTypeIds?: number[];\n initialParticipantIds?: string[];\n initialWorkingHoursView?: \"default\" | \"enabled\" | \"disabled\";\n initialSearchQuery?: string;\n\n /**\n * Preferences configuration for developer control\n */\n preferencesConfig?: IPreferencesConfig;\n\n /**\n * When set, the event with this ID will auto-open its popover after rendering.\n * Useful for deep-linking from other pages (e.g. clicking an appointment\n * in an overview card that navigates to the agenda with a specific event).\n */\n initialFocusEventId?: string | null;\n\n // Callbacks for parent sync\n onDateChange?: (date: Date, view: TCalendarView) => void;\n onViewChange?: (view: TCalendarView) => void;\n}\n\n// ============================================================================\n// PROVIDER COMPONENT\n// ============================================================================\n\nexport function InnoCalendarProvider<TEventData = Record<string, unknown>>({\n children,\n initialEvents = [],\n initialUsers = [],\n initialScheduleTypes = [],\n initialView,\n initialDate,\n initialSelectedUserId = \"all\",\n initialScheduleTypeIds = [],\n initialParticipantIds = [],\n initialWorkingHoursView = \"default\",\n initialSearchQuery = \"\",\n preferencesConfig,\n initialFocusEventId = null,\n onDateChange,\n onViewChange,\n}: InnoCalendarProviderProps<TEventData>) {\n // ========================================================================\n // PREFERENCES (persisted to localStorage)\n // ========================================================================\n const { preferences, setPreference, isLocked } =\n useCalendarPreferences(preferencesConfig);\n\n // ========================================================================\n // VIEW & NAVIGATION\n // ========================================================================\n const [view, setViewInternal] = useState<TCalendarView>(\n initialView ??\n (preferences.view as TCalendarView) ??\n readCalendarView() ??\n readResourceView() ??\n \"week\",\n );\n const [selectedDate, setSelectedDateInternal] = useState<Date>(\n initialDate ?? new Date(),\n );\n\n // User Filtering\n const [selectedUserId, setSelectedUserId] = useState<string | \"all\">(\n initialSelectedUserId,\n );\n\n // ========================================================================\n // PREFERENCE-BACKED STATE\n // ========================================================================\n const badgeVariant = (preferences.badgeVariant as TBadgeVariant) ?? \"colored\";\n const setBadgeVariant = useCallback(\n (variant: TBadgeVariant) => {\n if (!isLocked(\"badgeVariant\")) {\n setPreference(\"badgeVariant\", variant);\n }\n },\n [setPreference, isLocked],\n );\n\n const workingHours =\n (preferences.workingHours as TWorkingHours) ?? DEFAULT_WORKING_HOURS;\n const setWorkingHours: Dispatch<SetStateAction<TWorkingHours>> = useCallback(\n (action) => {\n if (!isLocked(\"workingHours\")) {\n const current =\n (preferences.workingHours as TWorkingHours) ?? DEFAULT_WORKING_HOURS;\n const newValue =\n typeof action === \"function\" ? action(current) : action;\n setPreference(\"workingHours\", newValue);\n }\n },\n [setPreference, isLocked, preferences.workingHours],\n );\n\n // Convert preferences format (from/to) to our format (start/end)\n const prefVisibleHours = preferences.visibleHours as\n | { from?: number; to?: number; start?: number; end?: number }\n | undefined;\n const visibleHours: TVisibleHours = useMemo(() => {\n if (!prefVisibleHours) return DEFAULT_VISIBLE_HOURS;\n // Handle both formats\n return {\n start: prefVisibleHours.start ?? prefVisibleHours.from ?? 0,\n end: prefVisibleHours.end ?? prefVisibleHours.to ?? 24,\n };\n }, [prefVisibleHours]);\n\n const setVisibleHours: Dispatch<SetStateAction<TVisibleHours>> = useCallback(\n (action) => {\n if (!isLocked(\"visibleHours\")) {\n const newValue =\n typeof action === \"function\" ? action(visibleHours) : action;\n // Store in preferences format (from/to)\n setPreference(\"visibleHours\", {\n from: newValue.start,\n to: newValue.end,\n });\n }\n },\n [setPreference, isLocked, visibleHours],\n );\n\n // showWorkingHoursOnly can be overridden by URL param\n const [showWorkingHoursOnlyState, setShowWorkingHoursOnlyState] = useState(\n initialWorkingHoursView === \"enabled\"\n ? true\n : initialWorkingHoursView === \"disabled\"\n ? false\n : ((preferences.showWorkingHoursOnly as boolean) ?? false),\n );\n const showWorkingHoursOnly = showWorkingHoursOnlyState;\n const setShowWorkingHoursOnly = useCallback(\n (show: boolean) => {\n setShowWorkingHoursOnlyState(show);\n if (!isLocked(\"showWorkingHoursOnly\")) {\n setPreference(\"showWorkingHoursOnly\", show);\n }\n },\n [setPreference, isLocked],\n );\n\n const slotDuration = (preferences.slotDuration as TSlotDuration) ?? 30;\n const setSlotDuration = useCallback(\n (duration: TSlotDuration) => {\n if (!isLocked(\"slotDuration\")) {\n setPreference(\"slotDuration\", duration);\n }\n },\n [setPreference, isLocked],\n );\n\n // ========================================================================\n // DATA\n // ========================================================================\n const [events, setEvents] =\n useState<CalendarEvent<TEventData>[]>(initialEvents);\n\n // Sync events when initialEvents changes (e.g., after navigation/refetch)\n useEffect(() => {\n setEvents(initialEvents);\n }, [initialEvents]);\n\n // Filter State\n const [selectedScheduleTypeIds, setSelectedScheduleTypeIds] = useState<\n number[]\n >(initialScheduleTypeIds);\n const [selectedParticipantIds, _setSelectedParticipantIds] = useState<\n string[]\n >(initialParticipantIds);\n\n // Search\n const [searchQuery, setSearchQuery] = useState(initialSearchQuery);\n\n // Focus Event (auto-open popover for a specific event)\n const [focusEventId, setFocusEventId] = useState<string | null>(\n initialFocusEventId ?? null,\n );\n\n // Sync focusEventId when the prop changes (e.g. URL navigation)\n useEffect(() => {\n if (initialFocusEventId) {\n setFocusEventId(initialFocusEventId);\n }\n }, [initialFocusEventId]);\n\n const clearFocusEventId = useCallback(() => setFocusEventId(null), []);\n\n // ========================================================================\n // VIEW & DATE SETTERS\n // ========================================================================\n const setView = useCallback(\n (newView: TCalendarView) => {\n setViewInternal(newView);\n if (!isLocked(\"view\")) {\n setPreference(\"view\", newView);\n }\n // Persist the last selection per family (calendar vs resource) so each\n // dropdown remembers its own most-recent value independently.\n saveViewByFamily(newView);\n onViewChange?.(newView);\n },\n [onViewChange, setPreference, isLocked],\n );\n\n const setSelectedDate = useCallback(\n (newDate: Date, forView?: TCalendarView) => {\n setSelectedDateInternal(newDate);\n onDateChange?.(newDate, forView ?? view);\n },\n [onDateChange, view],\n );\n\n // ========================================================================\n // FILTERED EVENTS\n // ========================================================================\n const filteredEvents = useMemo(() => {\n let result = events;\n\n // Filter by schedule type\n if (selectedScheduleTypeIds.length > 0) {\n result = result.filter(\n (event) =>\n event.scheduleTypeId !== undefined &&\n selectedScheduleTypeIds.includes(event.scheduleTypeId),\n );\n }\n\n // Filter by participant\n if (selectedParticipantIds.length > 0) {\n result = result.filter((event) => {\n // Check participants array (includes primary user in this model)\n if (event.participants?.length) {\n return event.participants.some((p) =>\n selectedParticipantIds.includes(p.id),\n );\n }\n return false;\n });\n }\n\n // Filter by selected user (single user view)\n if (selectedUserId !== \"all\") {\n result = result.filter((event) =>\n event.participants?.some((p) => p.id === selectedUserId),\n );\n }\n\n // Filter by search query\n if (searchQuery.trim()) {\n const query = searchQuery.toLowerCase();\n result = result.filter(\n (event) =>\n reactNodeToText(event.title).toLowerCase().includes(query) ||\n reactNodeToText(event.description).toLowerCase().includes(query) ||\n reactNodeToText(event.scheduleTypeName).toLowerCase().includes(query) ||\n event.participants?.some((p) => p.name.toLowerCase().includes(query)),\n );\n }\n\n return result;\n }, [\n events,\n selectedScheduleTypeIds,\n selectedParticipantIds,\n selectedUserId,\n searchQuery,\n ]);\n\n // ========================================================================\n // CONTEXT VALUE\n // ========================================================================\n const value = useMemo<IInnoCalendarContext<TEventData>>(\n () => ({\n // View & Navigation\n view,\n setView,\n selectedDate,\n setSelectedDate,\n\n // User Filtering\n selectedUserId,\n setSelectedUserId,\n\n // Visual Customization\n badgeVariant,\n setBadgeVariant,\n\n // Time Configuration\n workingHours,\n setWorkingHours,\n visibleHours,\n setVisibleHours,\n showWorkingHoursOnly,\n setShowWorkingHoursOnly,\n slotDuration,\n setSlotDuration,\n\n // Preferences\n isPreferenceLocked: isLocked,\n\n // Data\n events,\n setEvents,\n users: initialUsers,\n scheduleTypes: initialScheduleTypes,\n\n // Filters\n selectedScheduleTypeIds,\n setSelectedScheduleTypeIds,\n\n // Search\n searchQuery,\n setSearchQuery,\n\n // Computed\n filteredEvents,\n\n // Focus Event\n focusEventId,\n clearFocusEventId,\n }),\n [\n view,\n setView,\n selectedDate,\n setSelectedDate,\n selectedUserId,\n badgeVariant,\n setBadgeVariant,\n workingHours,\n setWorkingHours,\n visibleHours,\n setVisibleHours,\n showWorkingHoursOnly,\n setShowWorkingHoursOnly,\n slotDuration,\n setSlotDuration,\n isLocked,\n events,\n initialUsers,\n initialScheduleTypes,\n selectedScheduleTypeIds,\n searchQuery,\n filteredEvents,\n focusEventId,\n clearFocusEventId,\n ],\n );\n\n return (\n <InnoCalendarContext.Provider value={value}>\n {children}\n </InnoCalendarContext.Provider>\n );\n}\n\n// ============================================================================\n// HOOKS\n// ============================================================================\n\n/**\n * Get the full calendar context\n * This is the primary hook - use this in most components\n */\nexport function useInnoCalendar<\n TEventData = Record<string, unknown>,\n>(): IInnoCalendarContext<TEventData> {\n const context = useContext(InnoCalendarContext);\n if (!context) {\n throw new Error(\n \"useInnoCalendar must be used within an InnoCalendarProvider\",\n );\n }\n return context as IInnoCalendarContext<TEventData>;\n}\n\n/**\n * Optional calendar context hook\n */\nexport function useOptionalInnoCalendar<\n TEventData = Record<string, unknown>,\n>(): IInnoCalendarContext<TEventData> | undefined {\n return useContext(InnoCalendarContext) as\n | IInnoCalendarContext<TEventData>\n | undefined;\n}\n\n// ============================================================================\n// SELECTIVE HOOKS (for performance optimization)\n// ============================================================================\n\n/**\n * Access only view-related state\n */\nexport function useInnoCalendarView() {\n const { view, setView, selectedDate, setSelectedDate, slotDuration } =\n useInnoCalendar();\n return { view, setView, selectedDate, setSelectedDate, slotDuration };\n}\n\n/**\n * Access only events data\n */\nexport function useInnoCalendarEvents<TEventData = Record<string, unknown>>() {\n const { events, setEvents, filteredEvents } = useInnoCalendar<TEventData>();\n return { events, setEvents, filteredEvents };\n}\n\n/**\n * Access only filter state\n */\nexport function useInnoCalendarFilters() {\n const {\n selectedScheduleTypeIds,\n setSelectedScheduleTypeIds,\n selectedUserId,\n setSelectedUserId,\n searchQuery,\n setSearchQuery,\n } = useInnoCalendar();\n return {\n selectedScheduleTypeIds,\n setSelectedScheduleTypeIds,\n selectedUserId,\n setSelectedUserId,\n searchQuery,\n setSearchQuery,\n };\n}\n\n/**\n * Access only time configuration\n */\nexport function useInnoCalendarTimeConfig() {\n const {\n workingHours,\n setWorkingHours,\n visibleHours,\n setVisibleHours,\n showWorkingHoursOnly,\n setShowWorkingHoursOnly,\n } = useInnoCalendar();\n return {\n workingHours,\n setWorkingHours,\n visibleHours,\n setVisibleHours,\n showWorkingHoursOnly,\n setShowWorkingHoursOnly,\n };\n}\n","/**\n * Slot Selection Context\n *\n * Provides centralized selection state management across all calendar views.\n * This context handles:\n * - Drag-to-select functionality\n * - Selection visualization\n * - Callback integration with parent components\n *\n * Usage:\n * 1. Wrap calendar views with SlotSelectionProvider\n * 2. Use useSlotSelectionContext in child components\n * 3. Connect onSelect callback to event creation dialog\n */\n\nimport {\n\tcreateContext,\n\ttype ReactNode,\n\tuseCallback,\n\tuseContext,\n\tuseMemo,\n\tuseRef,\n\tuseState,\n} from 'react';\nimport type {\n\tISelectionResult,\n\tISlotSelectionState,\n\tITimeSlot,\n\tTOnSlotSelect,\n\tTSelectionMode,\n} from '../types';\n\n// ============================================================================\n// CONTEXT TYPES\n// ============================================================================\n\ninterface ISlotSelectionContext {\n\t/** Current selection state */\n\tisSelecting: boolean;\n\t/** Start selection at a slot */\n\tstartSelection: (slot: ITimeSlot) => void;\n\t/** Update selection end point */\n\tupdateSelection: (slot: ITimeSlot) => void;\n\t/** Complete the selection */\n\tendSelection: () => void;\n\t/** Cancel the selection */\n\tcancelSelection: () => void;\n\t/** Check if a slot is within the current selection */\n\tisSlotSelected: (slot: ITimeSlot) => boolean;\n\t/** Get the normalized selection result */\n\tgetSelectionResult: () => ISelectionResult | null;\n\t/** Current selection mode */\n\tmode: TSelectionMode;\n\t/** Temporarily suppress selection (e.g. after popover dismiss) */\n\tsuppressSelection: () => void;\n}\n\n// ============================================================================\n// CONTEXT\n// ============================================================================\n\nconst SlotSelectionContext = createContext<ISlotSelectionContext | undefined>(undefined);\n\n// ============================================================================\n// PROVIDER PROPS\n// ============================================================================\n\ninterface SlotSelectionProviderProps {\n\tchildren: ReactNode;\n\t/** Selection mode based on view */\n\tmode?: TSelectionMode;\n\t/** Callback when selection completes */\n\tonSelect?: TOnSlotSelect;\n\t/** Minimum duration in minutes (for time mode) */\n\tminDurationMinutes?: number;\n\t/** Slot duration in minutes (default: 30) */\n\tslotDurationMinutes?: number;\n\t/** Whether selection is enabled */\n\tenabled?: boolean;\n}\n\n// ============================================================================\n// HELPERS\n// ============================================================================\n\n/**\n * Convert a time slot to a Date object\n */\nfunction slotToDate(slot: ITimeSlot): Date {\n\tconst date = new Date(slot.date);\n\tif (typeof slot.hour === 'number') {\n\t\tdate.setHours(slot.hour, slot.minute ?? 0, 0, 0);\n\t}\n\treturn date;\n}\n\n/**\n * Round time to nearest 30-minute interval\n */\nfunction roundToNearest30Minutes(date: Date): Date {\n\tconst result = new Date(date);\n\tconst minutes = result.getMinutes();\n\tconst roundedMinutes = Math.floor(minutes / 30) * 30;\n\tresult.setMinutes(roundedMinutes, 0, 0);\n\treturn result;\n}\n\n/**\n * Check if two dates are the same calendar day\n */\nfunction isSameDay(date1: Date, date2: Date): boolean {\n\treturn (\n\t\tdate1.getFullYear() === date2.getFullYear() &&\n\t\tdate1.getMonth() === date2.getMonth() &&\n\t\tdate1.getDate() === date2.getDate()\n\t);\n}\n\n/**\n * Normalize selection so start is always before end\n */\nfunction normalizeSelection(\n\tstart: ITimeSlot,\n\tend: ITimeSlot,\n\tmode: TSelectionMode,\n\tminDurationMinutes: number,\n\tslotDurationMinutes = 30\n): ISelectionResult {\n\tconst startDate = slotToDate(start);\n\tconst endDate = slotToDate(end);\n\n\t// Swap if end is before start\n\tlet normalizedStart = startDate;\n\tlet normalizedEnd = endDate;\n\n\tif (endDate < startDate) {\n\t\tnormalizedStart = endDate;\n\t\tnormalizedEnd = startDate;\n\t}\n\n\t// Apply mode-specific logic\n\tif (mode === 'day') {\n\t\t// Check if single day selection (same day click)\n\t\tif (isSameDay(normalizedStart, normalizedEnd)) {\n\t\t\t// Single day: use current time rounded to nearest 30 min + default duration\n\t\t\tconst now = new Date();\n\t\t\tconst roundedNow = roundToNearest30Minutes(now);\n\n\t\t\t// Set the date portion from the selected day\n\t\t\tnormalizedStart = new Date(normalizedStart);\n\t\t\tnormalizedStart.setHours(roundedNow.getHours(), roundedNow.getMinutes(), 0, 0);\n\n\t\t\t// End time = start + minimum duration (default 30 min)\n\t\t\tnormalizedEnd = new Date(normalizedStart.getTime() + minDurationMinutes * 60 * 1000);\n\t\t} else {\n\t\t\t// Multi-day selection: full days\n\t\t\tnormalizedStart.setHours(0, 0, 0, 0);\n\t\t\tnormalizedEnd.setHours(23, 59, 59, 999);\n\t\t}\n\t} else {\n\t\t// Time mode: The end slot represents the START of that slot block,\n\t\t// so we need to add slot duration to get the actual end time\n\t\tconst slotDurationMs = slotDurationMinutes * 60 * 1000;\n\t\tnormalizedEnd = new Date(normalizedEnd.getTime() + slotDurationMs);\n\n\t\t// Ensure minimum duration\n\t\tconst durationMs = normalizedEnd.getTime() - normalizedStart.getTime();\n\t\tconst minDurationMs = minDurationMinutes * 60 * 1000;\n\n\t\tif (durationMs < minDurationMs) {\n\t\t\tnormalizedEnd = new Date(normalizedStart.getTime() + minDurationMs);\n\t\t}\n\t}\n\n\treturn {\n\t\tstartDate: normalizedStart,\n\t\tendDate: normalizedEnd,\n\t};\n}\n\n/**\n * Check if a slot is within a selection range\n */\nfunction isSlotWithinRange(\n\tslot: ITimeSlot,\n\tstart: ITimeSlot,\n\tend: ITimeSlot,\n\tmode: TSelectionMode\n): boolean {\n\tconst slotDate = slotToDate(slot);\n\tconst startDate = slotToDate(start);\n\tconst endDate = slotToDate(end);\n\n\t// Normalize the range\n\tconst rangeStart = startDate <= endDate ? startDate : endDate;\n\tconst rangeEnd = startDate <= endDate ? endDate : startDate;\n\n\tif (mode === 'day') {\n\t\t// Day mode: check if slot's date is within range (ignore time)\n\t\tconst slotDay = new Date(slotDate);\n\t\tslotDay.setHours(0, 0, 0, 0);\n\t\tconst startDay = new Date(rangeStart);\n\t\tstartDay.setHours(0, 0, 0, 0);\n\t\tconst endDay = new Date(rangeEnd);\n\t\tendDay.setHours(0, 0, 0, 0);\n\t\treturn slotDay >= startDay && slotDay <= endDay;\n\t}\n\n\t// Time mode: precise check\n\treturn slotDate >= rangeStart && slotDate <= rangeEnd;\n}\n\n// ============================================================================\n// PROVIDER\n// ============================================================================\n\nexport function SlotSelectionProvider({\n\tchildren,\n\tmode = 'time',\n\tonSelect,\n\tminDurationMinutes = 30,\n\tslotDurationMinutes = 30,\n\tenabled = true,\n}: SlotSelectionProviderProps) {\n\tconst [selection, setSelection] = useState<ISlotSelectionState | null>(null);\n\tconst isSelectingRef = useRef(false);\n\tconst isSuppressedRef = useRef(false);\n\tconst suppressTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n\t// Temporarily suppress selection (e.g. when a popover is dismissed by clicking outside)\n\tconst suppressSelection = useCallback(() => {\n\t\tisSuppressedRef.current = true;\n\t\tif (suppressTimerRef.current) clearTimeout(suppressTimerRef.current);\n\t\tsuppressTimerRef.current = setTimeout(() => {\n\t\t\tisSuppressedRef.current = false;\n\t\t}, 300);\n\t}, []);\n\n\t// Start selection\n\tconst startSelection = useCallback(\n\t\t(slot: ITimeSlot) => {\n\t\t\tif (!enabled || isSuppressedRef.current) return;\n\n\t\t\tisSelectingRef.current = true;\n\t\t\tsetSelection({\n\t\t\t\tstart: slot,\n\t\t\t\tend: slot,\n\t\t\t\tisSelecting: true,\n\t\t\t});\n\t\t},\n\t\t[enabled]\n\t);\n\n\t// Update selection end point\n\tconst updateSelection = useCallback(\n\t\t(slot: ITimeSlot) => {\n\t\t\tif (!(enabled && isSelectingRef.current)) return;\n\n\t\t\tsetSelection((prev) => {\n\t\t\t\tif (!prev) return null;\n\t\t\t\treturn {\n\t\t\t\t\t...prev,\n\t\t\t\t\tend: slot,\n\t\t\t\t};\n\t\t\t});\n\t\t},\n\t\t[enabled]\n\t);\n\n\t// Complete selection and fire callback\n\tconst endSelection = useCallback(() => {\n\t\tif (!isSelectingRef.current) return;\n\n\t\tisSelectingRef.current = false;\n\n\t\tsetSelection((prev) => {\n\t\t\tif (!prev) return null;\n\n\t\t\t// Calculate final result\n\t\t\tconst result = normalizeSelection(\n\t\t\t\tprev.start,\n\t\t\t\tprev.end,\n\t\t\t\tmode,\n\t\t\t\tminDurationMinutes,\n\t\t\t\tslotDurationMinutes\n\t\t\t);\n\n\t\t\t// Fire callback\n\t\t\tonSelect?.(result);\n\n\t\t\t// Clear selection after a brief delay for visual feedback\n\t\t\treturn {\n\t\t\t\t...prev,\n\t\t\t\tisSelecting: false,\n\t\t\t};\n\t\t});\n\n\t\t// Clear selection after callback\n\t\tsetTimeout(() => {\n\t\t\tsetSelection(null);\n\t\t}, 100);\n\t}, [mode, minDurationMinutes, slotDurationMinutes, onSelect]);\n\n\t// Cancel selection without firing callback\n\tconst cancelSelection = useCallback(() => {\n\t\tisSelectingRef.current = false;\n\t\tsetSelection(null);\n\t}, []);\n\n\t// Check if a slot is in the current selection\n\tconst isSlotInSelection = useCallback(\n\t\t(slot: ITimeSlot): boolean => {\n\t\t\tif (!selection) return false;\n\t\t\treturn isSlotWithinRange(slot, selection.start, selection.end, mode);\n\t\t},\n\t\t[selection, mode]\n\t);\n\n\t// Get the normalized selection result\n\tconst getSelectionResult = useCallback((): ISelectionResult | null => {\n\t\tif (!selection) return null;\n\t\treturn normalizeSelection(\n\t\t\tselection.start,\n\t\t\tselection.end,\n\t\t\tmode,\n\t\t\tminDurationMinutes,\n\t\t\tslotDurationMinutes\n\t\t);\n\t}, [selection, mode, minDurationMinutes, slotDurationMinutes]);\n\n\t// Memoize context value\n\tconst value = useMemo<ISlotSelectionContext>(\n\t\t() => ({\n\t\t\tisSelecting: selection?.isSelecting ?? false,\n\t\t\tstartSelection,\n\t\t\tupdateSelection,\n\t\t\tendSelection,\n\t\t\tcancelSelection,\n\t\t\tisSlotSelected: isSlotInSelection,\n\t\t\tgetSelectionResult,\n\t\t\tmode,\n\t\t\tsuppressSelection,\n\t\t}),\n\t\t[\n\t\t\tselection?.isSelecting,\n\t\t\tstartSelection,\n\t\t\tupdateSelection,\n\t\t\tendSelection,\n\t\t\tcancelSelection,\n\t\t\tisSlotInSelection,\n\t\t\tgetSelectionResult,\n\t\t\tmode,\n\t\t\tsuppressSelection,\n\t\t]\n\t);\n\n\treturn <SlotSelectionContext.Provider value={value}>{children}</SlotSelectionContext.Provider>;\n}\n\n// ============================================================================\n// HOOKS\n// ============================================================================\n\nexport function useSlotSelectionContext(): ISlotSelectionContext {\n\tconst context = useContext(SlotSelectionContext);\n\tif (!context) {\n\t\tthrow new Error('useSlotSelectionContext must be used within a SlotSelectionProvider');\n\t}\n\treturn context;\n}\n\n/**\n * Optional hook that returns undefined if not in provider\n * Useful for components that can work with or without selection\n */\nexport function useOptionalSlotSelection(): ISlotSelectionContext | undefined {\n\treturn useContext(SlotSelectionContext);\n}\n\n// ============================================================================\n// TYPE EXPORTS\n// ============================================================================\n\nexport type { ISlotSelectionContext, SlotSelectionProviderProps };\n"],"names":["DragDropContext","createContext","DragDropProvider","children","onEventDrop","dragState","setDragState","useState","startDrag","useCallback","event","startDate","endDate","updateDragPreview","date","hour","minute","resourceId","prev","endDrag","originalStartDate","originalEndDate","previewDate","previewHour","previewMinute","targetResourceId","durationMs","newStartDate","newEndDate","result","cancelDrag","jsx","useDragDrop","context","useContext","useOptionalDragDrop","PREFERENCES_STORAGE_KEY","DEFAULT_VISIBLE_HOURS","DEFAULT_WORKING_HOURS","DEFAULT_SLOT_DURATION","SYSTEM_DEFAULTS","loadFromStorage","key","stored","parsed","sanitized","saveToStorage","preferences","useCalendarPreferences","config","modes","locked","defaults","storageKey","disableStorage","effectiveDefaults","useMemo","storedPreferences","setStoredPreferences","sessionPreferences","setSessionPreferences","useEffect","getMode","isLocked","isPersisted","mode","setPreference","value","setPreferences","updates","sessionUpdates","storedUpdates","resetPreferences","resetPreference","_","rest","InnoCalendarContext","InnoCalendarProvider","initialEvents","initialUsers","initialScheduleTypes","initialView","initialDate","initialSelectedUserId","initialScheduleTypeIds","initialParticipantIds","initialWorkingHoursView","initialSearchQuery","preferencesConfig","initialFocusEventId","onDateChange","onViewChange","view","setViewInternal","readCalendarView","readResourceView","selectedDate","setSelectedDateInternal","selectedUserId","setSelectedUserId","badgeVariant","setBadgeVariant","variant","workingHours","setWorkingHours","action","current","newValue","prefVisibleHours","visibleHours","setVisibleHours","showWorkingHoursOnlyState","setShowWorkingHoursOnlyState","showWorkingHoursOnly","setShowWorkingHoursOnly","show","slotDuration","setSlotDuration","duration","events","setEvents","selectedScheduleTypeIds","setSelectedScheduleTypeIds","selectedParticipantIds","_setSelectedParticipantIds","searchQuery","setSearchQuery","focusEventId","setFocusEventId","clearFocusEventId","setView","newView","saveViewByFamily","setSelectedDate","newDate","forView","filteredEvents","p","query","reactNodeToText","useInnoCalendar","useOptionalInnoCalendar","useInnoCalendarView","useInnoCalendarEvents","useInnoCalendarFilters","useInnoCalendarTimeConfig","SlotSelectionContext","slotToDate","slot","roundToNearest30Minutes","minutes","roundedMinutes","isSameDay","date1","date2","normalizeSelection","start","end","minDurationMinutes","slotDurationMinutes","normalizedStart","normalizedEnd","roundedNow","slotDurationMs","minDurationMs","isSlotWithinRange","slotDate","rangeStart","rangeEnd","slotDay","startDay","endDay","SlotSelectionProvider","onSelect","enabled","selection","setSelection","isSelectingRef","useRef","isSuppressedRef","suppressTimerRef","suppressSelection","startSelection","updateSelection","endSelection","cancelSelection","isSlotInSelection","getSelectionResult","useSlotSelectionContext","useOptionalSlotSelection"],"mappings":"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"}
|