@open-mercato/core 0.4.2-canary-e6bf6a353e → 0.4.2-canary-49d47ff90e

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 (397) hide show
  1. package/dist/generated/entities/notification/index.js +57 -0
  2. package/dist/generated/entities/notification/index.js.map +7 -0
  3. package/dist/generated/entities.ids.generated.js +5 -1
  4. package/dist/generated/entities.ids.generated.js.map +2 -2
  5. package/dist/generated/entity-fields-registry.js +2 -0
  6. package/dist/generated/entity-fields-registry.js.map +2 -2
  7. package/dist/modules/api_docs/frontend/docs/api/page.js +3 -2
  8. package/dist/modules/api_docs/frontend/docs/api/page.js.map +2 -2
  9. package/dist/modules/api_keys/backend/api-keys/page.js +1 -1
  10. package/dist/modules/api_keys/backend/api-keys/page.js.map +2 -2
  11. package/dist/modules/attachments/components/AttachmentLibrary.js +4 -0
  12. package/dist/modules/attachments/components/AttachmentLibrary.js.map +2 -2
  13. package/dist/modules/attachments/components/AttachmentPartitionSettings.js +2 -0
  14. package/dist/modules/attachments/components/AttachmentPartitionSettings.js.map +2 -2
  15. package/dist/modules/auth/api/admin/nav.js +4 -3
  16. package/dist/modules/auth/api/admin/nav.js.map +2 -2
  17. package/dist/modules/auth/api/profile/route.js +157 -0
  18. package/dist/modules/auth/api/profile/route.js.map +7 -0
  19. package/dist/modules/auth/api/reset/confirm.js +25 -2
  20. package/dist/modules/auth/api/reset/confirm.js.map +2 -2
  21. package/dist/modules/auth/api/reset.js +23 -0
  22. package/dist/modules/auth/api/reset.js.map +2 -2
  23. package/dist/modules/auth/api/sidebar/preferences/route.js +14 -9
  24. package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
  25. package/dist/modules/auth/api/users/route.js +4 -2
  26. package/dist/modules/auth/api/users/route.js.map +2 -2
  27. package/dist/modules/auth/backend/auth/profile/page.js +141 -0
  28. package/dist/modules/auth/backend/auth/profile/page.js.map +7 -0
  29. package/dist/modules/auth/backend/auth/profile/page.meta.js +13 -0
  30. package/dist/modules/auth/backend/auth/profile/page.meta.js.map +7 -0
  31. package/dist/modules/auth/backend/roles/page.js +3 -3
  32. package/dist/modules/auth/backend/roles/page.js.map +2 -2
  33. package/dist/modules/auth/backend/users/[id]/edit/page.js +14 -2
  34. package/dist/modules/auth/backend/users/[id]/edit/page.js.map +2 -2
  35. package/dist/modules/auth/backend/users/create/page.js +15 -2
  36. package/dist/modules/auth/backend/users/create/page.js.map +2 -2
  37. package/dist/modules/auth/backend/users/page.js +3 -3
  38. package/dist/modules/auth/backend/users/page.js.map +2 -2
  39. package/dist/modules/auth/cli.js +13 -0
  40. package/dist/modules/auth/cli.js.map +2 -2
  41. package/dist/modules/auth/commands/users.js +59 -2
  42. package/dist/modules/auth/commands/users.js.map +2 -2
  43. package/dist/modules/auth/data/validators.js +4 -2
  44. package/dist/modules/auth/data/validators.js.map +2 -2
  45. package/dist/modules/auth/frontend/reset/[token]/page.js +20 -10
  46. package/dist/modules/auth/frontend/reset/[token]/page.js.map +2 -2
  47. package/dist/modules/auth/lib/setup-app.js +1 -0
  48. package/dist/modules/auth/lib/setup-app.js.map +2 -2
  49. package/dist/modules/auth/notifications.js +112 -0
  50. package/dist/modules/auth/notifications.js.map +7 -0
  51. package/dist/modules/auth/services/authService.js +3 -3
  52. package/dist/modules/auth/services/authService.js.map +2 -2
  53. package/dist/modules/business_rules/backend/rules/page.js +4 -0
  54. package/dist/modules/business_rules/backend/rules/page.js.map +2 -2
  55. package/dist/modules/business_rules/backend/sets/page.js +3 -0
  56. package/dist/modules/business_rules/backend/sets/page.js.map +2 -2
  57. package/dist/modules/business_rules/notifications.js +28 -0
  58. package/dist/modules/business_rules/notifications.js.map +7 -0
  59. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js +37 -0
  60. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js.map +7 -0
  61. package/dist/modules/catalog/components/PriceKindSettings.js +2 -0
  62. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  63. package/dist/modules/catalog/components/categories/CategoriesDataTable.js +2 -2
  64. package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
  65. package/dist/modules/catalog/components/products/ProductsDataTable.js +2 -0
  66. package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
  67. package/dist/modules/catalog/notifications.js +28 -0
  68. package/dist/modules/catalog/notifications.js.map +7 -0
  69. package/dist/modules/catalog/subscribers/low-stock-notification.js +38 -0
  70. package/dist/modules/catalog/subscribers/low-stock-notification.js.map +7 -0
  71. package/dist/modules/configs/cli.js +6 -0
  72. package/dist/modules/configs/cli.js.map +2 -2
  73. package/dist/modules/configs/lib/upgrade-actions.js +18 -0
  74. package/dist/modules/configs/lib/upgrade-actions.js.map +2 -2
  75. package/dist/modules/currencies/backend/currencies/page.js +3 -0
  76. package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
  77. package/dist/modules/currencies/backend/exchange-rates/page.js +2 -0
  78. package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
  79. package/dist/modules/customers/backend/customers/companies/page.js +3 -0
  80. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  81. package/dist/modules/customers/backend/customers/deals/page.js +3 -0
  82. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  83. package/dist/modules/customers/backend/customers/people/page.js +3 -0
  84. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  85. package/dist/modules/customers/commands/deals.js +31 -0
  86. package/dist/modules/customers/commands/deals.js.map +2 -2
  87. package/dist/modules/customers/components/CustomerTodosTable.js +1 -0
  88. package/dist/modules/customers/components/CustomerTodosTable.js.map +2 -2
  89. package/dist/modules/customers/notifications.js +48 -0
  90. package/dist/modules/customers/notifications.js.map +7 -0
  91. package/dist/modules/dashboards/cli.js +32 -1
  92. package/dist/modules/dashboards/cli.js.map +2 -2
  93. package/dist/modules/dashboards/lib/role-widgets.js +58 -0
  94. package/dist/modules/dashboards/lib/role-widgets.js.map +7 -0
  95. package/dist/modules/dictionaries/components/DictionaryTable.js +2 -0
  96. package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
  97. package/dist/modules/directory/backend/directory/organizations/page.js +2 -2
  98. package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
  99. package/dist/modules/directory/backend/directory/tenants/page.js +2 -2
  100. package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
  101. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +2 -2
  102. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  103. package/dist/modules/entities/components/SystemEntitiesTable.js +1 -1
  104. package/dist/modules/entities/components/SystemEntitiesTable.js.map +2 -2
  105. package/dist/modules/entities/components/UserEntitiesTable.js +2 -2
  106. package/dist/modules/entities/components/UserEntitiesTable.js.map +2 -2
  107. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +3 -3
  108. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
  109. package/dist/modules/feature_toggles/components/OverridesTable.js +1 -1
  110. package/dist/modules/feature_toggles/components/OverridesTable.js.map +2 -2
  111. package/dist/modules/notifications/acl.js +11 -0
  112. package/dist/modules/notifications/acl.js.map +7 -0
  113. package/dist/modules/notifications/api/[id]/action/route.js +74 -0
  114. package/dist/modules/notifications/api/[id]/action/route.js.map +7 -0
  115. package/dist/modules/notifications/api/[id]/dismiss/route.js +15 -0
  116. package/dist/modules/notifications/api/[id]/dismiss/route.js.map +7 -0
  117. package/dist/modules/notifications/api/[id]/read/route.js +15 -0
  118. package/dist/modules/notifications/api/[id]/read/route.js.map +7 -0
  119. package/dist/modules/notifications/api/[id]/restore/route.js +53 -0
  120. package/dist/modules/notifications/api/[id]/restore/route.js.map +7 -0
  121. package/dist/modules/notifications/api/batch/route.js +17 -0
  122. package/dist/modules/notifications/api/batch/route.js.map +7 -0
  123. package/dist/modules/notifications/api/feature/route.js +17 -0
  124. package/dist/modules/notifications/api/feature/route.js.map +7 -0
  125. package/dist/modules/notifications/api/mark-all-read/route.js +35 -0
  126. package/dist/modules/notifications/api/mark-all-read/route.js.map +7 -0
  127. package/dist/modules/notifications/api/openapi.js +76 -0
  128. package/dist/modules/notifications/api/openapi.js.map +7 -0
  129. package/dist/modules/notifications/api/role/route.js +17 -0
  130. package/dist/modules/notifications/api/role/route.js.map +7 -0
  131. package/dist/modules/notifications/api/route.js +85 -0
  132. package/dist/modules/notifications/api/route.js.map +7 -0
  133. package/dist/modules/notifications/api/settings/route.js +155 -0
  134. package/dist/modules/notifications/api/settings/route.js.map +7 -0
  135. package/dist/modules/notifications/api/unread-count/route.js +38 -0
  136. package/dist/modules/notifications/api/unread-count/route.js.map +7 -0
  137. package/dist/modules/notifications/backend/config/notifications/page.js +10 -0
  138. package/dist/modules/notifications/backend/config/notifications/page.js.map +7 -0
  139. package/dist/modules/notifications/backend/config/notifications/page.meta.js +24 -0
  140. package/dist/modules/notifications/backend/config/notifications/page.meta.js.map +7 -0
  141. package/dist/modules/notifications/cli.js +16 -0
  142. package/dist/modules/notifications/cli.js.map +7 -0
  143. package/dist/modules/notifications/data/entities.js +112 -0
  144. package/dist/modules/notifications/data/entities.js.map +7 -0
  145. package/dist/modules/notifications/data/validators.js +94 -0
  146. package/dist/modules/notifications/data/validators.js.map +7 -0
  147. package/dist/modules/notifications/di.js +13 -0
  148. package/dist/modules/notifications/di.js.map +7 -0
  149. package/dist/modules/notifications/emails/NotificationEmail.js +58 -0
  150. package/dist/modules/notifications/emails/NotificationEmail.js.map +7 -0
  151. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js +44 -0
  152. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js.map +7 -0
  153. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js +219 -0
  154. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js.map +7 -0
  155. package/dist/modules/notifications/index.js +14 -0
  156. package/dist/modules/notifications/index.js.map +7 -0
  157. package/dist/modules/notifications/lib/deliveryConfig.js +105 -0
  158. package/dist/modules/notifications/lib/deliveryConfig.js.map +7 -0
  159. package/dist/modules/notifications/lib/events.js +12 -0
  160. package/dist/modules/notifications/lib/events.js.map +7 -0
  161. package/dist/modules/notifications/lib/notificationBuilder.js +66 -0
  162. package/dist/modules/notifications/lib/notificationBuilder.js.map +7 -0
  163. package/dist/modules/notifications/lib/notificationFactory.js +54 -0
  164. package/dist/modules/notifications/lib/notificationFactory.js.map +7 -0
  165. package/dist/modules/notifications/lib/notificationMapper.js +34 -0
  166. package/dist/modules/notifications/lib/notificationMapper.js.map +7 -0
  167. package/dist/modules/notifications/lib/notificationRecipients.js +35 -0
  168. package/dist/modules/notifications/lib/notificationRecipients.js.map +7 -0
  169. package/dist/modules/notifications/lib/notificationService.js +279 -0
  170. package/dist/modules/notifications/lib/notificationService.js.map +7 -0
  171. package/dist/modules/notifications/lib/routeHelpers.js +101 -0
  172. package/dist/modules/notifications/lib/routeHelpers.js.map +7 -0
  173. package/dist/modules/notifications/lib/safeHref.js +24 -0
  174. package/dist/modules/notifications/lib/safeHref.js.map +7 -0
  175. package/dist/modules/notifications/migrations/Migration20260123000001.js +70 -0
  176. package/dist/modules/notifications/migrations/Migration20260123000001.js.map +7 -0
  177. package/dist/modules/notifications/migrations/Migration20260126150000.js +37 -0
  178. package/dist/modules/notifications/migrations/Migration20260126150000.js.map +7 -0
  179. package/dist/modules/notifications/subscribers/deliver-notification.js +139 -0
  180. package/dist/modules/notifications/subscribers/deliver-notification.js.map +7 -0
  181. package/dist/modules/notifications/workers/create-notification.worker.js +70 -0
  182. package/dist/modules/notifications/workers/create-notification.worker.js.map +7 -0
  183. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +2 -2
  184. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  185. package/dist/modules/query_index/components/QueryIndexesTable.js +7 -1
  186. package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
  187. package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
  188. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  189. package/dist/modules/resources/backend/resources/resources/page.js +2 -2
  190. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  191. package/dist/modules/sales/backend/sales/channels/offers/page.js +2 -0
  192. package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
  193. package/dist/modules/sales/backend/sales/channels/page.js +2 -0
  194. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  195. package/dist/modules/sales/commands/documents.js +53 -0
  196. package/dist/modules/sales/commands/documents.js.map +2 -2
  197. package/dist/modules/sales/commands/payments.js +26 -0
  198. package/dist/modules/sales/commands/payments.js.map +2 -2
  199. package/dist/modules/sales/components/AdjustmentKindSettings.js +2 -2
  200. package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
  201. package/dist/modules/sales/components/PaymentMethodsSettings.js +2 -2
  202. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  203. package/dist/modules/sales/components/ShippingMethodsSettings.js +2 -2
  204. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  205. package/dist/modules/sales/components/TaxRatesSettings.js +2 -2
  206. package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
  207. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +2 -0
  208. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
  209. package/dist/modules/sales/components/documents/AdjustmentsSection.js +2 -0
  210. package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
  211. package/dist/modules/sales/components/documents/PaymentsSection.js +2 -1
  212. package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
  213. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -0
  214. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  215. package/dist/modules/sales/notifications.client.js +51 -0
  216. package/dist/modules/sales/notifications.client.js.map +7 -0
  217. package/dist/modules/sales/notifications.js +88 -0
  218. package/dist/modules/sales/notifications.js.map +7 -0
  219. package/dist/modules/sales/subscribers/quote-expiring-notification.js +38 -0
  220. package/dist/modules/sales/subscribers/quote-expiring-notification.js.map +7 -0
  221. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +137 -0
  222. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +7 -0
  223. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +137 -0
  224. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +7 -0
  225. package/dist/modules/sales/widgets/notifications/index.js +7 -0
  226. package/dist/modules/sales/widgets/notifications/index.js.map +7 -0
  227. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js +60 -0
  228. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js.map +7 -0
  229. package/dist/modules/staff/backend/staff/team-members/page.js +1 -1
  230. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  231. package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
  232. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  233. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +2 -2
  234. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  235. package/dist/modules/staff/backend/staff/teams/page.js +2 -2
  236. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  237. package/dist/modules/staff/commands/leave-requests.js +79 -0
  238. package/dist/modules/staff/commands/leave-requests.js.map +2 -2
  239. package/dist/modules/staff/notifications.js +75 -0
  240. package/dist/modules/staff/notifications.js.map +7 -0
  241. package/dist/modules/workflows/backend/definitions/page.js +5 -0
  242. package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
  243. package/dist/modules/workflows/backend/instances/page.js +3 -0
  244. package/dist/modules/workflows/backend/instances/page.js.map +2 -2
  245. package/dist/modules/workflows/backend/tasks/page.js +3 -0
  246. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  247. package/dist/modules/workflows/notifications.js +28 -0
  248. package/dist/modules/workflows/notifications.js.map +7 -0
  249. package/dist/modules/workflows/subscribers/task-assigned-notification.js +38 -0
  250. package/dist/modules/workflows/subscribers/task-assigned-notification.js.map +7 -0
  251. package/generated/entities/notification/index.ts +27 -0
  252. package/generated/entities.ids.generated.ts +5 -1
  253. package/generated/entity-fields-registry.ts +2 -0
  254. package/package.json +2 -2
  255. package/src/modules/api_docs/frontend/docs/api/page.tsx +3 -2
  256. package/src/modules/api_keys/backend/api-keys/page.tsx +1 -1
  257. package/src/modules/attachments/components/AttachmentLibrary.tsx +4 -0
  258. package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +2 -0
  259. package/src/modules/auth/api/admin/nav.ts +10 -6
  260. package/src/modules/auth/api/profile/route.ts +163 -0
  261. package/src/modules/auth/api/reset/confirm.ts +25 -2
  262. package/src/modules/auth/api/reset.ts +23 -0
  263. package/src/modules/auth/api/sidebar/preferences/route.ts +21 -12
  264. package/src/modules/auth/api/users/route.ts +5 -2
  265. package/src/modules/auth/backend/auth/profile/page.meta.ts +9 -0
  266. package/src/modules/auth/backend/auth/profile/page.tsx +174 -0
  267. package/src/modules/auth/backend/roles/page.tsx +3 -3
  268. package/src/modules/auth/backend/users/[id]/edit/page.tsx +18 -2
  269. package/src/modules/auth/backend/users/create/page.tsx +19 -2
  270. package/src/modules/auth/backend/users/page.tsx +3 -3
  271. package/src/modules/auth/cli.ts +14 -0
  272. package/src/modules/auth/commands/users.ts +73 -2
  273. package/src/modules/auth/data/validators.ts +5 -2
  274. package/src/modules/auth/frontend/reset/[token]/page.tsx +24 -11
  275. package/src/modules/auth/i18n/de.json +43 -1
  276. package/src/modules/auth/i18n/en.json +43 -1
  277. package/src/modules/auth/i18n/es.json +43 -1
  278. package/src/modules/auth/i18n/pl.json +43 -1
  279. package/src/modules/auth/lib/setup-app.ts +1 -0
  280. package/src/modules/auth/notifications.ts +109 -0
  281. package/src/modules/auth/services/authService.ts +4 -4
  282. package/src/modules/business_rules/backend/rules/page.tsx +4 -0
  283. package/src/modules/business_rules/backend/sets/page.tsx +3 -0
  284. package/src/modules/business_rules/i18n/en.json +3 -1
  285. package/src/modules/business_rules/notifications.ts +25 -0
  286. package/src/modules/business_rules/subscribers/rule-execution-failed-notification.ts +50 -0
  287. package/src/modules/catalog/components/PriceKindSettings.tsx +2 -0
  288. package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +2 -2
  289. package/src/modules/catalog/components/products/ProductsDataTable.tsx +2 -0
  290. package/src/modules/catalog/i18n/en.json +3 -1
  291. package/src/modules/catalog/notifications.ts +25 -0
  292. package/src/modules/catalog/subscribers/low-stock-notification.ts +52 -0
  293. package/src/modules/configs/cli.ts +6 -0
  294. package/src/modules/configs/lib/upgrade-actions.ts +18 -0
  295. package/src/modules/currencies/backend/currencies/page.tsx +3 -0
  296. package/src/modules/currencies/backend/exchange-rates/page.tsx +2 -0
  297. package/src/modules/customers/backend/customers/companies/page.tsx +3 -0
  298. package/src/modules/customers/backend/customers/deals/page.tsx +3 -0
  299. package/src/modules/customers/backend/customers/people/page.tsx +3 -0
  300. package/src/modules/customers/commands/deals.ts +39 -0
  301. package/src/modules/customers/components/CustomerTodosTable.tsx +1 -0
  302. package/src/modules/customers/i18n/en.json +5 -1
  303. package/src/modules/customers/notifications.ts +44 -0
  304. package/src/modules/dashboards/cli.ts +41 -1
  305. package/src/modules/dashboards/lib/role-widgets.ts +80 -0
  306. package/src/modules/dictionaries/components/DictionaryTable.tsx +2 -0
  307. package/src/modules/directory/backend/directory/organizations/page.tsx +2 -2
  308. package/src/modules/directory/backend/directory/tenants/page.tsx +2 -2
  309. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +2 -2
  310. package/src/modules/entities/components/SystemEntitiesTable.tsx +1 -1
  311. package/src/modules/entities/components/UserEntitiesTable.tsx +2 -2
  312. package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +3 -4
  313. package/src/modules/feature_toggles/components/OverridesTable.tsx +1 -1
  314. package/src/modules/notifications/acl.ts +7 -0
  315. package/src/modules/notifications/api/[id]/action/route.ts +75 -0
  316. package/src/modules/notifications/api/[id]/dismiss/route.ts +12 -0
  317. package/src/modules/notifications/api/[id]/read/route.ts +12 -0
  318. package/src/modules/notifications/api/[id]/restore/route.ts +53 -0
  319. package/src/modules/notifications/api/batch/route.ts +14 -0
  320. package/src/modules/notifications/api/feature/route.ts +14 -0
  321. package/src/modules/notifications/api/mark-all-read/route.ts +34 -0
  322. package/src/modules/notifications/api/openapi.ts +76 -0
  323. package/src/modules/notifications/api/role/route.ts +14 -0
  324. package/src/modules/notifications/api/route.ts +92 -0
  325. package/src/modules/notifications/api/settings/route.ts +157 -0
  326. package/src/modules/notifications/api/unread-count/route.ts +38 -0
  327. package/src/modules/notifications/backend/config/notifications/page.meta.ts +22 -0
  328. package/src/modules/notifications/backend/config/notifications/page.tsx +12 -0
  329. package/src/modules/notifications/cli.ts +18 -0
  330. package/src/modules/notifications/data/entities.ts +99 -0
  331. package/src/modules/notifications/data/validators.ts +110 -0
  332. package/src/modules/notifications/di.ts +11 -0
  333. package/src/modules/notifications/emails/NotificationEmail.tsx +98 -0
  334. package/src/modules/notifications/frontend/NotificationInboxPageClient.tsx +42 -0
  335. package/src/modules/notifications/frontend/NotificationSettingsPageClient.tsx +231 -0
  336. package/src/modules/notifications/i18n/de.json +50 -0
  337. package/src/modules/notifications/i18n/en.json +50 -0
  338. package/src/modules/notifications/i18n/es.json +50 -0
  339. package/src/modules/notifications/i18n/pl.json +50 -0
  340. package/src/modules/notifications/index.ts +12 -0
  341. package/src/modules/notifications/lib/deliveryConfig.ts +145 -0
  342. package/src/modules/notifications/lib/events.ts +48 -0
  343. package/src/modules/notifications/lib/notificationBuilder.ts +121 -0
  344. package/src/modules/notifications/lib/notificationFactory.ts +76 -0
  345. package/src/modules/notifications/lib/notificationMapper.ts +33 -0
  346. package/src/modules/notifications/lib/notificationRecipients.ts +83 -0
  347. package/src/modules/notifications/lib/notificationService.ts +414 -0
  348. package/src/modules/notifications/lib/routeHelpers.ts +151 -0
  349. package/src/modules/notifications/lib/safeHref.ts +29 -0
  350. package/src/modules/notifications/migrations/.snapshot-open-mercato.json +300 -0
  351. package/src/modules/notifications/migrations/Migration20260123000001.ts +73 -0
  352. package/src/modules/notifications/migrations/Migration20260126150000.ts +39 -0
  353. package/src/modules/notifications/subscribers/deliver-notification.ts +175 -0
  354. package/src/modules/notifications/workers/create-notification.worker.ts +122 -0
  355. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +2 -2
  356. package/src/modules/query_index/components/QueryIndexesTable.tsx +8 -2
  357. package/src/modules/resources/backend/resources/resource-types/page.tsx +2 -2
  358. package/src/modules/resources/backend/resources/resources/page.tsx +2 -2
  359. package/src/modules/sales/backend/sales/channels/offers/page.tsx +2 -0
  360. package/src/modules/sales/backend/sales/channels/page.tsx +2 -0
  361. package/src/modules/sales/commands/documents.ts +65 -0
  362. package/src/modules/sales/commands/payments.ts +33 -0
  363. package/src/modules/sales/components/AdjustmentKindSettings.tsx +2 -2
  364. package/src/modules/sales/components/PaymentMethodsSettings.tsx +2 -2
  365. package/src/modules/sales/components/ShippingMethodsSettings.tsx +2 -2
  366. package/src/modules/sales/components/TaxRatesSettings.tsx +2 -2
  367. package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +2 -0
  368. package/src/modules/sales/components/documents/AdjustmentsSection.tsx +2 -0
  369. package/src/modules/sales/components/documents/PaymentsSection.tsx +2 -1
  370. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -0
  371. package/src/modules/sales/i18n/de.json +20 -0
  372. package/src/modules/sales/i18n/en.json +25 -1
  373. package/src/modules/sales/i18n/es.json +20 -0
  374. package/src/modules/sales/i18n/pl.json +20 -0
  375. package/src/modules/sales/notifications.client.ts +65 -0
  376. package/src/modules/sales/notifications.ts +82 -0
  377. package/src/modules/sales/subscribers/quote-expiring-notification.ts +53 -0
  378. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +156 -0
  379. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +156 -0
  380. package/src/modules/sales/widgets/notifications/index.ts +2 -0
  381. package/src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts +81 -0
  382. package/src/modules/staff/backend/staff/team-members/page.tsx +1 -1
  383. package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
  384. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +2 -2
  385. package/src/modules/staff/backend/staff/teams/page.tsx +2 -2
  386. package/src/modules/staff/commands/leave-requests.ts +94 -0
  387. package/src/modules/staff/i18n/de.json +4 -0
  388. package/src/modules/staff/i18n/en.json +9 -1
  389. package/src/modules/staff/i18n/es.json +4 -0
  390. package/src/modules/staff/i18n/pl.json +4 -0
  391. package/src/modules/staff/notifications.ts +71 -0
  392. package/src/modules/workflows/backend/definitions/page.tsx +5 -0
  393. package/src/modules/workflows/backend/instances/page.tsx +4 -1
  394. package/src/modules/workflows/backend/tasks/page.tsx +4 -1
  395. package/src/modules/workflows/i18n/en.json +3 -1
  396. package/src/modules/workflows/notifications.ts +25 -0
  397. package/src/modules/workflows/subscribers/task-assigned-notification.ts +53 -0
