@open-mercato/core 0.4.5-develop-754ef4d2f0 → 0.4.5-develop-9f9549ebc8

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 (434) hide show
  1. package/dist/generated/entities/message/index.js +65 -0
  2. package/dist/generated/entities/message/index.js.map +7 -0
  3. package/dist/generated/entities/message_access_token/index.js +19 -0
  4. package/dist/generated/entities/message_access_token/index.js.map +7 -0
  5. package/dist/generated/entities/message_confirmation/index.js +21 -0
  6. package/dist/generated/entities/message_confirmation/index.js.map +7 -0
  7. package/dist/generated/entities/message_object/index.js +23 -0
  8. package/dist/generated/entities/message_object/index.js.map +7 -0
  9. package/dist/generated/entities/message_recipient/index.js +31 -0
  10. package/dist/generated/entities/message_recipient/index.js.map +7 -0
  11. package/dist/generated/entities.ids.generated.js +8 -0
  12. package/dist/generated/entities.ids.generated.js.map +2 -2
  13. package/dist/generated/entity-fields-registry.js +10 -0
  14. package/dist/generated/entity-fields-registry.js.map +2 -2
  15. package/dist/modules/auth/components/AclEditor.js +4 -2
  16. package/dist/modules/auth/components/AclEditor.js.map +2 -2
  17. package/dist/modules/auth/frontend/reset.js +3 -3
  18. package/dist/modules/auth/frontend/reset.js.map +2 -2
  19. package/dist/modules/customers/backend/customers/deals/[id]/page.js +27 -8
  20. package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
  21. package/dist/modules/customers/lib/messageObjectPreviews.js +131 -0
  22. package/dist/modules/customers/lib/messageObjectPreviews.js.map +7 -0
  23. package/dist/modules/customers/message-objects.js +71 -0
  24. package/dist/modules/customers/message-objects.js.map +7 -0
  25. package/dist/modules/customers/widgets/messages/CustomerMessageObjectDetail.js +51 -0
  26. package/dist/modules/customers/widgets/messages/CustomerMessageObjectDetail.js.map +7 -0
  27. package/dist/modules/customers/widgets/messages/CustomerMessageObjectPreview.js +35 -0
  28. package/dist/modules/customers/widgets/messages/CustomerMessageObjectPreview.js.map +7 -0
  29. package/dist/modules/customers/widgets/messages/index.js +7 -0
  30. package/dist/modules/customers/widgets/messages/index.js.map +7 -0
  31. package/dist/modules/dashboards/components/WidgetVisibilityEditor.js +35 -6
  32. package/dist/modules/dashboards/components/WidgetVisibilityEditor.js.map +2 -2
  33. package/dist/modules/messages/acl.js +15 -0
  34. package/dist/modules/messages/acl.js.map +7 -0
  35. package/dist/modules/messages/api/[id]/actions/[actionId]/route.js +92 -0
  36. package/dist/modules/messages/api/[id]/actions/[actionId]/route.js.map +7 -0
  37. package/dist/modules/messages/api/[id]/archive/route.js +120 -0
  38. package/dist/modules/messages/api/[id]/archive/route.js.map +7 -0
  39. package/dist/modules/messages/api/[id]/attachments/route.js +195 -0
  40. package/dist/modules/messages/api/[id]/attachments/route.js.map +7 -0
  41. package/dist/modules/messages/api/[id]/confirmation/route.js +67 -0
  42. package/dist/modules/messages/api/[id]/confirmation/route.js.map +7 -0
  43. package/dist/modules/messages/api/[id]/conversation/archive/route.js +68 -0
  44. package/dist/modules/messages/api/[id]/conversation/archive/route.js.map +7 -0
  45. package/dist/modules/messages/api/[id]/conversation/read/route.js +68 -0
  46. package/dist/modules/messages/api/[id]/conversation/read/route.js.map +7 -0
  47. package/dist/modules/messages/api/[id]/conversation/route.js +68 -0
  48. package/dist/modules/messages/api/[id]/conversation/route.js.map +7 -0
  49. package/dist/modules/messages/api/[id]/forward/route.js +85 -0
  50. package/dist/modules/messages/api/[id]/forward/route.js.map +7 -0
  51. package/dist/modules/messages/api/[id]/forward-preview/route.js +70 -0
  52. package/dist/modules/messages/api/[id]/forward-preview/route.js.map +7 -0
  53. package/dist/modules/messages/api/[id]/read/route.js +120 -0
  54. package/dist/modules/messages/api/[id]/read/route.js.map +7 -0
  55. package/dist/modules/messages/api/[id]/reply/route.js +87 -0
  56. package/dist/modules/messages/api/[id]/reply/route.js.map +7 -0
  57. package/dist/modules/messages/api/[id]/route.js +350 -0
  58. package/dist/modules/messages/api/[id]/route.js.map +7 -0
  59. package/dist/modules/messages/api/object-types/route.js +54 -0
  60. package/dist/modules/messages/api/object-types/route.js.map +7 -0
  61. package/dist/modules/messages/api/openapi.js +261 -0
  62. package/dist/modules/messages/api/openapi.js.map +7 -0
  63. package/dist/modules/messages/api/route.js +262 -0
  64. package/dist/modules/messages/api/route.js.map +7 -0
  65. package/dist/modules/messages/api/token/[token]/route.js +99 -0
  66. package/dist/modules/messages/api/token/[token]/route.js.map +7 -0
  67. package/dist/modules/messages/api/types/route.js +40 -0
  68. package/dist/modules/messages/api/types/route.js.map +7 -0
  69. package/dist/modules/messages/api/unread-count/route.js +43 -0
  70. package/dist/modules/messages/api/unread-count/route.js.map +7 -0
  71. package/dist/modules/messages/backend/messages/[id]/page.js +10 -0
  72. package/dist/modules/messages/backend/messages/[id]/page.js.map +7 -0
  73. package/dist/modules/messages/backend/messages/[id]/page.meta.js +16 -0
  74. package/dist/modules/messages/backend/messages/[id]/page.meta.js.map +7 -0
  75. package/dist/modules/messages/backend/messages/compose/page.js +10 -0
  76. package/dist/modules/messages/backend/messages/compose/page.js.map +7 -0
  77. package/dist/modules/messages/backend/messages/compose/page.meta.js +17 -0
  78. package/dist/modules/messages/backend/messages/compose/page.meta.js.map +7 -0
  79. package/dist/modules/messages/backend/page.js +10 -0
  80. package/dist/modules/messages/backend/page.js.map +7 -0
  81. package/dist/modules/messages/backend/page.meta.js +33 -0
  82. package/dist/modules/messages/backend/page.meta.js.map +7 -0
  83. package/dist/modules/messages/commands/actions.js +265 -0
  84. package/dist/modules/messages/commands/actions.js.map +7 -0
  85. package/dist/modules/messages/commands/attachments.js +217 -0
  86. package/dist/modules/messages/commands/attachments.js.map +7 -0
  87. package/dist/modules/messages/commands/confirmations.js +151 -0
  88. package/dist/modules/messages/commands/confirmations.js.map +7 -0
  89. package/dist/modules/messages/commands/conversation.js +240 -0
  90. package/dist/modules/messages/commands/conversation.js.map +7 -0
  91. package/dist/modules/messages/commands/messages.js +748 -0
  92. package/dist/modules/messages/commands/messages.js.map +7 -0
  93. package/dist/modules/messages/commands/recipients.js +259 -0
  94. package/dist/modules/messages/commands/recipients.js.map +7 -0
  95. package/dist/modules/messages/commands/shared.js +258 -0
  96. package/dist/modules/messages/commands/shared.js.map +7 -0
  97. package/dist/modules/messages/commands/tokens.js +69 -0
  98. package/dist/modules/messages/commands/tokens.js.map +7 -0
  99. package/dist/modules/messages/components/ComposeMessagePageClient.js +24 -0
  100. package/dist/modules/messages/components/ComposeMessagePageClient.js.map +7 -0
  101. package/dist/modules/messages/components/MessageDetailPageClient.js +261 -0
  102. package/dist/modules/messages/components/MessageDetailPageClient.js.map +7 -0
  103. package/dist/modules/messages/components/MessagesInboxPageClient.js +390 -0
  104. package/dist/modules/messages/components/MessagesInboxPageClient.js.map +7 -0
  105. package/dist/modules/messages/components/confirmation/MessageConfirmationActions.js +31 -0
  106. package/dist/modules/messages/components/confirmation/MessageConfirmationActions.js.map +7 -0
  107. package/dist/modules/messages/components/confirmation/MessageConfirmationContent.js +69 -0
  108. package/dist/modules/messages/components/confirmation/MessageConfirmationContent.js.map +7 -0
  109. package/dist/modules/messages/components/defaults/DefaultMessageActions.js +31 -0
  110. package/dist/modules/messages/components/defaults/DefaultMessageActions.js.map +7 -0
  111. package/dist/modules/messages/components/defaults/DefaultMessageContent.js +19 -0
  112. package/dist/modules/messages/components/defaults/DefaultMessageContent.js.map +7 -0
  113. package/dist/modules/messages/components/defaults/DefaultMessageListItem.js +90 -0
  114. package/dist/modules/messages/components/defaults/DefaultMessageListItem.js.map +7 -0
  115. package/dist/modules/messages/components/defaults/MessageRecordObjectDetail.js +86 -0
  116. package/dist/modules/messages/components/defaults/MessageRecordObjectDetail.js.map +7 -0
  117. package/dist/modules/messages/components/defaults/MessageRecordObjectPreview.js +61 -0
  118. package/dist/modules/messages/components/defaults/MessageRecordObjectPreview.js.map +7 -0
  119. package/dist/modules/messages/components/message-detail/detail-panels.js +27 -0
  120. package/dist/modules/messages/components/message-detail/detail-panels.js.map +7 -0
  121. package/dist/modules/messages/components/message-detail/hooks/useMessageDetails.js +52 -0
  122. package/dist/modules/messages/components/message-detail/hooks/useMessageDetails.js.map +7 -0
  123. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsActions.js +289 -0
  124. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsActions.js.map +7 -0
  125. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsConversation.js +103 -0
  126. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsConversation.js.map +7 -0
  127. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsQueries.js +78 -0
  128. package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsQueries.js.map +7 -0
  129. package/dist/modules/messages/components/message-detail/panels/MainMessageHeader.js +94 -0
  130. package/dist/modules/messages/components/message-detail/panels/MainMessageHeader.js.map +7 -0
  131. package/dist/modules/messages/components/message-detail/panels/MessageHeader.js +110 -0
  132. package/dist/modules/messages/components/message-detail/panels/MessageHeader.js.map +7 -0
  133. package/dist/modules/messages/components/message-detail/panels/MessageListComponent.js +58 -0
  134. package/dist/modules/messages/components/message-detail/panels/MessageListComponent.js.map +7 -0
  135. package/dist/modules/messages/components/message-detail/panels/actions-panel.js +51 -0
  136. package/dist/modules/messages/components/message-detail/panels/actions-panel.js.map +7 -0
  137. package/dist/modules/messages/components/message-detail/panels/attachments-panel.js +66 -0
  138. package/dist/modules/messages/components/message-detail/panels/attachments-panel.js.map +7 -0
  139. package/dist/modules/messages/components/message-detail/panels/body-panel.js +20 -0
  140. package/dist/modules/messages/components/message-detail/panels/body-panel.js.map +7 -0
  141. package/dist/modules/messages/components/message-detail/panels/composer-dialogs.js +36 -0
  142. package/dist/modules/messages/components/message-detail/panels/composer-dialogs.js.map +7 -0
  143. package/dist/modules/messages/components/message-detail/panels/dialogs.js +96 -0
  144. package/dist/modules/messages/components/message-detail/panels/dialogs.js.map +7 -0
  145. package/dist/modules/messages/components/message-detail/panels/index.js +25 -0
  146. package/dist/modules/messages/components/message-detail/panels/index.js.map +7 -0
  147. package/dist/modules/messages/components/message-detail/panels/meta-panel.js +14 -0
  148. package/dist/modules/messages/components/message-detail/panels/meta-panel.js.map +7 -0
  149. package/dist/modules/messages/components/message-detail/panels/objects-panel.js +51 -0
  150. package/dist/modules/messages/components/message-detail/panels/objects-panel.js.map +7 -0
  151. package/dist/modules/messages/components/message-detail/panels/thread-panel.js +54 -0
  152. package/dist/modules/messages/components/message-detail/panels/thread-panel.js.map +7 -0
  153. package/dist/modules/messages/components/message-detail/types.js +1 -0
  154. package/dist/modules/messages/components/message-detail/types.js.map +7 -0
  155. package/dist/modules/messages/components/message-detail/utils.js +54 -0
  156. package/dist/modules/messages/components/message-detail/utils.js.map +7 -0
  157. package/dist/modules/messages/components/utils/PriorityBadge.js +52 -0
  158. package/dist/modules/messages/components/utils/PriorityBadge.js.map +7 -0
  159. package/dist/modules/messages/components/utils/typeUiRegistry.js +77 -0
  160. package/dist/modules/messages/components/utils/typeUiRegistry.js.map +7 -0
  161. package/dist/modules/messages/data/entities.js +309 -0
  162. package/dist/modules/messages/data/entities.js.map +7 -0
  163. package/dist/modules/messages/data/validators.js +272 -0
  164. package/dist/modules/messages/data/validators.js.map +7 -0
  165. package/dist/modules/messages/emails/MessageEmail.js +108 -0
  166. package/dist/modules/messages/emails/MessageEmail.js.map +7 -0
  167. package/dist/modules/messages/events.js +24 -0
  168. package/dist/modules/messages/events.js.map +7 -0
  169. package/dist/modules/messages/frontend/messages/view/[token]/page.js +247 -0
  170. package/dist/modules/messages/frontend/messages/view/[token]/page.js.map +7 -0
  171. package/dist/modules/messages/frontend/messages/view/[token]/page.meta.js +9 -0
  172. package/dist/modules/messages/frontend/messages/view/[token]/page.meta.js.map +7 -0
  173. package/dist/modules/messages/index.js +21 -0
  174. package/dist/modules/messages/index.js.map +7 -0
  175. package/dist/modules/messages/lib/actions.js +141 -0
  176. package/dist/modules/messages/lib/actions.js.map +7 -0
  177. package/dist/modules/messages/lib/attachments.js +131 -0
  178. package/dist/modules/messages/lib/attachments.js.map +7 -0
  179. package/dist/modules/messages/lib/constants.js +7 -0
  180. package/dist/modules/messages/lib/constants.js.map +7 -0
  181. package/dist/modules/messages/lib/email-sender.js +201 -0
  182. package/dist/modules/messages/lib/email-sender.js.map +7 -0
  183. package/dist/modules/messages/lib/forwarding.js +179 -0
  184. package/dist/modules/messages/lib/forwarding.js.map +7 -0
  185. package/dist/modules/messages/lib/message-objects-registry.js +49 -0
  186. package/dist/modules/messages/lib/message-objects-registry.js.map +7 -0
  187. package/dist/modules/messages/lib/message-types-registry.js +41 -0
  188. package/dist/modules/messages/lib/message-types-registry.js.map +7 -0
  189. package/dist/modules/messages/lib/object-validation.js +20 -0
  190. package/dist/modules/messages/lib/object-validation.js.map +7 -0
  191. package/dist/modules/messages/lib/operationMetadata.js +21 -0
  192. package/dist/modules/messages/lib/operationMetadata.js.map +7 -0
  193. package/dist/modules/messages/lib/priorityUtils.js +61 -0
  194. package/dist/modules/messages/lib/priorityUtils.js.map +7 -0
  195. package/dist/modules/messages/lib/routeHelpers.js +44 -0
  196. package/dist/modules/messages/lib/routeHelpers.js.map +7 -0
  197. package/dist/modules/messages/message-objects.js +7 -0
  198. package/dist/modules/messages/message-objects.js.map +7 -0
  199. package/dist/modules/messages/message-types.js +67 -0
  200. package/dist/modules/messages/message-types.js.map +7 -0
  201. package/dist/modules/messages/migrations/Migration20260213181243.js +31 -0
  202. package/dist/modules/messages/migrations/Migration20260213181243.js.map +7 -0
  203. package/dist/modules/messages/migrations/Migration20260215165126.js +16 -0
  204. package/dist/modules/messages/migrations/Migration20260215165126.js.map +7 -0
  205. package/dist/modules/messages/notifications.js +27 -0
  206. package/dist/modules/messages/notifications.js.map +7 -0
  207. package/dist/modules/messages/setup.js +21 -0
  208. package/dist/modules/messages/setup.js.map +7 -0
  209. package/dist/modules/messages/subscribers/message-notification.js +108 -0
  210. package/dist/modules/messages/subscribers/message-notification.js.map +7 -0
  211. package/dist/modules/messages/workers/send-email.worker.js +253 -0
  212. package/dist/modules/messages/workers/send-email.worker.js.map +7 -0
  213. package/dist/modules/sales/backend/sales/documents/[id]/page.js +30 -11
  214. package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
  215. package/dist/modules/sales/commands/payments.js +12 -6
  216. package/dist/modules/sales/commands/payments.js.map +2 -2
  217. package/dist/modules/sales/lib/messageObjectPreviews.js +114 -0
  218. package/dist/modules/sales/lib/messageObjectPreviews.js.map +7 -0
  219. package/dist/modules/sales/message-objects.js +57 -0
  220. package/dist/modules/sales/message-objects.js.map +7 -0
  221. package/dist/modules/sales/widgets/messages/SalesDocumentMessageDetail.js +51 -0
  222. package/dist/modules/sales/widgets/messages/SalesDocumentMessageDetail.js.map +7 -0
  223. package/dist/modules/sales/widgets/messages/SalesDocumentMessagePreview.js +36 -0
  224. package/dist/modules/sales/widgets/messages/SalesDocumentMessagePreview.js.map +7 -0
  225. package/dist/modules/sales/widgets/messages/index.js +7 -0
  226. package/dist/modules/sales/widgets/messages/index.js.map +7 -0
  227. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js +55 -1
  228. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js.map +2 -2
  229. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js +60 -1
  230. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js.map +2 -2
  231. package/dist/modules/staff/backend/staff/team-members/[id]/page.js +2 -19
  232. package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
  233. package/dist/modules/staff/components/LeaveRequestDetail.js +112 -0
  234. package/dist/modules/staff/components/LeaveRequestDetail.js.map +7 -0
  235. package/dist/modules/staff/components/LeaveRequestForm.js +3 -1
  236. package/dist/modules/staff/components/LeaveRequestForm.js.map +2 -2
  237. package/dist/modules/staff/components/LeaveRequestPreview.js +43 -0
  238. package/dist/modules/staff/components/LeaveRequestPreview.js.map +7 -0
  239. package/dist/modules/staff/lib/messageObjectPreviews.js +148 -0
  240. package/dist/modules/staff/lib/messageObjectPreviews.js.map +7 -0
  241. package/dist/modules/staff/message-objects.js +104 -0
  242. package/dist/modules/staff/message-objects.js.map +7 -0
  243. package/dist/modules/staff/message-types.js +23 -0
  244. package/dist/modules/staff/message-types.js.map +7 -0
  245. package/dist/modules/staff/widgets/messages/StaffMessageObjectDetail.js +51 -0
  246. package/dist/modules/staff/widgets/messages/StaffMessageObjectDetail.js.map +7 -0
  247. package/dist/modules/staff/widgets/messages/StaffMessageObjectPreview.js +34 -0
  248. package/dist/modules/staff/widgets/messages/StaffMessageObjectPreview.js.map +7 -0
  249. package/dist/modules/staff/widgets/messages/index.js +7 -0
  250. package/dist/modules/staff/widgets/messages/index.js.map +7 -0
  251. package/dist/modules/workflows/backend/definitions/visual-editor/page.js +6 -6
  252. package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
  253. package/generated/entities/message/index.ts +31 -0
  254. package/generated/entities/message_access_token/index.ts +8 -0
  255. package/generated/entities/message_confirmation/index.ts +9 -0
  256. package/generated/entities/message_object/index.ts +10 -0
  257. package/generated/entities/message_recipient/index.ts +14 -0
  258. package/generated/entities.ids.generated.ts +8 -0
  259. package/generated/entity-fields-registry.ts +10 -0
  260. package/jest.setup.ts +5 -0
  261. package/package.json +2 -2
  262. package/src/modules/attachments/i18n/de.json +4 -0
  263. package/src/modules/attachments/i18n/en.json +4 -0
  264. package/src/modules/attachments/i18n/es.json +4 -0
  265. package/src/modules/attachments/i18n/pl.json +4 -0
  266. package/src/modules/auth/components/AclEditor.tsx +4 -2
  267. package/src/modules/auth/frontend/reset.tsx +3 -3
  268. package/src/modules/auth/i18n/de.json +5 -0
  269. package/src/modules/auth/i18n/en.json +5 -0
  270. package/src/modules/auth/i18n/es.json +5 -0
  271. package/src/modules/auth/i18n/pl.json +5 -0
  272. package/src/modules/catalog/i18n/de.json +21 -0
  273. package/src/modules/catalog/i18n/en.json +21 -0
  274. package/src/modules/catalog/i18n/es.json +21 -0
  275. package/src/modules/catalog/i18n/pl.json +21 -0
  276. package/src/modules/currencies/i18n/de.json +6 -0
  277. package/src/modules/currencies/i18n/en.json +6 -0
  278. package/src/modules/currencies/i18n/es.json +6 -0
  279. package/src/modules/currencies/i18n/pl.json +6 -0
  280. package/src/modules/customers/backend/customers/deals/[id]/page.tsx +20 -4
  281. package/src/modules/customers/i18n/de.json +26 -1
  282. package/src/modules/customers/i18n/en.json +25 -0
  283. package/src/modules/customers/i18n/es.json +25 -0
  284. package/src/modules/customers/i18n/pl.json +25 -0
  285. package/src/modules/customers/lib/messageObjectPreviews.ts +154 -0
  286. package/src/modules/customers/message-objects.ts +70 -0
  287. package/src/modules/customers/widgets/messages/CustomerMessageObjectDetail.tsx +57 -0
  288. package/src/modules/customers/widgets/messages/CustomerMessageObjectPreview.tsx +49 -0
  289. package/src/modules/customers/widgets/messages/index.ts +2 -0
  290. package/src/modules/dashboards/components/WidgetVisibilityEditor.tsx +41 -5
  291. package/src/modules/dashboards/i18n/de.json +5 -1
  292. package/src/modules/dashboards/i18n/en.json +5 -1
  293. package/src/modules/dashboards/i18n/es.json +5 -1
  294. package/src/modules/dashboards/i18n/pl.json +5 -1
  295. package/src/modules/dictionaries/i18n/de.json +14 -1
  296. package/src/modules/dictionaries/i18n/en.json +14 -1
  297. package/src/modules/dictionaries/i18n/es.json +14 -1
  298. package/src/modules/dictionaries/i18n/pl.json +14 -1
  299. package/src/modules/feature_toggles/i18n/de.json +3 -0
  300. package/src/modules/feature_toggles/i18n/en.json +3 -0
  301. package/src/modules/feature_toggles/i18n/es.json +3 -0
  302. package/src/modules/feature_toggles/i18n/pl.json +3 -0
  303. package/src/modules/messages/acl.ts +11 -0
  304. package/src/modules/messages/api/[id]/actions/[actionId]/route.ts +103 -0
  305. package/src/modules/messages/api/[id]/archive/route.ts +138 -0
  306. package/src/modules/messages/api/[id]/attachments/route.ts +217 -0
  307. package/src/modules/messages/api/[id]/confirmation/route.ts +73 -0
  308. package/src/modules/messages/api/[id]/conversation/archive/route.ts +69 -0
  309. package/src/modules/messages/api/[id]/conversation/read/route.ts +69 -0
  310. package/src/modules/messages/api/[id]/conversation/route.ts +69 -0
  311. package/src/modules/messages/api/[id]/forward/route.ts +87 -0
  312. package/src/modules/messages/api/[id]/forward-preview/route.ts +75 -0
  313. package/src/modules/messages/api/[id]/read/route.ts +138 -0
  314. package/src/modules/messages/api/[id]/reply/route.ts +89 -0
  315. package/src/modules/messages/api/[id]/route.ts +401 -0
  316. package/src/modules/messages/api/object-types/route.ts +54 -0
  317. package/src/modules/messages/api/openapi.ts +261 -0
  318. package/src/modules/messages/api/route.ts +374 -0
  319. package/src/modules/messages/api/token/[token]/route.ts +103 -0
  320. package/src/modules/messages/api/types/route.ts +39 -0
  321. package/src/modules/messages/api/unread-count/route.ts +55 -0
  322. package/src/modules/messages/backend/messages/[id]/page.meta.ts +12 -0
  323. package/src/modules/messages/backend/messages/[id]/page.tsx +12 -0
  324. package/src/modules/messages/backend/messages/compose/page.meta.ts +13 -0
  325. package/src/modules/messages/backend/messages/compose/page.tsx +12 -0
  326. package/src/modules/messages/backend/page.meta.ts +31 -0
  327. package/src/modules/messages/backend/page.tsx +12 -0
  328. package/src/modules/messages/commands/actions.ts +307 -0
  329. package/src/modules/messages/commands/attachments.ts +227 -0
  330. package/src/modules/messages/commands/confirmations.ts +183 -0
  331. package/src/modules/messages/commands/conversation.ts +292 -0
  332. package/src/modules/messages/commands/messages.ts +845 -0
  333. package/src/modules/messages/commands/recipients.ts +281 -0
  334. package/src/modules/messages/commands/shared.ts +350 -0
  335. package/src/modules/messages/commands/tokens.ts +80 -0
  336. package/src/modules/messages/components/ComposeMessagePageClient.tsx +23 -0
  337. package/src/modules/messages/components/MessageDetailPageClient.tsx +287 -0
  338. package/src/modules/messages/components/MessagesInboxPageClient.tsx +469 -0
  339. package/src/modules/messages/components/confirmation/MessageConfirmationActions.tsx +35 -0
  340. package/src/modules/messages/components/confirmation/MessageConfirmationContent.tsx +88 -0
  341. package/src/modules/messages/components/defaults/DefaultMessageActions.tsx +37 -0
  342. package/src/modules/messages/components/defaults/DefaultMessageContent.tsx +21 -0
  343. package/src/modules/messages/components/defaults/DefaultMessageListItem.tsx +102 -0
  344. package/src/modules/messages/components/defaults/MessageRecordObjectDetail.tsx +114 -0
  345. package/src/modules/messages/components/defaults/MessageRecordObjectPreview.tsx +74 -0
  346. package/src/modules/messages/components/message-detail/detail-panels.ts +13 -0
  347. package/src/modules/messages/components/message-detail/hooks/useMessageDetails.ts +56 -0
  348. package/src/modules/messages/components/message-detail/hooks/useMessageDetailsActions.ts +367 -0
  349. package/src/modules/messages/components/message-detail/hooks/useMessageDetailsConversation.ts +134 -0
  350. package/src/modules/messages/components/message-detail/hooks/useMessageDetailsQueries.ts +102 -0
  351. package/src/modules/messages/components/message-detail/panels/MainMessageHeader.tsx +108 -0
  352. package/src/modules/messages/components/message-detail/panels/MessageHeader.tsx +144 -0
  353. package/src/modules/messages/components/message-detail/panels/MessageListComponent.tsx +63 -0
  354. package/src/modules/messages/components/message-detail/panels/actions-panel.tsx +66 -0
  355. package/src/modules/messages/components/message-detail/panels/attachments-panel.tsx +86 -0
  356. package/src/modules/messages/components/message-detail/panels/body-panel.tsx +32 -0
  357. package/src/modules/messages/components/message-detail/panels/composer-dialogs.tsx +42 -0
  358. package/src/modules/messages/components/message-detail/panels/dialogs.tsx +107 -0
  359. package/src/modules/messages/components/message-detail/panels/index.ts +11 -0
  360. package/src/modules/messages/components/message-detail/panels/meta-panel.tsx +19 -0
  361. package/src/modules/messages/components/message-detail/panels/objects-panel.tsx +65 -0
  362. package/src/modules/messages/components/message-detail/panels/thread-panel.tsx +65 -0
  363. package/src/modules/messages/components/message-detail/types.ts +114 -0
  364. package/src/modules/messages/components/message-detail/utils.ts +62 -0
  365. package/src/modules/messages/components/utils/PriorityBadge.tsx +63 -0
  366. package/src/modules/messages/components/utils/typeUiRegistry.ts +106 -0
  367. package/src/modules/messages/data/entities.ts +284 -0
  368. package/src/modules/messages/data/validators.ts +297 -0
  369. package/src/modules/messages/emails/MessageEmail.tsx +143 -0
  370. package/src/modules/messages/events.ts +24 -0
  371. package/src/modules/messages/frontend/messages/view/[token]/page.meta.ts +5 -0
  372. package/src/modules/messages/frontend/messages/view/[token]/page.tsx +389 -0
  373. package/src/modules/messages/i18n/de.json +240 -0
  374. package/src/modules/messages/i18n/en.json +240 -0
  375. package/src/modules/messages/i18n/es.json +240 -0
  376. package/src/modules/messages/i18n/pl.json +240 -0
  377. package/src/modules/messages/index.ts +19 -0
  378. package/src/modules/messages/lib/actions.ts +204 -0
  379. package/src/modules/messages/lib/attachments.ts +197 -0
  380. package/src/modules/messages/lib/constants.ts +2 -0
  381. package/src/modules/messages/lib/email-sender.ts +255 -0
  382. package/src/modules/messages/lib/forwarding.ts +240 -0
  383. package/src/modules/messages/lib/message-objects-registry.ts +60 -0
  384. package/src/modules/messages/lib/message-types-registry.ts +48 -0
  385. package/src/modules/messages/lib/object-validation.ts +26 -0
  386. package/src/modules/messages/lib/operationMetadata.ts +43 -0
  387. package/src/modules/messages/lib/priorityUtils.ts +76 -0
  388. package/src/modules/messages/lib/routeHelpers.ts +65 -0
  389. package/src/modules/messages/message-objects.ts +5 -0
  390. package/src/modules/messages/message-types.ts +65 -0
  391. package/src/modules/messages/migrations/.snapshot-open-mercato.json +957 -0
  392. package/src/modules/messages/migrations/Migration20260213181243.ts +34 -0
  393. package/src/modules/messages/migrations/Migration20260215165126.ts +16 -0
  394. package/src/modules/messages/notifications.ts +25 -0
  395. package/src/modules/messages/setup.ts +19 -0
  396. package/src/modules/messages/subscribers/message-notification.ts +138 -0
  397. package/src/modules/messages/workers/send-email.worker.ts +321 -0
  398. package/src/modules/query_index/i18n/es.json +2 -2
  399. package/src/modules/resources/i18n/de.json +57 -0
  400. package/src/modules/resources/i18n/en.json +57 -0
  401. package/src/modules/resources/i18n/es.json +57 -0
  402. package/src/modules/resources/i18n/pl.json +57 -0
  403. package/src/modules/sales/backend/sales/documents/[id]/page.tsx +23 -7
  404. package/src/modules/sales/commands/payments.ts +12 -6
  405. package/src/modules/sales/i18n/de.json +26 -0
  406. package/src/modules/sales/i18n/en.json +26 -0
  407. package/src/modules/sales/i18n/es.json +26 -0
  408. package/src/modules/sales/i18n/pl.json +26 -0
  409. package/src/modules/sales/lib/messageObjectPreviews.ts +150 -0
  410. package/src/modules/sales/message-objects.ts +56 -0
  411. package/src/modules/sales/widgets/messages/SalesDocumentMessageDetail.tsx +57 -0
  412. package/src/modules/sales/widgets/messages/SalesDocumentMessagePreview.tsx +46 -0
  413. package/src/modules/sales/widgets/messages/index.ts +2 -0
  414. package/src/modules/staff/backend/staff/leave-requests/[id]/page.tsx +54 -0
  415. package/src/modules/staff/backend/staff/my-leave-requests/[id]/page.tsx +58 -0
  416. package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +2 -32
  417. package/src/modules/staff/components/LeaveRequestDetail.tsx +135 -0
  418. package/src/modules/staff/components/LeaveRequestForm.tsx +3 -0
  419. package/src/modules/staff/components/LeaveRequestPreview.tsx +74 -0
  420. package/src/modules/staff/i18n/de.json +22 -0
  421. package/src/modules/staff/i18n/en.json +22 -0
  422. package/src/modules/staff/i18n/es.json +22 -0
  423. package/src/modules/staff/i18n/pl.json +22 -0
  424. package/src/modules/staff/lib/messageObjectPreviews.ts +182 -0
  425. package/src/modules/staff/message-objects.ts +102 -0
  426. package/src/modules/staff/message-types.ts +21 -0
  427. package/src/modules/staff/widgets/messages/StaffMessageObjectDetail.tsx +57 -0
  428. package/src/modules/staff/widgets/messages/StaffMessageObjectPreview.tsx +44 -0
  429. package/src/modules/staff/widgets/messages/index.ts +2 -0
  430. package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +6 -6
  431. package/src/modules/workflows/i18n/de.json +41 -0
  432. package/src/modules/workflows/i18n/en.json +41 -0
  433. package/src/modules/workflows/i18n/es.json +41 -0
  434. package/src/modules/workflows/i18n/pl.json +41 -0
