@tuturuuu/ui 0.8.0 → 0.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (245) hide show
  1. package/CHANGELOG.md +69 -0
  2. package/biome.json +1 -1
  3. package/package.json +74 -71
  4. package/src/components/ui/accordion.tsx +1 -1
  5. package/src/components/ui/breadcrumb.tsx +1 -1
  6. package/src/components/ui/calendar-app/calendar-page-shell.tsx +4 -0
  7. package/src/components/ui/calendar-app/components/calendar-connections-settings-content.tsx +239 -33
  8. package/src/components/ui/calendar-app/components/load-smart-scheduling-tasks.tsx +143 -0
  9. package/src/components/ui/calendar-app/components/priority-view.tsx +10 -3
  10. package/src/components/ui/calendar-app/components/tasks-sidebar.tsx +4 -116
  11. package/src/components/ui/calendar-app/components/use-calendar-connections-manager.ts +67 -2
  12. package/src/components/ui/calendar.tsx +1 -1
  13. package/src/components/ui/carousel.tsx +1 -1
  14. package/src/components/ui/chat/chat-agent-details-external-thread-panel.test.tsx +1 -1
  15. package/src/components/ui/chat/chat-agent-details-external-thread-panel.tsx +1 -1
  16. package/src/components/ui/chat/chat-agent-details-operations-panel.test.tsx +1 -1
  17. package/src/components/ui/chat/chat-agent-details-operations-panel.tsx +1 -1
  18. package/src/components/ui/chat/chat-agent-details-setup-panel.tsx +1 -1
  19. package/src/components/ui/chat/chat-agent-details-sidebar.test.tsx +1 -1
  20. package/src/components/ui/chat/chat-agent-details-sidebar.tsx +2 -2
  21. package/src/components/ui/chat/chat-agent-details-utils.test.ts +1 -1
  22. package/src/components/ui/chat/chat-agent-details-utils.tsx +1 -1
  23. package/src/components/ui/chat/chat-agent-details-zalo-personal-panel.tsx +2 -2
  24. package/src/components/ui/checkbox.tsx +1 -1
  25. package/src/components/ui/color-picker.tsx +1 -1
  26. package/src/components/ui/command.tsx +1 -1
  27. package/src/components/ui/context-menu.tsx +5 -1
  28. package/src/components/ui/custom/__tests__/settings-dialog-search.test.ts +78 -0
  29. package/src/components/ui/custom/__tests__/settings-dialog-shell-compile-graph.test.ts +76 -0
  30. package/src/components/ui/custom/__tests__/settings-dialog-shell.test.tsx +3 -0
  31. package/src/components/ui/custom/__tests__/workspace-select-helpers.test.ts +46 -1
  32. package/src/components/ui/custom/combobox.test.tsx +195 -0
  33. package/src/components/ui/custom/combobox.tsx +273 -156
  34. package/src/components/ui/custom/education/modules/youtube/delete-link-button.tsx +5 -13
  35. package/src/components/ui/custom/facebook-mockup/facebook-mockup.tsx +7 -1
  36. package/src/components/ui/custom/facebook-mockup/form.tsx +1 -1
  37. package/src/components/ui/custom/facebook-mockup/image-upload-field.tsx +1 -1
  38. package/src/components/ui/custom/facebook-mockup/preview.tsx +1 -1
  39. package/src/components/ui/custom/nav-link.test.tsx +165 -0
  40. package/src/components/ui/custom/nav-link.tsx +69 -11
  41. package/src/components/ui/custom/navigation.tsx +1 -0
  42. package/src/components/ui/custom/settings/task-settings.tsx +104 -0
  43. package/src/components/ui/custom/settings-dialog-search-loader.d.ts +5 -0
  44. package/src/components/ui/custom/settings-dialog-search-loader.js +3 -0
  45. package/src/components/ui/custom/settings-dialog-search.ts +75 -0
  46. package/src/components/ui/custom/settings-dialog-shell.tsx +65 -28
  47. package/src/components/ui/custom/theme-toggle.tsx +1 -1
  48. package/src/components/ui/custom/workspace-select-helpers.ts +23 -0
  49. package/src/components/ui/custom/workspace-select.tsx +25 -19
  50. package/src/components/ui/dialog.test.tsx +52 -0
  51. package/src/components/ui/dialog.tsx +6 -2
  52. package/src/components/ui/dropdown-menu.tsx +5 -1
  53. package/src/components/ui/finance/debts/debt-loan-form.tsx +12 -5
  54. package/src/components/ui/finance/debts/debt-loan-summary.tsx +3 -2
  55. package/src/components/ui/finance/debts/debts-page.test.tsx +54 -5
  56. package/src/components/ui/finance/debts/debts-page.tsx +15 -2
  57. package/src/components/ui/finance/invoices/components/subscription-group-selector.tsx +3 -5
  58. package/src/components/ui/finance/invoices/new-invoice-page.test.tsx +25 -5
  59. package/src/components/ui/finance/invoices/new-invoice-page.tsx +7 -2
  60. package/src/components/ui/finance/invoices/standard-invoice.tsx +4 -2
  61. package/src/components/ui/finance/invoices/subscription-invoice.tsx +4 -2
  62. package/src/components/ui/finance/invoices/utils.ts +3 -1
  63. package/src/components/ui/finance/transactions/form-content-dialog.tsx +3 -0
  64. package/src/components/ui/finance/transactions/form-types.ts +1 -0
  65. package/src/components/ui/finance/transactions/form.tsx +2 -0
  66. package/src/components/ui/finance/transactions/infinite-transactions-list.tsx +2 -0
  67. package/src/components/ui/finance/transactions/period-charts/category-breakdown-dialog.tsx +1 -1
  68. package/src/components/ui/finance/transactions/transaction-edit-dialog.tsx +1 -4
  69. package/src/components/ui/finance/transactions/transactions-create-summary.tsx +3 -0
  70. package/src/components/ui/finance/transactions/transactions-page.tsx +4 -1
  71. package/src/components/ui/finance/wallets/form.test.tsx +51 -3
  72. package/src/components/ui/finance/wallets/form.tsx +15 -4
  73. package/src/components/ui/finance/wallets/walletId/wallet-details-actions.tsx +4 -0
  74. package/src/components/ui/finance/wallets/walletId/wallet-details-page.tsx +4 -2
  75. package/src/components/ui/finance/wallets/wallets-data-table.tsx +1 -0
  76. package/src/components/ui/finance/wallets/wallets-page.tsx +5 -2
  77. package/src/components/ui/input-otp.tsx +1 -1
  78. package/src/components/ui/legacy/calendar/all-day-event-bar.tsx +28 -39
  79. package/src/components/ui/legacy/calendar/calendar-cell.tsx +2 -0
  80. package/src/components/ui/legacy/calendar/calendar-content.tsx +10 -6
  81. package/src/components/ui/legacy/calendar/calendar-header.tsx +23 -3
  82. package/src/components/ui/legacy/calendar/calendar-loading-skeleton.tsx +135 -0
  83. package/src/components/ui/legacy/calendar/calendar-matrix.tsx +175 -237
  84. package/src/components/ui/legacy/calendar/event-card.test.tsx +177 -0
  85. package/src/components/ui/legacy/calendar/event-card.tsx +220 -131
  86. package/src/components/ui/legacy/calendar/event-modal.tsx +17 -17
  87. package/src/components/ui/legacy/calendar/event-provider-display.tsx +69 -0
  88. package/src/components/ui/legacy/calendar/smart-calendar.test.tsx +86 -4
  89. package/src/components/ui/legacy/calendar/smart-calendar.tsx +32 -2
  90. package/src/components/ui/legacy/meet/create-plan-dialog.tsx +19 -10
  91. package/src/components/ui/navigation-menu.tsx +1 -1
  92. package/src/components/ui/pagination.tsx +1 -1
  93. package/src/components/ui/radio-group.tsx +1 -1
  94. package/src/components/ui/select.tsx +5 -1
  95. package/src/components/ui/sheet.tsx +1 -1
  96. package/src/components/ui/sidebar.tsx +1 -1
  97. package/src/components/ui/storefront/cart-popover.tsx +61 -0
  98. package/src/components/ui/storefront/cart-summary-parts.tsx +290 -0
  99. package/src/components/ui/storefront/cart-summary.tsx +93 -154
  100. package/src/components/ui/storefront/checkout-overlay.tsx +4 -5
  101. package/src/components/ui/storefront/listing-card.tsx +1 -1
  102. package/src/components/ui/storefront/merch-sections.tsx +70 -0
  103. package/src/components/ui/storefront/product-detail.tsx +1 -1
  104. package/src/components/ui/storefront/storefront-surface.test.tsx +106 -11
  105. package/src/components/ui/storefront/storefront-surface.tsx +101 -166
  106. package/src/components/ui/storefront/types.ts +4 -0
  107. package/src/components/ui/storefront/utils.ts +6 -0
  108. package/src/components/ui/text-editor/__tests__/extensions.test.ts +123 -0
  109. package/src/components/ui/text-editor/background-color-extension.ts +62 -0
  110. package/src/components/ui/text-editor/color-controls.tsx +284 -0
  111. package/src/components/ui/text-editor/editor.tsx +69 -14
  112. package/src/components/ui/text-editor/extensions.ts +8 -2
  113. package/src/components/ui/text-editor/highlight-extension.ts +22 -0
  114. package/src/components/ui/text-editor/tool-bar.tsx +9 -16
  115. package/src/components/ui/toast.tsx +1 -1
  116. package/src/components/ui/tu-do/boards/__tests__/board-share-dialog.test.tsx +286 -0
  117. package/src/components/ui/tu-do/boards/__tests__/task-board-form.test.tsx +12 -0
  118. package/src/components/ui/tu-do/boards/board-public-link-section.tsx +231 -0
  119. package/src/components/ui/tu-do/boards/board-share-dialog.tsx +15 -226
  120. package/src/components/ui/tu-do/boards/board-share-settings-panel.tsx +351 -0
  121. package/src/components/ui/tu-do/boards/boardId/board-column.tsx +121 -39
  122. package/src/components/ui/tu-do/boards/boardId/enhanced-task-list.tsx +7 -0
  123. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-clear-delete.ts +2 -0
  124. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-move.ts +5 -0
  125. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-updates.ts +3 -0
  126. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-operation-types.ts +3 -3
  127. package/src/components/ui/tu-do/boards/boardId/kanban/data/kanban-deadline-query.ts +50 -2
  128. package/src/components/ui/tu-do/boards/boardId/kanban/data/use-bulk-resources.ts +59 -5
  129. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/__tests__/column-reorder.test.ts +17 -0
  130. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/column-reorder.ts +4 -1
  131. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/drag-preview.tsx +20 -1
  132. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-cache.ts +38 -9
  133. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-order.ts +2 -8
  134. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-sort-key.ts +47 -0
  135. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/use-kanban-dnd.ts +81 -30
  136. package/src/components/ui/tu-do/boards/boardId/kanban/planner/__tests__/kanban-planner-island.test.tsx +380 -0
  137. package/src/components/ui/tu-do/boards/boardId/kanban/planner/kanban-planner-dialog.tsx +204 -0
  138. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-digest-panel.tsx +61 -0
  139. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-item-strip.tsx +54 -0
  140. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-plan-toolbar.tsx +251 -0
  141. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-scope-badge.tsx +27 -0
  142. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-section.tsx +58 -0
  143. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-share-dialog.tsx +238 -0
  144. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-target-controls.tsx +143 -0
  145. package/src/components/ui/tu-do/boards/boardId/kanban/planner/planner-utils.ts +65 -0
  146. package/src/components/ui/tu-do/boards/boardId/kanban/planner/use-kanban-planner-state.ts +234 -0
  147. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-columns.test.tsx +642 -5
  148. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-columns.tsx +224 -15
  149. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-deadline-panels.tsx +535 -53
  150. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-skeleton.tsx +101 -33
  151. package/src/components/ui/tu-do/boards/boardId/kanban.tsx +235 -113
  152. package/src/components/ui/tu-do/boards/boardId/task-board-server-page.test.tsx +50 -5
  153. package/src/components/ui/tu-do/boards/boardId/task-board-server-page.tsx +12 -2
  154. package/src/components/ui/tu-do/boards/boardId/task-card/measured-task-card.tsx +10 -1
  155. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-comparator.ts +3 -0
  156. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-open-options.test.ts +20 -0
  157. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-open-options.ts +10 -0
  158. package/src/components/ui/tu-do/boards/boardId/task-card/task-card.tsx +271 -36
  159. package/src/components/ui/tu-do/boards/boardId/task-filter.test.tsx +152 -0
  160. package/src/components/ui/tu-do/boards/boardId/task-filter.tsx +555 -545
  161. package/src/components/ui/tu-do/boards/boardId/task-list.tsx +22 -0
  162. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-grid.tsx +9 -0
  163. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-task-row.tsx +9 -0
  164. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-toolbar.tsx +9 -0
  165. package/src/components/ui/tu-do/boards/boardId/timeline-board.tsx +35 -3
  166. package/src/components/ui/tu-do/boards/form.tsx +1 -1
  167. package/src/components/ui/tu-do/boards/share-section.tsx +100 -0
  168. package/src/components/ui/tu-do/drafts/draft-convert-dialog.tsx +10 -12
  169. package/src/components/ui/tu-do/drafts/drafts-page.tsx +33 -16
  170. package/src/components/ui/tu-do/hooks/__tests__/useTaskLabelManagement.test.tsx +48 -0
  171. package/src/components/ui/tu-do/hooks/__tests__/useTaskProjectManagement.test.tsx +144 -0
  172. package/src/components/ui/tu-do/hooks/useTaskDialog.ts +7 -0
  173. package/src/components/ui/tu-do/hooks/useTaskLabelManagement.ts +115 -106
  174. package/src/components/ui/tu-do/hooks/useTaskProjectManagement.ts +115 -122
  175. package/src/components/ui/tu-do/initiatives/task-initiatives-client.tsx +56 -88
  176. package/src/components/ui/tu-do/my-tasks/my-tasks-content.tsx +26 -2
  177. package/src/components/ui/tu-do/my-tasks/use-my-tasks-state.ts +55 -8
  178. package/src/components/ui/tu-do/notes/note-edit-dialog.tsx +1 -4
  179. package/src/components/ui/tu-do/progress/task-progress-import-panel.tsx +60 -0
  180. package/src/components/ui/tu-do/progress/task-progress-leaderboards-panel.tsx +156 -0
  181. package/src/components/ui/tu-do/progress/task-progress-page.tsx +348 -0
  182. package/src/components/ui/tu-do/progress/task-progress-panels.tsx +301 -0
  183. package/src/components/ui/tu-do/providers/task-dialog-provider.tsx +26 -0
  184. package/src/components/ui/tu-do/shared/__tests__/assignee-select.test.tsx +81 -10
  185. package/src/components/ui/tu-do/shared/__tests__/board-client.test.tsx +141 -1
  186. package/src/components/ui/tu-do/shared/__tests__/board-header.test.tsx +377 -36
  187. package/src/components/ui/tu-do/shared/__tests__/board-switcher.test.tsx +374 -0
  188. package/src/components/ui/tu-do/shared/__tests__/board-views.test.tsx +419 -5
  189. package/src/components/ui/tu-do/shared/__tests__/task-board-loading-state.test.tsx +38 -0
  190. package/src/components/ui/tu-do/shared/__tests__/task-cache-patches.test.ts +147 -0
  191. package/src/components/ui/tu-do/shared/__tests__/task-legacy-route-recovery.test.tsx +16 -0
  192. package/src/components/ui/tu-do/shared/__tests__/use-progressive-board-loader.test.tsx +3 -0
  193. package/src/components/ui/tu-do/shared/assignee-select.tsx +77 -26
  194. package/src/components/ui/tu-do/shared/board-client.tsx +15 -10
  195. package/src/components/ui/tu-do/shared/board-config-storage.ts +7 -1
  196. package/src/components/ui/tu-do/shared/board-header.tsx +471 -975
  197. package/src/components/ui/tu-do/shared/board-layout-settings.tsx +165 -136
  198. package/src/components/ui/tu-do/shared/board-switcher.tsx +244 -220
  199. package/src/components/ui/tu-do/shared/board-user-presence-avatars.tsx +18 -12
  200. package/src/components/ui/tu-do/shared/board-views.tsx +577 -85
  201. package/src/components/ui/tu-do/shared/list-view.tsx +246 -2
  202. package/src/components/ui/tu-do/shared/recycle-bin-panel.tsx +142 -94
  203. package/src/components/ui/tu-do/shared/special-task-list-pins.ts +51 -0
  204. package/src/components/ui/tu-do/shared/task-board-loading-state.tsx +28 -0
  205. package/src/components/ui/tu-do/shared/task-cache-patches.ts +394 -0
  206. package/src/components/ui/tu-do/shared/task-dialog-manager.tsx +21 -1
  207. package/src/components/ui/tu-do/shared/task-edit-dialog/components/quick-settings-popover.tsx +5 -1
  208. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-dialog-header.tsx +25 -2
  209. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-list-selector.tsx +7 -1
  210. package/src/components/ui/tu-do/shared/task-edit-dialog/field-diff-viewer.tsx +3 -2
  211. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-data.ts +79 -10
  212. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-mutations.ts +76 -77
  213. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-relationships.test.tsx +63 -0
  214. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-relationships.ts +78 -69
  215. package/src/components/ui/tu-do/shared/task-edit-dialog/personal-overrides-section.tsx +28 -8
  216. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/dependencies-section.tsx +14 -3
  217. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/parent-section.tsx +6 -1
  218. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/related-section.tsx +6 -1
  219. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/subtasks-section.tsx +6 -1
  220. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/types/task-relationships.types.ts +8 -0
  221. package/src/components/ui/tu-do/shared/task-edit-dialog/selective-revert-panel.test.tsx +91 -0
  222. package/src/components/ui/tu-do/shared/task-edit-dialog/selective-revert-panel.tsx +123 -78
  223. package/src/components/ui/tu-do/shared/task-edit-dialog/task-activity-section.tsx +7 -1
  224. package/src/components/ui/tu-do/shared/task-edit-dialog/task-dialog-actions.tsx +8 -1
  225. package/src/components/ui/tu-do/shared/task-edit-dialog/task-properties-section.test.tsx +150 -0
  226. package/src/components/ui/tu-do/shared/task-edit-dialog/task-properties-section.tsx +61 -35
  227. package/src/components/ui/tu-do/shared/task-edit-dialog/task-relationships-properties.tsx +44 -2
  228. package/src/components/ui/tu-do/shared/task-edit-dialog/task-snapshot-dialog.tsx +8 -3
  229. package/src/components/ui/tu-do/shared/task-edit-dialog.tsx +11 -1
  230. package/src/components/ui/tu-do/shared/task-legacy-route-recovery.tsx +2 -9
  231. package/src/components/ui/tu-do/shared/task-row-actions-menu.tsx +11 -0
  232. package/src/components/ui/tu-do/shared/use-progressive-board-loader.ts +2 -0
  233. package/src/declarations.d.ts +1 -0
  234. package/src/hooks/__tests__/use-calendar-readonly.test.tsx +322 -2
  235. package/src/hooks/__tests__/use-calendar-sync.test.tsx +446 -0
  236. package/src/hooks/__tests__/useBoardPresence.test.tsx +191 -0
  237. package/src/hooks/__tests__/useBoardRealtime.test.tsx +24 -144
  238. package/src/hooks/use-calendar-sync.tsx +247 -243
  239. package/src/hooks/use-calendar.tsx +323 -138
  240. package/src/hooks/use-task-actions.ts +24 -0
  241. package/src/hooks/use-user-workspace-config.ts +75 -0
  242. package/src/hooks/use-workspace-currency.ts +8 -3
  243. package/src/hooks/useBoardPresence.ts +364 -0
  244. package/src/hooks/useBoardRealtimeEventHandler.ts +45 -90
  245. package/src/lib/workspace-actions.ts +2 -6
