@tuturuuu/ui 0.0.4 → 0.1.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 (1208) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/README.md +2 -2
  3. package/biome.json +5 -0
  4. package/components.json +1 -1
  5. package/jsr.json +9 -9
  6. package/package.json +422 -92
  7. package/postcss.config.mjs +1 -1
  8. package/src/components/ui/__tests__/avatar.test.tsx +53 -0
  9. package/src/components/ui/accordion.tsx +4 -4
  10. package/src/components/ui/alert-dialog.tsx +8 -6
  11. package/src/components/ui/alert.tsx +2 -2
  12. package/src/components/ui/avatar.tsx +6 -3
  13. package/src/components/ui/badge.tsx +6 -4
  14. package/src/components/ui/breadcrumb.tsx +3 -4
  15. package/src/components/ui/button.tsx +3 -3
  16. package/src/components/ui/calendar-app/calendar-client-page.tsx +69 -0
  17. package/src/components/ui/calendar-app/calendar-page-shell.tsx +59 -0
  18. package/src/components/ui/calendar-app/components/actions-dropdown.tsx +269 -0
  19. package/src/components/ui/calendar-app/components/auto-schedule-comprehensive-dialog.tsx +220 -0
  20. package/src/components/ui/calendar-app/components/calendar-connections-manager.tsx +2 -0
  21. package/src/components/ui/calendar-app/components/calendar-connections-unified.tsx +1433 -0
  22. package/src/components/ui/calendar-app/components/calendar-connections.tsx +843 -0
  23. package/src/components/ui/calendar-app/components/calendar-header-actions.tsx +67 -0
  24. package/src/components/ui/calendar-app/components/calendar-types.ts +46 -0
  25. package/src/components/ui/calendar-app/components/connected-accounts-dialog.tsx +353 -0
  26. package/src/components/ui/calendar-app/components/e2ee-status-badge.tsx +237 -0
  27. package/src/components/ui/calendar-app/components/experimental-feature-dialog.tsx +56 -0
  28. package/src/components/ui/calendar-app/components/habits-panel.tsx +332 -0
  29. package/src/components/ui/calendar-app/components/priority-dropdown.tsx +113 -0
  30. package/src/components/ui/calendar-app/components/priority-view.tsx +939 -0
  31. package/src/components/ui/calendar-app/components/quick-calendar-toggle.tsx +1 -0
  32. package/src/components/ui/calendar-app/components/quick-task-dialog.tsx +335 -0
  33. package/src/components/ui/calendar-app/components/quick-task-timer.tsx +112 -0
  34. package/src/components/ui/calendar-app/components/require-workspace-timezone-dialog.tsx +415 -0
  35. package/src/components/ui/calendar-app/components/scheduling-dialog.tsx +660 -0
  36. package/src/components/ui/calendar-app/components/sidebar/calendar-sidebar.tsx +128 -0
  37. package/src/components/ui/calendar-app/components/sidebar/index.tsx +1 -0
  38. package/src/components/ui/calendar-app/components/smart-schedule-button.tsx +117 -0
  39. package/src/components/ui/calendar-app/components/smart-schedule-preview-dialog.tsx +845 -0
  40. package/src/components/ui/calendar-app/components/smart-schedule-preview-panel.tsx +941 -0
  41. package/src/components/ui/calendar-app/components/sync-debug-panel.tsx +708 -0
  42. package/src/components/ui/calendar-app/components/task-fetcher.ts +51 -0
  43. package/src/components/ui/calendar-app/components/task-form.tsx +132 -0
  44. package/src/components/ui/calendar-app/components/task-list-form.tsx +84 -0
  45. package/src/components/ui/calendar-app/components/task-scheduler-panel.tsx +490 -0
  46. package/src/components/ui/calendar-app/components/tasks-sidebar-content.tsx +232 -0
  47. package/src/components/ui/calendar-app/components/tasks-sidebar.tsx +143 -0
  48. package/src/components/ui/calendar-app/components/test-event-generator-button.tsx +94 -0
  49. package/src/components/ui/calendar-app/components/time-tracker/components/active-session-card.tsx +113 -0
  50. package/src/components/ui/calendar-app/components/time-tracker/components/index.ts +3 -0
  51. package/src/components/ui/calendar-app/components/time-tracker/components/session-card.tsx +153 -0
  52. package/src/components/ui/calendar-app/components/time-tracker/components/stats-cards.tsx +58 -0
  53. package/src/components/ui/calendar-app/components/time-tracker/dialogs/create-task-dialog.tsx +190 -0
  54. package/src/components/ui/calendar-app/components/time-tracker/dialogs/delete-session-dialog.tsx +52 -0
  55. package/src/components/ui/calendar-app/components/time-tracker/dialogs/edit-session-dialog.tsx +173 -0
  56. package/src/components/ui/calendar-app/components/time-tracker/dialogs/index.ts +3 -0
  57. package/src/components/ui/calendar-app/components/time-tracker/hooks/index.ts +2 -0
  58. package/src/components/ui/calendar-app/components/time-tracker/hooks/use-sessions.ts +191 -0
  59. package/src/components/ui/calendar-app/components/time-tracker/hooks/use-time-tracker.ts +408 -0
  60. package/src/components/ui/calendar-app/components/time-tracker/index.tsx +902 -0
  61. package/src/components/ui/calendar-app/components/time-tracker/tabs/history-tab.tsx +28 -0
  62. package/src/components/ui/calendar-app/components/time-tracker/tabs/index.ts +2 -0
  63. package/src/components/ui/calendar-app/components/time-tracker/tabs/recent-sessions-tab.tsx +173 -0
  64. package/src/components/ui/calendar-app/components/time-tracker/time-tracker-trigger.tsx +50 -0
  65. package/src/components/ui/calendar-app/components/time-tracker/utils/category-colors.ts +38 -0
  66. package/src/components/ui/calendar-app/components/time-tracker/utils/format-time.ts +26 -0
  67. package/src/components/ui/calendar-app/components/time-tracker/utils/index.ts +2 -0
  68. package/src/components/ui/calendar-app/components/time-tracker.tsx +2040 -0
  69. package/src/components/ui/calendar-app/hooks/index.ts +9 -0
  70. package/src/components/ui/calendar-app/hooks/use-calendar-settings.ts +130 -0
  71. package/src/components/ui/calendar-app/hooks/use-e2ee.ts +283 -0
  72. package/src/components/ui/calendar.tsx +212 -64
  73. package/src/components/ui/card.tsx +3 -3
  74. package/src/components/ui/carousel.tsx +6 -5
  75. package/src/components/ui/chart.tsx +76 -24
  76. package/src/components/ui/chat/ai-message-markdown.tsx +59 -0
  77. package/src/components/ui/chat/ai-message-parts.tsx +176 -0
  78. package/src/components/ui/chat/ai-message-render-registry.tsx +203 -0
  79. package/src/components/ui/chat/ai-message-render-utils.ts +220 -0
  80. package/src/components/ui/chat/ai-message-tool-part.tsx +433 -0
  81. package/src/components/ui/chat/ai-message-tool-utils.ts +138 -0
  82. package/src/components/ui/chat/chat-agent-details-external-thread-panel.test.tsx +134 -0
  83. package/src/components/ui/chat/chat-agent-details-external-thread-panel.tsx +204 -0
  84. package/src/components/ui/chat/chat-agent-details-operations-panel.tsx +187 -0
  85. package/src/components/ui/chat/chat-agent-details-setup-panel.tsx +297 -0
  86. package/src/components/ui/chat/chat-agent-details-sidebar.test.tsx +136 -0
  87. package/src/components/ui/chat/chat-agent-details-sidebar.tsx +327 -0
  88. package/src/components/ui/chat/chat-agent-details-utils.test.ts +161 -0
  89. package/src/components/ui/chat/chat-agent-details-utils.tsx +280 -0
  90. package/src/components/ui/chat/chat-ai-credit-source-picker.tsx +210 -0
  91. package/src/components/ui/chat/chat-ai-details-panels.tsx +421 -0
  92. package/src/components/ui/chat/chat-ai-details-sidebar.tsx +206 -0
  93. package/src/components/ui/chat/chat-ai-models.test.ts +37 -0
  94. package/src/components/ui/chat/chat-ai-models.ts +23 -0
  95. package/src/components/ui/chat/chat-shared-content-sidebar.tsx +209 -0
  96. package/src/components/ui/chat/chat-sidebar-groups.test.ts +64 -0
  97. package/src/components/ui/chat/chat-sidebar-items.tsx +195 -0
  98. package/src/components/ui/chat/chat-sidebar-panel.tsx +183 -0
  99. package/src/components/ui/chat/chat-sidebar.tsx +505 -0
  100. package/src/components/ui/chat/chat-utils.test.ts +280 -0
  101. package/src/components/ui/chat/chat-workspace-header.tsx +429 -0
  102. package/src/components/ui/chat/chat-workspace.tsx +575 -0
  103. package/src/components/ui/chat/conversation-type-selector.tsx +51 -0
  104. package/src/components/ui/chat/create-conversation-dialog.test.tsx +89 -0
  105. package/src/components/ui/chat/create-conversation-dialog.tsx +377 -0
  106. package/src/components/ui/chat/directory-user-picker.tsx +160 -0
  107. package/src/components/ui/chat/friend-request-items.tsx +124 -0
  108. package/src/components/ui/chat/friend-requests-button.tsx +243 -0
  109. package/src/components/ui/chat/friend-requests-panel.tsx +116 -0
  110. package/src/components/ui/chat/hooks-ai.ts +73 -0
  111. package/src/components/ui/chat/hooks-attachments.ts +83 -0
  112. package/src/components/ui/chat/hooks-conversations.ts +205 -0
  113. package/src/components/ui/chat/hooks-directory.ts +45 -0
  114. package/src/components/ui/chat/hooks-friends.ts +74 -0
  115. package/src/components/ui/chat/hooks-messages.test.tsx +67 -0
  116. package/src/components/ui/chat/hooks-messages.ts +707 -0
  117. package/src/components/ui/chat/hooks-realtime.ts +157 -0
  118. package/src/components/ui/chat/hooks-shared-content.ts +56 -0
  119. package/src/components/ui/chat/hooks.ts +9 -0
  120. package/src/components/ui/chat/message-attachment-button.tsx +188 -0
  121. package/src/components/ui/chat/message-attachment-preview-dialog.tsx +141 -0
  122. package/src/components/ui/chat/message-bubble.tsx +372 -0
  123. package/src/components/ui/chat/message-composer.tsx +213 -0
  124. package/src/components/ui/chat/message-links.tsx +193 -0
  125. package/src/components/ui/chat/message-list.tsx +318 -0
  126. package/src/components/ui/chat/message-markdown.tsx +112 -0
  127. package/src/components/ui/chat/message-youtube.ts +48 -0
  128. package/src/components/ui/chat/query-keys.ts +48 -0
  129. package/src/components/ui/chat/utils.ts +284 -0
  130. package/src/components/ui/checkbox.tsx +24 -8
  131. package/src/components/ui/codeblock.tsx +7 -7
  132. package/src/components/ui/color-picker.tsx +9 -8
  133. package/src/components/ui/command.tsx +19 -11
  134. package/src/components/ui/context-menu.tsx +10 -10
  135. package/src/components/ui/currency-input.tsx +464 -0
  136. package/src/components/ui/custom/__tests__/facebook-mockup.test.tsx +291 -0
  137. package/src/components/ui/custom/__tests__/report-preview-pagination.test.ts +189 -0
  138. package/src/components/ui/custom/__tests__/report-preview.test.tsx +201 -0
  139. package/src/components/ui/custom/__tests__/settings-dialog-shell.test.tsx +144 -0
  140. package/src/components/ui/custom/__tests__/tuturuuu-logo.test.ts +10 -0
  141. package/src/components/ui/custom/autosize-textarea.tsx +3 -2
  142. package/src/components/ui/custom/calendar/core.tsx +11 -3
  143. package/src/components/ui/custom/calendar/day-cell.tsx +58 -44
  144. package/src/components/ui/custom/calendar/month-header.tsx +5 -7
  145. package/src/components/ui/custom/calendar/month-view.tsx +21 -9
  146. package/src/components/ui/custom/calendar/year-calendar.tsx +51 -10
  147. package/src/components/ui/custom/calendar/year-view.tsx +9 -5
  148. package/src/components/ui/custom/combobox.tsx +357 -89
  149. package/src/components/ui/custom/common-footer.tsx +89 -39
  150. package/src/components/ui/custom/compared-date-range-picker.tsx +478 -261
  151. package/src/components/ui/custom/data-pagination.tsx +446 -0
  152. package/src/components/ui/custom/date-input.tsx +4 -7
  153. package/src/components/ui/custom/education/certificates/certificate-viewer.tsx +98 -0
  154. package/src/components/ui/custom/education/certificates/download-button-pdf.tsx +82 -0
  155. package/src/components/ui/custom/education/certificates/types.ts +25 -0
  156. package/src/components/ui/custom/education/courses/course-card-view.tsx +252 -0
  157. package/src/components/ui/custom/education/courses/course-form.tsx +248 -0
  158. package/src/components/ui/custom/education/courses/course-header.tsx +38 -0
  159. package/src/components/ui/custom/education/courses/course-pagination.tsx +62 -0
  160. package/src/components/ui/custom/education/courses/course-row-actions.tsx +130 -0
  161. package/src/components/ui/custom/education/modules/content-section.tsx +73 -0
  162. package/src/components/ui/custom/education/modules/course-module-form.tsx +234 -0
  163. package/src/components/ui/custom/education/modules/course-module-row-actions.tsx +126 -0
  164. package/src/components/ui/custom/education/modules/link-button.tsx +66 -0
  165. package/src/components/ui/custom/education/modules/module-toggle.tsx +115 -0
  166. package/src/components/ui/custom/education/modules/resources/delete-resource.tsx +46 -0
  167. package/src/components/ui/custom/education/modules/resources/file-display.tsx +84 -0
  168. package/src/components/ui/custom/education/modules/resources/file-upload-form.tsx +544 -0
  169. package/src/components/ui/custom/education/modules/resources/pdf-viewer.tsx +119 -0
  170. package/src/components/ui/custom/education/modules/youtube/delete-link-button.tsx +62 -0
  171. package/src/components/ui/custom/education/modules/youtube/embed.tsx +20 -0
  172. package/src/components/ui/custom/education/modules/youtube/form.tsx +112 -0
  173. package/src/components/ui/custom/education/shell/education-content-surface.tsx +31 -0
  174. package/src/components/ui/custom/education/shell/education-kpi-strip.tsx +48 -0
  175. package/src/components/ui/custom/education/shell/education-page-header.tsx +56 -0
  176. package/src/components/ui/custom/empty-card.tsx +2 -2
  177. package/src/components/ui/custom/facebook-mockup/defaults.ts +81 -0
  178. package/src/components/ui/custom/facebook-mockup/facebook-mockup.tsx +313 -0
  179. package/src/components/ui/custom/facebook-mockup/form.tsx +629 -0
  180. package/src/components/ui/custom/facebook-mockup/image-upload-field.tsx +111 -0
  181. package/src/components/ui/custom/facebook-mockup/preview.tsx +760 -0
  182. package/src/components/ui/custom/facebook-mockup/types.ts +44 -0
  183. package/src/components/ui/custom/feature-summary.tsx +61 -7
  184. package/src/components/ui/custom/file-uploader.tsx +18 -15
  185. package/src/components/ui/custom/get-started-button.tsx +33 -0
  186. package/src/components/ui/custom/get-started-gradient-button.tsx +23 -0
  187. package/src/components/ui/custom/gradient-headline.tsx +25 -0
  188. package/src/components/ui/custom/icon-picker/icon-options.ts +24240 -0
  189. package/src/components/ui/custom/icon-picker/icon-picker.tsx +608 -0
  190. package/src/components/ui/custom/icon-picker/index.ts +45 -0
  191. package/src/components/ui/custom/icon-picker/types.ts +133 -0
  192. package/src/components/ui/custom/input-field.tsx +3 -1
  193. package/src/components/ui/custom/language-dropdown-item.tsx +55 -0
  194. package/src/components/ui/custom/language-dropdown-wrapper.tsx +34 -0
  195. package/src/components/ui/custom/language-toggle.tsx +43 -0
  196. package/src/components/ui/custom/language-wrapper.tsx +23 -0
  197. package/src/components/ui/custom/lead-generation-preview.tsx +310 -0
  198. package/src/components/ui/custom/loading-indicator.tsx +3 -0
  199. package/src/components/ui/custom/loading.tsx +8 -0
  200. package/src/components/ui/custom/logo-title.tsx +20 -0
  201. package/src/components/ui/custom/missed-entry/image-upload-section.tsx +192 -0
  202. package/src/components/ui/custom/modifiable-dialog-trigger.tsx +13 -7
  203. package/src/components/ui/custom/month-picker.tsx +8 -8
  204. package/src/components/ui/custom/nav-link.tsx +474 -0
  205. package/src/components/ui/custom/nav.tsx +86 -0
  206. package/src/components/ui/custom/navigation.tsx +192 -0
  207. package/src/components/ui/custom/notification-popover-client.tsx +758 -0
  208. package/src/components/ui/custom/production-indicator.tsx +22 -0
  209. package/src/components/ui/custom/qr/color.tsx +36 -0
  210. package/src/components/ui/custom/qr/display.tsx +82 -0
  211. package/src/components/ui/custom/qr/formats.tsx +52 -0
  212. package/src/components/ui/custom/qr/image-upload.tsx +440 -0
  213. package/src/components/ui/custom/qr/qr.tsx +170 -0
  214. package/src/components/ui/custom/qr/styles.tsx +133 -0
  215. package/src/components/ui/custom/qr/workspace-title.tsx +47 -0
  216. package/src/components/ui/custom/report-preview-pagination.ts +579 -0
  217. package/src/components/ui/custom/report-preview.tsx +745 -126
  218. package/src/components/ui/custom/search-bar.tsx +8 -12
  219. package/src/components/ui/custom/select-field.tsx +5 -3
  220. package/src/components/ui/custom/settings/appearance-settings.tsx +129 -0
  221. package/src/components/ui/custom/settings/lunar-calendar-settings.tsx +36 -0
  222. package/src/components/ui/custom/settings/sidebar-settings.tsx +135 -0
  223. package/src/components/ui/custom/settings/task-settings.tsx +236 -0
  224. package/src/components/ui/custom/settings-dialog-shell.tsx +481 -0
  225. package/src/components/ui/custom/settings-item-tab.tsx +29 -0
  226. package/src/components/ui/custom/sidebar-context.tsx +148 -0
  227. package/src/components/ui/custom/sidebar-footer-actions.tsx +337 -0
  228. package/src/components/ui/custom/staff-toolbar.tsx +16 -0
  229. package/src/components/ui/custom/structure.tsx +213 -0
  230. package/src/components/ui/custom/system-language-dropdown-item.tsx +46 -0
  231. package/src/components/ui/custom/system-language-wrapper.tsx +15 -0
  232. package/src/components/ui/custom/tables/custom-data-table.tsx +130 -0
  233. package/src/components/ui/custom/tables/data-table-column-header.tsx +9 -7
  234. package/src/components/ui/custom/tables/data-table-create-button.tsx +2 -2
  235. package/src/components/ui/custom/tables/data-table-faceted-filter.tsx +5 -5
  236. package/src/components/ui/custom/tables/data-table-pagination.tsx +69 -91
  237. package/src/components/ui/custom/tables/data-table-refresh-button.tsx +2 -2
  238. package/src/components/ui/custom/tables/data-table-toolbar.tsx +42 -19
  239. package/src/components/ui/custom/tables/data-table-view-options.tsx +27 -15
  240. package/src/components/ui/custom/tables/data-table.tsx +150 -56
  241. package/src/components/ui/custom/tailwind-indicator.tsx +14 -0
  242. package/src/components/ui/custom/theme-dropdown-items.tsx +90 -0
  243. package/src/components/ui/custom/theme-toggle.tsx +35 -0
  244. package/src/components/ui/custom/tuturuuu-logo.tsx +16 -0
  245. package/src/components/ui/custom/uploaded-files-card.tsx +1 -0
  246. package/src/components/ui/custom/user-filters.tsx +487 -0
  247. package/src/components/ui/custom/version-badge.test.tsx +102 -0
  248. package/src/components/ui/custom/version-badge.tsx +169 -0
  249. package/src/components/ui/custom/view-toggle.tsx +63 -0
  250. package/src/components/ui/custom/workspace-access/adapters.test.ts +31 -0
  251. package/src/components/ui/custom/workspace-access/adapters.ts +169 -0
  252. package/src/components/ui/custom/workspace-access/index.ts +32 -0
  253. package/src/components/ui/custom/workspace-access/member-filter-utils.test.ts +175 -0
  254. package/src/components/ui/custom/workspace-access/member-filter-utils.ts +190 -0
  255. package/src/components/ui/custom/workspace-access/types.ts +150 -0
  256. package/src/components/ui/custom/workspace-access/workspace-access-client-pages.tsx +26 -0
  257. package/src/components/ui/custom/workspace-access/workspace-access-default-role-card.tsx +92 -0
  258. package/src/components/ui/custom/workspace-access/workspace-access-invite-dialog.tsx +102 -0
  259. package/src/components/ui/custom/workspace-access/workspace-access-labels.ts +53 -0
  260. package/src/components/ui/custom/workspace-access/workspace-access-member-row.tsx +249 -0
  261. package/src/components/ui/custom/workspace-access/workspace-access-members.tsx +84 -0
  262. package/src/components/ui/custom/workspace-access/workspace-access-page-header.tsx +72 -0
  263. package/src/components/ui/custom/workspace-access/workspace-access-page.tsx +388 -0
  264. package/src/components/ui/custom/workspace-access/workspace-access-people-filters.tsx +200 -0
  265. package/src/components/ui/custom/workspace-access/workspace-access-permission-checklist.tsx +87 -0
  266. package/src/components/ui/custom/workspace-access/workspace-access-permission-preview.tsx +35 -0
  267. package/src/components/ui/custom/workspace-access/workspace-access-role-editor-dialog.tsx +180 -0
  268. package/src/components/ui/custom/workspace-access/workspace-access-role-editor-labels.ts +33 -0
  269. package/src/components/ui/custom/workspace-access/workspace-access-roles.tsx +171 -0
  270. package/src/components/ui/custom/workspace-access/workspace-access-tabs-toolbar.tsx +71 -0
  271. package/src/components/ui/custom/workspace-select.tsx +849 -0
  272. package/src/components/ui/custom/workspace-wrapper.tsx +142 -0
  273. package/src/components/ui/date-time-picker.tsx +706 -0
  274. package/src/components/ui/dialog.tsx +23 -14
  275. package/src/components/ui/diff-viewer.tsx +921 -0
  276. package/src/components/ui/drawer.tsx +5 -5
  277. package/src/components/ui/dropdown-menu.tsx +18 -10
  278. package/src/components/ui/finance/analytics/analytics-date-controls.tsx +173 -0
  279. package/src/components/ui/finance/analytics/analytics-page.test.tsx +182 -0
  280. package/src/components/ui/finance/analytics/analytics-page.tsx +296 -0
  281. package/src/components/ui/finance/analytics/balance-trend-chart.tsx +288 -0
  282. package/src/components/ui/finance/analytics/category-spending-chart.test.tsx +75 -0
  283. package/src/components/ui/finance/analytics/category-spending-chart.tsx +213 -0
  284. package/src/components/ui/finance/analytics/income-expense-chart-utils.ts +52 -0
  285. package/src/components/ui/finance/analytics/income-expense-chart.tsx +242 -0
  286. package/src/components/ui/finance/analytics/income-expense-view-mode-control.tsx +37 -0
  287. package/src/components/ui/finance/analytics/spending-trends-chart.tsx +156 -0
  288. package/src/components/ui/finance/budgets/budget-alerts.test.tsx +133 -0
  289. package/src/components/ui/finance/budgets/budget-alerts.tsx +111 -0
  290. package/src/components/ui/finance/budgets/budget-card.test.tsx +72 -0
  291. package/src/components/ui/finance/budgets/budget-card.tsx +233 -0
  292. package/src/components/ui/finance/budgets/budgets-page.test.tsx +56 -0
  293. package/src/components/ui/finance/budgets/budgets-page.tsx +179 -0
  294. package/src/components/ui/finance/budgets/form-fields.tsx +304 -0
  295. package/src/components/ui/finance/budgets/form-schema.ts +39 -0
  296. package/src/components/ui/finance/budgets/form.tsx +141 -0
  297. package/src/components/ui/finance/categories-tags-tabs.tsx +80 -0
  298. package/src/components/ui/finance/debts/debt-loan-card.test.tsx +75 -0
  299. package/src/components/ui/finance/debts/debt-loan-card.tsx +212 -0
  300. package/src/components/ui/finance/debts/debt-loan-detail-cards.test.tsx +89 -0
  301. package/src/components/ui/finance/debts/debt-loan-detail-cards.tsx +207 -0
  302. package/src/components/ui/finance/debts/debt-loan-detail-page.tsx +301 -0
  303. package/src/components/ui/finance/debts/debt-loan-form-schema.ts +30 -0
  304. package/src/components/ui/finance/debts/debt-loan-form.tsx +394 -0
  305. package/src/components/ui/finance/debts/debt-loan-list.tsx +58 -0
  306. package/src/components/ui/finance/debts/debt-loan-summary.tsx +94 -0
  307. package/src/components/ui/finance/debts/debts-page.test.tsx +97 -0
  308. package/src/components/ui/finance/debts/debts-page.tsx +230 -0
  309. package/src/components/ui/finance/debts/index.ts +6 -0
  310. package/src/components/ui/finance/debts/query-invalidation.test.ts +62 -0
  311. package/src/components/ui/finance/debts/query-invalidation.ts +44 -0
  312. package/src/components/ui/finance/finance-layout.tsx +98 -0
  313. package/src/components/ui/finance/finance-overview-metrics.tsx +120 -0
  314. package/src/components/ui/finance/finance-page.tsx +169 -0
  315. package/src/components/ui/finance/finance-route-context.tsx +50 -0
  316. package/src/components/ui/finance/invoices/attendance-calendar.tsx +247 -0
  317. package/src/components/ui/finance/invoices/charts/invoice-totals-chart.test.tsx +67 -0
  318. package/src/components/ui/finance/invoices/charts/invoice-totals-chart.tsx +868 -0
  319. package/src/components/ui/finance/invoices/columns.test.tsx +66 -0
  320. package/src/components/ui/finance/invoices/columns.tsx +365 -0
  321. package/src/components/ui/finance/invoices/components/invoice-blocked-state.tsx +30 -0
  322. package/src/components/ui/finance/invoices/components/invoice-checkout-summary.tsx +120 -0
  323. package/src/components/ui/finance/invoices/components/invoice-content-editor.tsx +71 -0
  324. package/src/components/ui/finance/invoices/components/invoice-customer-select-card.tsx +342 -0
  325. package/src/components/ui/finance/invoices/components/invoice-payment-settings.tsx +216 -0
  326. package/src/components/ui/finance/invoices/components/invoice-user-history-accordion.tsx +219 -0
  327. package/src/components/ui/finance/invoices/components/subscription-attendance-summary.tsx +272 -0
  328. package/src/components/ui/finance/invoices/components/subscription-group-selector.tsx +346 -0
  329. package/src/components/ui/finance/invoices/create-promotion-dialog.tsx +67 -0
  330. package/src/components/ui/finance/invoices/export-dialog-content.tsx +492 -0
  331. package/src/components/ui/finance/invoices/hooks/use-best-promotion-selection.ts +119 -0
  332. package/src/components/ui/finance/invoices/hooks/use-invoice-analytics.ts +79 -0
  333. package/src/components/ui/finance/invoices/hooks/use-invoice-rounding.ts +33 -0
  334. package/src/components/ui/finance/invoices/hooks/use-invoice-subtotal.ts +13 -0
  335. package/src/components/ui/finance/invoices/hooks/use-subscription-auto-selection.ts +412 -0
  336. package/src/components/ui/finance/invoices/hooks/use-subscription-invoice-content.ts +191 -0
  337. package/src/components/ui/finance/invoices/hooks.ts +878 -0
  338. package/src/components/ui/finance/invoices/internal-api.ts +466 -0
  339. package/src/components/ui/finance/invoices/invoice-analytics.tsx +167 -0
  340. package/src/components/ui/finance/invoices/invoice-page.tsx +468 -0
  341. package/src/components/ui/finance/invoices/invoice-visibility-format.test.ts +90 -0
  342. package/src/components/ui/finance/invoices/invoice-visibility-format.ts +67 -0
  343. package/src/components/ui/finance/invoices/invoiceId/compact-invoice-template.tsx +172 -0
  344. package/src/components/ui/finance/invoices/invoiceId/full-invoice-template.tsx +349 -0
  345. package/src/components/ui/finance/invoices/invoiceId/invoice-card.tsx +360 -0
  346. package/src/components/ui/finance/invoices/invoiceId/invoice-details-page.test.tsx +227 -0
  347. package/src/components/ui/finance/invoices/invoiceId/invoice-details-page.tsx +409 -0
  348. package/src/components/ui/finance/invoices/invoiceId/invoice-edit-form.tsx +161 -0
  349. package/src/components/ui/finance/invoices/invoiceId/product-card.tsx +90 -0
  350. package/src/components/ui/finance/invoices/invoiceId/promotion-card.tsx +76 -0
  351. package/src/components/ui/finance/invoices/invoices-table.tsx +284 -0
  352. package/src/components/ui/finance/invoices/new-invoice-page.tsx +215 -0
  353. package/src/components/ui/finance/invoices/pending-columns.tsx +308 -0
  354. package/src/components/ui/finance/invoices/pending-invoices-tab.tsx +43 -0
  355. package/src/components/ui/finance/invoices/pending-invoices-table.tsx +258 -0
  356. package/src/components/ui/finance/invoices/product-selection.test.tsx +78 -0
  357. package/src/components/ui/finance/invoices/product-selection.tsx +345 -0
  358. package/src/components/ui/finance/invoices/promotion-form.tsx +352 -0
  359. package/src/components/ui/finance/invoices/query-invalidation.ts +65 -0
  360. package/src/components/ui/finance/invoices/row-actions.tsx +148 -0
  361. package/src/components/ui/finance/invoices/standard-invoice.tsx +595 -0
  362. package/src/components/ui/finance/invoices/subscription-invoice.tsx +1009 -0
  363. package/src/components/ui/finance/invoices/types.ts +93 -0
  364. package/src/components/ui/finance/invoices/user-filter-wrapper.tsx +59 -0
  365. package/src/components/ui/finance/invoices/utils.test.ts +165 -0
  366. package/src/components/ui/finance/invoices/utils.ts +484 -0
  367. package/src/components/ui/finance/invoices/wallet-filter-wrapper.tsx +47 -0
  368. package/src/components/ui/finance/recurring/form-fields.tsx +142 -0
  369. package/src/components/ui/finance/recurring/form-schedule-fields.tsx +112 -0
  370. package/src/components/ui/finance/recurring/form-schema.ts +32 -0
  371. package/src/components/ui/finance/recurring/form.tsx +130 -0
  372. package/src/components/ui/finance/recurring/recurring-sections.tsx +141 -0
  373. package/src/components/ui/finance/recurring/recurring-transaction-card.tsx +152 -0
  374. package/src/components/ui/finance/recurring/recurring-transactions-page.test.tsx +87 -0
  375. package/src/components/ui/finance/recurring/recurring-transactions-page.tsx +154 -0
  376. package/src/components/ui/finance/recurring/upcoming-transaction-card.tsx +84 -0
  377. package/src/components/ui/finance/shared/charts/category-breakdown-chart-body.tsx +118 -0
  378. package/src/components/ui/finance/shared/charts/category-breakdown-chart-controls.tsx +146 -0
  379. package/src/components/ui/finance/shared/charts/category-breakdown-chart-legend.tsx +49 -0
  380. package/src/components/ui/finance/shared/charts/category-breakdown-chart-tooltip.tsx +109 -0
  381. package/src/components/ui/finance/shared/charts/category-breakdown-chart-types.ts +24 -0
  382. package/src/components/ui/finance/shared/charts/category-breakdown-chart-utils.ts +207 -0
  383. package/src/components/ui/finance/shared/charts/category-breakdown-chart.tsx +263 -0
  384. package/src/components/ui/finance/shared/charts/daily-total-chart-client.tsx +384 -0
  385. package/src/components/ui/finance/shared/charts/daily-total-chart.tsx +299 -0
  386. package/src/components/ui/finance/shared/charts/monthly-total-chart-client.tsx +391 -0
  387. package/src/components/ui/finance/shared/charts/monthly-total-chart.tsx +296 -0
  388. package/src/components/ui/finance/shared/charts/use-finance-confidential-visibility.ts +5 -0
  389. package/src/components/ui/finance/shared/confidential-toggle.tsx +78 -0
  390. package/src/components/ui/finance/shared/create-dialog-feature-summary.tsx +23 -0
  391. package/src/components/ui/finance/shared/dashboard-header.tsx +19 -0
  392. package/src/components/ui/finance/shared/date-picker.tsx +74 -0
  393. package/src/components/ui/finance/shared/date-range-filter-wrapper.tsx +86 -0
  394. package/src/components/ui/finance/shared/date-range-picker.tsx +42 -0
  395. package/src/components/ui/finance/shared/empty-state.tsx +41 -0
  396. package/src/components/ui/finance/shared/filter.tsx +267 -0
  397. package/src/components/ui/finance/shared/finance-display-amount.tsx +25 -0
  398. package/src/components/ui/finance/shared/loaders/statistics.tsx +18 -0
  399. package/src/components/ui/finance/shared/loaders/table-skeleton.tsx +5 -0
  400. package/src/components/ui/finance/shared/metrics.tsx +69 -0
  401. package/src/components/ui/finance/shared/month-picker.tsx +177 -0
  402. package/src/components/ui/finance/shared/month-range-picker.tsx +50 -0
  403. package/src/components/ui/finance/shared/next-charts.tsx +77 -0
  404. package/src/components/ui/finance/shared/numbers-visibility-toggle.test.tsx +38 -0
  405. package/src/components/ui/finance/shared/numbers-visibility-toggle.tsx +43 -0
  406. package/src/components/ui/finance/shared/quick-actions.tsx +143 -0
  407. package/src/components/ui/finance/shared/use-finance-confidential-visibility.ts +68 -0
  408. package/src/components/ui/finance/shared/year-picker.tsx +178 -0
  409. package/src/components/ui/finance/shared/year-range-picker.tsx +50 -0
  410. package/src/components/ui/finance/statistics/card.tsx +195 -0
  411. package/src/components/ui/finance/statistics/expense.tsx +63 -0
  412. package/src/components/ui/finance/statistics/income.tsx +63 -0
  413. package/src/components/ui/finance/statistics/invoices.tsx +77 -0
  414. package/src/components/ui/finance/statistics/monthly-expense.tsx +63 -0
  415. package/src/components/ui/finance/statistics/monthly-income.tsx +63 -0
  416. package/src/components/ui/finance/statistics/total-balance.test.tsx +79 -0
  417. package/src/components/ui/finance/statistics/total-balance.tsx +64 -0
  418. package/src/components/ui/finance/statistics/transaction-categories.tsx +59 -0
  419. package/src/components/ui/finance/statistics/transactions.tsx +90 -0
  420. package/src/components/ui/finance/statistics/wallets.tsx +55 -0
  421. package/src/components/ui/finance/tags/tag-badge.tsx +36 -0
  422. package/src/components/ui/finance/tags/tag-manager.test.tsx +128 -0
  423. package/src/components/ui/finance/tags/tag-manager.tsx +646 -0
  424. package/src/components/ui/finance/transactions/TRANSACTIONS_UI_REVAMP.md +328 -0
  425. package/src/components/ui/finance/transactions/UI_IMPROVEMENTS.md +444 -0
  426. package/src/components/ui/finance/transactions/__tests__/export-utils.test.ts +167 -0
  427. package/src/components/ui/finance/transactions/categories/amount-filter-wrapper.tsx +55 -0
  428. package/src/components/ui/finance/transactions/categories/amount-filter.tsx +135 -0
  429. package/src/components/ui/finance/transactions/categories/categories-data-table.tsx +274 -0
  430. package/src/components/ui/finance/transactions/categories/columns.test.tsx +113 -0
  431. package/src/components/ui/finance/transactions/categories/columns.tsx +253 -0
  432. package/src/components/ui/finance/transactions/categories/form.tsx +388 -0
  433. package/src/components/ui/finance/transactions/categories/hooks.test.tsx +72 -0
  434. package/src/components/ui/finance/transactions/categories/hooks.ts +95 -0
  435. package/src/components/ui/finance/transactions/categories/row-actions.tsx +132 -0
  436. package/src/components/ui/finance/transactions/categories/transactions-categories-page.tsx +96 -0
  437. package/src/components/ui/finance/transactions/categories/type-filter-wrapper.tsx +39 -0
  438. package/src/components/ui/finance/transactions/categories/type-filter.tsx +148 -0
  439. package/src/components/ui/finance/transactions/category-filter-wrapper.tsx +41 -0
  440. package/src/components/ui/finance/transactions/category-filter.tsx +252 -0
  441. package/src/components/ui/finance/transactions/columns.test.tsx +119 -0
  442. package/src/components/ui/finance/transactions/columns.tsx +313 -0
  443. package/src/components/ui/finance/transactions/confidential-field.test.tsx +32 -0
  444. package/src/components/ui/finance/transactions/confidential-field.tsx +289 -0
  445. package/src/components/ui/finance/transactions/export-dialog-content.tsx +393 -0
  446. package/src/components/ui/finance/transactions/export-utils.ts +99 -0
  447. package/src/components/ui/finance/transactions/form-basic-tab.tsx +360 -0
  448. package/src/components/ui/finance/transactions/form-content-dialog.tsx +163 -0
  449. package/src/components/ui/finance/transactions/form-more-tab.tsx +203 -0
  450. package/src/components/ui/finance/transactions/form-schema.ts +52 -0
  451. package/src/components/ui/finance/transactions/form-types.ts +39 -0
  452. package/src/components/ui/finance/transactions/form-utils.tsx +60 -0
  453. package/src/components/ui/finance/transactions/form.test.tsx +151 -0
  454. package/src/components/ui/finance/transactions/form.tsx +942 -0
  455. package/src/components/ui/finance/transactions/hooks/use-filter-reset.ts +27 -0
  456. package/src/components/ui/finance/transactions/infinite-transactions-list.tsx +1127 -0
  457. package/src/components/ui/finance/transactions/internal-api.ts +148 -0
  458. package/src/components/ui/finance/transactions/money-lover-import-dialog-utils.test.ts +16 -0
  459. package/src/components/ui/finance/transactions/money-lover-import-dialog-utils.ts +17 -0
  460. package/src/components/ui/finance/transactions/money-lover-import-dialog.tsx +908 -0
  461. package/src/components/ui/finance/transactions/period-charts/activity-distribution-chart.tsx +315 -0
  462. package/src/components/ui/finance/transactions/period-charts/category-breakdown-dialog.tsx +820 -0
  463. package/src/components/ui/finance/transactions/period-charts/category-donut-chart.tsx +609 -0
  464. package/src/components/ui/finance/transactions/period-charts/index.ts +4 -0
  465. package/src/components/ui/finance/transactions/period-charts/period-breakdown-panel.tsx +492 -0
  466. package/src/components/ui/finance/transactions/query-invalidation.test.ts +91 -0
  467. package/src/components/ui/finance/transactions/query-invalidation.ts +116 -0
  468. package/src/components/ui/finance/transactions/row-actions.tsx +145 -0
  469. package/src/components/ui/finance/transactions/tag-filter-wrapper.tsx +40 -0
  470. package/src/components/ui/finance/transactions/tag-filter.test.tsx +49 -0
  471. package/src/components/ui/finance/transactions/tag-filter.tsx +171 -0
  472. package/src/components/ui/finance/transactions/transaction-attachments-field.tsx +787 -0
  473. package/src/components/ui/finance/transactions/transaction-card.tsx +576 -0
  474. package/src/components/ui/finance/transactions/transaction-edit-dialog.test.tsx +101 -0
  475. package/src/components/ui/finance/transactions/transaction-edit-dialog.tsx +947 -0
  476. package/src/components/ui/finance/transactions/transaction-statistics.test.tsx +58 -0
  477. package/src/components/ui/finance/transactions/transaction-statistics.tsx +280 -0
  478. package/src/components/ui/finance/transactions/transaction-type-filter-wrapper.tsx +40 -0
  479. package/src/components/ui/finance/transactions/transaction-type-filter.tsx +95 -0
  480. package/src/components/ui/finance/transactions/transactionId/bill.tsx +307 -0
  481. package/src/components/ui/finance/transactions/transactionId/objects.tsx +149 -0
  482. package/src/components/ui/finance/transactions/transactionId/row-actions.tsx +263 -0
  483. package/src/components/ui/finance/transactions/transactionId/transaction-details-client-page.tsx +334 -0
  484. package/src/components/ui/finance/transactions/transactionId/transaction-details-page.tsx +76 -0
  485. package/src/components/ui/finance/transactions/transactions-create-summary.tsx +56 -0
  486. package/src/components/ui/finance/transactions/transactions-infinite-page.tsx +213 -0
  487. package/src/components/ui/finance/transactions/transactions-page.tsx +116 -0
  488. package/src/components/ui/finance/transactions/transfer-fields.tsx +194 -0
  489. package/src/components/ui/finance/transactions/user-filter-wrapper.tsx +42 -0
  490. package/src/components/ui/finance/transactions/user-filter.test.tsx +66 -0
  491. package/src/components/ui/finance/transactions/user-filter.tsx +282 -0
  492. package/src/components/ui/finance/transactions/view-mode-toggle.tsx +124 -0
  493. package/src/components/ui/finance/transactions/wallet-filter-wrapper.tsx +41 -0
  494. package/src/components/ui/finance/transactions/wallet-filter.test.tsx +52 -0
  495. package/src/components/ui/finance/transactions/wallet-filter.tsx +204 -0
  496. package/src/components/ui/finance/wallets/columns.tsx +324 -0
  497. package/src/components/ui/finance/wallets/form.test.tsx +78 -0
  498. package/src/components/ui/finance/wallets/form.tsx +491 -0
  499. package/src/components/ui/finance/wallets/query-invalidation.ts +58 -0
  500. package/src/components/ui/finance/wallets/row-actions.tsx +166 -0
  501. package/src/components/ui/finance/wallets/wallet-form-schema.ts +63 -0
  502. package/src/components/ui/finance/wallets/wallet-icon-display.tsx +71 -0
  503. package/src/components/ui/finance/wallets/wallet-icon-image-picker.tsx +369 -0
  504. package/src/components/ui/finance/wallets/wallet-images.ts +180 -0
  505. package/src/components/ui/finance/wallets/walletId/credit-wallet-summary.tsx +265 -0
  506. package/src/components/ui/finance/wallets/walletId/interest/index.ts +18 -0
  507. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-chart.tsx +296 -0
  508. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-detection-banner.tsx +77 -0
  509. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-disable-dialog.tsx +55 -0
  510. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-hero.tsx +149 -0
  511. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-pending-deposits.tsx +185 -0
  512. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-projections.tsx +259 -0
  513. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-rate-dialog.tsx +89 -0
  514. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-rate-history.tsx +116 -0
  515. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-section.tsx +260 -0
  516. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-settings.tsx +427 -0
  517. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-setup-dialog.tsx +206 -0
  518. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-summary.tsx +244 -0
  519. package/src/components/ui/finance/wallets/walletId/interest/wallet-interest-transparency.tsx +176 -0
  520. package/src/components/ui/finance/wallets/walletId/wallet-delete-button.test.tsx +78 -0
  521. package/src/components/ui/finance/wallets/walletId/wallet-delete-button.tsx +99 -0
  522. package/src/components/ui/finance/wallets/walletId/wallet-details-actions.tsx +119 -0
  523. package/src/components/ui/finance/wallets/walletId/wallet-details-amount.tsx +36 -0
  524. package/src/components/ui/finance/wallets/walletId/wallet-details-page.test.tsx +161 -0
  525. package/src/components/ui/finance/wallets/walletId/wallet-details-page.tsx +365 -0
  526. package/src/components/ui/finance/wallets/walletId/wallet-role-access-dialog.tsx +45 -0
  527. package/src/components/ui/finance/wallets/walletId/wallet-role-access.tsx +538 -0
  528. package/src/components/ui/finance/wallets/wallets-data-table.tsx +137 -0
  529. package/src/components/ui/finance/wallets/wallets-page.test.tsx +107 -0
  530. package/src/components/ui/finance/wallets/wallets-page.tsx +136 -0
  531. package/src/components/ui/form-required-indicator.tsx +31 -0
  532. package/src/components/ui/form.tsx +36 -13
  533. package/src/components/ui/hover-card.tsx +2 -2
  534. package/src/components/ui/input-otp.tsx +4 -4
  535. package/src/components/ui/input.tsx +2 -2
  536. package/src/components/ui/kbd.tsx +28 -0
  537. package/src/components/ui/label.tsx +2 -2
  538. package/src/components/ui/legacy/calendar/__tests__/event-utils.test.ts +247 -0
  539. package/src/components/ui/legacy/calendar/agenda-view.tsx +398 -0
  540. package/src/components/ui/legacy/calendar/all-day-event-bar.tsx +1033 -0
  541. package/src/components/ui/legacy/calendar/calendar-cell.tsx +1018 -0
  542. package/src/components/ui/legacy/calendar/calendar-column.tsx +31 -0
  543. package/src/components/ui/legacy/calendar/calendar-content.tsx +670 -0
  544. package/src/components/ui/legacy/calendar/calendar-header.tsx +195 -0
  545. package/src/components/ui/legacy/calendar/calendar-matrix.tsx +328 -0
  546. package/src/components/ui/legacy/calendar/calendar-settings-dialog.tsx +340 -0
  547. package/src/components/ui/legacy/calendar/calendar-view-with-trail.tsx +48 -0
  548. package/src/components/ui/legacy/calendar/calendar-view.tsx +28 -0
  549. package/src/components/ui/legacy/calendar/color-highlights.ts +55 -0
  550. package/src/components/ui/legacy/calendar/config.ts +8 -0
  551. package/src/components/ui/legacy/calendar/create-event-button.tsx +155 -0
  552. package/src/components/ui/legacy/calendar/day-title.tsx +68 -0
  553. package/src/components/ui/legacy/calendar/dynamic-island.tsx +245 -0
  554. package/src/components/ui/legacy/calendar/event-card.tsx +1430 -0
  555. package/src/components/ui/legacy/calendar/event-form-components.tsx +593 -0
  556. package/src/components/ui/legacy/calendar/event-modal.tsx +1544 -0
  557. package/src/components/ui/legacy/calendar/event-utils.ts +149 -0
  558. package/src/components/ui/legacy/calendar/location-timeline.tsx +1482 -0
  559. package/src/components/ui/legacy/calendar/month-calendar.tsx +986 -0
  560. package/src/components/ui/legacy/calendar/month-cell.tsx +33 -0
  561. package/src/components/ui/legacy/calendar/month-grid.tsx +105 -0
  562. package/src/components/ui/legacy/calendar/settings/analytics-charts.tsx +414 -0
  563. package/src/components/ui/legacy/calendar/settings/appearance-settings.tsx +218 -0
  564. package/src/components/ui/legacy/calendar/settings/calendar-sync-dashboard.tsx +217 -0
  565. package/src/components/ui/legacy/calendar/settings/category-color-settings.tsx +475 -0
  566. package/src/components/ui/legacy/calendar/settings/color-picker.tsx +177 -0
  567. package/src/components/ui/legacy/calendar/settings/google-calendar-settings.tsx +562 -0
  568. package/src/components/ui/legacy/calendar/settings/hour-settings.tsx +292 -0
  569. package/src/components/ui/legacy/calendar/settings/notification-settings.tsx +254 -0
  570. package/src/components/ui/legacy/calendar/settings/settings-context.tsx +257 -0
  571. package/src/components/ui/legacy/calendar/settings/smart-scheduling-settings.tsx +325 -0
  572. package/src/components/ui/legacy/calendar/settings/summary-cards.tsx +93 -0
  573. package/src/components/ui/legacy/calendar/settings/sync-logs-table.tsx +392 -0
  574. package/src/components/ui/legacy/calendar/settings/task-settings.tsx +330 -0
  575. package/src/components/ui/legacy/calendar/settings/time-range-picker.tsx +666 -0
  576. package/src/components/ui/legacy/calendar/settings/timezone-settings.tsx +580 -0
  577. package/src/components/ui/legacy/calendar/settings/types.ts +119 -0
  578. package/src/components/ui/legacy/calendar/settings-button.tsx +41 -0
  579. package/src/components/ui/legacy/calendar/smart-calendar.tsx +82 -0
  580. package/src/components/ui/legacy/calendar/time-indicator-line.tsx +61 -0
  581. package/src/components/ui/legacy/calendar/time-indicator-text.tsx +61 -0
  582. package/src/components/ui/legacy/calendar/time-indicator.tsx +19 -0
  583. package/src/components/ui/legacy/calendar/time-trail.tsx +52 -0
  584. package/src/components/ui/legacy/calendar/weekday-bar.tsx +78 -0
  585. package/src/components/ui/legacy/calendar/year-calendar.tsx +345 -0
  586. package/src/components/ui/legacy/meet/client-wrapper.tsx +74 -0
  587. package/src/components/ui/legacy/meet/create-plan-dialog.tsx +404 -0
  588. package/src/components/ui/legacy/meet/date-selector.tsx +46 -0
  589. package/src/components/ui/legacy/meet/edit-plan-dialog.tsx +501 -0
  590. package/src/components/ui/legacy/meet/experimental-notice.tsx +39 -0
  591. package/src/components/ui/legacy/meet/form.tsx +94 -0
  592. package/src/components/ui/legacy/meet/multiple-choice-vote.tsx +556 -0
  593. package/src/components/ui/legacy/meet/page.tsx +331 -0
  594. package/src/components/ui/legacy/meet/pagination.tsx +87 -0
  595. package/src/components/ui/legacy/meet/planId/account-badge.tsx +21 -0
  596. package/src/components/ui/legacy/meet/planId/agenda-details.tsx +131 -0
  597. package/src/components/ui/legacy/meet/planId/all-availabilities.tsx +105 -0
  598. package/src/components/ui/legacy/meet/planId/availability-planner.tsx +51 -0
  599. package/src/components/ui/legacy/meet/planId/confirm-button.tsx +53 -0
  600. package/src/components/ui/legacy/meet/planId/copy-link-button.tsx +218 -0
  601. package/src/components/ui/legacy/meet/planId/date-planner.tsx +243 -0
  602. package/src/components/ui/legacy/meet/planId/day-planner.tsx +78 -0
  603. package/src/components/ui/legacy/meet/planId/day-planners.tsx +151 -0
  604. package/src/components/ui/legacy/meet/planId/day-time.tsx +32 -0
  605. package/src/components/ui/legacy/meet/planId/download-as-png.tsx +16 -0
  606. package/src/components/ui/legacy/meet/planId/email-button.tsx +32 -0
  607. package/src/components/ui/legacy/meet/planId/logged-in-as-button.tsx +56 -0
  608. package/src/components/ui/legacy/meet/planId/page.tsx +236 -0
  609. package/src/components/ui/legacy/meet/planId/plan-details-client.tsx +205 -0
  610. package/src/components/ui/legacy/meet/planId/plan-login.tsx +342 -0
  611. package/src/components/ui/legacy/meet/planId/plan-user-filter-accordion.tsx +227 -0
  612. package/src/components/ui/legacy/meet/planId/plan-user-filter.tsx +68 -0
  613. package/src/components/ui/legacy/meet/planId/preview-day-time.tsx +303 -0
  614. package/src/components/ui/legacy/meet/planId/selectable-day-time.tsx +247 -0
  615. package/src/components/ui/legacy/meet/planId/show-qr-button.tsx +53 -0
  616. package/src/components/ui/legacy/meet/planId/sidebar-display.tsx +68 -0
  617. package/src/components/ui/legacy/meet/planId/sticky-bottom-indicator.tsx +28 -0
  618. package/src/components/ui/legacy/meet/planId/time-column.tsx +57 -0
  619. package/src/components/ui/legacy/meet/planId/unified-availability.tsx +93 -0
  620. package/src/components/ui/legacy/meet/planId/utility-buttons.tsx +54 -0
  621. package/src/components/ui/legacy/meet/plans-grid.tsx +192 -0
  622. package/src/components/ui/legacy/meet/plans-list-view.tsx +148 -0
  623. package/src/components/ui/legacy/meet/plans-loading-skeleton.tsx +61 -0
  624. package/src/components/ui/legacy/meet/time-selector.tsx +67 -0
  625. package/src/components/ui/legacy/meet/timezone-selector.tsx +72 -0
  626. package/src/components/ui/legacy/meet/user-time.tsx +7 -0
  627. package/src/components/ui/legacy/meet/view-toggle.tsx +44 -0
  628. package/src/components/ui/legacy/polls/poll-display.tsx +388 -0
  629. package/src/components/ui/legacy/polls/where-tu-meet.tsx +80 -0
  630. package/src/components/ui/markdown.tsx +2 -2
  631. package/src/components/ui/menubar.tsx +11 -11
  632. package/src/components/ui/navbar.tsx +150 -0
  633. package/src/components/ui/navigation-menu.tsx +9 -9
  634. package/src/components/ui/pagination.tsx +4 -5
  635. package/src/components/ui/popover.tsx +2 -2
  636. package/src/components/ui/progress.tsx +15 -4
  637. package/src/components/ui/radio-group.tsx +3 -3
  638. package/src/components/ui/report-problem-dialog.tsx +980 -0
  639. package/src/components/ui/resizable.tsx +35 -13
  640. package/src/components/ui/scroll-area.tsx +3 -3
  641. package/src/components/ui/select.tsx +9 -8
  642. package/src/components/ui/separator.tsx +2 -2
  643. package/src/components/ui/sheet.tsx +13 -11
  644. package/src/components/ui/sidebar.tsx +18 -17
  645. package/src/components/ui/slider.tsx +3 -3
  646. package/src/components/ui/sonner.tsx +2 -2
  647. package/src/components/ui/sticky-bottom-bar.tsx +51 -0
  648. package/src/components/ui/switch.tsx +2 -2
  649. package/src/components/ui/table.tsx +2 -2
  650. package/src/components/ui/tabs.tsx +3 -3
  651. package/src/components/ui/text-editor/__tests__/content-migration.test.ts +757 -0
  652. package/src/components/ui/text-editor/__tests__/extensions-integration.test.ts +131 -0
  653. package/src/components/ui/text-editor/__tests__/extensions.test.ts +69 -0
  654. package/src/components/ui/text-editor/__tests__/image-extension.test.ts +799 -0
  655. package/src/components/ui/text-editor/__tests__/inline-task-conversion.test.tsx +46 -0
  656. package/src/components/ui/text-editor/__tests__/keyboard.test.ts +176 -0
  657. package/src/components/ui/text-editor/__tests__/list-converter-extension.test.ts +51 -0
  658. package/src/components/ui/text-editor/__tests__/markdown-paste-extension.test.ts +266 -0
  659. package/src/components/ui/text-editor/__tests__/media-utils.test.ts +425 -0
  660. package/src/components/ui/text-editor/__tests__/task-mention-checkbox.test.ts +251 -0
  661. package/src/components/ui/text-editor/__tests__/task-mention-chip.test.tsx +279 -0
  662. package/src/components/ui/text-editor/__tests__/text-replacements.test.ts +39 -0
  663. package/src/components/ui/text-editor/__tests__/upload-placeholder.test.ts +279 -0
  664. package/src/components/ui/text-editor/__tests__/video-extension.test.ts +711 -0
  665. package/src/components/ui/text-editor/content-migration.ts +199 -0
  666. package/src/components/ui/text-editor/draggable-node-container.tsx +164 -0
  667. package/src/components/ui/text-editor/editor.tsx +707 -0
  668. package/src/components/ui/text-editor/extensions.ts +232 -0
  669. package/src/components/ui/text-editor/image-extension.ts +907 -0
  670. package/src/components/ui/text-editor/keyboard.ts +58 -0
  671. package/src/components/ui/text-editor/list-converter-extension.ts +55 -0
  672. package/src/components/ui/text-editor/list-item-extension.ts +13 -0
  673. package/src/components/ui/text-editor/list-item-view.tsx +17 -0
  674. package/src/components/ui/text-editor/markdown-paste-extension.ts +610 -0
  675. package/src/components/ui/text-editor/media-utils.ts +198 -0
  676. package/src/components/ui/text-editor/mention-extension.tsx +604 -0
  677. package/src/components/ui/text-editor/node-drag-extension.ts +300 -0
  678. package/src/components/ui/text-editor/task-item-checkbox-extension.ts +32 -0
  679. package/src/components/ui/text-editor/task-item-checkbox-view.tsx +227 -0
  680. package/src/components/ui/text-editor/task-item-checkbox.ts +100 -0
  681. package/src/components/ui/text-editor/task-mention-chip.tsx +1375 -0
  682. package/src/components/ui/text-editor/task-mention-resolution.ts +235 -0
  683. package/src/components/ui/text-editor/task-summary-popover.tsx +360 -0
  684. package/src/components/ui/text-editor/text-replacements.ts +65 -0
  685. package/src/components/ui/text-editor/text-shortcuts-extension.ts +60 -0
  686. package/src/components/ui/text-editor/tool-bar.tsx +1294 -0
  687. package/src/components/ui/text-editor/types.ts +1 -0
  688. package/src/components/ui/text-editor/upload-errors.ts +93 -0
  689. package/src/components/ui/text-editor/upload-placeholder.ts +124 -0
  690. package/src/components/ui/text-editor/video-extension.ts +435 -0
  691. package/src/components/ui/textarea.tsx +2 -2
  692. package/src/components/ui/time-picker-input.tsx +6 -6
  693. package/src/components/ui/time-picker-utils.tsx +2 -2
  694. package/src/components/ui/time-tracker/types.ts +112 -0
  695. package/src/components/ui/toast.tsx +9 -9
  696. package/src/components/ui/toaster.tsx +10 -14
  697. package/src/components/ui/toggle-group.tsx +2 -2
  698. package/src/components/ui/toggle.tsx +2 -2
  699. package/src/components/ui/tooltip.tsx +3 -3
  700. package/src/components/ui/tu-do/boards/__tests__/quick-create-board-dialog.test.tsx +52 -0
  701. package/src/components/ui/tu-do/boards/__tests__/task-board-form.test.tsx +130 -0
  702. package/src/components/ui/tu-do/boards/__tests__/workspace-projects-client-page.test.tsx +100 -0
  703. package/src/components/ui/tu-do/boards/analytics/GanttChart.tsx +419 -0
  704. package/src/components/ui/tu-do/boards/analytics/GanttControls.tsx +123 -0
  705. package/src/components/ui/tu-do/boards/analytics/GanttHeader.tsx +48 -0
  706. package/src/components/ui/tu-do/boards/analytics/GanttTimeline.tsx +355 -0
  707. package/src/components/ui/tu-do/boards/analytics/StatusDistribution.tsx +130 -0
  708. package/src/components/ui/tu-do/boards/analytics/TaskCreationAnalytics.tsx +285 -0
  709. package/src/components/ui/tu-do/boards/analytics/TaskDetailCard.tsx +270 -0
  710. package/src/components/ui/tu-do/boards/analytics/TaskGroup.tsx +282 -0
  711. package/src/components/ui/tu-do/boards/analytics/TaskWorkflowAnalytics.tsx +280 -0
  712. package/src/components/ui/tu-do/boards/board-selector.tsx +301 -0
  713. package/src/components/ui/tu-do/boards/board-share-dialog.tsx +280 -0
  714. package/src/components/ui/tu-do/boards/boardId/__tests__/list-actions.test.tsx +199 -0
  715. package/src/components/ui/tu-do/boards/boardId/board-column-initial-load.test.ts +38 -0
  716. package/src/components/ui/tu-do/boards/boardId/board-column.tsx +755 -0
  717. package/src/components/ui/tu-do/boards/boardId/board-text-utils.test.ts +24 -0
  718. package/src/components/ui/tu-do/boards/boardId/board-text-utils.ts +11 -0
  719. package/src/components/ui/tu-do/boards/boardId/enhanced-task-list.tsx +536 -0
  720. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/__tests__/bulk-mutations-move.test.tsx +176 -0
  721. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-actions-bar.tsx +94 -0
  722. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-actions-menu.tsx +689 -0
  723. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-custom-date-dialog.tsx +82 -0
  724. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-delete-dialog.tsx +58 -0
  725. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-clear-delete.ts +471 -0
  726. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-move.ts +657 -0
  727. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-relations-assignees.ts +429 -0
  728. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-relations-labels.ts +294 -0
  729. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-relations-projects.ts +316 -0
  730. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-relations.ts +12 -0
  731. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-mutations-updates.ts +499 -0
  732. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-operation-i18n.ts +524 -0
  733. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-operation-types.ts +34 -0
  734. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-operation-utils.ts +60 -0
  735. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-operations.ts +312 -0
  736. package/src/components/ui/tu-do/boards/boardId/kanban/bulk/bulk-personal-external-move.ts +132 -0
  737. package/src/components/ui/tu-do/boards/boardId/kanban/data/kanban-deadline-query.ts +42 -0
  738. package/src/components/ui/tu-do/boards/boardId/kanban/data/use-applied-sets.ts +79 -0
  739. package/src/components/ui/tu-do/boards/boardId/kanban/data/use-bulk-resources.ts +58 -0
  740. package/src/components/ui/tu-do/boards/boardId/kanban/data/use-filtered-resources.ts +54 -0
  741. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/__tests__/column-reorder.test.ts +63 -0
  742. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/auto-scroll.ts +86 -0
  743. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/column-reorder.ts +62 -0
  744. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/drag-preview.tsx +98 -0
  745. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/kanban-sort-helpers.ts +76 -0
  746. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-cache.ts +325 -0
  747. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-geometry.test.ts +67 -0
  748. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-geometry.ts +94 -0
  749. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-order.ts +156 -0
  750. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-pending.test.ts +37 -0
  751. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-pending.ts +42 -0
  752. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-preview.test.ts +318 -0
  753. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-preview.ts +260 -0
  754. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/task-drag-types.ts +39 -0
  755. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/use-kanban-dnd.test.ts +686 -0
  756. package/src/components/ui/tu-do/boards/boardId/kanban/dnd/use-kanban-dnd.ts +1503 -0
  757. package/src/components/ui/tu-do/boards/boardId/kanban/kanban-constants.ts +43 -0
  758. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-column-width.ts +30 -0
  759. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-columns.test.tsx +383 -0
  760. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-columns.tsx +228 -0
  761. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-deadline-panels.tsx +221 -0
  762. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-deadline-tasks.test.ts +267 -0
  763. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-deadline-tasks.ts +112 -0
  764. package/src/components/ui/tu-do/boards/boardId/kanban/rendering/kanban-skeleton.tsx +45 -0
  765. package/src/components/ui/tu-do/boards/boardId/kanban/selection/use-keyboard-shortcuts.ts +137 -0
  766. package/src/components/ui/tu-do/boards/boardId/kanban/selection/use-multi-select.ts +119 -0
  767. package/src/components/ui/tu-do/boards/boardId/kanban.tsx +495 -0
  768. package/src/components/ui/tu-do/boards/boardId/list-actions.tsx +522 -0
  769. package/src/components/ui/tu-do/boards/boardId/menus/__tests__/task-estimation-menu.test.tsx +261 -0
  770. package/src/components/ui/tu-do/boards/boardId/menus/__tests__/task-menus.test.tsx +491 -0
  771. package/src/components/ui/tu-do/boards/boardId/menus/index.ts +11 -0
  772. package/src/components/ui/tu-do/boards/boardId/menus/task-assignees-menu.tsx +176 -0
  773. package/src/components/ui/tu-do/boards/boardId/menus/task-blocking-menu.tsx +304 -0
  774. package/src/components/ui/tu-do/boards/boardId/menus/task-due-date-menu.tsx +180 -0
  775. package/src/components/ui/tu-do/boards/boardId/menus/task-estimation-menu.tsx +103 -0
  776. package/src/components/ui/tu-do/boards/boardId/menus/task-labels-menu.tsx +134 -0
  777. package/src/components/ui/tu-do/boards/boardId/menus/task-move-menu.tsx +66 -0
  778. package/src/components/ui/tu-do/boards/boardId/menus/task-parent-menu.tsx +228 -0
  779. package/src/components/ui/tu-do/boards/boardId/menus/task-picker-popover.tsx +300 -0
  780. package/src/components/ui/tu-do/boards/boardId/menus/task-priority-menu.tsx +176 -0
  781. package/src/components/ui/tu-do/boards/boardId/menus/task-projects-menu.tsx +143 -0
  782. package/src/components/ui/tu-do/boards/boardId/menus/task-related-menu.tsx +242 -0
  783. package/src/components/ui/tu-do/boards/boardId/status-grouped-board.tsx +533 -0
  784. package/src/components/ui/tu-do/boards/boardId/status-section.tsx +274 -0
  785. package/src/components/ui/tu-do/boards/boardId/task-actions.tsx +675 -0
  786. package/src/components/ui/tu-do/boards/boardId/task-board-server-page.tsx +41 -0
  787. package/src/components/ui/tu-do/boards/boardId/task-card/TaskCardCheckbox.tsx +47 -0
  788. package/src/components/ui/tu-do/boards/boardId/task-card/TaskCardDates.tsx +106 -0
  789. package/src/components/ui/tu-do/boards/boardId/task-card/TaskCardHeader.tsx +97 -0
  790. package/src/components/ui/tu-do/boards/boardId/task-card/TaskCardMetadata.tsx +152 -0
  791. package/src/components/ui/tu-do/boards/boardId/task-card/measured-task-card.tsx +109 -0
  792. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-comparator.test.ts +61 -0
  793. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-comparator.ts +87 -0
  794. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-label-options.ts +24 -0
  795. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-visibility.test.ts +41 -0
  796. package/src/components/ui/tu-do/boards/boardId/task-card/task-card-visibility.ts +14 -0
  797. package/src/components/ui/tu-do/boards/boardId/task-card/task-card.tsx +2543 -0
  798. package/src/components/ui/tu-do/boards/boardId/task-dialogs/BoardEstimationConfigDialog.tsx +499 -0
  799. package/src/components/ui/tu-do/boards/boardId/task-dialogs/TaskCustomDateDialog.tsx +109 -0
  800. package/src/components/ui/tu-do/boards/boardId/task-dialogs/TaskDeleteDialog.tsx +96 -0
  801. package/src/components/ui/tu-do/boards/boardId/task-dialogs/TaskNewLabelDialog.tsx +178 -0
  802. package/src/components/ui/tu-do/boards/boardId/task-dialogs/TaskNewProjectDialog.tsx +127 -0
  803. package/src/components/ui/tu-do/boards/boardId/task-filter.tsx +986 -0
  804. package/src/components/ui/tu-do/boards/boardId/task-form.tsx +498 -0
  805. package/src/components/ui/tu-do/boards/boardId/task-list-drag-preview.test.ts +101 -0
  806. package/src/components/ui/tu-do/boards/boardId/task-list-form.tsx +91 -0
  807. package/src/components/ui/tu-do/boards/boardId/task-list.tsx +513 -0
  808. package/src/components/ui/tu-do/boards/boardId/task-parent-badge-state.test.ts +64 -0
  809. package/src/components/ui/tu-do/boards/boardId/task-parent-badge-state.ts +31 -0
  810. package/src/components/ui/tu-do/boards/boardId/task.tsx +2 -0
  811. package/src/components/ui/tu-do/boards/boardId/timeline/task-edit-dialog.tsx +109 -0
  812. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-display.ts +94 -0
  813. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-grid.tsx +380 -0
  814. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-task-row.tsx +350 -0
  815. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-toolbar.tsx +347 -0
  816. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-utils.test.ts +134 -0
  817. package/src/components/ui/tu-do/boards/boardId/timeline/timeline-utils.ts +468 -0
  818. package/src/components/ui/tu-do/boards/boardId/timeline-board.test.tsx +217 -0
  819. package/src/components/ui/tu-do/boards/boardId/timeline-board.tsx +793 -0
  820. package/src/components/ui/tu-do/boards/boards-list-skeleton.tsx +62 -0
  821. package/src/components/ui/tu-do/boards/columns.tsx +159 -0
  822. package/src/components/ui/tu-do/boards/copy-board-dialog.tsx +156 -0
  823. package/src/components/ui/tu-do/boards/enhanced-boards-view.tsx +347 -0
  824. package/src/components/ui/tu-do/boards/form.tsx +264 -0
  825. package/src/components/ui/tu-do/boards/quick-create-board-dialog.tsx +75 -0
  826. package/src/components/ui/tu-do/boards/row-actions.tsx +397 -0
  827. package/src/components/ui/tu-do/boards/workspace-projects-client-page.tsx +182 -0
  828. package/src/components/ui/tu-do/boards/workspace-projects-page.tsx +155 -0
  829. package/src/components/ui/tu-do/cycles/task-cycles-client.tsx +744 -0
  830. package/src/components/ui/tu-do/cycles/task-cycles-page.tsx +103 -0
  831. package/src/components/ui/tu-do/drafts/draft-card.tsx +204 -0
  832. package/src/components/ui/tu-do/drafts/draft-convert-dialog.tsx +197 -0
  833. package/src/components/ui/tu-do/drafts/drafts-page.tsx +105 -0
  834. package/src/components/ui/tu-do/drafts/task-drafts-page.tsx +90 -0
  835. package/src/components/ui/tu-do/estimates/client.tsx +292 -0
  836. package/src/components/ui/tu-do/estimates/edit-estimation-dialog.tsx +374 -0
  837. package/src/components/ui/tu-do/estimates/task-estimates-page.tsx +57 -0
  838. package/src/components/ui/tu-do/estimates/use-task-estimates.ts +319 -0
  839. package/src/components/ui/tu-do/habits/client.tsx +260 -0
  840. package/src/components/ui/tu-do/habits/habit-card.tsx +183 -0
  841. package/src/components/ui/tu-do/habits/habit-form-dialog.tsx +701 -0
  842. package/src/components/ui/tu-do/habits/habits-page.tsx +27 -0
  843. package/src/components/ui/tu-do/hooks/__tests__/useDraftPersistence.test.ts +494 -0
  844. package/src/components/ui/tu-do/hooks/__tests__/useTaskDialog.test.tsx +127 -0
  845. package/src/components/ui/tu-do/hooks/__tests__/useTaskDialogState.test.ts +217 -0
  846. package/src/components/ui/tu-do/hooks/__tests__/useTaskLabelManagement.test.tsx +524 -0
  847. package/src/components/ui/tu-do/hooks/useDraftPersistence.ts +199 -0
  848. package/src/components/ui/tu-do/hooks/useTaskCardRelationships.ts +340 -0
  849. package/src/components/ui/tu-do/hooks/useTaskDialog.ts +101 -0
  850. package/src/components/ui/tu-do/hooks/useTaskDialogState.ts +145 -0
  851. package/src/components/ui/tu-do/hooks/useTaskFormState.ts +206 -0
  852. package/src/components/ui/tu-do/hooks/useTaskKeyboardShortcuts.ts +58 -0
  853. package/src/components/ui/tu-do/hooks/useTaskLabelManagement.ts +416 -0
  854. package/src/components/ui/tu-do/hooks/useTaskProjectManagement.ts +461 -0
  855. package/src/components/ui/tu-do/initiatives/task-initiatives-client.tsx +921 -0
  856. package/src/components/ui/tu-do/initiatives/task-initiatives-page.tsx +116 -0
  857. package/src/components/ui/tu-do/labels/client.tsx +201 -0
  858. package/src/components/ui/tu-do/labels/components/delete-label-dialog.tsx +57 -0
  859. package/src/components/ui/tu-do/labels/components/label-card.tsx +86 -0
  860. package/src/components/ui/tu-do/labels/components/label-dialog.tsx +259 -0
  861. package/src/components/ui/tu-do/labels/components/label-list.tsx +83 -0
  862. package/src/components/ui/tu-do/labels/hooks/use-task-labels.ts +170 -0
  863. package/src/components/ui/tu-do/labels/task-labels-page.tsx +53 -0
  864. package/src/components/ui/tu-do/labels/types.ts +6 -0
  865. package/src/components/ui/tu-do/logs/columns.tsx +755 -0
  866. package/src/components/ui/tu-do/logs/logs-client.tsx +1006 -0
  867. package/src/components/ui/tu-do/logs/logs-timeline.tsx +2483 -0
  868. package/src/components/ui/tu-do/logs/task-logs-page.tsx +59 -0
  869. package/src/components/ui/tu-do/my-tasks/__tests__/my-tasks-content.test.tsx +408 -0
  870. package/src/components/ui/tu-do/my-tasks/__tests__/use-my-tasks-query.test.ts +469 -0
  871. package/src/components/ui/tu-do/my-tasks/__tests__/use-my-tasks-state.test.ts +1017 -0
  872. package/src/components/ui/tu-do/my-tasks/__tests__/use-task-context-actions.test.ts +516 -0
  873. package/src/components/ui/tu-do/my-tasks/ai-credit-indicator.tsx +143 -0
  874. package/src/components/ui/tu-do/my-tasks/board-selector-dialog.tsx +239 -0
  875. package/src/components/ui/tu-do/my-tasks/command-bar.tsx +1351 -0
  876. package/src/components/ui/tu-do/my-tasks/label-project-filter.tsx +194 -0
  877. package/src/components/ui/tu-do/my-tasks/my-task-context-menu.tsx +356 -0
  878. package/src/components/ui/tu-do/my-tasks/my-tasks-content.tsx +304 -0
  879. package/src/components/ui/tu-do/my-tasks/my-tasks-data-loader.tsx +17 -0
  880. package/src/components/ui/tu-do/my-tasks/my-tasks-filters.tsx +258 -0
  881. package/src/components/ui/tu-do/my-tasks/my-tasks-header.tsx +121 -0
  882. package/src/components/ui/tu-do/my-tasks/my-tasks-page.tsx +40 -0
  883. package/src/components/ui/tu-do/my-tasks/personal-placement-dialog.tsx +188 -0
  884. package/src/components/ui/tu-do/my-tasks/task-filter.tsx +235 -0
  885. package/src/components/ui/tu-do/my-tasks/task-list-with-completion.tsx +782 -0
  886. package/src/components/ui/tu-do/my-tasks/task-list.tsx +258 -0
  887. package/src/components/ui/tu-do/my-tasks/task-preview-dialog.tsx +1597 -0
  888. package/src/components/ui/tu-do/my-tasks/task-section.tsx +261 -0
  889. package/src/components/ui/tu-do/my-tasks/use-my-tasks-query.ts +140 -0
  890. package/src/components/ui/tu-do/my-tasks/use-my-tasks-state.ts +983 -0
  891. package/src/components/ui/tu-do/my-tasks/use-task-context-actions.ts +332 -0
  892. package/src/components/ui/tu-do/notes/note-edit-dialog.tsx +121 -0
  893. package/src/components/ui/tu-do/notes/note-list.tsx +799 -0
  894. package/src/components/ui/tu-do/notes/notes-content.tsx +178 -0
  895. package/src/components/ui/tu-do/notes/notes-page.tsx +27 -0
  896. package/src/components/ui/tu-do/projects/components/index.ts +11 -0
  897. package/src/components/ui/tu-do/projects/components/project-actions-menu.tsx +66 -0
  898. package/src/components/ui/tu-do/projects/components/project-badges.tsx +84 -0
  899. package/src/components/ui/tu-do/projects/components/project-filter-menu.tsx +170 -0
  900. package/src/components/ui/tu-do/projects/components/project-grid-card.tsx +182 -0
  901. package/src/components/ui/tu-do/projects/components/project-list-item.tsx +179 -0
  902. package/src/components/ui/tu-do/projects/components/project-metrics.tsx +87 -0
  903. package/src/components/ui/tu-do/projects/components/project-progress-meter.tsx +33 -0
  904. package/src/components/ui/tu-do/projects/components/project-sort-menu.tsx +71 -0
  905. package/src/components/ui/tu-do/projects/components/projects-empty-state.tsx +38 -0
  906. package/src/components/ui/tu-do/projects/components/projects-loading-state.tsx +26 -0
  907. package/src/components/ui/tu-do/projects/components/projects-toolbar.tsx +179 -0
  908. package/src/components/ui/tu-do/projects/dialogs/create-project-dialog.tsx +117 -0
  909. package/src/components/ui/tu-do/projects/dialogs/edit-project-dialog.tsx +125 -0
  910. package/src/components/ui/tu-do/projects/dialogs/index.ts +3 -0
  911. package/src/components/ui/tu-do/projects/dialogs/manage-tasks-dialog.tsx +205 -0
  912. package/src/components/ui/tu-do/projects/hooks/index.ts +2 -0
  913. package/src/components/ui/tu-do/projects/hooks/use-project-filters.ts +248 -0
  914. package/src/components/ui/tu-do/projects/hooks/use-task-projects.ts +238 -0
  915. package/src/components/ui/tu-do/projects/projectId/components/__tests__/tasks-tab-layout.test.tsx +163 -0
  916. package/src/components/ui/tu-do/projects/projectId/components/documents-tab.tsx +90 -0
  917. package/src/components/ui/tu-do/projects/projectId/components/index.ts +9 -0
  918. package/src/components/ui/tu-do/projects/projectId/components/overview/description-card.tsx +77 -0
  919. package/src/components/ui/tu-do/projects/projectId/components/overview/linked-documents-card.tsx +81 -0
  920. package/src/components/ui/tu-do/projects/projectId/components/overview/linked-tasks-card.tsx +85 -0
  921. package/src/components/ui/tu-do/projects/projectId/components/overview/updates-card.tsx +85 -0
  922. package/src/components/ui/tu-do/projects/projectId/components/overview-tab.tsx +35 -0
  923. package/src/components/ui/tu-do/projects/projectId/components/project-configuration.tsx +229 -0
  924. package/src/components/ui/tu-do/projects/projectId/components/project-header.tsx +115 -0
  925. package/src/components/ui/tu-do/projects/projectId/components/project-lead-selector.tsx +55 -0
  926. package/src/components/ui/tu-do/projects/projectId/components/project-overview-context.tsx +82 -0
  927. package/src/components/ui/tu-do/projects/projectId/components/project-sidebar.tsx +246 -0
  928. package/src/components/ui/tu-do/projects/projectId/components/tasks-tab.tsx +264 -0
  929. package/src/components/ui/tu-do/projects/projectId/components/update-card.tsx +162 -0
  930. package/src/components/ui/tu-do/projects/projectId/components/updates-tab.tsx +129 -0
  931. package/src/components/ui/tu-do/projects/projectId/dialogs/index.ts +1 -0
  932. package/src/components/ui/tu-do/projects/projectId/dialogs/link-task-dialog.tsx +114 -0
  933. package/src/components/ui/tu-do/projects/projectId/hooks/index.ts +4 -0
  934. package/src/components/ui/tu-do/projects/projectId/hooks/use-animation-variants.ts +68 -0
  935. package/src/components/ui/tu-do/projects/projectId/hooks/use-project-form.ts +214 -0
  936. package/src/components/ui/tu-do/projects/projectId/hooks/use-project-updates.ts +209 -0
  937. package/src/components/ui/tu-do/projects/projectId/hooks/use-task-linking.ts +101 -0
  938. package/src/components/ui/tu-do/projects/projectId/task-project-detail-page-client.tsx +85 -0
  939. package/src/components/ui/tu-do/projects/projectId/task-project-detail-page.tsx +78 -0
  940. package/src/components/ui/tu-do/projects/projectId/task-project-detail.tsx +365 -0
  941. package/src/components/ui/tu-do/projects/projectId/types.ts +50 -0
  942. package/src/components/ui/tu-do/projects/task-projects-client.tsx +226 -0
  943. package/src/components/ui/tu-do/projects/task-projects-page.tsx +97 -0
  944. package/src/components/ui/tu-do/projects/types.ts +59 -0
  945. package/src/components/ui/tu-do/providers/__tests__/task-dialog-provider.test.tsx +404 -0
  946. package/src/components/ui/tu-do/providers/task-dialog-provider.tsx +721 -0
  947. package/src/components/ui/tu-do/providers/workspace-presence-provider.tsx +88 -0
  948. package/src/components/ui/tu-do/shared/AccessibleButton.tsx +79 -0
  949. package/src/components/ui/tu-do/shared/__tests__/assignee-select.test.tsx +72 -0
  950. package/src/components/ui/tu-do/shared/__tests__/board-client.test.tsx +152 -0
  951. package/src/components/ui/tu-do/shared/__tests__/board-header.test.tsx +172 -0
  952. package/src/components/ui/tu-do/shared/__tests__/board-query-cache.test.ts +219 -0
  953. package/src/components/ui/tu-do/shared/__tests__/board-views.test.tsx +613 -0
  954. package/src/components/ui/tu-do/shared/__tests__/create-list-dialog.test.tsx +118 -0
  955. package/src/components/ui/tu-do/shared/__tests__/task-detail-page.test.tsx +170 -0
  956. package/src/components/ui/tu-do/shared/__tests__/task-dialog-manager.test.tsx +708 -0
  957. package/src/components/ui/tu-do/shared/__tests__/task-legacy-route-recovery.test.tsx +85 -0
  958. package/src/components/ui/tu-do/shared/__tests__/use-progressive-board-loader.test.tsx +588 -0
  959. package/src/components/ui/tu-do/shared/assignee-select.tsx +541 -0
  960. package/src/components/ui/tu-do/shared/board-broadcast-context.tsx +45 -0
  961. package/src/components/ui/tu-do/shared/board-client.tsx +268 -0
  962. package/src/components/ui/tu-do/shared/board-config-storage.ts +88 -0
  963. package/src/components/ui/tu-do/shared/board-header.tsx +1248 -0
  964. package/src/components/ui/tu-do/shared/board-layout-settings.tsx +1105 -0
  965. package/src/components/ui/tu-do/shared/board-query-cache.ts +157 -0
  966. package/src/components/ui/tu-do/shared/board-switcher.tsx +298 -0
  967. package/src/components/ui/tu-do/shared/board-user-presence-avatars.tsx +572 -0
  968. package/src/components/ui/tu-do/shared/board-views.tsx +847 -0
  969. package/src/components/ui/tu-do/shared/clear-menu-item.tsx +34 -0
  970. package/src/components/ui/tu-do/shared/create-list-dialog.tsx +401 -0
  971. package/src/components/ui/tu-do/shared/cursor-overlay-multi-wrapper.tsx +204 -0
  972. package/src/components/ui/tu-do/shared/cursor-overlay.tsx +78 -0
  973. package/src/components/ui/tu-do/shared/custom-date-picker/custom-date-picker-dialog.tsx +157 -0
  974. package/src/components/ui/tu-do/shared/description-overflow-warning-dialog.tsx +68 -0
  975. package/src/components/ui/tu-do/shared/edit-list-dialog.tsx +383 -0
  976. package/src/components/ui/tu-do/shared/empty-state-card.tsx +46 -0
  977. package/src/components/ui/tu-do/shared/estimation-mapping.test.ts +32 -0
  978. package/src/components/ui/tu-do/shared/estimation-mapping.ts +74 -0
  979. package/src/components/ui/tu-do/shared/estimation-utils.ts +54 -0
  980. package/src/components/ui/tu-do/shared/fade-setting-initializer.tsx +59 -0
  981. package/src/components/ui/tu-do/shared/label-chip.tsx +46 -0
  982. package/src/components/ui/tu-do/shared/list-view-context-menu.test.tsx +150 -0
  983. package/src/components/ui/tu-do/shared/list-view-sorting.test.ts +50 -0
  984. package/src/components/ui/tu-do/shared/list-view-sorting.ts +108 -0
  985. package/src/components/ui/tu-do/shared/list-view.tsx +987 -0
  986. package/src/components/ui/tu-do/shared/mention-system/__tests__/mention-system.test.ts +250 -0
  987. package/src/components/ui/tu-do/shared/mention-system/mention-menu.tsx +217 -0
  988. package/src/components/ui/tu-do/shared/mention-system/types.ts +141 -0
  989. package/src/components/ui/tu-do/shared/mention-system/use-mention-suggestions.ts +244 -0
  990. package/src/components/ui/tu-do/shared/progressive-loader-context.tsx +45 -0
  991. package/src/components/ui/tu-do/shared/recent-sidebar-events.ts +58 -0
  992. package/src/components/ui/tu-do/shared/recycle-bin-panel.tsx +603 -0
  993. package/src/components/ui/tu-do/shared/relationship-task-identifier.ts +14 -0
  994. package/src/components/ui/tu-do/shared/slash-commands/__tests__/slash-commands.test.ts +315 -0
  995. package/src/components/ui/tu-do/shared/slash-commands/definitions.ts +167 -0
  996. package/src/components/ui/tu-do/shared/slash-commands/slash-command-menu.tsx +114 -0
  997. package/src/components/ui/tu-do/shared/sync-warning-dialog.tsx +133 -0
  998. package/src/components/ui/tu-do/shared/task-board-errors.ts +19 -0
  999. package/src/components/ui/tu-do/shared/task-detail-page.tsx +176 -0
  1000. package/src/components/ui/tu-do/shared/task-detail-server-page.tsx +37 -0
  1001. package/src/components/ui/tu-do/shared/task-dialog-manager.tsx +557 -0
  1002. package/src/components/ui/tu-do/shared/task-dialog-wrapper.tsx +28 -0
  1003. package/src/components/ui/tu-do/shared/task-edit-dialog/components/mobile-floating-save-button.tsx +46 -0
  1004. package/src/components/ui/tu-do/shared/task-edit-dialog/components/quick-settings-popover.tsx +164 -0
  1005. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-description-editor.tsx +491 -0
  1006. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-dialog-header.tsx +511 -0
  1007. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-list-picker-panel.tsx +220 -0
  1008. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-list-selector.tsx +127 -0
  1009. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-list-trigger-styles.ts +69 -0
  1010. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-name-input.test.tsx +140 -0
  1011. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-name-input.tsx +163 -0
  1012. package/src/components/ui/tu-do/shared/task-edit-dialog/components/task-suggestion-menus.tsx +148 -0
  1013. package/src/components/ui/tu-do/shared/task-edit-dialog/constants.ts +18 -0
  1014. package/src/components/ui/tu-do/shared/task-edit-dialog/context-menu-guard.test.ts +37 -0
  1015. package/src/components/ui/tu-do/shared/task-edit-dialog/description-diff-viewer.tsx +1665 -0
  1016. package/src/components/ui/tu-do/shared/task-edit-dialog/field-diff-viewer.tsx +715 -0
  1017. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-task-dialog-close.test.ts +218 -0
  1018. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-task-dialog-keyboard-shortcuts.test.ts +140 -0
  1019. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-task-overrides.test.ts +412 -0
  1020. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-task-realtime-sync.test.tsx +209 -0
  1021. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-task-yjs-sync.test.ts +190 -0
  1022. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/__tests__/use-update-shared-task.test.ts +103 -0
  1023. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/task-api.ts +127 -0
  1024. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-editor-commands.ts +314 -0
  1025. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-suggestion-menus.ts +503 -0
  1026. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-change-detection.ts +175 -0
  1027. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-data.ts +332 -0
  1028. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-dependencies.ts +773 -0
  1029. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-dialog-close.ts +193 -0
  1030. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-dialog-keyboard-shortcuts.ts +365 -0
  1031. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-form-reset.ts +215 -0
  1032. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-form-state.ts +307 -0
  1033. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-mutations.ts +536 -0
  1034. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-overrides.ts +122 -0
  1035. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-realtime-sync.ts +252 -0
  1036. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-relationships.test.tsx +149 -0
  1037. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-relationships.ts +545 -0
  1038. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-revert.ts +140 -0
  1039. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-save.test.ts +206 -0
  1040. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-save.ts +1450 -0
  1041. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-snapshot.ts +59 -0
  1042. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-task-yjs-sync.ts +224 -0
  1043. package/src/components/ui/tu-do/shared/task-edit-dialog/hooks/use-update-shared-task.ts +55 -0
  1044. package/src/components/ui/tu-do/shared/task-edit-dialog/personal-overrides-section.tsx +348 -0
  1045. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/components/clickable-task-item.tsx +90 -0
  1046. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/components/index.ts +4 -0
  1047. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/components/subtask-action-buttons.tsx +140 -0
  1048. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/components/tab-button.tsx +76 -0
  1049. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/components/task-relationship-action-buttons.tsx +178 -0
  1050. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/dependencies-section.tsx +129 -0
  1051. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/index.ts +35 -0
  1052. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/parent-section.tsx +65 -0
  1053. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/related-section.tsx +76 -0
  1054. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/subtasks-section.tsx +82 -0
  1055. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/task-search-popover.tsx +229 -0
  1056. package/src/components/ui/tu-do/shared/task-edit-dialog/relationships/types/task-relationships.types.ts +186 -0
  1057. package/src/components/ui/tu-do/shared/task-edit-dialog/selective-revert-panel.tsx +320 -0
  1058. package/src/components/ui/tu-do/shared/task-edit-dialog/task-activity-section.tsx +748 -0
  1059. package/src/components/ui/tu-do/shared/task-edit-dialog/task-delete-dialog.tsx +104 -0
  1060. package/src/components/ui/tu-do/shared/task-edit-dialog/task-dialog-actions.tsx +213 -0
  1061. package/src/components/ui/tu-do/shared/task-edit-dialog/task-instances-section.tsx +173 -0
  1062. package/src/components/ui/tu-do/shared/task-edit-dialog/task-properties-section.tsx +1895 -0
  1063. package/src/components/ui/tu-do/shared/task-edit-dialog/task-relationships-properties.tsx +273 -0
  1064. package/src/components/ui/tu-do/shared/task-edit-dialog/task-snapshot-dialog.tsx +162 -0
  1065. package/src/components/ui/tu-do/shared/task-edit-dialog/types/pending-relationship.test.ts +52 -0
  1066. package/src/components/ui/tu-do/shared/task-edit-dialog/types/pending-relationship.ts +134 -0
  1067. package/src/components/ui/tu-do/shared/task-edit-dialog/types.ts +64 -0
  1068. package/src/components/ui/tu-do/shared/task-edit-dialog/user-display.test.ts +58 -0
  1069. package/src/components/ui/tu-do/shared/task-edit-dialog/user-display.ts +45 -0
  1070. package/src/components/ui/tu-do/shared/task-edit-dialog/utils/inline-task-target-list.test.ts +98 -0
  1071. package/src/components/ui/tu-do/shared/task-edit-dialog/utils/inline-task-target-list.ts +39 -0
  1072. package/src/components/ui/tu-do/shared/task-edit-dialog/utils.test.ts +404 -0
  1073. package/src/components/ui/tu-do/shared/task-edit-dialog/utils.ts +473 -0
  1074. package/src/components/ui/tu-do/shared/task-edit-dialog/yjs-prosemirror-compat.test.ts +65 -0
  1075. package/src/components/ui/tu-do/shared/task-edit-dialog.tsx +2168 -0
  1076. package/src/components/ui/tu-do/shared/task-estimation-display.tsx +52 -0
  1077. package/src/components/ui/tu-do/shared/task-estimation-picker.tsx +223 -0
  1078. package/src/components/ui/tu-do/shared/task-filter.types.ts +65 -0
  1079. package/src/components/ui/tu-do/shared/task-label-selector.tsx +217 -0
  1080. package/src/components/ui/tu-do/shared/task-labels-display.test.tsx +51 -0
  1081. package/src/components/ui/tu-do/shared/task-labels-display.tsx +119 -0
  1082. package/src/components/ui/tu-do/shared/task-legacy-route-recovery.tsx +72 -0
  1083. package/src/components/ui/tu-do/shared/task-open-events.ts +94 -0
  1084. package/src/components/ui/tu-do/shared/task-projects-display.tsx +80 -0
  1085. package/src/components/ui/tu-do/shared/task-resource-search-field.tsx +49 -0
  1086. package/src/components/ui/tu-do/shared/task-resource-search-filters.ts +27 -0
  1087. package/src/components/ui/tu-do/shared/task-row-actions-menu.tsx +352 -0
  1088. package/src/components/ui/tu-do/shared/task-share-dialog/components/share-link-settings.tsx +86 -0
  1089. package/src/components/ui/tu-do/shared/task-share-dialog/components/shares-list.tsx +85 -0
  1090. package/src/components/ui/tu-do/shared/task-share-dialog/hooks/use-task-sharing.ts +296 -0
  1091. package/src/components/ui/tu-do/shared/task-share-dialog.tsx +144 -0
  1092. package/src/components/ui/tu-do/shared/task-url.ts +56 -0
  1093. package/src/components/ui/tu-do/shared/text-diff-viewer.tsx +55 -0
  1094. package/src/components/ui/tu-do/shared/types.ts +14 -0
  1095. package/src/components/ui/tu-do/shared/unsaved-changes-warning-dialog.tsx +71 -0
  1096. package/src/components/ui/tu-do/shared/use-progressive-board-loader.ts +310 -0
  1097. package/src/components/ui/tu-do/shared/user-avatar.tsx +46 -0
  1098. package/src/components/ui/tu-do/shared/user-presence-avatars.tsx +415 -0
  1099. package/src/components/ui/tu-do/shared/utils/translate-task-list-display-name.ts +28 -0
  1100. package/src/components/ui/tu-do/tasks-route-context.tsx +50 -0
  1101. package/src/components/ui/tu-do/templates/client.tsx +486 -0
  1102. package/src/components/ui/tu-do/templates/marketplace/client.tsx +451 -0
  1103. package/src/components/ui/tu-do/templates/marketplace/task-marketplace-page.tsx +169 -0
  1104. package/src/components/ui/tu-do/templates/save-as-template-dialog.tsx +381 -0
  1105. package/src/components/ui/tu-do/templates/task-templates-page.tsx +162 -0
  1106. package/src/components/ui/tu-do/templates/templateId/client.tsx +659 -0
  1107. package/src/components/ui/tu-do/templates/templateId/edit-template-dialog.tsx +300 -0
  1108. package/src/components/ui/tu-do/templates/templateId/share-template-dialog.tsx +261 -0
  1109. package/src/components/ui/tu-do/templates/templateId/task-template-detail-page-client.tsx +121 -0
  1110. package/src/components/ui/tu-do/templates/templateId/task-template-detail-page.tsx +89 -0
  1111. package/src/components/ui/tu-do/templates/templateId/use-template-dialog.tsx +165 -0
  1112. package/src/components/ui/tu-do/templates/types.ts +53 -0
  1113. package/src/components/ui/tu-do/utils/__tests__/label-colors.test.ts +43 -0
  1114. package/src/components/ui/tu-do/utils/__tests__/taskColorUtils.test.ts +267 -0
  1115. package/src/components/ui/tu-do/utils/__tests__/taskConstants.test.ts +178 -0
  1116. package/src/components/ui/tu-do/utils/__tests__/taskDateUtils.test.ts +226 -0
  1117. package/src/components/ui/tu-do/utils/__tests__/weekDateUtils.test.ts +182 -0
  1118. package/src/components/ui/tu-do/utils/label-colors.test.ts +15 -0
  1119. package/src/components/ui/tu-do/utils/label-colors.ts +59 -0
  1120. package/src/components/ui/tu-do/utils/taskColorUtils.ts +116 -0
  1121. package/src/components/ui/tu-do/utils/taskConstants.ts +102 -0
  1122. package/src/components/ui/tu-do/utils/taskDateUtils.ts +81 -0
  1123. package/src/components/ui/tu-do/utils/taskPriorityUtils.tsx +80 -0
  1124. package/src/components/ui/tu-do/utils/weekDateUtils.ts +64 -0
  1125. package/src/constants/boards.ts +9 -0
  1126. package/src/declarations.d.ts +2 -0
  1127. package/src/globals.css +609 -200
  1128. package/src/gsap.ts +2 -0
  1129. package/src/hooks/__tests__/supabase-provider.test.ts +346 -0
  1130. package/src/hooks/__tests__/use-calendar-readonly.test.tsx +90 -0
  1131. package/src/hooks/__tests__/use-forwarded-ref.test.tsx +127 -0
  1132. package/src/hooks/__tests__/use-local-storage.test.tsx +243 -0
  1133. package/src/hooks/__tests__/use-mobile.test.tsx +222 -0
  1134. package/src/hooks/__tests__/use-notifications-subscription.test.tsx +163 -0
  1135. package/src/hooks/__tests__/use-notifications.test.ts +59 -0
  1136. package/src/hooks/__tests__/use-stable-array.test.tsx +214 -0
  1137. package/src/hooks/__tests__/use-task-actions.test.tsx +1893 -0
  1138. package/src/hooks/__tests__/use-workspace-presence.test.tsx +217 -0
  1139. package/src/hooks/__tests__/useBoardRealtime.test.tsx +1156 -0
  1140. package/src/hooks/supabase-provider.ts +725 -0
  1141. package/src/hooks/task-actions-personal-external.ts +308 -0
  1142. package/src/hooks/time-blocking-provider.tsx +756 -0
  1143. package/src/hooks/use-ai-credits.ts +78 -0
  1144. package/src/hooks/use-analytics-filters.ts +269 -0
  1145. package/src/hooks/use-at-bottom.tsx +64 -0
  1146. package/src/hooks/use-board-actions.ts +213 -0
  1147. package/src/hooks/use-calendar-preferences.tsx +24 -0
  1148. package/src/hooks/use-calendar-sync.tsx +1184 -0
  1149. package/src/hooks/use-calendar.tsx +1972 -0
  1150. package/src/hooks/use-callback-ref.ts +1 -1
  1151. package/src/hooks/use-controllable-state.ts +3 -4
  1152. package/src/hooks/use-copy-to-clipboard.ts +1 -3
  1153. package/src/hooks/use-currency-formatter.ts +81 -0
  1154. package/src/hooks/use-debounce.ts +15 -0
  1155. package/src/hooks/use-dom-resolved-theme.ts +81 -0
  1156. package/src/hooks/use-enter-submit.tsx +23 -0
  1157. package/src/hooks/use-exchange-rates.ts +23 -0
  1158. package/src/hooks/use-finance-transaction-preferences.ts +58 -0
  1159. package/src/hooks/use-form.ts +8 -6
  1160. package/src/hooks/use-interest-preferences.ts +104 -0
  1161. package/src/hooks/use-local-storage.ts +42 -0
  1162. package/src/hooks/use-notifications.ts +704 -0
  1163. package/src/hooks/use-page-visibility.ts +27 -0
  1164. package/src/hooks/use-popover-manager.tsx +98 -0
  1165. package/src/hooks/use-semantic-task-search.ts +76 -0
  1166. package/src/hooks/use-session-history-query.ts +1 -0
  1167. package/src/hooks/use-stable-array.ts +23 -0
  1168. package/src/hooks/use-task-actions.ts +1915 -0
  1169. package/src/hooks/use-task-analytics.ts +354 -0
  1170. package/src/hooks/use-toast.ts +2 -3
  1171. package/src/hooks/use-user-config.ts +109 -0
  1172. package/src/hooks/use-view-transition.ts +69 -0
  1173. package/src/hooks/use-workspace-config.ts +49 -0
  1174. package/src/hooks/use-workspace-currency.ts +26 -0
  1175. package/src/hooks/use-workspace-members.ts +58 -0
  1176. package/src/hooks/use-workspace-permission.ts +61 -0
  1177. package/src/hooks/use-workspace-presence.ts +554 -0
  1178. package/src/hooks/use-workspace-user.ts +33 -0
  1179. package/src/hooks/use-yjs-collaboration.ts +297 -0
  1180. package/src/hooks/useBoardRealtime.ts +372 -0
  1181. package/src/hooks/useBoardRealtime.types.ts +45 -0
  1182. package/src/hooks/useBoardRealtimeEventHandler.ts +332 -0
  1183. package/src/hooks/useCursorTracking.ts +338 -0
  1184. package/src/hooks/useHorizontalScroll.ts +215 -0
  1185. package/src/hooks/usePresence.ts +272 -0
  1186. package/src/hooks/useSearchParams.tsx +133 -0
  1187. package/src/lib/__tests__/lunar-calendar.test.ts +137 -0
  1188. package/src/lib/calendar/planning-query-client.ts +197 -0
  1189. package/src/lib/calendar/preview-engine.ts +14 -0
  1190. package/src/lib/calendar-settings-resolver.ts +200 -0
  1191. package/src/lib/lunar-calendar.ts +71 -0
  1192. package/src/lib/template-background.ts +159 -0
  1193. package/src/lib/workspace-actions.ts +457 -0
  1194. package/src/utils/__tests__/label-colors.test.ts +219 -0
  1195. package/src/utils/__tests__/priority-styles.test.ts +78 -0
  1196. package/src/utils/label-colors.ts +4 -0
  1197. package/src/utils/priority-styles.ts +42 -0
  1198. package/src/xlsx.ts +3 -0
  1199. package/tsconfig.json +3 -11
  1200. package/tsconfig.typecheck.json +9 -0
  1201. package/vendor/xlsx-0.20.3.tgz +0 -0
  1202. package/vitest.config.ts +42 -0
  1203. package/vitest.setup.ts +37 -0
  1204. package/eslint.config.mjs +0 -20
  1205. package/rollup.config.js +0 -40
  1206. package/src/components/ui/icons.tsx +0 -506
  1207. package/src/components/ui/tag-input.tsx +0 -141
  1208. /package/src/hooks/{use-mobile.tsx → use-mobile.ts} +0 -0