@@ -0,0 +1,65 @@
1
+ 'use client'
2
+
3
+ import type { NotificationTypeDefinition } from '@open-mercato/shared/modules/notifications/types'
4
+ import { SalesOrderCreatedRenderer } from './widgets/notifications/SalesOrderCreatedRenderer'
5
+ import { SalesQuoteCreatedRenderer } from './widgets/notifications/SalesQuoteCreatedRenderer'
6
+
7
+ /**
8
+ * Client-side notification type definitions with custom renderers.
9
+ * These should be used in client components where custom rendering is needed.
10
+ *
11
+ * Example usage:
12
+ * ```tsx
13
+ * import { salesNotificationTypes } from '@open-mercato/core/modules/sales/notifications.client'
14
+ *
15
+ * // Use in NotificationPanel or NotificationItem
16
+ * const renderer = salesNotificationTypes.find(t => t.type === notification.type)?.Renderer
17
+ * if (renderer) {
18
+ * return <renderer notification={notification} onAction={...} onDismiss={...} actions={...} />
19
+ * }
20
+ * ```
21
+ */
22
+ export const salesNotificationTypes: NotificationTypeDefinition[] = [
23
+ {
24
+ type: 'sales.order.created',
25
+ module: 'sales',
26
+ titleKey: 'sales.notifications.order.created.title',
27
+ bodyKey: 'sales.notifications.order.created.body',
28
+ icon: 'shopping-cart',
29
+ severity: 'info',
30
+ actions: [
31
+ {
32
+ id: 'view',
33
+ labelKey: 'common.view',
34
+ variant: 'outline',
35
+ href: '/backend/sales/orders/{sourceEntityId}',
36
+ icon: 'external-link',
37
+ },
38
+ ],
39
+ linkHref: '/backend/sales/orders/{sourceEntityId}',
40
+ Renderer: SalesOrderCreatedRenderer,
41
+ expiresAfterHours: 168,
42
+ },
43
+ {
44
+ type: 'sales.quote.created',
45
+ module: 'sales',
46
+ titleKey: 'sales.notifications.quote.created.title',
47
+ bodyKey: 'sales.notifications.quote.created.body',
48
+ icon: 'file-text',
49
+ severity: 'info',
50
+ actions: [
51
+ {
52
+ id: 'view',
53
+ labelKey: 'common.view',
54
+ variant: 'outline',
55
+ href: '/backend/sales/quotes/{sourceEntityId}',
56
+ icon: 'external-link',
57
+ },
58
+ ],
59
+ linkHref: '/backend/sales/quotes/{sourceEntityId}',
60
+ Renderer: SalesQuoteCreatedRenderer,
61
+ expiresAfterHours: 168,
62
+ },
63
+ ]
64
+
65
+ export default salesNotificationTypes
@@ -0,0 +1,82 @@
1
+ import type { NotificationTypeDefinition } from '@open-mercato/shared/modules/notifications/types'
2
+
3
+ export const notificationTypes: NotificationTypeDefinition[] = [
4
+ {
5
+ type: 'sales.order.created',
6
+ module: 'sales',
7
+ titleKey: 'sales.notifications.order.created.title',
8
+ bodyKey: 'sales.notifications.order.created.body',
9
+ icon: 'shopping-cart',
10
+ severity: 'info',
11
+ actions: [
12
+ {
13
+ id: 'view',
14
+ labelKey: 'common.view',
15
+ variant: 'outline',
16
+ href: '/backend/sales/orders/{sourceEntityId}',
17
+ icon: 'external-link',
18
+ },
19
+ ],
20
+ linkHref: '/backend/sales/orders/{sourceEntityId}',
21
+ expiresAfterHours: 168, // 7 days
22
+ },
23
+ {
24
+ type: 'sales.quote.created',
25
+ module: 'sales',
26
+ titleKey: 'sales.notifications.quote.created.title',
27
+ bodyKey: 'sales.notifications.quote.created.body',
28
+ icon: 'file-text',
29
+ severity: 'info',
30
+ actions: [
31
+ {
32
+ id: 'view',
33
+ labelKey: 'common.view',
34
+ variant: 'outline',
35
+ href: '/backend/sales/quotes/{sourceEntityId}',
36
+ icon: 'external-link',
37
+ },
38
+ ],
39
+ linkHref: '/backend/sales/quotes/{sourceEntityId}',
40
+ expiresAfterHours: 168, // 7 days
41
+ },
42
+ {
43
+ type: 'sales.payment.received',
44
+ module: 'sales',
45
+ titleKey: 'sales.notifications.payment.received.title',
46
+ bodyKey: 'sales.notifications.payment.received.body',
47
+ icon: 'credit-card',
48
+ severity: 'success',
49
+ actions: [
50
+ {
51
+ id: 'view',
52
+ labelKey: 'common.view',
53
+ variant: 'outline',
54
+ href: '/backend/sales/orders/{sourceEntityId}',
55
+ icon: 'external-link',
56
+ },
57
+ ],
58
+ linkHref: '/backend/sales/orders/{sourceEntityId}',
59
+ expiresAfterHours: 168, // 7 days
60
+ },
61
+ {
62
+ type: 'sales.quote.expiring',
63
+ module: 'sales',
64
+ titleKey: 'sales.notifications.quote.expiring.title',
65
+ bodyKey: 'sales.notifications.quote.expiring.body',
66
+ icon: 'clock',
67
+ severity: 'warning',
68
+ actions: [
69
+ {
70
+ id: 'view',
71
+ labelKey: 'common.view',
72
+ variant: 'outline',
73
+ href: '/backend/sales/quotes/{sourceEntityId}',
74
+ icon: 'external-link',
75
+ },
76
+ ],
77
+ linkHref: '/backend/sales/quotes/{sourceEntityId}',
78
+ expiresAfterHours: 72, // 3 days
79
+ },
80
+ ]
81
+
82
+ export default notificationTypes
@@ -0,0 +1,53 @@
1
+ import type { EntityManager } from '@mikro-orm/postgresql'
2
+ import { resolveNotificationService } from '../../notifications/lib/notificationService'
3
+ import { buildFeatureNotificationFromType } from '../../notifications/lib/notificationBuilder'
4
+ import { notificationTypes } from '../notifications'
5
+
6
+ export const metadata = {
7
+ event: 'sales.quote.expiring',
8
+ persistent: true,
9
+ id: 'sales:quote-expiring-notification',
10
+ }
11
+
12
+ type QuoteExpiringPayload = {
13
+ quoteId: string
14
+ quoteNumber: string
15
+ expiresAt: string
16
+ daysUntilExpiry: number
17
+ customerName?: string | null
18
+ totalAmount?: string | null
19
+ tenantId: string
20
+ organizationId?: string | null
21
+ }
22
+
23
+ type ResolverContext = {
24
+ resolve: <T = unknown>(name: string) => T
25
+ }
26
+
27
+ export default async function handle(payload: QuoteExpiringPayload, ctx: ResolverContext) {
28
+ try {
29
+ const notificationService = resolveNotificationService(ctx)
30
+ const typeDef = notificationTypes.find((type) => type.type === 'sales.quote.expiring')
31
+ if (!typeDef) return
32
+
33
+ const notificationInput = buildFeatureNotificationFromType(typeDef, {
34
+ requiredFeature: 'sales.quotes.manage',
35
+ bodyVariables: {
36
+ quoteNumber: payload.quoteNumber,
37
+ expiresAt: payload.expiresAt,
38
+ daysUntilExpiry: String(payload.daysUntilExpiry),
39
+ customerName: payload.customerName ?? '',
40
+ },
41
+ sourceEntityType: 'sales:quote',
42
+ sourceEntityId: payload.quoteId,
43
+ linkHref: `/backend/sales/quotes/${payload.quoteId}`,
44
+ })
45
+
46
+ await notificationService.createForFeature(notificationInput, {
47
+ tenantId: payload.tenantId,
48
+ organizationId: payload.organizationId ?? null,
49
+ })
50
+ } catch (err) {
51
+ console.error('[sales:quote-expiring-notification] Failed to create notification:', err)
52
+ }
53
+ }
@@ -0,0 +1,156 @@
1
+ 'use client'
2
+
3
+ import * as React from 'react'
4
+ import { ShoppingCart, ExternalLink, DollarSign, User, Calendar } from 'lucide-react'
5
+ import { useRouter } from 'next/navigation'
6
+ import { Button } from '@open-mercato/ui/primitives/button'
7
+ import { cn } from '@open-mercato/shared/lib/utils'
8
+ import { useT } from '@open-mercato/shared/lib/i18n/context'
9
+ import type { NotificationRendererProps } from '@open-mercato/shared/modules/notifications/types'
10
+ import { formatMoney } from '../../components/documents/lineItemUtils'
11
+ import { useSalesDocumentTotals } from './useSalesDocumentTotals'
12
+
13
+ function formatTimeAgo(dateString: string, t: (key: string, fallback?: string) => string): string {
14
+ const date = new Date(dateString)
15
+ const now = new Date()
16
+ const diffMs = now.getTime() - date.getTime()
17
+ const diffMins = Math.floor(diffMs / 60000)
18
+ const diffHours = Math.floor(diffMs / 3600000)
19
+ const diffDays = Math.floor(diffMs / 86400000)
20
+
21
+ if (diffMins < 1) return t('common.time.justNow', 'just now')
22
+ if (diffMins < 60) return t('common.time.minutesAgo', '{count}m ago').replace('{count}', String(diffMins))
23
+ if (diffHours < 24) return t('common.time.hoursAgo', '{count}h ago').replace('{count}', String(diffHours))
24
+ if (diffDays < 7) return t('common.time.daysAgo', '{count}d ago').replace('{count}', String(diffDays))
25
+ return date.toLocaleDateString()
26
+ }
27
+
28
+ function normalizeTotal(value?: string | null): string | null {
29
+ if (!value) return null
30
+ let trimmed = value.trim()
31
+ if (trimmed.startsWith('(') && trimmed.endsWith(')')) {
32
+ trimmed = trimmed.slice(1, -1).trim()
33
+ }
34
+ return trimmed.length ? trimmed : null
35
+ }
36
+
37
+ export function SalesOrderCreatedRenderer({
38
+ notification,
39
+ onAction,
40
+ onDismiss,
41
+ actions = [],
42
+ }: NotificationRendererProps) {
43
+ const t = useT()
44
+ const router = useRouter()
45
+ const [executing, setExecuting] = React.useState(false)
46
+ const isUnread = notification.status === 'unread'
47
+ const orderNumber = notification.bodyVariables?.orderNumber ?? notification.titleVariables?.orderNumber
48
+ const fallbackTotal =
49
+ normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ??
50
+ normalizeTotal(notification.bodyVariables?.total ?? null)
51
+ const { totals } = useSalesDocumentTotals('order', notification.sourceEntityId)
52
+
53
+ const currentTotal =
54
+ totals && typeof totals.grandTotalGrossAmount === 'number'
55
+ ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode)
56
+ : fallbackTotal
57
+
58
+ const viewAction = actions.find((action) => action.id === 'view') ?? actions[0] ?? null
59
+
60
+ const handleView = async () => {
61
+ if (!viewAction) {
62
+ if (notification.linkHref) router.push(notification.linkHref)
63
+ return
64
+ }
65
+ setExecuting(true)
66
+ try {
67
+ await onAction(viewAction.id)
68
+ } finally {
69
+ setExecuting(false)
70
+ }
71
+ }
72
+
73
+ return (
74
+ <div
75
+ className={cn(
76
+ 'group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-blue-500',
77
+ isUnread && 'bg-blue-50/50 dark:bg-blue-950/20'
78
+ )}
79
+ onClick={handleView}
80
+ >
81
+ {isUnread && (
82
+ <div className="absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary" />
83
+ )}
84
+
85
+ <div className="flex gap-3">
86
+ <div className="flex-shrink-0 mt-0.5">
87
+ <div className="h-10 w-10 rounded-lg bg-blue-100 dark:bg-blue-900/40 flex items-center justify-center">
88
+ <ShoppingCart className="h-5 w-5 text-blue-600 dark:text-blue-400" />
89
+ </div>
90
+ </div>
91
+
92
+ <div className="flex-1 min-w-0">
93
+ <div className="flex items-start justify-between gap-2">
94
+ <div>
95
+ <h4 className={cn('text-sm font-medium', isUnread && 'font-semibold')}>
96
+ {notification.title}
97
+ </h4>
98
+ {orderNumber && (
99
+ <div className="flex items-center gap-1 mt-0.5">
100
+ <span className="text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded">
101
+ #{orderNumber}
102
+ </span>
103
+ </div>
104
+ )}
105
+ </div>
106
+ <span className="flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1">
107
+ <Calendar className="h-3 w-3" />
108
+ {formatTimeAgo(notification.createdAt, t)}
109
+ </span>
110
+ </div>
111
+
112
+ <div className="mt-2 flex items-center gap-4 text-xs text-muted-foreground">
113
+ {currentTotal && (
114
+ <div className="flex items-center gap-1">
115
+ <DollarSign className="h-3 w-3" />
116
+ <span className="font-medium text-foreground">{currentTotal}</span>
117
+ </div>
118
+ )}
119
+ <div className="flex items-center gap-1">
120
+ <User className="h-3 w-3" />
121
+ <span>{t('sales.notifications.renderer.assignedToYou', 'Assigned to you')}</span>
122
+ </div>
123
+ </div>
124
+
125
+ <div className="mt-3 flex gap-2">
126
+ <Button
127
+ variant="default"
128
+ size="sm"
129
+ onClick={(e) => {
130
+ e.stopPropagation()
131
+ handleView()
132
+ }}
133
+ disabled={executing || (!viewAction && !notification.linkHref)}
134
+ className="gap-1"
135
+ >
136
+ <ExternalLink className="h-3 w-3" />
137
+ {t('sales.notifications.renderer.viewOrder', 'View Order')}
138
+ </Button>
139
+ <Button
140
+ variant="ghost"
141
+ size="sm"
142
+ onClick={(e) => {
143
+ e.stopPropagation()
144
+ onDismiss()
145
+ }}
146
+ >
147
+ {t('notifications.actions.dismiss', 'Dismiss')}
148
+ </Button>
149
+ </div>
150
+ </div>
151
+ </div>
152
+ </div>
153
+ )
154
+ }
155
+
156
+ export default SalesOrderCreatedRenderer
@@ -0,0 +1,156 @@
1
+ 'use client'
2
+
3
+ import * as React from 'react'
4
+ import { FileText, ExternalLink, DollarSign, User, Calendar } from 'lucide-react'
5
+ import { useRouter } from 'next/navigation'
6
+ import { Button } from '@open-mercato/ui/primitives/button'
7
+ import { cn } from '@open-mercato/shared/lib/utils'
8
+ import { useT } from '@open-mercato/shared/lib/i18n/context'
9
+ import type { NotificationRendererProps } from '@open-mercato/shared/modules/notifications/types'
10
+ import { formatMoney } from '../../components/documents/lineItemUtils'
11
+ import { useSalesDocumentTotals } from './useSalesDocumentTotals'
12
+
13
+ function formatTimeAgo(dateString: string, t: (key: string, fallback?: string) => string): string {
14
+ const date = new Date(dateString)
15
+ const now = new Date()
16
+ const diffMs = now.getTime() - date.getTime()
17
+ const diffMins = Math.floor(diffMs / 60000)
18
+ const diffHours = Math.floor(diffMs / 3600000)
19
+ const diffDays = Math.floor(diffMs / 86400000)
20
+
21
+ if (diffMins < 1) return t('common.time.justNow', 'just now')
22
+ if (diffMins < 60) return t('common.time.minutesAgo', '{count}m ago').replace('{count}', String(diffMins))
23
+ if (diffHours < 24) return t('common.time.hoursAgo', '{count}h ago').replace('{count}', String(diffHours))
24
+ if (diffDays < 7) return t('common.time.daysAgo', '{count}d ago').replace('{count}', String(diffDays))
25
+ return date.toLocaleDateString()
26
+ }
27
+
28
+ function normalizeTotal(value?: string | null): string | null {
29
+ if (!value) return null
30
+ let trimmed = value.trim()
31
+ if (trimmed.startsWith('(') && trimmed.endsWith(')')) {
32
+ trimmed = trimmed.slice(1, -1).trim()
33
+ }
34
+ return trimmed.length ? trimmed : null
35
+ }
36
+
37
+ export function SalesQuoteCreatedRenderer({
38
+ notification,
39
+ onAction,
40
+ onDismiss,
41
+ actions = [],
42
+ }: NotificationRendererProps) {
43
+ const t = useT()
44
+ const router = useRouter()
45
+ const [executing, setExecuting] = React.useState(false)
46
+ const isUnread = notification.status === 'unread'
47
+ const quoteNumber = notification.bodyVariables?.quoteNumber ?? notification.titleVariables?.quoteNumber
48
+ const fallbackTotal =
49
+ normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ??
50
+ normalizeTotal(notification.bodyVariables?.total ?? null)
51
+ const { totals } = useSalesDocumentTotals('quote', notification.sourceEntityId)
52
+
53
+ const currentTotal =
54
+ totals && typeof totals.grandTotalGrossAmount === 'number'
55
+ ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode)
56
+ : fallbackTotal
57
+
58
+ const viewAction = actions.find((action) => action.id === 'view') ?? actions[0] ?? null
59
+
60
+ const handleView = async () => {
61
+ if (!viewAction) {
62
+ if (notification.linkHref) router.push(notification.linkHref)
63
+ return
64
+ }
65
+ setExecuting(true)
66
+ try {
67
+ await onAction(viewAction.id)
68
+ } finally {
69
+ setExecuting(false)
70
+ }
71
+ }
72
+
73
+ return (
74
+ <div
75
+ className={cn(
76
+ 'group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-amber-500',
77
+ isUnread && 'bg-amber-50/50 dark:bg-amber-950/20'
78
+ )}
79
+ onClick={handleView}
80
+ >
81
+ {isUnread && (
82
+ <div className="absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary" />
83
+ )}
84
+
85
+ <div className="flex gap-3">
86
+ <div className="flex-shrink-0 mt-0.5">
87
+ <div className="h-10 w-10 rounded-lg bg-amber-100 dark:bg-amber-900/40 flex items-center justify-center">
88
+ <FileText className="h-5 w-5 text-amber-600 dark:text-amber-400" />
89
+ </div>
90
+ </div>
91
+
92
+ <div className="flex-1 min-w-0">
93
+ <div className="flex items-start justify-between gap-2">
94
+ <div>
95
+ <h4 className={cn('text-sm font-medium', isUnread && 'font-semibold')}>
96
+ {notification.title}
97
+ </h4>
98
+ {quoteNumber && (
99
+ <div className="flex items-center gap-1 mt-0.5">
100
+ <span className="text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded">
101
+ #{quoteNumber}
102
+ </span>
103
+ </div>
104
+ )}
105
+ </div>
106
+ <span className="flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1">
107
+ <Calendar className="h-3 w-3" />
108
+ {formatTimeAgo(notification.createdAt, t)}
109
+ </span>
110
+ </div>
111
+
112
+ <div className="mt-2 flex items-center gap-4 text-xs text-muted-foreground">
113
+ {currentTotal && (
114
+ <div className="flex items-center gap-1">
115
+ <DollarSign className="h-3 w-3" />
116
+ <span className="font-medium text-foreground">{currentTotal}</span>
117
+ </div>
118
+ )}
119
+ <div className="flex items-center gap-1">
120
+ <User className="h-3 w-3" />
121
+ <span>{t('sales.notifications.renderer.pendingReview', 'Pending review')}</span>
122
+ </div>
123
+ </div>
124
+
125
+ <div className="mt-3 flex gap-2">
126
+ <Button
127
+ variant="default"
128
+ size="sm"
129
+ onClick={(e) => {
130
+ e.stopPropagation()
131
+ handleView()
132
+ }}
133
+ disabled={executing || (!viewAction && !notification.linkHref)}
134
+ className="gap-1"
135
+ >
136
+ <ExternalLink className="h-3 w-3" />
137
+ {t('sales.notifications.renderer.viewQuote', 'View Quote')}
138
+ </Button>
139
+ <Button
140
+ variant="ghost"
141
+ size="sm"
142
+ onClick={(e) => {
143
+ e.stopPropagation()
144
+ onDismiss()
145
+ }}
146
+ >
147
+ {t('notifications.actions.dismiss', 'Dismiss')}
148
+ </Button>
149
+ </div>
150
+ </div>
151
+ </div>
152
+ </div>
153
+ )
154
+ }
155
+
156
+ export default SalesQuoteCreatedRenderer
@@ -0,0 +1,2 @@
1
+ export { SalesOrderCreatedRenderer } from './SalesOrderCreatedRenderer'
2
+ export { SalesQuoteCreatedRenderer } from './SalesQuoteCreatedRenderer'
@@ -0,0 +1,81 @@
1
+ 'use client'
2
+
3
+ import * as React from 'react'
4
+ import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
5
+
6
+ type DocumentKind = 'order' | 'quote'
7
+
8
+ type DocumentTotals = {
9
+ grandTotalGrossAmount: number | null
10
+ currencyCode: string | null
11
+ }
12
+
13
+ type DocumentListResponse = {
14
+ items?: Array<{
15
+ grandTotalGrossAmount?: number | string | null
16
+ currencyCode?: string | null
17
+ }>
18
+ }
19
+
20
+ const REFRESH_INTERVAL_MS = 30000
21
+
22
+ function buildDocumentTotalsUrl(kind: DocumentKind, documentId: string) {
23
+ const params = new URLSearchParams({ id: documentId, page: '1', pageSize: '1' })
24
+ const collection = kind === 'order' ? 'orders' : 'quotes'
25
+ return `/api/sales/${collection}?${params.toString()}`
26
+ }
27
+
28
+ function extractTotals(payload: DocumentListResponse | null): DocumentTotals | null {
29
+ const item = payload?.items?.[0]
30
+ if (!item) return null
31
+ const rawAmount = item.grandTotalGrossAmount
32
+ let grandTotalGrossAmount: number | null = null
33
+ if (typeof rawAmount === 'number') {
34
+ grandTotalGrossAmount = Number.isNaN(rawAmount) ? null : rawAmount
35
+ } else if (typeof rawAmount === 'string' && rawAmount.trim().length) {
36
+ const parsed = Number(rawAmount)
37
+ grandTotalGrossAmount = Number.isNaN(parsed) ? null : parsed
38
+ }
39
+ return {
40
+ grandTotalGrossAmount,
41
+ currencyCode: typeof item.currencyCode === 'string' ? item.currencyCode : null,
42
+ }
43
+ }
44
+
45
+ export function useSalesDocumentTotals(kind: DocumentKind, documentId?: string | null) {
46
+ const [totals, setTotals] = React.useState<DocumentTotals | null>(null)
47
+
48
+ React.useEffect(() => {
49
+ if (!documentId) {
50
+ setTotals(null)
51
+ return
52
+ }
53
+
54
+ let active = true
55
+
56
+ const loadTotals = async () => {
57
+ try {
58
+ const call = await apiCall<DocumentListResponse>(buildDocumentTotalsUrl(kind, documentId))
59
+ if (!active) return
60
+ if (call.ok) {
61
+ const nextTotals = extractTotals(call.result ?? null)
62
+ setTotals(nextTotals)
63
+ }
64
+ } catch {
65
+ if (active) {
66
+ setTotals(null)
67
+ }
68
+ }
69
+ }
70
+
71
+ loadTotals()
72
+ const interval = setInterval(loadTotals, REFRESH_INTERVAL_MS)
73
+
74
+ return () => {
75
+ active = false
76
+ clearInterval(interval)
77
+ }
78
+ }, [kind, documentId])
79
+
80
+ return { totals }
81
+ }
@@ -434,7 +434,7 @@ export default function StaffTeamMembersPage() {
434
434
  <RowActions
435
435
  items={[
436
436
  { id: 'edit', label: labels.actions.edit, onSelect: () => { router.push(`/backend/staff/team-members/${row.id}`) } },
437
- { label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } },
437
+ { id: 'delete', label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } },
438
438
  ]}