@@ -0,0 +1,446 @@
1
+ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
2
+ import { act, renderHook, waitFor } from '@testing-library/react';
3
+ import type { CalendarEvent } from '@tuturuuu/types/primitives/calendar-event';
4
+ import type { ReactNode } from 'react';
5
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
6
+ import { CalendarSyncProvider, useCalendarSync } from '../use-calendar-sync';
7
+
8
+ vi.mock('../../components/ui/sonner', () => ({
9
+ toast: {
10
+ error: vi.fn(),
11
+ info: vi.fn(),
12
+ },
13
+ }));
14
+
15
+ const createQueryClient = () =>
16
+ new QueryClient({
17
+ defaultOptions: {
18
+ queries: {
19
+ retry: false,
20
+ },
21
+ },
22
+ });
23
+
24
+ const createEvent = (
25
+ id: string,
26
+ overrides: Partial<CalendarEvent> = {}
27
+ ): CalendarEvent => ({
28
+ id,
29
+ title: 'Planning',
30
+ description: '',
31
+ start_at: '2026-06-22T09:00:00.000Z',
32
+ end_at: '2026-06-22T10:00:00.000Z',
33
+ color: 'BLUE',
34
+ ws_id: 'workspace-1',
35
+ ...overrides,
36
+ });
37
+
38
+ function renderCalendarSync({
39
+ externalEvents,
40
+ }: {
41
+ externalEvents?: CalendarEvent[];
42
+ } = {}) {
43
+ const queryClient = createQueryClient();
44
+
45
+ const wrapper = ({ children }: { children: ReactNode }) => (
46
+ <QueryClientProvider client={queryClient}>
47
+ <CalendarSyncProvider externalEvents={externalEvents} wsId="workspace-1">
48
+ {children}
49
+ </CalendarSyncProvider>
50
+ </QueryClientProvider>
51
+ );
52
+
53
+ return renderHook(() => useCalendarSync(), { wrapper });
54
+ }
55
+
56
+ function mockCalendarFetch(events: CalendarEvent[]) {
57
+ return vi.fn(async (input: RequestInfo | URL) => {
58
+ const url = String(input);
59
+
60
+ if (url.includes('/calendar/habit-events')) {
61
+ return {
62
+ ok: true,
63
+ json: async () => ({
64
+ completedHabitEventIds: [],
65
+ habitEventIds: [],
66
+ }),
67
+ } as Response;
68
+ }
69
+
70
+ if (url.includes('/calendar/events')) {
71
+ return {
72
+ ok: true,
73
+ json: async () => ({
74
+ count: events.length,
75
+ data: events,
76
+ }),
77
+ } as Response;
78
+ }
79
+
80
+ throw new Error(`Unexpected fetch ${url}`);
81
+ });
82
+ }
83
+
84
+ function createWeekDates(startIsoDate: string) {
85
+ return Array.from({ length: 7 }, (_, index) => {
86
+ const date = new Date(`${startIsoDate}T00:00:00.000Z`);
87
+ date.setUTCDate(date.getUTCDate() + index);
88
+ return date;
89
+ });
90
+ }
91
+
92
+ function mockCalendarFetchByWeek(
93
+ eventsByWeekStart: Record<string, CalendarEvent[]>
94
+ ) {
95
+ return vi.fn(async (input: RequestInfo | URL) => {
96
+ const url = new URL(String(input), 'http://localhost');
97
+
98
+ if (url.pathname.includes('/calendar/habit-events')) {
99
+ return {
100
+ ok: true,
101
+ json: async () => ({
102
+ completedHabitEventIds: [],
103
+ habitEventIds: [],
104
+ }),
105
+ } as Response;
106
+ }
107
+
108
+ if (url.pathname.includes('/calendar/events')) {
109
+ const startAt = url.searchParams.get('start_at');
110
+ const startAtTime = startAt ? new Date(startAt).getTime() : NaN;
111
+ const matchingEvents = Object.entries(eventsByWeekStart).find(
112
+ ([weekStart]) => {
113
+ const expectedStart = new Date(
114
+ `${weekStart}T00:00:00.000Z`
115
+ ).getTime();
116
+
117
+ return Math.abs(startAtTime - expectedStart) < 24 * 60 * 60 * 1000;
118
+ }
119
+ )?.[1];
120
+
121
+ return {
122
+ ok: true,
123
+ json: async () => ({
124
+ count: matchingEvents?.length ?? 0,
125
+ data: matchingEvents ?? [],
126
+ }),
127
+ } as Response;
128
+ }
129
+
130
+ throw new Error(`Unexpected fetch ${url}`);
131
+ });
132
+ }
133
+
134
+ describe('CalendarSyncProvider optimistic visible events', () => {
135
+ beforeEach(() => {
136
+ vi.clearAllMocks();
137
+ vi.stubGlobal('fetch', vi.fn());
138
+ });
139
+
140
+ it('keeps last successful events visible while a refetch is pending', async () => {
141
+ const initialEvent = createEvent('event-1');
142
+ let databaseCalls = 0;
143
+ let resolveRefetch:
144
+ | ((response: Pick<Response, 'json' | 'ok'>) => void)
145
+ | undefined;
146
+ const fetchMock = vi.fn(async (input: RequestInfo | URL) => {
147
+ const url = String(input);
148
+
149
+ if (url.includes('/calendar/habit-events')) {
150
+ return {
151
+ ok: true,
152
+ json: async () => ({
153
+ completedHabitEventIds: [],
154
+ habitEventIds: [],
155
+ }),
156
+ } as Response;
157
+ }
158
+
159
+ if (url.includes('/calendar/events')) {
160
+ databaseCalls += 1;
161
+
162
+ if (databaseCalls === 1) {
163
+ return {
164
+ ok: true,
165
+ json: async () => ({
166
+ count: 1,
167
+ data: [initialEvent],
168
+ }),
169
+ } as Response;
170
+ }
171
+
172
+ return new Promise((resolve) => {
173
+ resolveRefetch = resolve;
174
+ });
175
+ }
176
+
177
+ throw new Error(`Unexpected fetch ${url}`);
178
+ });
179
+ vi.stubGlobal('fetch', fetchMock);
180
+
181
+ const { result } = renderCalendarSync();
182
+
183
+ act(() => {
184
+ result.current.setDates([new Date('2026-06-22T00:00:00.000Z')]);
185
+ });
186
+
187
+ await waitFor(() => {
188
+ expect(result.current.events.map((event) => event.id)).toEqual([
189
+ 'event-1',
190
+ ]);
191
+ });
192
+
193
+ act(() => {
194
+ result.current.refresh();
195
+ });
196
+
197
+ await waitFor(() => {
198
+ expect(databaseCalls).toBe(2);
199
+ });
200
+ expect(result.current.events.map((event) => event.id)).toEqual(['event-1']);
201
+
202
+ act(() => {
203
+ resolveRefetch?.({
204
+ ok: true,
205
+ json: async () => ({
206
+ count: 1,
207
+ data: [initialEvent],
208
+ }),
209
+ });
210
+ });
211
+ });
212
+
213
+ it('keeps last successful events visible and exposes error state on fetch failure', async () => {
214
+ const initialEvent = createEvent('event-1');
215
+ const fetchMock = mockCalendarFetch([initialEvent]);
216
+ vi.stubGlobal('fetch', fetchMock);
217
+
218
+ const { result } = renderCalendarSync();
219
+
220
+ act(() => {
221
+ result.current.setDates([new Date('2026-06-22T00:00:00.000Z')]);
222
+ });
223
+
224
+ await waitFor(() => {
225
+ expect(result.current.events).toHaveLength(1);
226
+ });
227
+
228
+ fetchMock.mockImplementation(async (input: RequestInfo | URL) => {
229
+ if (String(input).includes('/calendar/habit-events')) {
230
+ return {
231
+ ok: true,
232
+ json: async () => ({
233
+ completedHabitEventIds: [],
234
+ habitEventIds: [],
235
+ }),
236
+ } as Response;
237
+ }
238
+
239
+ return {
240
+ ok: false,
241
+ json: async () => ({ error: 'Nope' }),
242
+ } as Response;
243
+ });
244
+
245
+ act(() => {
246
+ result.current.refresh();
247
+ });
248
+
249
+ await waitFor(() => {
250
+ expect(result.current.syncStatus.state).toBe('error');
251
+ });
252
+ expect(result.current.events.map((event) => event.id)).toEqual(['event-1']);
253
+ });
254
+
255
+ it('shows the active week events when navigating away and back', async () => {
256
+ const currentWeekEvent = createEvent('current-week-event', {
257
+ start_at: '2026-06-22T09:00:00.000Z',
258
+ end_at: '2026-06-22T10:00:00.000Z',
259
+ });
260
+ const nextWeekEvent = createEvent('next-week-event', {
261
+ start_at: '2026-06-29T09:00:00.000Z',
262
+ end_at: '2026-06-29T10:00:00.000Z',
263
+ });
264
+ const fetchMock = mockCalendarFetchByWeek({
265
+ '2026-06-22': [currentWeekEvent],
266
+ '2026-06-29': [nextWeekEvent],
267
+ });
268
+ vi.stubGlobal('fetch', fetchMock);
269
+
270
+ const { result } = renderCalendarSync();
271
+
272
+ act(() => {
273
+ result.current.setDates(createWeekDates('2026-06-22'));
274
+ });
275
+
276
+ await waitFor(() => {
277
+ expect(result.current.events.map((event) => event.id)).toEqual([
278
+ 'current-week-event',
279
+ ]);
280
+ });
281
+
282
+ act(() => {
283
+ result.current.setDates(createWeekDates('2026-06-29'));
284
+ });
285
+
286
+ await waitFor(() => {
287
+ expect(result.current.events.map((event) => event.id)).toEqual([
288
+ 'next-week-event',
289
+ ]);
290
+ });
291
+
292
+ act(() => {
293
+ result.current.setDates(createWeekDates('2026-06-22'));
294
+ });
295
+
296
+ await waitFor(() => {
297
+ expect(result.current.events.map((event) => event.id)).toEqual([
298
+ 'current-week-event',
299
+ ]);
300
+ });
301
+
302
+ act(() => {
303
+ result.current.setDates(createWeekDates('2026-06-29'));
304
+ });
305
+
306
+ await waitFor(() => {
307
+ expect(result.current.events.map((event) => event.id)).toEqual([
308
+ 'next-week-event',
309
+ ]);
310
+ });
311
+ });
312
+
313
+ it('applies optimistic insert, update, and delete deltas immediately', async () => {
314
+ const baseEvent = createEvent('event-1');
315
+ const { result } = renderCalendarSync({ externalEvents: [baseEvent] });
316
+
317
+ await waitFor(() => {
318
+ expect(result.current.events).toHaveLength(1);
319
+ });
320
+
321
+ act(() => {
322
+ result.current.patchVisibleEvents(
323
+ [
324
+ createEvent('optimistic-1', {
325
+ title: 'Draft',
326
+ }),
327
+ ],
328
+ { status: 'creating' }
329
+ );
330
+ });
331
+
332
+ expect(result.current.events.map((event) => event.id)).toContain(
333
+ 'optimistic-1'
334
+ );
335
+ expect(
336
+ result.current.events.find((event) => event.id === 'optimistic-1')
337
+ ?._optimisticStatus
338
+ ).toBe('creating');
339
+
340
+ act(() => {
341
+ result.current.patchVisibleEvents(
342
+ [
343
+ {
344
+ ...baseEvent,
345
+ title: 'Moved',
346
+ },
347
+ ],
348
+ { status: 'updating' }
349
+ );
350
+ });
351
+
352
+ expect(
353
+ result.current.events.find((event) => event.id === 'event-1')
354
+ ).toMatchObject({
355
+ _optimisticStatus: 'updating',
356
+ title: 'Moved',
357
+ });
358
+
359
+ act(() => {
360
+ result.current.patchVisibleEvents([], { removeIds: ['event-1'] });
361
+ });
362
+
363
+ expect(result.current.events.some((event) => event.id === 'event-1')).toBe(
364
+ false
365
+ );
366
+
367
+ act(() => {
368
+ result.current.patchVisibleEvents([baseEvent]);
369
+ });
370
+
371
+ expect(
372
+ result.current.events.find((event) => event.id === 'event-1')
373
+ ).toMatchObject({
374
+ title: 'Planning',
375
+ });
376
+ });
377
+
378
+ it('reconciles optimistic updates with later server data', async () => {
379
+ let externalEvents = [createEvent('event-1')];
380
+ const queryClient = createQueryClient();
381
+ const wrapper = ({ children }: { children: ReactNode }) => (
382
+ <QueryClientProvider client={queryClient}>
383
+ <CalendarSyncProvider
384
+ externalEvents={externalEvents}
385
+ wsId="workspace-1"
386
+ >
387
+ {children}
388
+ </CalendarSyncProvider>
389
+ </QueryClientProvider>
390
+ );
391
+ const { rerender, result } = renderHook(() => useCalendarSync(), {
392
+ wrapper,
393
+ });
394
+
395
+ await waitFor(() => {
396
+ expect(result.current.events).toHaveLength(1);
397
+ });
398
+
399
+ act(() => {
400
+ result.current.patchVisibleEvents([
401
+ createEvent('event-1', { title: 'Server-confirmed' }),
402
+ ]);
403
+ });
404
+
405
+ expect(
406
+ result.current.events.find((event) => event.id === 'event-1')?.title
407
+ ).toBe('Server-confirmed');
408
+
409
+ externalEvents = [createEvent('event-1', { title: 'Server-confirmed' })];
410
+ rerender();
411
+
412
+ await waitFor(() => {
413
+ expect(
414
+ result.current.events.find((event) => event.id === 'event-1')
415
+ ?._optimisticStatus
416
+ ).toBeUndefined();
417
+ expect(
418
+ result.current.events.find((event) => event.id === 'event-1')?.title
419
+ ).toBe('Server-confirmed');
420
+ });
421
+ });
422
+
423
+ it('keeps identical title and time events as distinct IDs without deleting either', async () => {
424
+ const duplicateA = createEvent('event-a');
425
+ const duplicateB = createEvent('event-b');
426
+ const fetchMock = mockCalendarFetch([duplicateA, duplicateB]);
427
+ vi.stubGlobal('fetch', fetchMock);
428
+
429
+ const { result } = renderCalendarSync();
430
+
431
+ act(() => {
432
+ result.current.setDates([new Date('2026-06-22T00:00:00.000Z')]);
433
+ });
434
+
435
+ await waitFor(() => {
436
+ expect(result.current.events.map((event) => event.id).sort()).toEqual([
437
+ 'event-a',
438
+ 'event-b',
439
+ ]);
440
+ });
441
+ expect(fetchMock).not.toHaveBeenCalledWith(
442
+ expect.stringContaining('/calendar/events/event-b'),
443
+ expect.objectContaining({ method: 'DELETE' })
444
+ );
445
+ });
446
+ });
@@ -0,0 +1,191 @@
1
+ /**
2
+ * @vitest-environment jsdom
3
+ */
4
+
5
+ import { act, renderHook } from '@testing-library/react';
6
+ import { getCurrentUserProfile } from '@tuturuuu/internal-api/users';
7
+ import { createClient } from '@tuturuuu/supabase/next/client';
8
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
9
+ import { useBoardPresence } from '../useBoardPresence';
10
+
11
+ type PresenceListener = () => void;
12
+
13
+ type MockChannel = {
14
+ on: ReturnType<typeof vi.fn>;
15
+ presenceState: ReturnType<typeof vi.fn>;
16
+ subscribe: ReturnType<typeof vi.fn>;
17
+ track: ReturnType<typeof vi.fn>;
18
+ untrack: ReturnType<typeof vi.fn>;
19
+ };
20
+
21
+ type MockSupabaseClient = {
22
+ auth: {
23
+ getUser: ReturnType<typeof vi.fn>;
24
+ };
25
+ channel: ReturnType<typeof vi.fn>;
26
+ removeChannel: ReturnType<typeof vi.fn>;
27
+ };
28
+
29
+ type MockCreateClientFn = {
30
+ (): MockSupabaseClient;
31
+ mockReturnValue: (value: MockSupabaseClient) => void;
32
+ };
33
+
34
+ vi.mock('@tuturuuu/supabase/next/client', () => ({
35
+ createClient: vi.fn(),
36
+ }));
37
+
38
+ vi.mock('@tuturuuu/internal-api/users', () => ({
39
+ getCurrentUserProfile: vi.fn(),
40
+ }));
41
+
42
+ vi.mock('@tuturuuu/utils/constants', () => ({
43
+ DEV_MODE: false,
44
+ }));
45
+
46
+ describe('useBoardPresence', () => {
47
+ let mockChannel: MockChannel;
48
+ let mockClient: MockSupabaseClient;
49
+ let presenceListeners: Map<string, PresenceListener>;
50
+
51
+ beforeEach(() => {
52
+ presenceListeners = new Map();
53
+
54
+ mockChannel = {
55
+ on: vi.fn(
56
+ (
57
+ type: string,
58
+ config: { event?: string },
59
+ callback: PresenceListener
60
+ ) => {
61
+ if (type === 'presence' && config.event) {
62
+ presenceListeners.set(config.event, callback);
63
+ }
64
+ return mockChannel;
65
+ }
66
+ ),
67
+ presenceState: vi.fn(() => ({})),
68
+ subscribe: vi.fn((callback?: (status: string) => void) => {
69
+ callback?.('SUBSCRIBED');
70
+ return mockChannel;
71
+ }),
72
+ track: vi.fn(() => Promise.resolve('ok')),
73
+ untrack: vi.fn(() => Promise.resolve('ok')),
74
+ };
75
+
76
+ mockClient = {
77
+ auth: {
78
+ getUser: vi.fn(() =>
79
+ Promise.resolve({
80
+ data: {
81
+ user: {
82
+ email: 'ada.auth@example.com',
83
+ id: 'user-1',
84
+ },
85
+ },
86
+ })
87
+ ),
88
+ },
89
+ channel: vi.fn(() => mockChannel),
90
+ removeChannel: vi.fn(() => Promise.resolve()),
91
+ };
92
+
93
+ (createClient as unknown as MockCreateClientFn).mockReturnValue(mockClient);
94
+ vi.mocked(getCurrentUserProfile).mockResolvedValue({
95
+ avatar_url: 'https://example.com/ada.png',
96
+ created_at: '2026-01-01T00:00:00.000Z',
97
+ default_workspace_id: 'user-1',
98
+ display_name: 'Ada Lovelace',
99
+ email: 'ada@example.com',
100
+ full_name: 'Ada Lovelace',
101
+ id: 'user-1',
102
+ new_email: null,
103
+ });
104
+ });
105
+
106
+ afterEach(() => {
107
+ vi.clearAllMocks();
108
+ });
109
+
110
+ it('tracks sanitized profile data on a private board realtime presence channel', async () => {
111
+ const { result } = renderHook(() => useBoardPresence('board-1'));
112
+
113
+ await act(async () => {
114
+ await (result.current.updateLocation(
115
+ { boardId: 'board-1', type: 'board' },
116
+ { listStatusFilter: 'active' }
117
+ ) as unknown as Promise<void>);
118
+ });
119
+
120
+ expect(mockClient.channel).toHaveBeenCalledWith('board-realtime-board-1', {
121
+ config: {
122
+ presence: {
123
+ enabled: true,
124
+ key: 'user-1',
125
+ },
126
+ private: true,
127
+ },
128
+ });
129
+ expect(mockChannel.track).toHaveBeenCalledWith(
130
+ expect.objectContaining({
131
+ away: false,
132
+ location: { boardId: 'board-1', type: 'board' },
133
+ metadata: { listStatusFilter: 'active' },
134
+ session_id: expect.any(String),
135
+ user: {
136
+ avatar_url: 'https://example.com/ada.png',
137
+ display_name: 'Ada Lovelace',
138
+ email: 'ada@example.com',
139
+ id: 'user-1',
140
+ },
141
+ })
142
+ );
143
+ });
144
+
145
+ it('does not create a channel while disabled', async () => {
146
+ const { result } = renderHook(() =>
147
+ useBoardPresence('board-1', { enabled: false })
148
+ );
149
+
150
+ await act(async () => {
151
+ await (result.current.updateLocation({
152
+ boardId: 'board-1',
153
+ type: 'board',
154
+ }) as unknown as Promise<void>);
155
+ });
156
+
157
+ expect(mockClient.channel).not.toHaveBeenCalled();
158
+ expect(mockChannel.track).not.toHaveBeenCalled();
159
+ });
160
+
161
+ it('exposes board viewers from presence sync events', async () => {
162
+ const viewerPresence = {
163
+ location: { boardId: 'board-1', type: 'board' as const },
164
+ online_at: '2026-01-01T00:00:00.000Z',
165
+ session_id: 'session-2',
166
+ user: {
167
+ avatar_url: null,
168
+ display_name: 'Guest Reviewer',
169
+ email: 'guest@example.com',
170
+ id: 'user-2',
171
+ },
172
+ };
173
+ mockChannel.presenceState.mockReturnValue({
174
+ 'user-2': [viewerPresence],
175
+ });
176
+ const { result } = renderHook(() => useBoardPresence('board-1'));
177
+
178
+ await act(async () => {
179
+ await (result.current.updateLocation({
180
+ boardId: 'board-1',
181
+ type: 'board',
182
+ }) as unknown as Promise<void>);
183
+ });
184
+
185
+ act(() => {
186
+ presenceListeners.get('sync')?.();
187
+ });
188
+
189
+ expect(result.current.getBoardViewers('board-1')).toEqual([viewerPresence]);
190
+ });
191
+ });