@@ -0,0 +1,367 @@
1
+ "use client"
2
+
3
+ import * as React from 'react'
4
+ import type { UseQueryResult } from '@tanstack/react-query'
5
+ import type { TranslateFn } from '@open-mercato/shared/lib/i18n/context'
6
+ import type { MessageActionsProps, MessageContentProps } from '@open-mercato/shared/modules/messages/types'
7
+ import { flash } from '@open-mercato/ui/backend/FlashMessages'
8
+ import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
9
+ import type {
10
+ ActionResult,
11
+ MessageAction,
12
+ MessageAttachment,
13
+ MessageDetail,
14
+ PendingActionConfirmation,
15
+ } from '../types'
16
+ import { parseObjectActionId, toErrorMessage } from '../utils'
17
+
18
+ type RequestAndRefreshOptions = {
19
+ skipDetailAutoMarkRead?: boolean
20
+ }
21
+
22
+ type ConversationActionKind = 'archiveConversation' | 'deleteConversation' | 'markAllUnread'
23
+
24
+ type UseMessageDetailsActionsInput = {
25
+ id: string
26
+ t: TranslateFn
27
+ detail: MessageDetail | null
28
+ detailQuery: UseQueryResult<MessageDetail, Error>
29
+ attachments: MessageAttachment[] | undefined
30
+ isArchived: boolean
31
+ onDeleted: () => void
32
+ refreshDetailWithoutAutoMarkRead: () => Promise<MessageDetail>
33
+ }
34
+
35
+ export function useMessageDetailsActions({
36
+ id,
37
+ t,
38
+ detail,
39
+ detailQuery,
40
+ attachments,
41
+ isArchived,
42
+ onDeleted,
43
+ refreshDetailWithoutAutoMarkRead,
44
+ }: UseMessageDetailsActionsInput) {
45
+ const [replyOpen, setReplyOpen] = React.useState(false)
46
+ const [forwardOpen, setForwardOpen] = React.useState(false)
47
+ const [editOpen, setEditOpen] = React.useState(false)
48
+ const [updatingState, setUpdatingState] = React.useState(false)
49
+ const [executingActionId, setExecutingActionId] = React.useState<string | null>(null)
50
+ const [pendingActionConfirmation, setPendingActionConfirmation] = React.useState<PendingActionConfirmation | null>(null)
51
+ const [deleteConfirmationOpen, setDeleteConfirmationOpen] = React.useState(false)
52
+ const [activeConversationAction, setActiveConversationAction] = React.useState<ConversationActionKind | null>(null)
53
+
54
+ const requestAndRefresh = React.useCallback(async (
55
+ url: string,
56
+ method: 'PUT' | 'DELETE',
57
+ options?: RequestAndRefreshOptions,
58
+ ) => {
59
+ setUpdatingState(true)
60
+ try {
61
+ const call = await apiCall<{ ok?: boolean }>(url, { method })
62
+ if (!call.ok) {
63
+ throw new Error(
64
+ toErrorMessage(call.result)
65
+ ?? t('messages.errors.stateChangeFailed', 'Failed to update message state.'),
66
+ )
67
+ }
68
+ if (options?.skipDetailAutoMarkRead) {
69
+ await refreshDetailWithoutAutoMarkRead()
70
+ } else {
71
+ await detailQuery.refetch()
72
+ }
73
+ } catch (err) {
74
+ flash(
75
+ err instanceof Error
76
+ ? err.message
77
+ : t('messages.errors.stateChangeFailed', 'Failed to update message state.'),
78
+ 'error',
79
+ )
80
+ } finally {
81
+ setUpdatingState(false)
82
+ }
83
+ }, [detailQuery, refreshDetailWithoutAutoMarkRead, t])
84
+
85
+ const handleDelete = React.useCallback(async () => {
86
+ setUpdatingState(true)
87
+ try {
88
+ const call = await apiCall<{ ok?: boolean }>(`/api/messages/${encodeURIComponent(id)}`, {
89
+ method: 'DELETE',
90
+ })
91
+ if (!call.ok) {
92
+ throw new Error(toErrorMessage(call.result) ?? t('messages.errors.deleteFailed', 'Failed to delete message.'))
93
+ }
94
+ flash(t('messages.flash.deleted', 'Message deleted.'), 'success')
95
+ onDeleted()
96
+ } catch (err) {
97
+ flash(
98
+ err instanceof Error
99
+ ? err.message
100
+ : t('messages.errors.deleteFailed', 'Failed to delete message.'),
101
+ 'error',
102
+ )
103
+ } finally {
104
+ setUpdatingState(false)
105
+ }
106
+ }, [id, onDeleted, t])
107
+
108
+ const runConversationAction = React.useCallback(async (
109
+ action: ConversationActionKind,
110
+ options: {
111
+ url: string
112
+ method: 'PUT' | 'DELETE'
113
+ successMessage: string
114
+ skipDetailAutoMarkRead?: boolean
115
+ onSuccess?: () => void
116
+ },
117
+ ) => {
118
+ setActiveConversationAction(action)
119
+ try {
120
+ const call = await apiCall<{ ok?: boolean; affectedCount?: number }>(options.url, { method: options.method })
121
+ if (!call.ok) {
122
+ throw new Error(
123
+ toErrorMessage(call.result)
124
+ ?? t('messages.errors.conversationActionFailed', 'Failed to update conversation.'),
125
+ )
126
+ }
127
+
128
+ flash(options.successMessage, 'success')
129
+
130
+ if (options.onSuccess) {
131
+ options.onSuccess()
132
+ return
133
+ }
134
+
135
+ if (options.skipDetailAutoMarkRead) {
136
+ await refreshDetailWithoutAutoMarkRead()
137
+ } else {
138
+ await detailQuery.refetch()
139
+ }
140
+ } catch (err) {
141
+ flash(
142
+ err instanceof Error
143
+ ? err.message
144
+ : t('messages.errors.conversationActionFailed', 'Failed to update conversation.'),
145
+ 'error',
146
+ )
147
+ } finally {
148
+ setActiveConversationAction(null)
149
+ }
150
+ }, [detailQuery, refreshDetailWithoutAutoMarkRead, t])
151
+
152
+ const archiveConversation = React.useCallback(async (messageId?: string) => {
153
+ const targetMessageId = messageId ?? id
154
+ await runConversationAction('archiveConversation', {
155
+ url: `/api/messages/${encodeURIComponent(targetMessageId)}/conversation/archive`,
156
+ method: 'PUT',
157
+ successMessage: t('messages.flash.conversationArchived', 'Conversation archived.'),
158
+ })
159
+ }, [id, runConversationAction, t])
160
+
161
+ const markConversationUnread = React.useCallback(async (messageId?: string) => {
162
+ const targetMessageId = messageId ?? id
163
+ await runConversationAction('markAllUnread', {
164
+ url: `/api/messages/${encodeURIComponent(targetMessageId)}/conversation/read`,
165
+ method: 'DELETE',
166
+ successMessage: t('messages.flash.conversationMarkedUnread', 'Conversation marked unread.'),
167
+ skipDetailAutoMarkRead: true,
168
+ })
169
+ }, [id, runConversationAction, t])
170
+
171
+ const deleteConversation = React.useCallback(async (messageId?: string) => {
172
+ const targetMessageId = messageId ?? id
173
+ await runConversationAction('deleteConversation', {
174
+ url: `/api/messages/${encodeURIComponent(targetMessageId)}/conversation`,
175
+ method: 'DELETE',
176
+ successMessage: t('messages.flash.conversationDeleted', 'Conversation deleted.'),
177
+ onSuccess: onDeleted,
178
+ })
179
+ }, [id, onDeleted, runConversationAction, t])
180
+
181
+ const handleDeleteDialogKeyDown = React.useCallback((event: React.KeyboardEvent<HTMLDivElement>) => {
182
+ if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {
183
+ event.preventDefault()
184
+ void handleDelete()
185
+ return
186
+ }
187
+
188
+ if (event.key === 'Escape') {
189
+ event.preventDefault()
190
+ setDeleteConfirmationOpen(false)
191
+ }
192
+ }, [handleDelete])
193
+
194
+ const executeAction = React.useCallback(async (action: MessageAction, payload?: Record<string, unknown>) => {
195
+ setExecutingActionId(action.id)
196
+ try {
197
+ const call = await apiCall<ActionResult>(
198
+ `/api/messages/${encodeURIComponent(id)}/actions/${encodeURIComponent(action.id)}`,
199
+ {
200
+ method: 'POST',
201
+ headers: { 'Content-Type': 'application/json' },
202
+ body: JSON.stringify(payload ?? {}),
203
+ },
204
+ )
205
+
206
+ if (!call.ok) {
207
+ throw new Error(
208
+ toErrorMessage(call.result)
209
+ ?? t('messages.errors.actionFailed', 'Failed to execute action.'),
210
+ )
211
+ }
212
+
213
+ flash(t('messages.flash.actionSuccess', 'Action completed.'), 'success')
214
+
215
+ const redirectTarget = call.result?.result?.redirect
216
+ if (typeof redirectTarget === 'string' && redirectTarget.trim().length > 0) {
217
+ window.location.href = redirectTarget
218
+ return
219
+ }
220
+
221
+ await detailQuery.refetch()
222
+ } catch (err) {
223
+ flash(
224
+ err instanceof Error
225
+ ? err.message
226
+ : t('messages.errors.actionFailed', 'Failed to execute action.'),
227
+ 'error',
228
+ )
229
+ } finally {
230
+ setExecutingActionId(null)
231
+ }
232
+ }, [detailQuery, id, t])
233
+
234
+ const handleExecuteAction = React.useCallback(async (action: MessageAction, payload?: Record<string, unknown>) => {
235
+ if (executingActionId || detail?.actionTaken) return
236
+
237
+ if (action.confirmRequired) {
238
+ setPendingActionConfirmation({ action, payload })
239
+ return
240
+ }
241
+
242
+ await executeAction(action, payload)
243
+ }, [detail?.actionTaken, executeAction, executingActionId])
244
+
245
+ const handleConfirmPendingAction = React.useCallback(async () => {
246
+ if (!pendingActionConfirmation || executingActionId || detail?.actionTaken) return
247
+ const { action, payload } = pendingActionConfirmation
248
+ setPendingActionConfirmation(null)
249
+ await executeAction(action, payload)
250
+ }, [detail?.actionTaken, executeAction, executingActionId, pendingActionConfirmation])
251
+
252
+ const handleActionConfirmDialogKeyDown = React.useCallback((event: React.KeyboardEvent<HTMLDivElement>) => {
253
+ if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {
254
+ event.preventDefault()
255
+ void handleConfirmPendingAction()
256
+ return
257
+ }
258
+
259
+ if (event.key === 'Escape') {
260
+ event.preventDefault()
261
+ setPendingActionConfirmation(null)
262
+ }
263
+ }, [handleConfirmPendingAction])
264
+
265
+ const contentProps = React.useMemo<MessageContentProps>(() => ({
266
+ message: {
267
+ id: detail?.id ?? '',
268
+ type: detail?.type ?? 'default',
269
+ subject: detail?.subject ?? '',
270
+ body: detail?.body ?? '',
271
+ bodyFormat: detail?.bodyFormat ?? 'text',
272
+ priority: (detail?.priority ?? 'normal') as 'low' | 'normal' | 'high' | 'urgent',
273
+ sentAt: detail?.sentAt ? new Date(detail.sentAt) : null,
274
+ senderName: detail?.senderName ?? undefined,
275
+ senderUserId: detail?.senderUserId ?? '',
276
+ actionData: detail?.actionData ?? null,
277
+ actionTaken: detail?.actionTaken ?? null,
278
+ actionTakenAt: detail?.actionTakenAt ? new Date(detail.actionTakenAt) : null,
279
+ },
280
+ objects: (detail?.objects ?? []).map((item) => ({
281
+ id: item.id,
282
+ entityModule: item.entityModule,
283
+ entityType: item.entityType,
284
+ entityId: item.entityId,
285
+ actionRequired: item.actionRequired,
286
+ snapshot: item.snapshot ?? undefined,
287
+ })),
288
+ attachments: (attachments ?? []).map((item) => ({
289
+ id: item.id,
290
+ fileName: item.fileName,
291
+ fileSize: item.fileSize,
292
+ mimeType: item.mimeType,
293
+ url: item.url,
294
+ })),
295
+ }), [attachments, detail])
296
+
297
+ const handleExecuteActionById = React.useCallback<MessageActionsProps['onExecuteAction']>(async (actionId, payload) => {
298
+ const action = detail?.actionData?.actions?.find((entry) => {
299
+ if (parseObjectActionId(entry.id)) return false
300
+ return entry.id === actionId
301
+ })
302
+ if (!action) return
303
+ await handleExecuteAction(action, payload)
304
+ }, [detail?.actionData?.actions, handleExecuteAction])
305
+
306
+ const messageActions = React.useMemo(() => {
307
+ return (detail?.actionData?.actions ?? []).filter((action) => !parseObjectActionId(action.id))
308
+ }, [detail?.actionData?.actions])
309
+
310
+ const objectActionsByObjectId = React.useMemo(() => {
311
+ const byObjectId = new Map<string, Map<string, MessageAction>>()
312
+ for (const action of detail?.actionData?.actions ?? []) {
313
+ const parsed = parseObjectActionId(action.id)
314
+ if (!parsed) continue
315
+ const current = byObjectId.get(parsed.objectId) ?? new Map<string, MessageAction>()
316
+ current.set(parsed.actionId, action)
317
+ byObjectId.set(parsed.objectId, current)
318
+ }
319
+ return byObjectId
320
+ }, [detail?.actionData?.actions])
321
+
322
+ const toggleRead = React.useCallback(async () => {
323
+ await requestAndRefresh(
324
+ `/api/messages/${encodeURIComponent(id)}/read`,
325
+ detail?.isRead ? 'DELETE' : 'PUT',
326
+ detail?.isRead ? { skipDetailAutoMarkRead: true } : undefined,
327
+ )
328
+ }, [detail?.isRead, id, requestAndRefresh])
329
+
330
+ const toggleArchive = React.useCallback(async () => {
331
+ await requestAndRefresh(
332
+ `/api/messages/${encodeURIComponent(id)}/archive`,
333
+ isArchived ? 'DELETE' : 'PUT',
334
+ )
335
+ }, [id, isArchived, requestAndRefresh])
336
+
337
+ return {
338
+ replyOpen,
339
+ setReplyOpen,
340
+ forwardOpen,
341
+ setForwardOpen,
342
+ editOpen,
343
+ setEditOpen,
344
+ updatingState,
345
+ executingActionId,
346
+ pendingActionConfirmation,
347
+ setPendingActionConfirmation,
348
+ deleteConfirmationOpen,
349
+ setDeleteConfirmationOpen,
350
+ activeConversationAction,
351
+ toggleRead,
352
+ toggleArchive,
353
+ archiveConversation,
354
+ deleteConversation,
355
+ markConversationUnread,
356
+ requestAndRefresh,
357
+ handleDelete,
358
+ handleDeleteDialogKeyDown,
359
+ handleExecuteAction,
360
+ handleConfirmPendingAction,
361
+ handleActionConfirmDialogKeyDown,
362
+ contentProps,
363
+ handleExecuteActionById,
364
+ messageActions,
365
+ objectActionsByObjectId,
366
+ }
367
+ }
@@ -0,0 +1,134 @@
1
+ "use client"
2
+
3
+ import * as React from 'react'
4
+ import type { TranslateFn } from '@open-mercato/shared/lib/i18n/context'
5
+ import type { MessageListItemProps } from '@open-mercato/shared/modules/messages/types'
6
+ import type { MessageDetail } from '../types'
7
+
8
+ export type ConversationItem = {
9
+ id: string
10
+ senderUserId: string
11
+ senderName?: string | null
12
+ senderEmail?: string | null
13
+ body: string
14
+ bodyFormat?: 'text' | 'markdown'
15
+ sentAt?: string | null
16
+ objects?: Array<{ id: string }>
17
+ }
18
+
19
+ type UseMessageDetailsConversationInput = {
20
+ detail: MessageDetail | null
21
+ t: TranslateFn
22
+ }
23
+
24
+ export function useMessageDetailsConversation({
25
+ detail,
26
+ t,
27
+ }: UseMessageDetailsConversationInput) {
28
+ const conversationItems = React.useMemo<ConversationItem[]>(() => {
29
+ const thread = detail?.thread ?? []
30
+ if (thread.length === 0 && detail) {
31
+ return [{
32
+ id: detail.id,
33
+ senderUserId: detail.senderUserId,
34
+ senderName: detail.senderName,
35
+ senderEmail: detail.senderEmail,
36
+ body: detail.body,
37
+ bodyFormat: detail.bodyFormat,
38
+ sentAt: detail.sentAt,
39
+ objects: detail.objects.map((item) => ({ id: item.id })),
40
+ }]
41
+ }
42
+
43
+ const items = thread
44
+ .map((item) => ({
45
+ id: item.id,
46
+ senderUserId: item.senderUserId,
47
+ senderName: item.senderName,
48
+ senderEmail: item.senderEmail,
49
+ body: item.body,
50
+ bodyFormat: item.bodyFormat,
51
+ sentAt: item.sentAt,
52
+ objects: item.objects?.map((entry) => ({ id: entry.id })) ?? [],
53
+ }))
54
+
55
+ if (detail && !items.some((item) => item.id === detail.id)) {
56
+ items.push({
57
+ id: detail.id,
58
+ senderUserId: detail.senderUserId,
59
+ senderName: detail.senderName,
60
+ senderEmail: detail.senderEmail,
61
+ body: detail.body,
62
+ bodyFormat: detail.bodyFormat,
63
+ sentAt: detail.sentAt,
64
+ objects: detail.objects.map((entry) => ({ id: entry.id })),
65
+ })
66
+ }
67
+
68
+ items.sort((a, b) => {
69
+ const aTime = a.sentAt ? new Date(a.sentAt).getTime() : 0
70
+ const bTime = b.sentAt ? new Date(b.sentAt).getTime() : 0
71
+ return aTime - bTime
72
+ })
73
+
74
+ return items
75
+ }, [detail])
76
+
77
+ const forcedExpandedItemId = conversationItems.length > 0
78
+ ? conversationItems[conversationItems.length - 1].id
79
+ : null
80
+
81
+ const [expandedById, setExpandedById] = React.useState<Record<string, boolean>>({})
82
+ const conversationIdsKey = React.useMemo(
83
+ () => conversationItems.map((item) => item.id).join('|'),
84
+ [conversationItems],
85
+ )
86
+
87
+ React.useEffect(() => {
88
+ const nextState: Record<string, boolean> = {}
89
+ for (const item of conversationItems) {
90
+ nextState[item.id] = forcedExpandedItemId === item.id
91
+ }
92
+ setExpandedById(nextState)
93
+ }, [conversationIdsKey, conversationItems, forcedExpandedItemId])
94
+
95
+ const toggleConversationItem = React.useCallback((itemId: string) => {
96
+ if (itemId === forcedExpandedItemId) return
97
+ setExpandedById((previous) => ({
98
+ ...previous,
99
+ [itemId]: !previous[itemId],
100
+ }))
101
+ }, [forcedExpandedItemId])
102
+
103
+ const isConversationItemExpanded = React.useCallback((itemId: string) => {
104
+ if (itemId === forcedExpandedItemId) return true
105
+ return Boolean(expandedById[itemId])
106
+ }, [expandedById, forcedExpandedItemId])
107
+
108
+ const buildConversationListItemMessage = React.useCallback((item: ConversationItem): MessageListItemProps['message'] => ({
109
+ id: item.id,
110
+ type: detail?.type ?? 'default',
111
+ typeLabel: detail ? t(detail.typeDefinition.labelKey, detail.type) : undefined,
112
+ subject: detail?.subject ?? '',
113
+ body: item.body,
114
+ bodyFormat: item.bodyFormat ?? 'text',
115
+ priority: 'normal',
116
+ sentAt: item.sentAt ? new Date(item.sentAt) : null,
117
+ senderName: item.senderName || item.senderEmail || item.senderUserId,
118
+ hasObjects: (item.objects ?? []).length > 0,
119
+ objectCount: (item.objects ?? []).length,
120
+ hasAttachments: false,
121
+ attachmentCount: 0,
122
+ hasActions: false,
123
+ actionTaken: null,
124
+ unread: false,
125
+ }), [detail, t])
126
+
127
+ return {
128
+ conversationItems,
129
+ forcedExpandedItemId,
130
+ toggleConversationItem,
131
+ isConversationItemExpanded,
132
+ buildConversationListItemMessage,
133
+ }
134
+ }
@@ -0,0 +1,102 @@
1
+ "use client"
2
+
3
+ import * as React from 'react'
4
+ import { useQuery } from '@tanstack/react-query'
5
+ import type { QueryClient } from '@tanstack/react-query'
6
+ import type { TranslateFn } from '@open-mercato/shared/lib/i18n/context'
7
+ import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
8
+ import type { MessageAttachment, MessageDetail } from '../types'
9
+ import { toErrorMessage } from '../utils'
10
+
11
+ type UseMessageDetailsQueriesInput = {
12
+ id: string
13
+ t: TranslateFn
14
+ scopeVersion: unknown
15
+ queryClient: QueryClient
16
+ }
17
+
18
+ export function useMessageDetailsQueries({
19
+ id,
20
+ t,
21
+ scopeVersion,
22
+ queryClient,
23
+ }: UseMessageDetailsQueriesInput) {
24
+ const detailQueryKey = React.useMemo(
25
+ () => ['messages', 'detail', id, scopeVersion],
26
+ [id, scopeVersion],
27
+ )
28
+
29
+ const detailQuery = useQuery({
30
+ queryKey: detailQueryKey,
31
+ queryFn: async () => {
32
+ const call = await apiCall<MessageDetail>(`/api/messages/${encodeURIComponent(id)}`)
33
+ if (!call.ok || !call.result) {
34
+ throw new Error(
35
+ toErrorMessage(call.result)
36
+ ?? t('messages.errors.loadDetailFailed', 'Failed to load message details.'),
37
+ )
38
+ }
39
+ return call.result
40
+ },
41
+ })
42
+
43
+ const detail = detailQuery.data ?? null
44
+ const isLoadingDetail = detailQuery.isLoading
45
+ const loadErrorMessage = detailQuery.error instanceof Error
46
+ ? detailQuery.error.message
47
+ : t('messages.errors.loadDetailFailed', 'Failed to load message details.')
48
+
49
+ const attachmentsQuery = useQuery({
50
+ queryKey: ['messages', 'detail', id, 'attachments', scopeVersion],
51
+ queryFn: async () => {
52
+ const call = await apiCall<{ attachments?: MessageAttachment[] }>(
53
+ `/api/messages/${encodeURIComponent(id)}/attachments`,
54
+ )
55
+
56
+ if (!call.ok) {
57
+ throw new Error(
58
+ toErrorMessage(call.result)
59
+ ?? t('messages.errors.loadDetailFailed', 'Failed to load message details.'),
60
+ )
61
+ }
62
+
63
+ return call.result?.attachments ?? []
64
+ },
65
+ })
66
+
67
+ const attachments = attachmentsQuery.data
68
+
69
+ const refreshDetailWithoutAutoMarkRead = React.useCallback(async () => {
70
+ const call = await apiCall<MessageDetail>(
71
+ `/api/messages/${encodeURIComponent(id)}?skipMarkRead=1`,
72
+ )
73
+
74
+ if (!call.ok || !call.result) {
75
+ throw new Error(
76
+ toErrorMessage(call.result)
77
+ ?? t('messages.errors.stateChangeFailed', 'Failed to update message state.'),
78
+ )
79
+ }
80
+
81
+ queryClient.setQueryData(detailQueryKey, call.result)
82
+ return call.result
83
+ }, [detailQueryKey, id, queryClient, t])
84
+
85
+ const listItemComponentKey = detail?.typeDefinition.ui?.listItemComponent ?? null
86
+ const contentComponentKey = detail?.typeDefinition.ui?.contentComponent ?? null
87
+ const actionsComponentKey = detail?.typeDefinition.ui?.actionsComponent ?? null
88
+
89
+ return {
90
+ detailQueryKey,
91
+ detailQuery,
92
+ detail,
93
+ isLoadingDetail,
94
+ loadErrorMessage,
95
+ attachmentsQuery,
96
+ attachments,
97
+ refreshDetailWithoutAutoMarkRead,
98
+ listItemComponentKey,
99
+ contentComponentKey,
100
+ actionsComponentKey,
101
+ }
102
+ }