439
439
  />
440
440
  ) : null}
@@ -345,8 +345,8 @@ export default function StaffTeamRolesPage() {
345
345
  rowActions={(row) => row.kind === 'role' ? (
346
346
  <RowActions
347
347
  items={[
348
- { label: labels.actions.edit, onSelect: () => { router.push(`/backend/staff/team-roles/${row.id}/edit`) } },
349
- { label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } },
348
+ { id: 'edit', label: labels.actions.edit, onSelect: () => { router.push(`/backend/staff/team-roles/${row.id}/edit`) } },
349
+ { id: 'delete', label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } },
350
350
  ]}
351
351
  />
352
352
  ) : null}
@@ -369,8 +369,8 @@ export default function StaffTeamEditPage({ params }: { params?: { id?: string }
369
369
  rowActions={(row) => (
370
370
  <RowActions
371
371
  items={[
372
- { label: memberLabels.actions.edit, onSelect: () => { router.push(`/backend/staff/team-members/${row.id}`) } },
373
- { label: memberLabels.actions.unassign, onSelect: () => { void handleUnassignMember(row) } },
372
+ { id: 'edit', label: memberLabels.actions.edit, onSelect: () => { router.push(`/backend/staff/team-members/${row.id}`) } },
373
+ { id: 'unassign', label: memberLabels.actions.unassign, onSelect: () => { void handleUnassignMember(row) } },
374
374
  ]}
375
375
  />
376
376
  )}
@@ -275,10 +275,10 @@ export default function StaffTeamsPage() {
275
275
  rowActions={(row) => (
276
276
  <RowActions
277
277
  items={[
278
- { label: labels.actions.edit, href: `/backend/staff/teams/${row.id}/edit` },
278
+ { id: 'edit', label: labels.actions.edit, href: `/backend/staff/teams/${row.id}/edit` },
279
279
  ...(row.memberCount > 0
280
280
  ? []
281
- : [{ label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } }]),
281
+ : [{ id: 'delete', label: labels.actions.delete, destructive: true, onSelect: () => { void handleDelete(row) } }]),
282
282
  ]}
283
283
  />
284
284
  )}