@@ -0,0 +1,2483 @@
1
+ 'use client';
2
+
3
+ import {
4
+ ArrowRight,
5
+ Calendar,
6
+ CheckCircle2,
7
+ ChevronDown,
8
+ CircleDot,
9
+ Clock,
10
+ ExternalLink,
11
+ Eye,
12
+ FileText,
13
+ Flag,
14
+ FolderKanban,
15
+ horseHead,
16
+ Icon,
17
+ Layers,
18
+ LayoutGrid,
19
+ Plus,
20
+ Rabbit,
21
+ RotateCcw,
22
+ Tag,
23
+ Target,
24
+ Trash2,
25
+ Turtle,
26
+ UserMinus,
27
+ UserPlus,
28
+ unicornHead,
29
+ } from '@tuturuuu/icons';
30
+ import { Avatar, AvatarFallback, AvatarImage } from '@tuturuuu/ui/avatar';
31
+ import { Badge } from '@tuturuuu/ui/badge';
32
+ import { Tooltip, TooltipContent, TooltipTrigger } from '@tuturuuu/ui/tooltip';
33
+ import {
34
+ type EstimationType,
35
+ mapEstimationPoints,
36
+ } from '@tuturuuu/ui/tu-do/shared/estimation-mapping';
37
+ import { DescriptionDiffViewer } from '@tuturuuu/ui/tu-do/shared/task-edit-dialog/description-diff-viewer';
38
+ import { TextDiffViewer } from '@tuturuuu/ui/tu-do/shared/text-diff-viewer';
39
+ import { useTasksHref } from '@tuturuuu/ui/tu-do/tasks-route-context';
40
+ import { cn } from '@tuturuuu/utils/format';
41
+ import { getDescriptionText } from '@tuturuuu/utils/text-helper';
42
+ import { format, formatDistanceToNow, isToday, isYesterday } from 'date-fns';
43
+ import { enUS, vi } from 'date-fns/locale';
44
+ import { AnimatePresence, motion } from 'motion/react';
45
+ import Link from 'next/link';
46
+ import type React from 'react';
47
+ import { useMemo, useState } from 'react';
48
+ import type { TaskHistoryLogEntry } from './columns';
49
+
50
+ // Priority styling constants (matching task card)
51
+ const PRIORITY_LABELS: Record<string, string> = {
52
+ critical: 'Urgent',
53
+ high: 'High',
54
+ normal: 'Medium',
55
+ low: 'Low',
56
+ };
57
+
58
+ const PRIORITY_BADGE_COLORS: Record<string, string> = {
59
+ critical:
60
+ 'bg-dynamic-red/20 border-dynamic-red/50 text-dynamic-red shadow-sm shadow-dynamic-red/50',
61
+ high: 'bg-dynamic-orange/10 border-dynamic-orange/30 text-dynamic-orange',
62
+ normal: 'bg-dynamic-yellow/10 border-dynamic-yellow/30 text-dynamic-yellow',
63
+ low: 'bg-dynamic-blue/10 border-dynamic-blue/30 text-dynamic-blue',
64
+ };
65
+
66
+ function getPriorityIcon(
67
+ priority: string,
68
+ className?: string
69
+ ): React.ReactNode {
70
+ const icons: Record<string, React.ReactElement> = {
71
+ critical: <Icon iconNode={unicornHead} className={className} />,
72
+ high: <Icon iconNode={horseHead} className={className} />,
73
+ normal: <Rabbit className={className} />,
74
+ low: <Turtle className={className} />,
75
+ };
76
+ return icons[priority] || null;
77
+ }
78
+
79
+ function renderPriorityBadge(
80
+ priority: string | number | null
81
+ ): React.ReactNode {
82
+ if (priority === null || priority === undefined) return null;
83
+
84
+ // Handle numeric priority (legacy)
85
+ const priorityMap: Record<number, string> = {
86
+ 1: 'low',
87
+ 2: 'normal',
88
+ 3: 'high',
89
+ 4: 'critical',
90
+ };
91
+
92
+ const priorityKey =
93
+ typeof priority === 'number' ? priorityMap[priority] : priority;
94
+ if (!priorityKey || !PRIORITY_LABELS[priorityKey]) return null;
95
+
96
+ return (
97
+ <Badge
98
+ variant="secondary"
99
+ className={cn('gap-1 p-0.75 text-xs', PRIORITY_BADGE_COLORS[priorityKey])}
100
+ >
101
+ {getPriorityIcon(priorityKey, 'size-3')}
102
+ {PRIORITY_LABELS[priorityKey]}
103
+ </Badge>
104
+ );
105
+ }
106
+
107
+ interface LogsTimelineProps {
108
+ entries: TaskHistoryLogEntry[];
109
+ wsId: string;
110
+ locale: string;
111
+ t: (key: string, options?: { defaultValue?: string }) => string;
112
+ className?: string;
113
+ /** Time window in minutes for grouping rapid successive changes (default: 5) */
114
+ rapidChangeWindowMinutes?: number;
115
+ /** Map of board_id -> estimation_type for proper estimation display */
116
+ estimationTypes?: Record<string, string | null>;
117
+ }
118
+
119
+ /** Represents a group of rapid successive changes to the same task by the same user */
120
+ interface RapidChangeGroup {
121
+ id: string;
122
+ task_id: string | null;
123
+ task_name: string;
124
+ task_deleted_at?: string;
125
+ task_permanently_deleted?: boolean;
126
+ board_id?: string;
127
+ board_name?: string;
128
+ user: TaskHistoryLogEntry['user'];
129
+ first_changed_at: string;
130
+ last_changed_at: string;
131
+ entries: TaskHistoryLogEntry[];
132
+ is_grouped: true;
133
+ change_count: number;
134
+ }
135
+
136
+ /** Represents an aggregated group of same-type actions (e.g., multiple assignees added at once) */
137
+ interface AggregatedActionGroup {
138
+ id: string;
139
+ task_id: string | null;
140
+ task_name: string;
141
+ task_deleted_at?: string;
142
+ task_permanently_deleted?: boolean;
143
+ board_id?: string;
144
+ board_name?: string;
145
+ user: TaskHistoryLogEntry['user'];
146
+ changed_at: string;
147
+ change_type: TaskHistoryLogEntry['change_type'];
148
+ entries: TaskHistoryLogEntry[];
149
+ is_aggregated: true;
150
+ aggregated_items: Array<{
151
+ name: string;
152
+ avatar_url?: string;
153
+ color?: string;
154
+ changed_at: string;
155
+ }>;
156
+ }
157
+
158
+ type TimelineItem =
159
+ | TaskHistoryLogEntry
160
+ | RapidChangeGroup
161
+ | AggregatedActionGroup;
162
+
163
+ function isRapidChangeGroup(item: TimelineItem): item is RapidChangeGroup {
164
+ return 'is_grouped' in item && item.is_grouped === true;
165
+ }
166
+
167
+ function isAggregatedActionGroup(
168
+ item: TimelineItem
169
+ ): item is AggregatedActionGroup {
170
+ return 'is_aggregated' in item && item.is_aggregated === true;
171
+ }
172
+
173
+ /** Check if a change type should be aggregated (multiple items shown as one) */
174
+ function isAggregatableChangeType(changeType: string): boolean {
175
+ return [
176
+ 'assignee_added',
177
+ 'assignee_removed',
178
+ 'label_added',
179
+ 'label_removed',
180
+ ].includes(changeType);
181
+ }
182
+
183
+ /** Extract item details from an entry for aggregation */
184
+ function extractAggregatedItem(entry: TaskHistoryLogEntry): {
185
+ name: string;
186
+ avatar_url?: string;
187
+ color?: string;
188
+ changed_at: string;
189
+ } | null {
190
+ const { change_type, old_value, new_value, metadata, changed_at } = entry;
191
+
192
+ switch (change_type) {
193
+ case 'assignee_added': {
194
+ const data = new_value as {
195
+ user_name?: string;
196
+ avatar_url?: string;
197
+ } | null;
198
+ return {
199
+ name:
200
+ data?.user_name || (metadata?.assignee_name as string) || 'Unknown',
201
+ avatar_url: data?.avatar_url,
202
+ changed_at,
203
+ };
204
+ }
205
+ case 'assignee_removed': {
206
+ const data = old_value as {
207
+ user_name?: string;
208
+ avatar_url?: string;
209
+ } | null;
210
+ return {
211
+ name:
212
+ data?.user_name || (metadata?.assignee_name as string) || 'Unknown',
213
+ avatar_url: data?.avatar_url,
214
+ changed_at,
215
+ };
216
+ }
217
+ case 'label_added': {
218
+ const data = new_value as { name?: string; color?: string } | null;
219
+ return {
220
+ name: data?.name || (metadata?.label_name as string) || 'Unknown',
221
+ color: data?.color || (metadata?.label_color as string),
222
+ changed_at,
223
+ };
224
+ }
225
+ case 'label_removed': {
226
+ const data = old_value as { name?: string; color?: string } | null;
227
+ return {
228
+ name: data?.name || (metadata?.label_name as string) || 'Unknown',
229
+ color: data?.color || (metadata?.label_color as string),
230
+ changed_at,
231
+ };
232
+ }
233
+ default:
234
+ return null;
235
+ }
236
+ }
237
+
238
+ interface GroupedEntries {
239
+ date: string;
240
+ dateLabel: string;
241
+ entries: TimelineItem[];
242
+ }
243
+
244
+ /**
245
+ * Groups rapid successive changes to the same task by the same user within a time window.
246
+ * Also aggregates same-type actions (like multiple assignee_added) into single entries.
247
+ */
248
+ function groupRapidSuccessiveChanges(
249
+ entries: TaskHistoryLogEntry[],
250
+ timeWindowMinutes: number
251
+ ): TimelineItem[] {
252
+ if (entries.length <= 1) {
253
+ // Even single entries might need aggregation check
254
+ if (entries.length === 1) return entries;
255
+ return entries;
256
+ }
257
+
258
+ // Sort by changed_at descending (newest first)
259
+ const sorted = [...entries].sort(
260
+ (a, b) =>
261
+ new Date(b.changed_at).getTime() - new Date(a.changed_at).getTime()
262
+ );
263
+
264
+ const result: TimelineItem[] = [];
265
+ const processed = new Set<string>();
266
+
267
+ // Helper to check if two entries refer to the same task
268
+ const isSameTask = (a: TaskHistoryLogEntry, b: TaskHistoryLogEntry) => {
269
+ if (a.task_id && b.task_id) return a.task_id === b.task_id;
270
+ // Fallback for permanently deleted tasks (null task_id)
271
+ return !a.task_id && !b.task_id && a.task_name === b.task_name;
272
+ };
273
+
274
+ for (const entry of sorted) {
275
+ if (processed.has(entry.id)) continue;
276
+
277
+ const entryTime = new Date(entry.changed_at).getTime();
278
+
279
+ // First, try to aggregate same-type actions for the same task by the same user
280
+ if (isAggregatableChangeType(entry.change_type)) {
281
+ const sameTypeEntries: TaskHistoryLogEntry[] = [entry];
282
+
283
+ for (const otherEntry of sorted) {
284
+ if (processed.has(otherEntry.id) || otherEntry.id === entry.id)
285
+ continue;
286
+
287
+ const otherTime = new Date(otherEntry.changed_at).getTime();
288
+ const diffMinutes = Math.abs(entryTime - otherTime) / (1000 * 60);
289
+
290
+ // Aggregate same-type actions for the same task by the same user within the time window
291
+ if (
292
+ isSameTask(otherEntry, entry) &&
293
+ otherEntry.changed_by === entry.changed_by &&
294
+ otherEntry.change_type === entry.change_type &&
295
+ diffMinutes <= timeWindowMinutes
296
+ ) {
297
+ sameTypeEntries.push(otherEntry);
298
+ }
299
+ }
300
+
301
+ if (sameTypeEntries.length > 1) {
302
+ // Create an aggregated action group
303
+ const aggregatedItems = sameTypeEntries
304
+ .map(extractAggregatedItem)
305
+ .filter((item): item is NonNullable<typeof item> => item !== null);
306
+
307
+ // Sort chronologically
308
+ const chronological = [...sameTypeEntries].sort(
309
+ (a, b) =>
310
+ new Date(a.changed_at).getTime() - new Date(b.changed_at).getTime()
311
+ );
312
+
313
+ result.push({
314
+ id: entry.id,
315
+ task_id: entry.task_id,
316
+ task_name: entry.task_name,
317
+ task_deleted_at: entry.task_deleted_at,
318
+ task_permanently_deleted: entry.task_permanently_deleted,
319
+ board_id: entry.board_id,
320
+ board_name: entry.board_name,
321
+ user: entry.user,
322
+ changed_at: chronological[chronological.length - 1]!.changed_at,
323
+ change_type: entry.change_type,
324
+ entries: chronological,
325
+ is_aggregated: true,
326
+ aggregated_items: aggregatedItems,
327
+ });
328
+
329
+ sameTypeEntries.forEach((e) => {
330
+ processed.add(e.id);
331
+ });
332
+ continue;
333
+ }
334
+ }
335
+
336
+ // Find all entries for the same task by the same user within the time window (different types)
337
+ const groupEntries: TaskHistoryLogEntry[] = [entry];
338
+
339
+ for (const otherEntry of sorted) {
340
+ if (processed.has(otherEntry.id) || otherEntry.id === entry.id) continue;
341
+
342
+ const otherTime = new Date(otherEntry.changed_at).getTime();
343
+ const diffMinutes = Math.abs(entryTime - otherTime) / (1000 * 60);
344
+
345
+ if (
346
+ isSameTask(otherEntry, entry) &&
347
+ otherEntry.changed_by === entry.changed_by &&
348
+ diffMinutes <= timeWindowMinutes
349
+ ) {
350
+ groupEntries.push(otherEntry);
351
+ }
352
+ }
353
+
354
+ if (groupEntries.length > 1) {
355
+ // Sort chronologically (oldest first) for display
356
+ const chronological = [...groupEntries].sort(
357
+ (a, b) =>
358
+ new Date(a.changed_at).getTime() - new Date(b.changed_at).getTime()
359
+ );
360
+
361
+ result.push({
362
+ id: entry.id,
363
+ task_id: entry.task_id,
364
+ task_name: entry.task_name,
365
+ task_deleted_at: entry.task_deleted_at,
366
+ task_permanently_deleted: entry.task_permanently_deleted,
367
+ board_id: entry.board_id,
368
+ board_name: entry.board_name,
369
+ user: entry.user,
370
+ first_changed_at: chronological[0]!.changed_at,
371
+ last_changed_at: chronological[chronological.length - 1]!.changed_at,
372
+ entries: chronological,
373
+ is_grouped: true,
374
+ change_count: groupEntries.length,
375
+ });
376
+
377
+ groupEntries.forEach((e) => {
378
+ processed.add(e.id);
379
+ });
380
+ } else {
381
+ result.push(entry);
382
+ processed.add(entry.id);
383
+ }
384
+ }
385
+
386
+ // Sort result by most recent timestamp
387
+ return result.sort((a, b) => {
388
+ const aTime = isRapidChangeGroup(a)
389
+ ? new Date(a.last_changed_at).getTime()
390
+ : isAggregatedActionGroup(a)
391
+ ? new Date(a.changed_at).getTime()
392
+ : new Date((a as TaskHistoryLogEntry).changed_at).getTime();
393
+ const bTime = isRapidChangeGroup(b)
394
+ ? new Date(b.last_changed_at).getTime()
395
+ : isAggregatedActionGroup(b)
396
+ ? new Date(b.changed_at).getTime()
397
+ : new Date((b as TaskHistoryLogEntry).changed_at).getTime();
398
+ return bTime - aTime;
399
+ });
400
+ }
401
+
402
+ export default function LogsTimeline({
403
+ entries,
404
+ wsId,
405
+ locale,
406
+ t,
407
+ className,
408
+ rapidChangeWindowMinutes = 5,
409
+ estimationTypes,
410
+ }: LogsTimelineProps) {
411
+ const dateLocale = locale === 'vi' ? vi : enUS;
412
+
413
+ // Calculate latest deletion entries for permanently deleted tasks
414
+ const latestDeletions = useMemo(() => {
415
+ const deletions = new Set<string>();
416
+ const processedTasks = new Set<string>();
417
+
418
+ // Entries are typically sorted by date descending, but let's be safe
419
+ // We want the NEWEST 'deleted_at' entry for each task
420
+ const sortedEntries = [...entries].sort(
421
+ (a, b) =>
422
+ new Date(b.changed_at).getTime() - new Date(a.changed_at).getTime()
423
+ );
424
+
425
+ for (const entry of sortedEntries) {
426
+ if (
427
+ entry.task_permanently_deleted &&
428
+ entry.change_type === 'field_updated' &&
429
+ entry.field_name === 'deleted_at' &&
430
+ // Check if it's a delete action (not restore)
431
+ entry.new_value !== null &&
432
+ entry.new_value !== undefined &&
433
+ entry.new_value !== '' &&
434
+ entry.new_value !== 'null'
435
+ ) {
436
+ // For permanently deleted tasks, task_id is null.
437
+ // We use task_name as a fallback key to distinguish between different deleted tasks.
438
+ const taskKey = entry.task_id || `null-${entry.task_name}`;
439
+
440
+ if (!processedTasks.has(taskKey)) {
441
+ deletions.add(entry.id);
442
+ processedTasks.add(taskKey);
443
+ }
444
+ }
445
+ }
446
+ return deletions;
447
+ }, [entries]);
448
+
449
+ // Group entries by date, then apply rapid change grouping
450
+ const groupedEntries = useMemo(() => {
451
+ const groups = new Map<string, TaskHistoryLogEntry[]>();
452
+
453
+ entries.forEach((entry) => {
454
+ const date = new Date(entry.changed_at);
455
+ const dateKey = format(date, 'yyyy-MM-dd');
456
+
457
+ if (!groups.has(dateKey)) {
458
+ groups.set(dateKey, []);
459
+ }
460
+ groups.get(dateKey)!.push(entry);
461
+ });
462
+
463
+ const result: GroupedEntries[] = [];
464
+ groups.forEach((dayEntries, dateKey) => {
465
+ const date = new Date(dateKey);
466
+ let dateLabel: string;
467
+
468
+ if (isToday(date)) {
469
+ dateLabel = t('date.today', { defaultValue: 'Today' });
470
+ } else if (isYesterday(date)) {
471
+ dateLabel = t('date.yesterday', { defaultValue: 'Yesterday' });
472
+ } else {
473
+ dateLabel = format(date, 'EEEE, MMMM d, yyyy', { locale: dateLocale });
474
+ }
475
+
476
+ // Apply rapid successive change grouping to each day's entries
477
+ const groupedDayEntries = groupRapidSuccessiveChanges(
478
+ dayEntries,
479
+ rapidChangeWindowMinutes
480
+ );
481
+
482
+ result.push({
483
+ date: dateKey,
484
+ dateLabel,
485
+ entries: groupedDayEntries,
486
+ });
487
+ });
488
+
489
+ return result;
490
+ }, [entries, t, dateLocale, rapidChangeWindowMinutes]);
491
+
492
+ if (entries.length === 0) {
493
+ return null;
494
+ }
495
+
496
+ return (
497
+ <div className={cn('space-y-8 overflow-hidden', className)}>
498
+ {groupedEntries.map((group, groupIndex) => (
499
+ <motion.div
500
+ key={group.date}
501
+ initial={{ opacity: 0, y: 20 }}
502
+ animate={{ opacity: 1, y: 0 }}
503
+ transition={{ delay: groupIndex * 0.1 }}
504
+ className="space-y-3"
505
+ >
506
+ {/* Date header */}
507
+ <div className="sticky top-0 z-10 flex items-center gap-3 py-2 backdrop-blur-sm">
508
+ <div className="h-px flex-1 bg-border/50" />
509
+ <div className="flex items-center gap-1.5 rounded-full border bg-background/80 px-3 py-1 shadow-sm">
510
+ <Calendar className="h-3 w-3 text-muted-foreground" />
511
+ <span className="font-medium text-xs">{group.dateLabel}</span>
512
+ </div>
513
+ <div className="h-px flex-1 bg-border/50" />
514
+ </div>
515
+
516
+ {/* Entries for this date */}
517
+ <div className="space-y-2">
518
+ {group.entries.map((item, index) =>
519
+ isRapidChangeGroup(item) ? (
520
+ <RapidChangeGroupEntry
521
+ key={item.id}
522
+ group={item}
523
+ wsId={wsId}
524
+ locale={locale}
525
+ t={t}
526
+ index={index}
527
+ dateLocale={dateLocale}
528
+ estimationType={
529
+ item.board_id ? estimationTypes?.[item.board_id] : undefined
530
+ }
531
+ latestDeletions={latestDeletions}
532
+ />
533
+ ) : isAggregatedActionGroup(item) ? (
534
+ <AggregatedActionEntry
535
+ key={item.id}
536
+ group={item}
537
+ wsId={wsId}
538
+ t={t}
539
+ index={index}
540
+ dateLocale={dateLocale}
541
+ />
542
+ ) : (
543
+ <TimelineEntry
544
+ key={item.id}
545
+ entry={item}
546
+ wsId={wsId}
547
+ locale={locale}
548
+ t={t}
549
+ index={index}
550
+ dateLocale={dateLocale}
551
+ estimationType={
552
+ item.board_id ? estimationTypes?.[item.board_id] : undefined
553
+ }
554
+ isLatestDeletion={latestDeletions.has(item.id)}
555
+ />
556
+ )
557
+ )}
558
+ </div>
559
+ </motion.div>
560
+ ))}
561
+ </div>
562
+ );
563
+ }
564
+
565
+ interface RapidChangeGroupEntryProps {
566
+ group: RapidChangeGroup;
567
+ wsId: string;
568
+ locale: string;
569
+ t: (key: string, options?: { defaultValue?: string }) => string;
570
+ index: number;
571
+ dateLocale: typeof enUS | typeof vi;
572
+ /** Estimation type for proper points display */
573
+ estimationType?: EstimationType;
574
+ /** Set of IDs for entries that are the latest deletion for a permanently deleted task */
575
+ latestDeletions?: Set<string>;
576
+ }
577
+
578
+ function RapidChangeGroupEntry({
579
+ group,
580
+ wsId,
581
+ locale,
582
+ t,
583
+ index,
584
+ dateLocale,
585
+ estimationType,
586
+ latestDeletions,
587
+ }: RapidChangeGroupEntryProps) {
588
+ const tasksHref = useTasksHref();
589
+ const [expanded, setExpanded] = useState(false);
590
+
591
+ const startTime = new Date(group.first_changed_at);
592
+ const endTime = new Date(group.last_changed_at);
593
+
594
+ const userName =
595
+ group.user?.name || t('unknown_user', { defaultValue: 'Unknown user' });
596
+ const userInitials = userName
597
+ .split(' ')
598
+ .map((n) => n[0])
599
+ .join('')
600
+ .toUpperCase()
601
+ .slice(0, 2);
602
+
603
+ // Aggregate same-type entries within this group (e.g., multiple assignee_added)
604
+ const aggregatedEntries = useMemo(() => {
605
+ const result: Array<TaskHistoryLogEntry | AggregatedActionGroup> = [];
606
+ const processed = new Set<string>();
607
+
608
+ for (const entry of group.entries) {
609
+ if (processed.has(entry.id)) continue;
610
+
611
+ if (isAggregatableChangeType(entry.change_type)) {
612
+ // Find all entries of the same type
613
+ const sameTypeEntries = group.entries.filter(
614
+ (e) => !processed.has(e.id) && e.change_type === entry.change_type
615
+ );
616
+
617
+ if (sameTypeEntries.length > 1) {
618
+ // Create an aggregated action group
619
+ const aggregatedItems = sameTypeEntries
620
+ .map(extractAggregatedItem)
621
+ .filter((item): item is NonNullable<typeof item> => item !== null);
622
+
623
+ const chronological = [...sameTypeEntries].sort(
624
+ (a, b) =>
625
+ new Date(a.changed_at).getTime() -
626
+ new Date(b.changed_at).getTime()
627
+ );
628
+
629
+ result.push({
630
+ id: entry.id,
631
+ task_id: entry.task_id,
632
+ task_name: entry.task_name,
633
+ task_deleted_at: entry.task_deleted_at,
634
+ task_permanently_deleted: entry.task_permanently_deleted,
635
+ board_id: entry.board_id,
636
+ board_name: entry.board_name,
637
+ user: entry.user,
638
+ changed_at: chronological[chronological.length - 1]!.changed_at,
639
+ change_type: entry.change_type,
640
+ entries: chronological,
641
+ is_aggregated: true,
642
+ aggregated_items: aggregatedItems,
643
+ });
644
+
645
+ sameTypeEntries.forEach((e) => {
646
+ processed.add(e.id);
647
+ });
648
+ continue;
649
+ }
650
+ }
651
+
652
+ result.push(entry);
653
+ processed.add(entry.id);
654
+ }
655
+
656
+ return result;
657
+ }, [group.entries]);
658
+
659
+ // Get change type distribution for summary
660
+ const changeTypeCounts = group.entries.reduce(
661
+ (acc, e) => {
662
+ const key =
663
+ e.change_type === 'field_updated'
664
+ ? `field:${e.field_name}`
665
+ : e.change_type;
666
+ acc[key] = (acc[key] || 0) + 1;
667
+ return acc;
668
+ },
669
+ {} as Record<string, number>
670
+ );
671
+
672
+ const changeTypeSummary = Object.entries(changeTypeCounts)
673
+ .slice(0, 3)
674
+ .map(([type, count]) => {
675
+ let label: string;
676
+ if (type.startsWith('field:')) {
677
+ const fieldName = type.replace('field:', '');
678
+ label = t(`field_name.${fieldName}`, { defaultValue: fieldName });
679
+ } else {
680
+ label = t(`change_type.${type}`, { defaultValue: type });
681
+ }
682
+ return count > 1 ? `${label} (${count})` : label;
683
+ })
684
+ .join(', ');
685
+
686
+ const timeAgo = formatDistanceToNow(endTime, {
687
+ addSuffix: true,
688
+ locale: dateLocale,
689
+ });
690
+
691
+ // Find description changes in the group for showing diff viewers
692
+ const descriptionChanges = group.entries
693
+ .filter(
694
+ (e) => e.change_type === 'field_updated' && e.field_name === 'description'
695
+ )
696
+ .map((e) => ({
697
+ id: e.id,
698
+ oldValue: e.old_value,
699
+ newValue: e.new_value,
700
+ }));
701
+
702
+ return (
703
+ <motion.div
704
+ initial={{ opacity: 0, x: -10 }}
705
+ animate={{ opacity: 1, x: 0 }}
706
+ transition={{ delay: index * 0.03 }}
707
+ className="group relative overflow-hidden rounded-lg border border-l-2 border-l-dynamic-indigo bg-card transition-all hover:bg-accent/50"
708
+ >
709
+ {/* Header - clickable to expand */}
710
+ <button
711
+ type="button"
712
+ onClick={() => setExpanded(!expanded)}
713
+ className="flex w-full gap-3 overflow-hidden p-4 text-left"
714
+ >
715
+ {/* Icon indicator */}
716
+ <div className="flex h-9 w-9 shrink-0 items-center justify-center rounded-full bg-dynamic-indigo/10">
717
+ <Layers className="h-4 w-4 text-dynamic-indigo" />
718
+ </div>
719
+
720
+ {/* Content */}
721
+ <div className="min-w-0 flex-1">
722
+ {/* Header row */}
723
+ <div className="mb-1.5 flex flex-wrap items-center gap-x-2 gap-y-1">
724
+ {/* User info */}
725
+ <div className="flex items-center gap-1.5">
726
+ <Avatar className="h-5 w-5">
727
+ <AvatarImage
728
+ src={group.user?.avatar_url || undefined}
729
+ alt={userName}
730
+ />
731
+ <AvatarFallback className="text-[10px]">
732
+ {userInitials}
733
+ </AvatarFallback>
734
+ </Avatar>
735
+ <span className="font-medium text-sm">{userName}</span>
736
+ </div>
737
+
738
+ {/* Change count badge */}
739
+ <Badge variant="secondary" className="gap-1 text-xs">
740
+ <Layers className="h-3 w-3" />
741
+ {group.change_count} {t('changes', { defaultValue: 'changes' })}
742
+ </Badge>
743
+
744
+ {/* Task link */}
745
+ <div className="flex flex-wrap items-center gap-1.5">
746
+ <Tooltip>
747
+ <TooltipTrigger asChild>
748
+ {group.task_permanently_deleted ? (
749
+ <span className="wrap-break-word line-clamp-1 max-w-37.5 font-medium text-muted-foreground text-sm line-through md:max-w-62.5">
750
+ {group.task_name}
751
+ </span>
752
+ ) : (
753
+ <Link
754
+ href={`/${wsId}/tasks/${group.task_id}`}
755
+ onClick={(e) => e.stopPropagation()}
756
+ className={cn(
757
+ 'wrap-break-word line-clamp-1 max-w-37.5 font-medium text-sm hover:underline md:max-w-62.5',
758
+ group.task_deleted_at
759
+ ? 'text-muted-foreground line-through'
760
+ : 'text-foreground'
761
+ )}
762
+ >
763
+ {group.task_name}
764
+ </Link>
765
+ )}
766
+ </TooltipTrigger>
767
+ {group.task_name.length > 25 && (
768
+ <TooltipContent
769
+ side="bottom"
770
+ className="wrap-break-word line-clamp-1 max-w-md text-sm"
771
+ >
772
+ {group.task_name}
773
+ </TooltipContent>
774
+ )}
775
+ </Tooltip>
776
+ {group.task_permanently_deleted ? (
777
+ <Badge
778
+ variant="outline"
779
+ className="gap-1 border-dynamic-red/50 bg-dynamic-red/20 px-1.5 py-0.5 text-dynamic-red text-xs"
780
+ >
781
+ <Trash2 className="h-3 w-3" />
782
+ {t('permanently_deleted', {
783
+ defaultValue: 'Permanently Deleted',
784
+ })}
785
+ </Badge>
786
+ ) : group.task_deleted_at ? (
787
+ <Badge
788
+ variant="outline"
789
+ className="gap-1 border-dynamic-orange/30 bg-dynamic-orange/10 px-1.5 py-0.5 text-dynamic-orange text-xs"
790
+ >
791
+ <RotateCcw className="h-3 w-3" />
792
+ {t('in_trash', { defaultValue: 'In Trash' })}
793
+ </Badge>
794
+ ) : null}
795
+ {/* Board link */}
796
+ {group.board_id && group.board_name && (
797
+ <>
798
+ <span className="text-muted-foreground text-xs">
799
+ {t('in_board', { defaultValue: 'in' })}
800
+ </span>
801
+ <Tooltip>
802
+ <TooltipTrigger asChild>
803
+ <Link
804
+ href={`/${wsId}${tasksHref(`/boards/${group.board_id}`)}`}
805
+ onClick={(e) => e.stopPropagation()}
806
+ className="group/board inline-flex items-center gap-1 text-xs hover:text-foreground"
807
+ >
808
+ <LayoutGrid className="h-3 w-3 shrink-0 text-muted-foreground" />
809
+ <span className="wrap-break-word line-clamp-1 text-muted-foreground group-hover/board:text-foreground group-hover/board:underline">
810
+ {group.board_name}
811
+ </span>
812
+ <ExternalLink className="h-2.5 w-2.5 shrink-0 text-muted-foreground opacity-0 transition-opacity group-hover/board:opacity-100" />
813
+ </Link>
814
+ </TooltipTrigger>
815
+ <TooltipContent side="bottom" className="text-xs">
816
+ <p>
817
+ {t('open_board', { defaultValue: 'Open board' })}:{' '}
818
+ {group.board_name}
819
+ </p>
820
+ </TooltipContent>
821
+ </Tooltip>
822
+ </>
823
+ )}
824
+ </div>
825
+
826
+ {/* Expand indicator */}
827
+ <ChevronDown
828
+ className={cn(
829
+ 'ml-auto h-4 w-4 shrink-0 text-muted-foreground transition-transform',
830
+ expanded && 'rotate-180'
831
+ )}
832
+ />
833
+ </div>
834
+
835
+ {/* Change type summary */}
836
+ <p className="mb-1 text-muted-foreground text-xs">
837
+ {changeTypeSummary}
838
+ </p>
839
+
840
+ {/* Timestamp row */}
841
+ <p className="text-muted-foreground text-xs">
842
+ {(() => {
843
+ const startFormatted = format(startTime, 'HH:mm', {
844
+ locale: dateLocale,
845
+ });
846
+ const endFormatted = format(endTime, 'HH:mm', {
847
+ locale: dateLocale,
848
+ });
849
+ const isSameTime = startFormatted === endFormatted;
850
+
851
+ return (
852
+ <>
853
+ <span className="font-medium">{startFormatted}</span>
854
+ {!isSameTime && (
855
+ <>
856
+ <span className="mx-1 opacity-50">-</span>
857
+ <span className="font-medium">{endFormatted}</span>
858
+ </>
859
+ )}
860
+ </>
861
+ );
862
+ })()}
863
+ <span className="mx-1.5 opacity-50">·</span>
864
+ <span>{timeAgo}</span>
865
+ </p>
866
+ </div>
867
+ </button>
868
+
869
+ {/* Show diff viewers for description changes in the group - outside button to avoid nesting */}
870
+ {descriptionChanges.length > 0 && (
871
+ <div className="flex flex-wrap gap-2 border-t px-4 py-2">
872
+ {descriptionChanges.map((change, idx) => (
873
+ <DescriptionDiffViewer
874
+ key={change.id}
875
+ oldValue={change.oldValue}
876
+ newValue={change.newValue}
877
+ t={t}
878
+ triggerVariant="inline"
879
+ trigger={
880
+ descriptionChanges.length > 1 ? (
881
+ <span className="inline-flex cursor-pointer items-center gap-1 text-dynamic-blue text-xs hover:underline">
882
+ <Eye className="h-3 w-3" />
883
+ {t('view_changes', {
884
+ defaultValue: 'View changes',
885
+ })}{' '}
886
+ #{idx + 1}
887
+ </span>
888
+ ) : undefined
889
+ }
890
+ />
891
+ ))}
892
+ </div>
893
+ )}
894
+
895
+ {/* Expanded content */}
896
+ <AnimatePresence>
897
+ {expanded && (
898
+ <motion.div
899
+ initial={{ opacity: 0, height: 0 }}
900
+ animate={{ opacity: 1, height: 'auto' }}
901
+ exit={{ opacity: 0, height: 0 }}
902
+ transition={{ duration: 0.2 }}
903
+ className="border-t"
904
+ >
905
+ <div className="space-y-2 p-4 pt-3">
906
+ {aggregatedEntries.map((item, i) =>
907
+ isAggregatedActionGroup(item) ? (
908
+ <CompactAggregatedEntry
909
+ key={item.id}
910
+ group={item}
911
+ t={t}
912
+ index={i}
913
+ dateLocale={dateLocale}
914
+ />
915
+ ) : (
916
+ <TimelineEntry
917
+ key={item.id}
918
+ entry={item}
919
+ wsId={wsId}
920
+ locale={locale}
921
+ t={t}
922
+ index={i}
923
+ dateLocale={dateLocale}
924
+ compact
925
+ estimationType={estimationType}
926
+ isLatestDeletion={latestDeletions?.has(item.id)}
927
+ />
928
+ )
929
+ )}
930
+ </div>
931
+ </motion.div>
932
+ )}
933
+ </AnimatePresence>
934
+ </motion.div>
935
+ );
936
+ }
937
+
938
+ /** Compact version of aggregated entry for display within RapidChangeGroup */
939
+ interface CompactAggregatedEntryProps {
940
+ group: AggregatedActionGroup;
941
+ t: (key: string, options?: { defaultValue?: string }) => string;
942
+ index: number;
943
+ dateLocale: typeof enUS | typeof vi;
944
+ }
945
+
946
+ function CompactAggregatedEntry({
947
+ group,
948
+ t,
949
+ index,
950
+ dateLocale,
951
+ }: CompactAggregatedEntryProps) {
952
+ const { icon, color } = getChangeIcon(group.change_type, null);
953
+ const exactTime = format(new Date(group.changed_at), 'HH:mm', {
954
+ locale: dateLocale,
955
+ });
956
+
957
+ const isAssignee =
958
+ group.change_type === 'assignee_added' ||
959
+ group.change_type === 'assignee_removed';
960
+ const isLabel =
961
+ group.change_type === 'label_added' ||
962
+ group.change_type === 'label_removed';
963
+ const isRemoved =
964
+ group.change_type === 'assignee_removed' ||
965
+ group.change_type === 'label_removed';
966
+
967
+ const formatItemDate = (changedAt: string) => {
968
+ const date = new Date(changedAt);
969
+ return format(date, 'HH:mm, EEEE, d MMMM yyyy', { locale: dateLocale });
970
+ };
971
+
972
+ const getActionLabel = () => {
973
+ switch (group.change_type) {
974
+ case 'assignee_added':
975
+ return t('assigned_multiple', { defaultValue: 'Assigned' });
976
+ case 'assignee_removed':
977
+ return t('unassigned_multiple', { defaultValue: 'Unassigned' });
978
+ case 'label_added':
979
+ return t('added_labels', { defaultValue: 'Added labels' });
980
+ case 'label_removed':
981
+ return t('removed_labels', { defaultValue: 'Removed labels' });
982
+ default:
983
+ return t('updated', { defaultValue: 'Updated' });
984
+ }
985
+ };
986
+
987
+ return (
988
+ <motion.div
989
+ initial={{ opacity: 0, x: -10 }}
990
+ animate={{ opacity: 1, x: 0 }}
991
+ transition={{ delay: index * 0.03 }}
992
+ className="group relative flex gap-3 rounded-lg bg-muted/30 p-3 transition-all"
993
+ >
994
+ {/* Icon indicator */}
995
+ <div
996
+ className={cn(
997
+ 'flex h-7 w-7 shrink-0 items-center justify-center rounded-full',
998
+ color
999
+ )}
1000
+ >
1001
+ {icon}
1002
+ </div>
1003
+
1004
+ {/* Content */}
1005
+ <div className="min-w-0 flex-1">
1006
+ {/* Action label */}
1007
+ <p className="mb-1.5 text-muted-foreground text-sm capitalize">
1008
+ {getActionLabel()}
1009
+ </p>
1010
+
1011
+ {/* Aggregated items with tooltips */}
1012
+ <div className="mb-2 flex flex-wrap items-center gap-1.5">
1013
+ {isAssignee &&
1014
+ group.aggregated_items.map((item, i) => (
1015
+ <Tooltip key={i}>
1016
+ <TooltipTrigger asChild>
1017
+ <Badge
1018
+ variant={isRemoved ? 'outline' : 'secondary'}
1019
+ className={cn(
1020
+ 'cursor-default gap-1.5 text-xs',
1021
+ isRemoved && 'opacity-70'
1022
+ )}
1023
+ >
1024
+ <Avatar className="h-4 w-4">
1025
+ <AvatarImage
1026
+ src={item.avatar_url || undefined}
1027
+ alt={item.name}
1028
+ />
1029
+ <AvatarFallback className="text-[8px]">
1030
+ {item.name
1031
+ .split(' ')
1032
+ .map((n) => n[0])
1033
+ .join('')
1034
+ .toUpperCase()
1035
+ .slice(0, 2)}
1036
+ </AvatarFallback>
1037
+ </Avatar>
1038
+ <span className={cn(isRemoved && 'line-through')}>
1039
+ {item.name}
1040
+ </span>
1041
+ </Badge>
1042
+ </TooltipTrigger>
1043
+ <TooltipContent side="bottom" className="text-xs">
1044
+ <div className="font-medium">{item.name}</div>
1045
+ <div className="text-muted-foreground">
1046
+ {isRemoved
1047
+ ? t('unassigned_at', { defaultValue: 'Unassigned at' })
1048
+ : t('assigned_at', { defaultValue: 'Assigned at' })}
1049
+ : {formatItemDate(item.changed_at)}
1050
+ </div>
1051
+ </TooltipContent>
1052
+ </Tooltip>
1053
+ ))}
1054
+ {isLabel &&
1055
+ group.aggregated_items.map((item, i) => (
1056
+ <Tooltip key={i}>
1057
+ <TooltipTrigger asChild>
1058
+ <Badge
1059
+ variant={isRemoved ? 'outline' : 'secondary'}
1060
+ className={cn(
1061
+ 'cursor-default gap-1.5 text-xs',
1062
+ isRemoved && 'opacity-70'
1063
+ )}
1064
+ style={
1065
+ item.color
1066
+ ? isRemoved
1067
+ ? {
1068
+ borderColor: `${item.color}60`,
1069
+ color: item.color,
1070
+ }
1071
+ : {
1072
+ backgroundColor: `${item.color}20`,
1073
+ borderColor: `${item.color}40`,
1074
+ color: item.color,
1075
+ }
1076
+ : undefined
1077
+ }
1078
+ >
1079
+ {item.color && (
1080
+ <span
1081
+ className={cn(
1082
+ 'h-2 w-2 rounded-full',
1083
+ isRemoved && 'opacity-60'
1084
+ )}
1085
+ style={{ backgroundColor: item.color }}
1086
+ />
1087
+ )}
1088
+ {item.name}
1089
+ </Badge>
1090
+ </TooltipTrigger>
1091
+ <TooltipContent side="bottom" className="text-xs">
1092
+ <div className="font-medium">{item.name}</div>
1093
+ <div className="text-muted-foreground">
1094
+ {isRemoved
1095
+ ? t('removed_at', { defaultValue: 'Removed at' })
1096
+ : t('added_at', { defaultValue: 'Added at' })}
1097
+ : {formatItemDate(item.changed_at)}
1098
+ </div>
1099
+ </TooltipContent>
1100
+ </Tooltip>
1101
+ ))}
1102
+ </div>
1103
+
1104
+ {/* Timestamp */}
1105
+ <p className="text-muted-foreground text-xs">
1106
+ <span className="font-medium">{exactTime}</span>
1107
+ </p>
1108
+ </div>
1109
+ </motion.div>
1110
+ );
1111
+ }
1112
+
1113
+ interface AggregatedActionEntryProps {
1114
+ group: AggregatedActionGroup;
1115
+ wsId: string;
1116
+ t: (key: string, options?: { defaultValue?: string }) => string;
1117
+ index: number;
1118
+ dateLocale: typeof enUS | typeof vi;
1119
+ }
1120
+
1121
+ function AggregatedActionEntry({
1122
+ group,
1123
+ wsId,
1124
+ t,
1125
+ index,
1126
+ dateLocale,
1127
+ }: AggregatedActionEntryProps) {
1128
+ const tasksHref = useTasksHref();
1129
+ const { icon, color, borderColor } = getChangeIconWithBorder(
1130
+ group.change_type,
1131
+ null
1132
+ );
1133
+ const timeAgo = formatDistanceToNow(new Date(group.changed_at), {
1134
+ addSuffix: true,
1135
+ locale: dateLocale,
1136
+ });
1137
+ const exactTime = format(new Date(group.changed_at), 'HH:mm', {
1138
+ locale: dateLocale,
1139
+ });
1140
+
1141
+ const userName =
1142
+ group.user?.name || t('unknown_user', { defaultValue: 'Unknown user' });
1143
+ const userInitials = userName
1144
+ .split(' ')
1145
+ .map((n) => n[0])
1146
+ .join('')
1147
+ .toUpperCase()
1148
+ .slice(0, 2);
1149
+
1150
+ // Get action label based on change type
1151
+ const getActionLabel = () => {
1152
+ switch (group.change_type) {
1153
+ case 'assignee_added':
1154
+ return t('assigned_multiple', { defaultValue: 'assigned' });
1155
+ case 'assignee_removed':
1156
+ return t('unassigned_multiple', { defaultValue: 'unassigned' });
1157
+ case 'label_added':
1158
+ return t('added_labels', { defaultValue: 'added labels' });
1159
+ case 'label_removed':
1160
+ return t('removed_labels', { defaultValue: 'removed labels' });
1161
+ default:
1162
+ return t('updated', { defaultValue: 'updated' });
1163
+ }
1164
+ };
1165
+
1166
+ // Format aggregated items list
1167
+ const formatItemsList = () => {
1168
+ const items = group.aggregated_items;
1169
+ if (items.length === 0) return null;
1170
+
1171
+ const isAssignee =
1172
+ group.change_type === 'assignee_added' ||
1173
+ group.change_type === 'assignee_removed';
1174
+ const isLabel =
1175
+ group.change_type === 'label_added' ||
1176
+ group.change_type === 'label_removed';
1177
+ const isRemoved =
1178
+ group.change_type === 'assignee_removed' ||
1179
+ group.change_type === 'label_removed';
1180
+
1181
+ const formatItemDate = (changedAt: string) => {
1182
+ const date = new Date(changedAt);
1183
+ return format(date, 'HH:mm, EEEE, d MMMM yyyy', { locale: dateLocale });
1184
+ };
1185
+
1186
+ if (isAssignee) {
1187
+ return (
1188
+ <div className="mt-2 flex flex-wrap items-center gap-2">
1189
+ {items.slice(0, 5).map((item) => (
1190
+ <Tooltip key={item.name}>
1191
+ <TooltipTrigger asChild>
1192
+ <div
1193
+ className={cn(
1194
+ 'flex items-center gap-1.5 rounded-full bg-muted/50 px-2 py-0.5',
1195
+ isRemoved && 'line-through opacity-70'
1196
+ )}
1197
+ >
1198
+ <Avatar className="h-4 w-4">
1199
+ <AvatarImage src={item.avatar_url} alt={item.name} />
1200
+ <AvatarFallback className="text-[8px]">
1201
+ {item.name
1202
+ .split(' ')
1203
+ .map((n) => n[0])
1204
+ .join('')
1205
+ .toUpperCase()
1206
+ .slice(0, 2)}
1207
+ </AvatarFallback>
1208
+ </Avatar>
1209
+ <span className="wrap-break-word line-clamp-1 max-w-25 text-xs md:max-w-37.5">
1210
+ {item.name}
1211
+ </span>
1212
+ </div>
1213
+ </TooltipTrigger>
1214
+ <TooltipContent className="text-xs">
1215
+ {isRemoved
1216
+ ? t('unassigned_at', { defaultValue: 'Unassigned at' })
1217
+ : t('assigned_at', { defaultValue: 'Assigned at' })}
1218
+ : {formatItemDate(item.changed_at)}
1219
+ </TooltipContent>
1220
+ </Tooltip>
1221
+ ))}
1222
+ {items.length > 5 && (
1223
+ <span className="text-muted-foreground text-xs">
1224
+ +{items.length - 5}
1225
+ </span>
1226
+ )}
1227
+ </div>
1228
+ );
1229
+ }
1230
+
1231
+ if (isLabel) {
1232
+ return (
1233
+ <div className="mt-2 flex flex-wrap items-center gap-2">
1234
+ {items.slice(0, 5).map((item) => (
1235
+ <Tooltip key={item.name}>
1236
+ <TooltipTrigger asChild>
1237
+ <Badge
1238
+ variant="secondary"
1239
+ className={cn(
1240
+ 'gap-1 text-[11px]',
1241
+ isRemoved && 'line-through opacity-70'
1242
+ )}
1243
+ style={
1244
+ item.color
1245
+ ? {
1246
+ backgroundColor: `${item.color}20`,
1247
+ color: item.color,
1248
+ }
1249
+ : undefined
1250
+ }
1251
+ >
1252
+ <Tag className="h-2.5 w-2.5" />
1253
+ {item.name}
1254
+ </Badge>
1255
+ </TooltipTrigger>
1256
+ <TooltipContent className="text-xs">
1257
+ {isRemoved
1258
+ ? t('removed_at', { defaultValue: 'Removed at' })
1259
+ : t('added_at', { defaultValue: 'Added at' })}
1260
+ : {formatItemDate(item.changed_at)}
1261
+ </TooltipContent>
1262
+ </Tooltip>
1263
+ ))}
1264
+ {items.length > 5 && (
1265
+ <span className="text-muted-foreground text-xs">
1266
+ +{items.length - 5}
1267
+ </span>
1268
+ )}
1269
+ </div>
1270
+ );
1271
+ }
1272
+
1273
+ return null;
1274
+ };
1275
+
1276
+ return (
1277
+ <motion.div
1278
+ initial={{ opacity: 0, x: -10 }}
1279
+ animate={{ opacity: 1, x: 0 }}
1280
+ transition={{ delay: index * 0.03 }}
1281
+ className={cn(
1282
+ 'group relative flex gap-3 overflow-hidden rounded-lg border border-l-2 bg-card p-4 transition-all hover:bg-accent/50',
1283
+ borderColor
1284
+ )}
1285
+ >
1286
+ {/* Icon indicator */}
1287
+ <div
1288
+ className={cn(
1289
+ 'flex h-9 w-9 shrink-0 items-center justify-center rounded-full',
1290
+ color
1291
+ )}
1292
+ >
1293
+ {icon}
1294
+ </div>
1295
+
1296
+ {/* Content */}
1297
+ <div className="min-w-0 flex-1">
1298
+ {/* Header row */}
1299
+ <div className="mb-1.5 flex flex-wrap items-center gap-x-2 gap-y-1">
1300
+ {/* User info */}
1301
+ <div className="flex items-center gap-1.5">
1302
+ <Avatar className="h-5 w-5">
1303
+ <AvatarImage
1304
+ src={group.user?.avatar_url || undefined}
1305
+ alt={userName}
1306
+ />
1307
+ <AvatarFallback className="text-[10px]">
1308
+ {userInitials}
1309
+ </AvatarFallback>
1310
+ </Avatar>
1311
+ <span className="font-medium text-sm">{userName}</span>
1312
+ </div>
1313
+
1314
+ {/* Action */}
1315
+ <span className="text-muted-foreground text-sm">
1316
+ {getActionLabel()}
1317
+ </span>
1318
+
1319
+ {/* Task link */}
1320
+ <div className="flex flex-wrap items-center gap-1.5">
1321
+ <Tooltip>
1322
+ <TooltipTrigger asChild>
1323
+ {group.task_permanently_deleted ? (
1324
+ <span className="wrap-break-word line-clamp-1 max-w-50 font-medium text-muted-foreground text-sm line-through md:max-w-75">
1325
+ {group.task_name}
1326
+ </span>
1327
+ ) : (
1328
+ <Link
1329
+ href={`/${wsId}/tasks/${group.task_id}`}
1330
+ className={cn(
1331
+ 'wrap-break-word line-clamp-1 max-w-50 font-medium text-sm hover:underline md:max-w-75',
1332
+ group.task_deleted_at
1333
+ ? 'text-muted-foreground line-through'
1334
+ : 'text-foreground'
1335
+ )}
1336
+ >
1337
+ {group.task_name}
1338
+ </Link>
1339
+ )}
1340
+ </TooltipTrigger>
1341
+ {group.task_name.length > 30 && (
1342
+ <TooltipContent
1343
+ side="bottom"
1344
+ className="wrap-break-word line-clamp-1 max-w-md text-sm"
1345
+ >
1346
+ {group.task_name}
1347
+ </TooltipContent>
1348
+ )}
1349
+ </Tooltip>
1350
+ {group.task_permanently_deleted ? (
1351
+ <Badge
1352
+ variant="outline"
1353
+ className="gap-1 border-dynamic-red/50 bg-dynamic-red/20 px-1.5 py-0.5 text-dynamic-red text-xs"
1354
+ >
1355
+ <Trash2 className="h-3 w-3" />
1356
+ {t('permanently_deleted', {
1357
+ defaultValue: 'Permanently Deleted',
1358
+ })}
1359
+ </Badge>
1360
+ ) : group.task_deleted_at ? (
1361
+ <Badge
1362
+ variant="outline"
1363
+ className="gap-1 border-dynamic-orange/30 bg-dynamic-orange/10 px-1.5 py-0.5 text-dynamic-orange text-xs"
1364
+ >
1365
+ <RotateCcw className="h-3 w-3" />
1366
+ {t('in_trash', { defaultValue: 'In Trash' })}
1367
+ </Badge>
1368
+ ) : null}
1369
+ {/* Board link */}
1370
+ {group.board_id && group.board_name && (
1371
+ <>
1372
+ <span className="text-muted-foreground text-xs">
1373
+ {t('in_board', { defaultValue: 'in' })}
1374
+ </span>
1375
+ <Tooltip>
1376
+ <TooltipTrigger asChild>
1377
+ <Link
1378
+ href={`/${wsId}${tasksHref(`/boards/${group.board_id}`)}`}
1379
+ className="group/board inline-flex items-center gap-1 text-xs hover:text-foreground"
1380
+ >
1381
+ <LayoutGrid className="h-3 w-3 shrink-0 text-muted-foreground" />
1382
+ <span className="text-muted-foreground group-hover/board:text-foreground group-hover/board:underline">
1383
+ {group.board_name}
1384
+ </span>
1385
+ <ExternalLink className="h-2.5 w-2.5 shrink-0 text-muted-foreground opacity-0 transition-opacity group-hover/board:opacity-100" />
1386
+ </Link>
1387
+ </TooltipTrigger>
1388
+ <TooltipContent side="bottom" className="text-xs">
1389
+ <p>
1390
+ {t('open_board', { defaultValue: 'Open board' })}:{' '}
1391
+ {group.board_name}
1392
+ </p>
1393
+ </TooltipContent>
1394
+ </Tooltip>
1395
+ </>
1396
+ )}
1397
+ </div>
1398
+ </div>
1399
+
1400
+ {/* Aggregated items */}
1401
+ <div className="mb-2">{formatItemsList()}</div>
1402
+
1403
+ {/* Timestamp */}
1404
+ <p className="text-muted-foreground text-xs">
1405
+ <span className="font-medium">{exactTime}</span>
1406
+ <span className="mx-1.5 opacity-50">·</span>
1407
+ <span>{timeAgo}</span>
1408
+ </p>
1409
+ </div>
1410
+ </motion.div>
1411
+ );
1412
+ }
1413
+
1414
+ interface TimelineEntryProps {
1415
+ entry: TaskHistoryLogEntry;
1416
+ wsId: string;
1417
+ locale: string;
1418
+ t: (key: string, options?: { defaultValue?: string }) => string;
1419
+ index: number;
1420
+ dateLocale: typeof enUS | typeof vi;
1421
+ /** Compact mode for nested display within groups */
1422
+ compact?: boolean;
1423
+ /** Estimation type for proper points display */
1424
+ estimationType?: EstimationType;
1425
+ /** Whether this entry is the latest deletion for a permanently deleted task */
1426
+ isLatestDeletion?: boolean;
1427
+ }
1428
+
1429
+ function TimelineEntry({
1430
+ entry,
1431
+ wsId,
1432
+ t,
1433
+ index,
1434
+ dateLocale,
1435
+ compact = false,
1436
+ estimationType,
1437
+ isLatestDeletion,
1438
+ }: TimelineEntryProps) {
1439
+ const tasksHref = useTasksHref();
1440
+ const { icon, color, borderColor } = getChangeIconWithBorder(
1441
+ entry.change_type,
1442
+ entry.field_name,
1443
+ entry.new_value
1444
+ );
1445
+ const description = getChangeDescription(
1446
+ entry,
1447
+ t,
1448
+ estimationType,
1449
+ isLatestDeletion,
1450
+ dateLocale
1451
+ );
1452
+ const timeAgo = formatDistanceToNow(new Date(entry.changed_at), {
1453
+ addSuffix: true,
1454
+ locale: dateLocale,
1455
+ });
1456
+ const exactTime = format(new Date(entry.changed_at), 'HH:mm', {
1457
+ locale: dateLocale,
1458
+ });
1459
+
1460
+ const userName =
1461
+ entry.user?.name || t('unknown_user', { defaultValue: 'Unknown user' });
1462
+ const userInitials = userName
1463
+ .split(' ')
1464
+ .map((n) => n[0])
1465
+ .join('')
1466
+ .toUpperCase()
1467
+ .slice(0, 2);
1468
+
1469
+ return (
1470
+ <motion.div
1471
+ initial={{ opacity: 0, x: -10 }}
1472
+ animate={{ opacity: 1, x: 0 }}
1473
+ transition={{ delay: index * 0.03 }}
1474
+ className={cn(
1475
+ 'group relative flex gap-3 rounded-lg transition-all',
1476
+ compact
1477
+ ? 'bg-muted/30 p-3'
1478
+ : 'border border-l-2 bg-card p-4 hover:bg-accent/50',
1479
+ !compact && borderColor
1480
+ )}
1481
+ >
1482
+ {/* Icon indicator */}
1483
+ <div
1484
+ className={cn(
1485
+ 'flex shrink-0 items-center justify-center rounded-full',
1486
+ compact ? 'h-7 w-7' : 'h-9 w-9',
1487
+ color
1488
+ )}
1489
+ >
1490
+ {icon}
1491
+ </div>
1492
+
1493
+ {/* Content */}
1494
+ <div className="min-w-0 flex-1">
1495
+ {/* Header row */}
1496
+ <div className="mb-1.5 flex flex-wrap items-center gap-x-2 gap-y-1">
1497
+ {/* User info - hidden in compact mode since parent shows it */}
1498
+ {!compact && (
1499
+ <div className="flex items-center gap-1.5">
1500
+ <Avatar className="h-5 w-5">
1501
+ <AvatarImage
1502
+ src={entry.user?.avatar_url || undefined}
1503
+ alt={userName}
1504
+ />
1505
+ <AvatarFallback className="text-[10px]">
1506
+ {userInitials}
1507
+ </AvatarFallback>
1508
+ </Avatar>
1509
+ <span className="font-medium text-sm">{userName}</span>
1510
+ </div>
1511
+ )}
1512
+
1513
+ {/* Action */}
1514
+ <span
1515
+ className={cn(
1516
+ 'text-muted-foreground text-sm',
1517
+ compact && 'capitalize'
1518
+ )}
1519
+ >
1520
+ {description.action}
1521
+ </span>
1522
+
1523
+ {/* Task link - hidden in compact mode since parent shows it */}
1524
+ {!compact && (
1525
+ <div className="flex flex-wrap items-center gap-1.5">
1526
+ <Tooltip>
1527
+ <TooltipTrigger asChild>
1528
+ {entry.task_permanently_deleted ? (
1529
+ <span className="wrap-break-word line-clamp-1 max-w-50 font-medium text-muted-foreground text-sm line-through md:max-w-75">
1530
+ {entry.task_name}
1531
+ </span>
1532
+ ) : (
1533
+ <Link
1534
+ href={`/${wsId}/tasks/${entry.task_id}`}
1535
+ className={cn(
1536
+ 'wrap-break-word line-clamp-1 max-w-50 font-medium text-sm hover:underline md:max-w-75',
1537
+ entry.task_deleted_at
1538
+ ? 'text-muted-foreground line-through'
1539
+ : 'text-foreground'
1540
+ )}
1541
+ >
1542
+ {entry.task_name}
1543
+ </Link>
1544
+ )}
1545
+ </TooltipTrigger>
1546
+ {entry.task_name.length > 30 && (
1547
+ <TooltipContent
1548
+ side="bottom"
1549
+ className="wrap-break-word line-clamp-1 max-w-md text-sm"
1550
+ >
1551
+ {entry.task_name}
1552
+ </TooltipContent>
1553
+ )}
1554
+ </Tooltip>
1555
+ {entry.task_permanently_deleted ? (
1556
+ <Badge
1557
+ variant="outline"
1558
+ className="gap-1 border-dynamic-red/50 bg-dynamic-red/20 px-1.5 py-0.5 text-dynamic-red text-xs"
1559
+ >
1560
+ <Trash2 className="h-3 w-3" />
1561
+ {t('permanently_deleted', {
1562
+ defaultValue: 'Permanently Deleted',
1563
+ })}
1564
+ </Badge>
1565
+ ) : entry.task_deleted_at ? (
1566
+ <Badge
1567
+ variant="outline"
1568
+ className="gap-1 border-dynamic-orange/30 bg-dynamic-orange/10 px-1.5 py-0.5 text-dynamic-orange text-xs"
1569
+ >
1570
+ <RotateCcw className="h-3 w-3" />
1571
+ {t('in_trash', { defaultValue: 'In Trash' })}
1572
+ </Badge>
1573
+ ) : null}
1574
+ {/* Board link */}
1575
+ {entry.board_id && entry.board_name && (
1576
+ <>
1577
+ <span className="text-muted-foreground text-xs">
1578
+ {t('in_board', { defaultValue: 'in' })}
1579
+ </span>
1580
+ <Tooltip>
1581
+ <TooltipTrigger asChild>
1582
+ <Link
1583
+ href={`/${wsId}${tasksHref(`/boards/${entry.board_id}`)}`}
1584
+ className="group/board inline-flex items-center gap-1 text-xs hover:text-foreground"
1585
+ >
1586
+ <LayoutGrid className="h-3 w-3 shrink-0 text-muted-foreground" />
1587
+ <span className="text-muted-foreground group-hover/board:text-foreground group-hover/board:underline">
1588
+ {entry.board_name}
1589
+ </span>
1590
+ <ExternalLink className="h-2.5 w-2.5 shrink-0 text-muted-foreground opacity-0 transition-opacity group-hover/board:opacity-100" />
1591
+ </Link>
1592
+ </TooltipTrigger>
1593
+ <TooltipContent side="bottom" className="text-xs">
1594
+ <p>
1595
+ {t('open_board', { defaultValue: 'Open board' })}:{' '}
1596
+ {entry.board_name}
1597
+ </p>
1598
+ </TooltipContent>
1599
+ </Tooltip>
1600
+ </>
1601
+ )}
1602
+ </div>
1603
+ )}
1604
+ </div>
1605
+
1606
+ {/* Change details */}
1607
+ {description.details && (
1608
+ <div className="mb-2 flex flex-wrap items-center gap-2">
1609
+ {description.noDetailsWrapper ? (
1610
+ <div className="inline-flex items-center gap-2 text-sm">
1611
+ {description.details}
1612
+ </div>
1613
+ ) : (
1614
+ <div className="inline-flex items-center gap-2 rounded-md bg-muted/50 px-2.5 py-1.5 text-sm">
1615
+ {description.details}
1616
+ </div>
1617
+ )}
1618
+ {description.showDescriptionDiff && (
1619
+ <DescriptionDiffViewer
1620
+ oldValue={entry.old_value}
1621
+ newValue={entry.new_value}
1622
+ t={t}
1623
+ />
1624
+ )}
1625
+ {description.showNameDiff && (
1626
+ <TextDiffViewer
1627
+ oldValue={
1628
+ typeof entry.old_value === 'string'
1629
+ ? entry.old_value
1630
+ : String(entry.old_value || '')
1631
+ }
1632
+ newValue={
1633
+ typeof entry.new_value === 'string'
1634
+ ? entry.new_value
1635
+ : String(entry.new_value || '')
1636
+ }
1637
+ t={t}
1638
+ fieldLabel={t('name_changes', {
1639
+ defaultValue: 'Task Name Changes',
1640
+ })}
1641
+ />
1642
+ )}
1643
+ </div>
1644
+ )}
1645
+
1646
+ {/* Timestamp */}
1647
+ <p className="text-muted-foreground text-xs">
1648
+ <span className="font-medium">{exactTime}</span>
1649
+ {!compact && (
1650
+ <>
1651
+ <span className="mx-1.5 opacity-50">·</span>
1652
+ <span>{timeAgo}</span>
1653
+ </>
1654
+ )}
1655
+ </p>
1656
+ </div>
1657
+ </motion.div>
1658
+ );
1659
+ }
1660
+
1661
+ /** Returns icon, background color, and left border color for a change type */
1662
+ function getChangeIconWithBorder(
1663
+ changeType: string,
1664
+ fieldName?: string | null,
1665
+ newValue?: unknown
1666
+ ): { icon: React.ReactNode; color: string; borderColor: string } {
1667
+ const base = getChangeIcon(changeType, fieldName, newValue);
1668
+
1669
+ // Map bg colors to border colors
1670
+ const borderColorMap: Record<string, string> = {
1671
+ 'bg-dynamic-blue/10': 'border-l-dynamic-blue',
1672
+ 'bg-dynamic-purple/10': 'border-l-dynamic-purple',
1673
+ 'bg-dynamic-orange/10': 'border-l-dynamic-orange',
1674
+ 'bg-dynamic-red/10': 'border-l-dynamic-red',
1675
+ 'bg-dynamic-cyan/10': 'border-l-dynamic-cyan',
1676
+ 'bg-dynamic-pink/10': 'border-l-dynamic-pink',
1677
+ 'bg-dynamic-indigo/10': 'border-l-dynamic-indigo',
1678
+ 'bg-dynamic-green/10': 'border-l-dynamic-green',
1679
+ 'bg-dynamic-yellow/10': 'border-l-dynamic-yellow',
1680
+ 'bg-muted': 'border-l-muted-foreground/30',
1681
+ };
1682
+
1683
+ return {
1684
+ ...base,
1685
+ borderColor: borderColorMap[base.color] || 'border-l-border',
1686
+ };
1687
+ }
1688
+
1689
+ function getChangeIcon(
1690
+ changeType: string,
1691
+ fieldName?: string | null,
1692
+ newValue?: unknown
1693
+ ): { icon: React.ReactNode; color: string } {
1694
+ if (changeType === 'field_updated') {
1695
+ switch (fieldName) {
1696
+ case 'name':
1697
+ return {
1698
+ icon: <FileText className="h-4 w-4 text-dynamic-blue" />,
1699
+ color: 'bg-dynamic-blue/10',
1700
+ };
1701
+ case 'description':
1702
+ return {
1703
+ icon: <FileText className="h-4 w-4 text-dynamic-purple" />,
1704
+ color: 'bg-dynamic-purple/10',
1705
+ };
1706
+ case 'priority':
1707
+ return {
1708
+ icon: <Flag className="h-4 w-4 text-dynamic-orange" />,
1709
+ color: 'bg-dynamic-orange/10',
1710
+ };
1711
+ case 'end_date':
1712
+ return {
1713
+ icon: <Calendar className="h-4 w-4 text-dynamic-red" />,
1714
+ color: 'bg-dynamic-red/10',
1715
+ };
1716
+ case 'start_date':
1717
+ return {
1718
+ icon: <Clock className="h-4 w-4 text-dynamic-cyan" />,
1719
+ color: 'bg-dynamic-cyan/10',
1720
+ };
1721
+ case 'estimation_points':
1722
+ return {
1723
+ icon: <Target className="h-4 w-4 text-dynamic-pink" />,
1724
+ color: 'bg-dynamic-pink/10',
1725
+ };
1726
+ case 'list_id':
1727
+ return {
1728
+ icon: <ArrowRight className="h-4 w-4 text-dynamic-indigo" />,
1729
+ color: 'bg-dynamic-indigo/10',
1730
+ };
1731
+ case 'completed':
1732
+ return {
1733
+ icon: <CheckCircle2 className="h-4 w-4 text-dynamic-green" />,
1734
+ color: 'bg-dynamic-green/10',
1735
+ };
1736
+ case 'deleted_at': {
1737
+ // Check if task was deleted or restored based on new_value
1738
+ const wasDeleted =
1739
+ newValue !== null &&
1740
+ newValue !== undefined &&
1741
+ newValue !== '' &&
1742
+ newValue !== 'null';
1743
+ return wasDeleted
1744
+ ? {
1745
+ icon: <RotateCcw className="h-4 w-4 text-dynamic-orange" />,
1746
+ color: 'bg-dynamic-orange/10',
1747
+ }
1748
+ : {
1749
+ icon: <RotateCcw className="h-4 w-4 text-dynamic-green" />,
1750
+ color: 'bg-dynamic-green/10',
1751
+ };
1752
+ }
1753
+ default:
1754
+ return {
1755
+ icon: <CircleDot className="h-4 w-4 text-muted-foreground" />,
1756
+ color: 'bg-muted',
1757
+ };
1758
+ }
1759
+ }
1760
+
1761
+ switch (changeType) {
1762
+ case 'task_created':
1763
+ return {
1764
+ icon: <Plus className="h-4 w-4 text-dynamic-green" />,
1765
+ color: 'bg-dynamic-green/10',
1766
+ };
1767
+ case 'assignee_added':
1768
+ return {
1769
+ icon: <UserPlus className="h-4 w-4 text-dynamic-green" />,
1770
+ color: 'bg-dynamic-green/10',
1771
+ };
1772
+ case 'assignee_removed':
1773
+ return {
1774
+ icon: <UserMinus className="h-4 w-4 text-dynamic-red" />,
1775
+ color: 'bg-dynamic-red/10',
1776
+ };
1777
+ case 'label_added':
1778
+ return {
1779
+ icon: <Tag className="h-4 w-4 text-dynamic-yellow" />,
1780
+ color: 'bg-dynamic-yellow/10',
1781
+ };
1782
+ case 'label_removed':
1783
+ return {
1784
+ icon: <Tag className="h-4 w-4 text-dynamic-orange" />,
1785
+ color: 'bg-dynamic-orange/10',
1786
+ };
1787
+ case 'project_linked':
1788
+ return {
1789
+ icon: <FolderKanban className="h-4 w-4 text-dynamic-purple" />,
1790
+ color: 'bg-dynamic-purple/10',
1791
+ };
1792
+ case 'project_unlinked':
1793
+ return {
1794
+ icon: <FolderKanban className="h-4 w-4 text-dynamic-pink" />,
1795
+ color: 'bg-dynamic-pink/10',
1796
+ };
1797
+ default:
1798
+ return {
1799
+ icon: <CircleDot className="h-4 w-4 text-muted-foreground" />,
1800
+ color: 'bg-muted',
1801
+ };
1802
+ }
1803
+ }
1804
+
1805
+ interface ChangeDescription {
1806
+ action: string;
1807
+ details?: React.ReactNode;
1808
+ showDescriptionDiff?: boolean;
1809
+ showNameDiff?: boolean;
1810
+ /** If true, details won't be wrapped with background/padding (for items with their own styling) */
1811
+ noDetailsWrapper?: boolean;
1812
+ }
1813
+
1814
+ function getChangeDescription(
1815
+ entry: TaskHistoryLogEntry,
1816
+ t: (key: string, options?: { defaultValue?: string }) => string,
1817
+ estimationType?: EstimationType,
1818
+ isLatestDeletion?: boolean,
1819
+ dateLocale?: typeof enUS | typeof vi
1820
+ ): ChangeDescription {
1821
+ // Handle task_created
1822
+ if (entry.change_type === 'task_created') {
1823
+ const metadata = entry.metadata as Record<string, unknown> | null;
1824
+ const hasDescription = !!metadata?.description;
1825
+ const hasEstimation = metadata?.estimation_points != null;
1826
+ const hasStartDate = !!metadata?.start_date;
1827
+ const hasEndDate = !!metadata?.end_date;
1828
+ const listName = metadata?.list_name as string | undefined;
1829
+
1830
+ const badges: React.ReactNode[] = [];
1831
+ if (listName) {
1832
+ badges.push(
1833
+ <Badge key="list" variant="outline" className="gap-1 text-xs">
1834
+ <ArrowRight className="h-3 w-3" />
1835
+ {listName}
1836
+ </Badge>
1837
+ );
1838
+ }
1839
+ if (hasDescription) {
1840
+ badges.push(
1841
+ <Badge key="desc" variant="secondary" className="text-xs">
1842
+ {t('with_description', { defaultValue: 'with description' })}
1843
+ </Badge>
1844
+ );
1845
+ }
1846
+ if (hasEstimation) {
1847
+ const points = metadata.estimation_points as number;
1848
+ const displayPoints = estimationType
1849
+ ? mapEstimationPoints(points, estimationType)
1850
+ : `${points} ${t('points', { defaultValue: 'pts' })}`;
1851
+ badges.push(
1852
+ <Badge key="est" variant="secondary" className="gap-1 text-xs">
1853
+ <Target className="h-3 w-3" />
1854
+ {displayPoints}
1855
+ </Badge>
1856
+ );
1857
+ }
1858
+ if (hasStartDate) {
1859
+ const startDate = new Date(metadata.start_date as string);
1860
+ badges.push(
1861
+ <Tooltip key="start">
1862
+ <TooltipTrigger asChild>
1863
+ <Badge variant="secondary" className="cursor-default gap-1 text-xs">
1864
+ <Clock className="h-3 w-3" />
1865
+ {format(startDate, 'd MMM', { locale: dateLocale })}
1866
+ </Badge>
1867
+ </TooltipTrigger>
1868
+ <TooltipContent side="bottom" className="text-xs">
1869
+ <div className="font-medium">
1870
+ {t('field_name.start_date', { defaultValue: 'Start Date' })}
1871
+ </div>
1872
+ <div className="text-muted-foreground">
1873
+ {format(startDate, 'HH:mm, EEEE, d MMMM yyyy', {
1874
+ locale: dateLocale,
1875
+ })}
1876
+ </div>
1877
+ </TooltipContent>
1878
+ </Tooltip>
1879
+ );
1880
+ }
1881
+ if (hasEndDate) {
1882
+ const endDate = new Date(metadata.end_date as string);
1883
+ badges.push(
1884
+ <Tooltip key="end">
1885
+ <TooltipTrigger asChild>
1886
+ <Badge variant="secondary" className="cursor-default gap-1 text-xs">
1887
+ <Calendar className="h-3 w-3" />
1888
+ {format(endDate, 'd MMM', { locale: dateLocale })}
1889
+ </Badge>
1890
+ </TooltipTrigger>
1891
+ <TooltipContent side="bottom" className="text-xs">
1892
+ <div className="font-medium">
1893
+ {t('field_name.end_date', { defaultValue: 'Due Date' })}
1894
+ </div>
1895
+ <div className="text-muted-foreground">
1896
+ {format(endDate, 'HH:mm, EEEE, d MMMM yyyy', {
1897
+ locale: dateLocale,
1898
+ })}
1899
+ </div>
1900
+ </TooltipContent>
1901
+ </Tooltip>
1902
+ );
1903
+ }
1904
+
1905
+ return {
1906
+ action: t('task_created', { defaultValue: 'created' }),
1907
+ details:
1908
+ badges.length > 0 ? (
1909
+ <div className="flex flex-wrap items-center gap-1.5">{badges}</div>
1910
+ ) : undefined,
1911
+ noDetailsWrapper: true,
1912
+ };
1913
+ }
1914
+
1915
+ if (entry.change_type === 'field_updated') {
1916
+ const fieldLabels: Record<string, string> = {
1917
+ name: t('field_updated.name', { defaultValue: 'updated the name of' }),
1918
+ description: t('field_updated.description', {
1919
+ defaultValue: 'updated description for',
1920
+ }),
1921
+ priority: t('field_updated.priority', {
1922
+ defaultValue: 'changed priority of',
1923
+ }),
1924
+ end_date: t('field_updated.end_date', {
1925
+ defaultValue: 'updated due date for',
1926
+ }),
1927
+ start_date: t('field_updated.start_date', {
1928
+ defaultValue: 'set start date for',
1929
+ }),
1930
+ estimation_points: t('field_updated.estimation_points', {
1931
+ defaultValue: 'updated points for',
1932
+ }),
1933
+ list_id: t('field_updated.list_id', { defaultValue: 'moved' }),
1934
+ completed: t('field_updated.completed', {
1935
+ defaultValue: 'changed status of',
1936
+ }),
1937
+ deleted_at: t('field_updated.deleted_at', {
1938
+ defaultValue: 'deleted',
1939
+ }),
1940
+ };
1941
+
1942
+ const action =
1943
+ fieldLabels[entry.field_name || ''] ||
1944
+ t('field_updated.unknown', { defaultValue: 'updated' });
1945
+
1946
+ // For description changes, show a simplified view with diff button
1947
+ if (entry.field_name === 'description') {
1948
+ const oldText = getDescriptionText(entry.old_value);
1949
+ const newText = getDescriptionText(entry.new_value);
1950
+ const oldSummary = oldText
1951
+ ? oldText.length > 30
1952
+ ? `${oldText.slice(0, 27)}...`
1953
+ : oldText
1954
+ : t('value.empty', { defaultValue: 'Empty' });
1955
+ const newSummary = newText
1956
+ ? newText.length > 30
1957
+ ? `${newText.slice(0, 27)}...`
1958
+ : newText
1959
+ : t('value.empty', { defaultValue: 'Empty' });
1960
+
1961
+ const details = (
1962
+ <div className="flex flex-wrap items-center gap-1.5">
1963
+ <span className="text-muted-foreground line-through">
1964
+ {oldSummary}
1965
+ </span>
1966
+ <ArrowRight className="h-3 w-3 shrink-0 text-muted-foreground" />
1967
+ <span className="font-medium">{newSummary}</span>
1968
+ </div>
1969
+ );
1970
+
1971
+ return { action, details, showDescriptionDiff: true };
1972
+ }
1973
+
1974
+ // For name changes, show line-clamp-1 wrap-break-wordd values with diff button for long names
1975
+ if (entry.field_name === 'name') {
1976
+ const oldName = String(entry.old_value || '');
1977
+ const newName = String(entry.new_value || '');
1978
+ const isLongChange = oldName.length > 40 || newName.length > 40;
1979
+
1980
+ const oldSummary =
1981
+ oldName.length > 40 ? `${oldName.slice(0, 37)}...` : oldName;
1982
+ const newSummary =
1983
+ newName.length > 40 ? `${newName.slice(0, 37)}...` : newName;
1984
+
1985
+ const details = (
1986
+ <>
1987
+ <Tooltip>
1988
+ <TooltipTrigger asChild>
1989
+ <span className="wrap-break-word line-clamp-1 max-w-37.5 text-muted-foreground line-through">
1990
+ {oldSummary}
1991
+ </span>
1992
+ </TooltipTrigger>
1993
+ {oldName.length > 40 && (
1994
+ <TooltipContent
1995
+ side="bottom"
1996
+ className="wrap-break-word max-w-md truncate"
1997
+ >
1998
+ {oldName}
1999
+ </TooltipContent>
2000
+ )}
2001
+ </Tooltip>
2002
+ <ArrowRight className="h-3 w-3 shrink-0 text-muted-foreground" />
2003
+ <Tooltip>
2004
+ <TooltipTrigger asChild>
2005
+ <span className="wrap-break-word line-clamp-1 max-w-37.5 font-medium">
2006
+ {newSummary}
2007
+ </span>
2008
+ </TooltipTrigger>
2009
+ {newName.length > 40 && (
2010
+ <TooltipContent
2011
+ side="bottom"
2012
+ className="wrap-break-word max-w-md truncate"
2013
+ >
2014
+ {newName}
2015
+ </TooltipContent>
2016
+ )}
2017
+ </Tooltip>
2018
+ </>
2019
+ );
2020
+
2021
+ return { action, details, showNameDiff: isLongChange };
2022
+ }
2023
+
2024
+ // For priority changes, show styled priority badges
2025
+ if (entry.field_name === 'priority') {
2026
+ const oldPriorityBadge = renderPriorityBadge(
2027
+ entry.old_value as string | number | null
2028
+ );
2029
+ const newPriorityBadge = renderPriorityBadge(
2030
+ entry.new_value as string | number | null
2031
+ );
2032
+
2033
+ const details = (
2034
+ <div className="flex items-center gap-2">
2035
+ {oldPriorityBadge ? (
2036
+ <span className="opacity-60">{oldPriorityBadge}</span>
2037
+ ) : (
2038
+ <span className="text-muted-foreground text-xs">
2039
+ {t('value.none', { defaultValue: 'None' })}
2040
+ </span>
2041
+ )}
2042
+ <ArrowRight className="h-3 w-3 text-muted-foreground" />
2043
+ {newPriorityBadge || (
2044
+ <span className="text-muted-foreground text-xs">
2045
+ {t('value.none', { defaultValue: 'None' })}
2046
+ </span>
2047
+ )}
2048
+ </div>
2049
+ );
2050
+
2051
+ return { action, details, noDetailsWrapper: true };
2052
+ }
2053
+
2054
+ // For list_id changes, show actual column names from metadata
2055
+ if (entry.field_name === 'list_id') {
2056
+ const metadata = entry.metadata as Record<string, unknown> | null;
2057
+ const oldListName =
2058
+ (metadata?.old_list_name as string) ||
2059
+ t('value.unknown_column', { defaultValue: 'Unknown' });
2060
+ const newListName =
2061
+ (metadata?.new_list_name as string) ||
2062
+ t('value.unknown_column', { defaultValue: 'Unknown' });
2063
+
2064
+ const details = (
2065
+ <div className="flex flex-wrap items-center gap-1.5">
2066
+ <Badge variant="outline" className="text-xs opacity-70">
2067
+ {oldListName}
2068
+ </Badge>
2069
+ <ArrowRight className="h-3 w-3 shrink-0 text-muted-foreground" />
2070
+ <Badge variant="secondary" className="text-xs">
2071
+ {newListName}
2072
+ </Badge>
2073
+ </div>
2074
+ );
2075
+
2076
+ return { action, details, noDetailsWrapper: true };
2077
+ }
2078
+
2079
+ // For estimation_points changes, use proper estimation type formatting
2080
+ if (entry.field_name === 'estimation_points') {
2081
+ const oldPoints = entry.old_value as number | null;
2082
+ const newPoints = entry.new_value as number | null;
2083
+
2084
+ const formatPoints = (points: number | null) => {
2085
+ if (points === null || points === undefined) {
2086
+ return t('value.none', { defaultValue: 'None' });
2087
+ }
2088
+ if (estimationType) {
2089
+ return mapEstimationPoints(points, estimationType);
2090
+ }
2091
+ return `${points} ${t('points', { defaultValue: 'pts' })}`;
2092
+ };
2093
+
2094
+ const details = (
2095
+ <div className="flex items-center gap-2">
2096
+ <Badge variant="outline" className="gap-1 text-xs opacity-70">
2097
+ <Target className="h-3 w-3" />
2098
+ {formatPoints(oldPoints)}
2099
+ </Badge>
2100
+ <ArrowRight className="h-3 w-3 text-muted-foreground" />
2101
+ <Badge variant="secondary" className="gap-1 text-xs">
2102
+ <Target className="h-3 w-3" />
2103
+ {formatPoints(newPoints)}
2104
+ </Badge>
2105
+ </div>
2106
+ );
2107
+
2108
+ return { action, details, noDetailsWrapper: true };
2109
+ }
2110
+
2111
+ // For deleted_at changes, show task moved to trash/restored status
2112
+ // If the task is now permanently deleted, show that instead of "In Trash"
2113
+ if (entry.field_name === 'deleted_at') {
2114
+ // Check if task was deleted (new_value has a timestamp) or restored (new_value is null/undefined/empty)
2115
+ // Handle various representations of null from JSONB: null, undefined, empty string, "null" string
2116
+ const newValue = entry.new_value;
2117
+ const wasDeleted =
2118
+ newValue !== null &&
2119
+ newValue !== undefined &&
2120
+ newValue !== '' &&
2121
+ newValue !== 'null';
2122
+
2123
+ // If the task was subsequently permanently deleted, show that status
2124
+ // Otherwise show the soft-delete (moved to trash) status
2125
+ const showPermanentlyDeleted = wasDeleted && isLatestDeletion;
2126
+
2127
+ const action = wasDeleted
2128
+ ? showPermanentlyDeleted
2129
+ ? t('field_updated.deleted_at', {
2130
+ defaultValue: 'permanently deleted',
2131
+ })
2132
+ : t('field_updated.deleted_at', { defaultValue: 'moved to trash' })
2133
+ : t('task_restored', { defaultValue: 'restored' });
2134
+
2135
+ const details = (
2136
+ <Badge
2137
+ variant="outline"
2138
+ className={cn(
2139
+ 'gap-1 text-xs',
2140
+ showPermanentlyDeleted
2141
+ ? 'border-dynamic-red/50 bg-dynamic-red/20 text-dynamic-red'
2142
+ : wasDeleted
2143
+ ? 'border-dynamic-orange/30 bg-dynamic-orange/10 text-dynamic-orange'
2144
+ : 'border-dynamic-green/30 bg-dynamic-green/10 text-dynamic-green'
2145
+ )}
2146
+ >
2147
+ {showPermanentlyDeleted ? (
2148
+ <Trash2 className="h-3 w-3" />
2149
+ ) : wasDeleted ? (
2150
+ <RotateCcw className="h-3 w-3" />
2151
+ ) : (
2152
+ <RotateCcw className="h-3 w-3" />
2153
+ )}
2154
+ {showPermanentlyDeleted
2155
+ ? t('permanently_deleted', { defaultValue: 'Permanently Deleted' })
2156
+ : wasDeleted
2157
+ ? t('in_trash', { defaultValue: 'In Trash' })
2158
+ : t('status.restored', { defaultValue: 'Restored' })}
2159
+ </Badge>
2160
+ );
2161
+
2162
+ return { action, details, noDetailsWrapper: true };
2163
+ }
2164
+
2165
+ // Format the old and new values
2166
+ const oldValue = formatValue(entry.old_value, entry.field_name, t);
2167
+ const newValue = formatValue(entry.new_value, entry.field_name, t);
2168
+
2169
+ const details = (
2170
+ <>
2171
+ <span className="text-muted-foreground line-through">{oldValue}</span>
2172
+ <ArrowRight className="h-3 w-3 text-muted-foreground" />
2173
+ <span className="font-medium">{newValue}</span>
2174
+ </>
2175
+ );
2176
+
2177
+ return { action, details };
2178
+ }
2179
+
2180
+ // Handle relationship changes
2181
+ switch (entry.change_type) {
2182
+ case 'assignee_added': {
2183
+ // Extract assignee data from new_value or metadata
2184
+ const assigneeData = entry.new_value as {
2185
+ user_id?: string;
2186
+ user_name?: string;
2187
+ avatar_url?: string;
2188
+ } | null;
2189
+ const assigneeName =
2190
+ assigneeData?.user_name ||
2191
+ entry.metadata?.assignee_name ||
2192
+ t('unknown_user', { defaultValue: 'Unknown user' });
2193
+ const assigneeAvatar = assigneeData?.avatar_url;
2194
+ const assigneeInitials = assigneeName
2195
+ .split(' ')
2196
+ .map((n: string) => n[0])
2197
+ .join('')
2198
+ .toUpperCase()
2199
+ .slice(0, 2);
2200
+
2201
+ return {
2202
+ action: t('assignee_added', { defaultValue: 'assigned' }),
2203
+ details: (
2204
+ <Badge variant="secondary" className="gap-1.5 text-xs">
2205
+ <Avatar className="h-4 w-4">
2206
+ <AvatarImage
2207
+ src={assigneeAvatar || undefined}
2208
+ alt={assigneeName}
2209
+ />
2210
+ <AvatarFallback className="text-[8px]">
2211
+ {assigneeInitials}
2212
+ </AvatarFallback>
2213
+ </Avatar>
2214
+ {assigneeName}
2215
+ </Badge>
2216
+ ),
2217
+ noDetailsWrapper: true,
2218
+ };
2219
+ }
2220
+ case 'assignee_removed': {
2221
+ // Extract assignee data from old_value or metadata
2222
+ const assigneeData = entry.old_value as {
2223
+ user_id?: string;
2224
+ user_name?: string;
2225
+ avatar_url?: string;
2226
+ } | null;
2227
+ const assigneeName =
2228
+ assigneeData?.user_name ||
2229
+ entry.metadata?.assignee_name ||
2230
+ t('unknown_user', { defaultValue: 'Unknown user' });
2231
+ const assigneeAvatar = assigneeData?.avatar_url;
2232
+ const assigneeInitials = assigneeName
2233
+ .split(' ')
2234
+ .map((n: string) => n[0])
2235
+ .join('')
2236
+ .toUpperCase()
2237
+ .slice(0, 2);
2238
+
2239
+ return {
2240
+ action: t('assignee_removed', { defaultValue: 'unassigned' }),
2241
+ details: (
2242
+ <Badge variant="outline" className="gap-1.5 text-xs">
2243
+ <Avatar className="h-4 w-4 opacity-60">
2244
+ <AvatarImage
2245
+ src={assigneeAvatar || undefined}
2246
+ alt={assigneeName}
2247
+ />
2248
+ <AvatarFallback className="text-[8px]">
2249
+ {assigneeInitials}
2250
+ </AvatarFallback>
2251
+ </Avatar>
2252
+ {assigneeName}
2253
+ </Badge>
2254
+ ),
2255
+ noDetailsWrapper: true,
2256
+ };
2257
+ }
2258
+ case 'label_added': {
2259
+ // Extract label data from new_value or metadata
2260
+ const labelData = entry.new_value as {
2261
+ id?: string;
2262
+ name?: string;
2263
+ color?: string;
2264
+ } | null;
2265
+ const labelName =
2266
+ labelData?.name ||
2267
+ entry.metadata?.label_name ||
2268
+ t('unknown_label', { defaultValue: 'Unknown label' });
2269
+ const labelColor = labelData?.color || entry.metadata?.label_color;
2270
+
2271
+ return {
2272
+ action: t('label_added', { defaultValue: 'added label to' }),
2273
+ details: (
2274
+ <Badge
2275
+ variant="secondary"
2276
+ className="gap-1.5 text-xs"
2277
+ style={
2278
+ labelColor
2279
+ ? {
2280
+ backgroundColor: `${labelColor}20`,
2281
+ borderColor: `${labelColor}40`,
2282
+ color: labelColor,
2283
+ }
2284
+ : undefined
2285
+ }
2286
+ >
2287
+ {labelColor && (
2288
+ <span
2289
+ className="h-2 w-2 rounded-full"
2290
+ style={{ backgroundColor: labelColor }}
2291
+ />
2292
+ )}
2293
+ {labelName}
2294
+ </Badge>
2295
+ ),
2296
+ noDetailsWrapper: true,
2297
+ };
2298
+ }
2299
+ case 'label_removed': {
2300
+ // Extract label data from old_value or metadata
2301
+ const labelData = entry.old_value as {
2302
+ id?: string;
2303
+ name?: string;
2304
+ color?: string;
2305
+ } | null;
2306
+ const labelName =
2307
+ labelData?.name ||
2308
+ entry.metadata?.label_name ||
2309
+ t('unknown_label', { defaultValue: 'Unknown label' });
2310
+ const labelColor = labelData?.color || entry.metadata?.label_color;
2311
+
2312
+ return {
2313
+ action: t('label_removed', { defaultValue: 'removed label from' }),
2314
+ details: (
2315
+ <Badge
2316
+ variant="outline"
2317
+ className="gap-1.5 text-xs opacity-70"
2318
+ style={
2319
+ labelColor
2320
+ ? {
2321
+ borderColor: `${labelColor}60`,
2322
+ color: labelColor,
2323
+ }
2324
+ : undefined
2325
+ }
2326
+ >
2327
+ {labelColor && (
2328
+ <span
2329
+ className="h-2 w-2 rounded-full opacity-60"
2330
+ style={{ backgroundColor: labelColor }}
2331
+ />
2332
+ )}
2333
+ {labelName}
2334
+ </Badge>
2335
+ ),
2336
+ noDetailsWrapper: true,
2337
+ };
2338
+ }
2339
+ case 'project_linked': {
2340
+ // Extract project name from various possible locations
2341
+ const newValueData =
2342
+ typeof entry.new_value === 'string'
2343
+ ? (() => {
2344
+ try {
2345
+ return JSON.parse(entry.new_value);
2346
+ } catch {
2347
+ return null;
2348
+ }
2349
+ })()
2350
+ : (entry.new_value as Record<string, unknown> | null);
2351
+
2352
+ const projectName =
2353
+ (entry.metadata?.project_name as string) ||
2354
+ newValueData?.project_name ||
2355
+ newValueData?.name ||
2356
+ // If new_value is a plain string (not JSON), use it directly
2357
+ (typeof entry.new_value === 'string' && !newValueData
2358
+ ? entry.new_value
2359
+ : null) ||
2360
+ t('unknown_project', { defaultValue: 'Unknown project' });
2361
+
2362
+ return {
2363
+ action: t('project_linked', { defaultValue: 'linked' }),
2364
+ details: (
2365
+ <>
2366
+ <span className="text-muted-foreground">
2367
+ {t('to_project', { defaultValue: 'to' })}
2368
+ </span>
2369
+ <Badge variant="secondary" className="gap-1 text-xs">
2370
+ <FolderKanban className="h-3 w-3" />
2371
+ {projectName}
2372
+ </Badge>
2373
+ </>
2374
+ ),
2375
+ noDetailsWrapper: true,
2376
+ };
2377
+ }
2378
+ case 'project_unlinked': {
2379
+ // Extract project name from various possible locations
2380
+ const oldValueData =
2381
+ typeof entry.old_value === 'string'
2382
+ ? (() => {
2383
+ try {
2384
+ return JSON.parse(entry.old_value);
2385
+ } catch {
2386
+ return null;
2387
+ }
2388
+ })()
2389
+ : (entry.old_value as Record<string, unknown> | null);
2390
+
2391
+ const projectName =
2392
+ (entry.metadata?.project_name as string) ||
2393
+ oldValueData?.project_name ||
2394
+ oldValueData?.name ||
2395
+ // If old_value is a plain string (not JSON), use it directly
2396
+ (typeof entry.old_value === 'string' && !oldValueData
2397
+ ? entry.old_value
2398
+ : null) ||
2399
+ t('unknown_project', { defaultValue: 'Unknown project' });
2400
+
2401
+ return {
2402
+ action: t('project_unlinked', { defaultValue: 'unlinked' }),
2403
+ details: (
2404
+ <>
2405
+ <span className="text-muted-foreground">
2406
+ {t('from_project', { defaultValue: 'from' })}
2407
+ </span>
2408
+ <Badge variant="outline" className="gap-1 text-xs opacity-70">
2409
+ <FolderKanban className="h-3 w-3" />
2410
+ {projectName}
2411
+ </Badge>
2412
+ </>
2413
+ ),
2414
+ noDetailsWrapper: true,
2415
+ };
2416
+ }
2417
+ default:
2418
+ return {
2419
+ action: t('unknown_change', { defaultValue: 'made a change to' }),
2420
+ };
2421
+ }
2422
+ }
2423
+
2424
+ function formatValue(
2425
+ value: unknown,
2426
+ fieldName?: string | null,
2427
+ t?: (key: string, options?: { defaultValue?: string }) => string
2428
+ ): string {
2429
+ const translate =
2430
+ t ||
2431
+ ((key: string, opts?: { defaultValue?: string }) =>
2432
+ opts?.defaultValue || key);
2433
+
2434
+ if (value === null || value === undefined) {
2435
+ return translate('value.none', { defaultValue: 'None' });
2436
+ }
2437
+
2438
+ // Handle boolean values
2439
+ if (typeof value === 'boolean') {
2440
+ return value
2441
+ ? translate('value.yes', { defaultValue: 'Yes' })
2442
+ : translate('value.no', { defaultValue: 'No' });
2443
+ }
2444
+
2445
+ // Handle dates
2446
+ if (fieldName === 'end_date' || fieldName === 'start_date') {
2447
+ try {
2448
+ return format(new Date(value as string), 'MMM d, yyyy');
2449
+ } catch {
2450
+ return String(value);
2451
+ }
2452
+ }
2453
+
2454
+ // Handle priority
2455
+ if (fieldName === 'priority') {
2456
+ const priorityLabels: Record<number, string> = {
2457
+ 1: translate('priority.low', { defaultValue: 'Low' }),
2458
+ 2: translate('priority.medium', { defaultValue: 'Medium' }),
2459
+ 3: translate('priority.high', { defaultValue: 'High' }),
2460
+ 4: translate('priority.urgent', { defaultValue: 'Urgent' }),
2461
+ };
2462
+ return priorityLabels[value as number] || String(value);
2463
+ }
2464
+
2465
+ // Handle completed status
2466
+ if (fieldName === 'completed') {
2467
+ return value
2468
+ ? translate('status.completed', { defaultValue: 'Completed' })
2469
+ : translate('status.pending', { defaultValue: 'Pending' });
2470
+ }
2471
+
2472
+ // Handle list_id
2473
+ if (fieldName === 'list_id') {
2474
+ return translate('value.column', { defaultValue: 'Column' });
2475
+ }
2476
+
2477
+ // Default: convert to string
2478
+ const strValue = String(value);
2479
+ if (strValue.length > 40) {
2480
+ return `${strValue.slice(0, 37)}...`;
2481
+ }
2482
+ return strValue;
2483
+ }