@open-mercato/core 0.4.2-canary-36ab8921da → 0.4.2-canary-07dbc98202

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 (235) 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 +63 -59
  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/auth/api/admin/nav.js +4 -3
  10. package/dist/modules/auth/api/admin/nav.js.map +2 -2
  11. package/dist/modules/auth/api/profile/route.js +155 -0
  12. package/dist/modules/auth/api/profile/route.js.map +7 -0
  13. package/dist/modules/auth/api/reset/confirm.js +25 -2
  14. package/dist/modules/auth/api/reset/confirm.js.map +2 -2
  15. package/dist/modules/auth/api/reset.js +23 -0
  16. package/dist/modules/auth/api/reset.js.map +2 -2
  17. package/dist/modules/auth/api/sidebar/preferences/route.js +14 -9
  18. package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
  19. package/dist/modules/auth/backend/auth/profile/page.js +99 -0
  20. package/dist/modules/auth/backend/auth/profile/page.js.map +7 -0
  21. package/dist/modules/auth/backend/auth/profile/page.meta.js +12 -0
  22. package/dist/modules/auth/backend/auth/profile/page.meta.js.map +7 -0
  23. package/dist/modules/auth/commands/users.js +55 -0
  24. package/dist/modules/auth/commands/users.js.map +2 -2
  25. package/dist/modules/auth/lib/setup-app.js +1 -0
  26. package/dist/modules/auth/lib/setup-app.js.map +2 -2
  27. package/dist/modules/auth/notifications.js +112 -0
  28. package/dist/modules/auth/notifications.js.map +7 -0
  29. package/dist/modules/auth/services/authService.js +3 -3
  30. package/dist/modules/auth/services/authService.js.map +2 -2
  31. package/dist/modules/business_rules/notifications.js +28 -0
  32. package/dist/modules/business_rules/notifications.js.map +7 -0
  33. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js +37 -0
  34. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js.map +7 -0
  35. package/dist/modules/catalog/notifications.js +28 -0
  36. package/dist/modules/catalog/notifications.js.map +7 -0
  37. package/dist/modules/catalog/subscribers/low-stock-notification.js +38 -0
  38. package/dist/modules/catalog/subscribers/low-stock-notification.js.map +7 -0
  39. package/dist/modules/configs/cli.js +6 -0
  40. package/dist/modules/configs/cli.js.map +2 -2
  41. package/dist/modules/customers/commands/deals.js +31 -0
  42. package/dist/modules/customers/commands/deals.js.map +2 -2
  43. package/dist/modules/customers/notifications.js +48 -0
  44. package/dist/modules/customers/notifications.js.map +7 -0
  45. package/dist/modules/notifications/acl.js +11 -0
  46. package/dist/modules/notifications/acl.js.map +7 -0
  47. package/dist/modules/notifications/api/[id]/action/route.js +74 -0
  48. package/dist/modules/notifications/api/[id]/action/route.js.map +7 -0
  49. package/dist/modules/notifications/api/[id]/dismiss/route.js +15 -0
  50. package/dist/modules/notifications/api/[id]/dismiss/route.js.map +7 -0
  51. package/dist/modules/notifications/api/[id]/read/route.js +15 -0
  52. package/dist/modules/notifications/api/[id]/read/route.js.map +7 -0
  53. package/dist/modules/notifications/api/[id]/restore/route.js +53 -0
  54. package/dist/modules/notifications/api/[id]/restore/route.js.map +7 -0
  55. package/dist/modules/notifications/api/batch/route.js +17 -0
  56. package/dist/modules/notifications/api/batch/route.js.map +7 -0
  57. package/dist/modules/notifications/api/feature/route.js +17 -0
  58. package/dist/modules/notifications/api/feature/route.js.map +7 -0
  59. package/dist/modules/notifications/api/mark-all-read/route.js +35 -0
  60. package/dist/modules/notifications/api/mark-all-read/route.js.map +7 -0
  61. package/dist/modules/notifications/api/openapi.js +76 -0
  62. package/dist/modules/notifications/api/openapi.js.map +7 -0
  63. package/dist/modules/notifications/api/role/route.js +17 -0
  64. package/dist/modules/notifications/api/role/route.js.map +7 -0
  65. package/dist/modules/notifications/api/route.js +85 -0
  66. package/dist/modules/notifications/api/route.js.map +7 -0
  67. package/dist/modules/notifications/api/settings/route.js +155 -0
  68. package/dist/modules/notifications/api/settings/route.js.map +7 -0
  69. package/dist/modules/notifications/api/unread-count/route.js +38 -0
  70. package/dist/modules/notifications/api/unread-count/route.js.map +7 -0
  71. package/dist/modules/notifications/backend/config/notifications/page.js +10 -0
  72. package/dist/modules/notifications/backend/config/notifications/page.js.map +7 -0
  73. package/dist/modules/notifications/backend/config/notifications/page.meta.js +24 -0
  74. package/dist/modules/notifications/backend/config/notifications/page.meta.js.map +7 -0
  75. package/dist/modules/notifications/cli.js +16 -0
  76. package/dist/modules/notifications/cli.js.map +7 -0
  77. package/dist/modules/notifications/data/entities.js +112 -0
  78. package/dist/modules/notifications/data/entities.js.map +7 -0
  79. package/dist/modules/notifications/data/validators.js +94 -0
  80. package/dist/modules/notifications/data/validators.js.map +7 -0
  81. package/dist/modules/notifications/di.js +13 -0
  82. package/dist/modules/notifications/di.js.map +7 -0
  83. package/dist/modules/notifications/emails/NotificationEmail.js +58 -0
  84. package/dist/modules/notifications/emails/NotificationEmail.js.map +7 -0
  85. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js +44 -0
  86. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js.map +7 -0
  87. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js +219 -0
  88. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js.map +7 -0
  89. package/dist/modules/notifications/index.js +14 -0
  90. package/dist/modules/notifications/index.js.map +7 -0
  91. package/dist/modules/notifications/lib/deliveryConfig.js +105 -0
  92. package/dist/modules/notifications/lib/deliveryConfig.js.map +7 -0
  93. package/dist/modules/notifications/lib/events.js +12 -0
  94. package/dist/modules/notifications/lib/events.js.map +7 -0
  95. package/dist/modules/notifications/lib/notificationBuilder.js +66 -0
  96. package/dist/modules/notifications/lib/notificationBuilder.js.map +7 -0
  97. package/dist/modules/notifications/lib/notificationFactory.js +54 -0
  98. package/dist/modules/notifications/lib/notificationFactory.js.map +7 -0
  99. package/dist/modules/notifications/lib/notificationMapper.js +34 -0
  100. package/dist/modules/notifications/lib/notificationMapper.js.map +7 -0
  101. package/dist/modules/notifications/lib/notificationRecipients.js +35 -0
  102. package/dist/modules/notifications/lib/notificationRecipients.js.map +7 -0
  103. package/dist/modules/notifications/lib/notificationService.js +279 -0
  104. package/dist/modules/notifications/lib/notificationService.js.map +7 -0
  105. package/dist/modules/notifications/lib/routeHelpers.js +101 -0
  106. package/dist/modules/notifications/lib/routeHelpers.js.map +7 -0
  107. package/dist/modules/notifications/lib/safeHref.js +24 -0
  108. package/dist/modules/notifications/lib/safeHref.js.map +7 -0
  109. package/dist/modules/notifications/migrations/Migration20260123000001.js +70 -0
  110. package/dist/modules/notifications/migrations/Migration20260123000001.js.map +7 -0
  111. package/dist/modules/notifications/migrations/Migration20260126150000.js +37 -0
  112. package/dist/modules/notifications/migrations/Migration20260126150000.js.map +7 -0
  113. package/dist/modules/notifications/subscribers/deliver-notification.js +139 -0
  114. package/dist/modules/notifications/subscribers/deliver-notification.js.map +7 -0
  115. package/dist/modules/notifications/workers/create-notification.worker.js +70 -0
  116. package/dist/modules/notifications/workers/create-notification.worker.js.map +7 -0
  117. package/dist/modules/sales/commands/documents.js +53 -0
  118. package/dist/modules/sales/commands/documents.js.map +2 -2
  119. package/dist/modules/sales/commands/payments.js +26 -0
  120. package/dist/modules/sales/commands/payments.js.map +2 -2
  121. package/dist/modules/sales/notifications.client.js +51 -0
  122. package/dist/modules/sales/notifications.client.js.map +7 -0
  123. package/dist/modules/sales/notifications.js +88 -0
  124. package/dist/modules/sales/notifications.js.map +7 -0
  125. package/dist/modules/sales/subscribers/quote-expiring-notification.js +38 -0
  126. package/dist/modules/sales/subscribers/quote-expiring-notification.js.map +7 -0
  127. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +137 -0
  128. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +7 -0
  129. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +137 -0
  130. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +7 -0
  131. package/dist/modules/sales/widgets/notifications/index.js +7 -0
  132. package/dist/modules/sales/widgets/notifications/index.js.map +7 -0
  133. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js +60 -0
  134. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js.map +7 -0
  135. package/dist/modules/staff/commands/leave-requests.js +79 -0
  136. package/dist/modules/staff/commands/leave-requests.js.map +2 -2
  137. package/dist/modules/staff/notifications.js +75 -0
  138. package/dist/modules/staff/notifications.js.map +7 -0
  139. package/dist/modules/workflows/notifications.js +28 -0
  140. package/dist/modules/workflows/notifications.js.map +7 -0
  141. package/dist/modules/workflows/subscribers/task-assigned-notification.js +38 -0
  142. package/dist/modules/workflows/subscribers/task-assigned-notification.js.map +7 -0
  143. package/generated/entities/notification/index.ts +27 -0
  144. package/generated/entities.ids.generated.ts +63 -59
  145. package/generated/entity-fields-registry.ts +2 -0
  146. package/package.json +2 -2
  147. package/src/modules/api_docs/frontend/docs/api/page.tsx +3 -2
  148. package/src/modules/auth/api/admin/nav.ts +10 -6
  149. package/src/modules/auth/api/profile/route.ts +160 -0
  150. package/src/modules/auth/api/reset/confirm.ts +25 -2
  151. package/src/modules/auth/api/reset.ts +23 -0
  152. package/src/modules/auth/api/sidebar/preferences/route.ts +21 -12
  153. package/src/modules/auth/backend/auth/profile/page.meta.ts +8 -0
  154. package/src/modules/auth/backend/auth/profile/page.tsx +127 -0
  155. package/src/modules/auth/commands/users.ts +68 -0
  156. package/src/modules/auth/i18n/de.json +29 -1
  157. package/src/modules/auth/i18n/en.json +29 -1
  158. package/src/modules/auth/i18n/es.json +29 -1
  159. package/src/modules/auth/i18n/pl.json +29 -1
  160. package/src/modules/auth/lib/setup-app.ts +1 -0
  161. package/src/modules/auth/notifications.ts +109 -0
  162. package/src/modules/auth/services/authService.ts +4 -4
  163. package/src/modules/business_rules/i18n/en.json +3 -1
  164. package/src/modules/business_rules/notifications.ts +25 -0
  165. package/src/modules/business_rules/subscribers/rule-execution-failed-notification.ts +50 -0
  166. package/src/modules/catalog/i18n/en.json +3 -1
  167. package/src/modules/catalog/notifications.ts +25 -0
  168. package/src/modules/catalog/subscribers/low-stock-notification.ts +52 -0
  169. package/src/modules/configs/cli.ts +6 -0
  170. package/src/modules/customers/commands/deals.ts +39 -0
  171. package/src/modules/customers/i18n/en.json +5 -1
  172. package/src/modules/customers/notifications.ts +44 -0
  173. package/src/modules/notifications/acl.ts +7 -0
  174. package/src/modules/notifications/api/[id]/action/route.ts +75 -0
  175. package/src/modules/notifications/api/[id]/dismiss/route.ts +12 -0
  176. package/src/modules/notifications/api/[id]/read/route.ts +12 -0
  177. package/src/modules/notifications/api/[id]/restore/route.ts +53 -0
  178. package/src/modules/notifications/api/batch/route.ts +14 -0
  179. package/src/modules/notifications/api/feature/route.ts +14 -0
  180. package/src/modules/notifications/api/mark-all-read/route.ts +34 -0
  181. package/src/modules/notifications/api/openapi.ts +76 -0
  182. package/src/modules/notifications/api/role/route.ts +14 -0
  183. package/src/modules/notifications/api/route.ts +92 -0
  184. package/src/modules/notifications/api/settings/route.ts +157 -0
  185. package/src/modules/notifications/api/unread-count/route.ts +38 -0
  186. package/src/modules/notifications/backend/config/notifications/page.meta.ts +22 -0
  187. package/src/modules/notifications/backend/config/notifications/page.tsx +12 -0
  188. package/src/modules/notifications/cli.ts +18 -0
  189. package/src/modules/notifications/data/entities.ts +99 -0
  190. package/src/modules/notifications/data/validators.ts +110 -0
  191. package/src/modules/notifications/di.ts +11 -0
  192. package/src/modules/notifications/emails/NotificationEmail.tsx +98 -0
  193. package/src/modules/notifications/frontend/NotificationInboxPageClient.tsx +42 -0
  194. package/src/modules/notifications/frontend/NotificationSettingsPageClient.tsx +231 -0
  195. package/src/modules/notifications/i18n/de.json +50 -0
  196. package/src/modules/notifications/i18n/en.json +50 -0
  197. package/src/modules/notifications/i18n/es.json +50 -0
  198. package/src/modules/notifications/i18n/pl.json +50 -0
  199. package/src/modules/notifications/index.ts +12 -0
  200. package/src/modules/notifications/lib/deliveryConfig.ts +145 -0
  201. package/src/modules/notifications/lib/events.ts +48 -0
  202. package/src/modules/notifications/lib/notificationBuilder.ts +121 -0
  203. package/src/modules/notifications/lib/notificationFactory.ts +76 -0
  204. package/src/modules/notifications/lib/notificationMapper.ts +33 -0
  205. package/src/modules/notifications/lib/notificationRecipients.ts +83 -0
  206. package/src/modules/notifications/lib/notificationService.ts +414 -0
  207. package/src/modules/notifications/lib/routeHelpers.ts +151 -0
  208. package/src/modules/notifications/lib/safeHref.ts +29 -0
  209. package/src/modules/notifications/migrations/.snapshot-open-mercato.json +300 -0
  210. package/src/modules/notifications/migrations/Migration20260123000001.ts +73 -0
  211. package/src/modules/notifications/migrations/Migration20260126150000.ts +39 -0
  212. package/src/modules/notifications/subscribers/deliver-notification.ts +175 -0
  213. package/src/modules/notifications/workers/create-notification.worker.ts +122 -0
  214. package/src/modules/sales/commands/documents.ts +65 -0
  215. package/src/modules/sales/commands/payments.ts +33 -0
  216. package/src/modules/sales/i18n/de.json +20 -0
  217. package/src/modules/sales/i18n/en.json +25 -1
  218. package/src/modules/sales/i18n/es.json +20 -0
  219. package/src/modules/sales/i18n/pl.json +20 -0
  220. package/src/modules/sales/notifications.client.ts +65 -0
  221. package/src/modules/sales/notifications.ts +82 -0
  222. package/src/modules/sales/subscribers/quote-expiring-notification.ts +53 -0
  223. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +156 -0
  224. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +156 -0
  225. package/src/modules/sales/widgets/notifications/index.ts +2 -0
  226. package/src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts +81 -0
  227. package/src/modules/staff/commands/leave-requests.ts +94 -0
  228. package/src/modules/staff/i18n/de.json +4 -0
  229. package/src/modules/staff/i18n/en.json +9 -1
  230. package/src/modules/staff/i18n/es.json +4 -0
  231. package/src/modules/staff/i18n/pl.json +4 -0
  232. package/src/modules/staff/notifications.ts +71 -0
  233. package/src/modules/workflows/i18n/en.json +3 -1
  234. package/src/modules/workflows/notifications.ts +25 -0
  235. package/src/modules/workflows/subscribers/task-assigned-notification.ts +53 -0
@@ -0,0 +1,137 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
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 { formatMoney } from "../../components/documents/lineItemUtils.js";
10
+ import { useSalesDocumentTotals } from "./useSalesDocumentTotals.js";
11
+ function formatTimeAgo(dateString, t) {
12
+ const date = new Date(dateString);
13
+ const now = /* @__PURE__ */ new Date();
14
+ const diffMs = now.getTime() - date.getTime();
15
+ const diffMins = Math.floor(diffMs / 6e4);
16
+ const diffHours = Math.floor(diffMs / 36e5);
17
+ const diffDays = Math.floor(diffMs / 864e5);
18
+ if (diffMins < 1) return t("common.time.justNow", "just now");
19
+ if (diffMins < 60) return t("common.time.minutesAgo", "{count}m ago").replace("{count}", String(diffMins));
20
+ if (diffHours < 24) return t("common.time.hoursAgo", "{count}h ago").replace("{count}", String(diffHours));
21
+ if (diffDays < 7) return t("common.time.daysAgo", "{count}d ago").replace("{count}", String(diffDays));
22
+ return date.toLocaleDateString();
23
+ }
24
+ function normalizeTotal(value) {
25
+ if (!value) return null;
26
+ let trimmed = value.trim();
27
+ if (trimmed.startsWith("(") && trimmed.endsWith(")")) {
28
+ trimmed = trimmed.slice(1, -1).trim();
29
+ }
30
+ return trimmed.length ? trimmed : null;
31
+ }
32
+ function SalesOrderCreatedRenderer({
33
+ notification,
34
+ onAction,
35
+ onDismiss,
36
+ actions = []
37
+ }) {
38
+ const t = useT();
39
+ const router = useRouter();
40
+ const [executing, setExecuting] = React.useState(false);
41
+ const isUnread = notification.status === "unread";
42
+ const orderNumber = notification.bodyVariables?.orderNumber ?? notification.titleVariables?.orderNumber;
43
+ const fallbackTotal = normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ?? normalizeTotal(notification.bodyVariables?.total ?? null);
44
+ const { totals } = useSalesDocumentTotals("order", notification.sourceEntityId);
45
+ const currentTotal = totals && typeof totals.grandTotalGrossAmount === "number" ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode) : fallbackTotal;
46
+ const viewAction = actions.find((action) => action.id === "view") ?? actions[0] ?? null;
47
+ const handleView = async () => {
48
+ if (!viewAction) {
49
+ if (notification.linkHref) router.push(notification.linkHref);
50
+ return;
51
+ }
52
+ setExecuting(true);
53
+ try {
54
+ await onAction(viewAction.id);
55
+ } finally {
56
+ setExecuting(false);
57
+ }
58
+ };
59
+ return /* @__PURE__ */ jsxs(
60
+ "div",
61
+ {
62
+ className: cn(
63
+ "group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-blue-500",
64
+ isUnread && "bg-blue-50/50 dark:bg-blue-950/20"
65
+ ),
66
+ onClick: handleView,
67
+ children: [
68
+ isUnread && /* @__PURE__ */ jsx("div", { className: "absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary" }),
69
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-3", children: [
70
+ /* @__PURE__ */ jsx("div", { className: "flex-shrink-0 mt-0.5", children: /* @__PURE__ */ jsx("div", { className: "h-10 w-10 rounded-lg bg-blue-100 dark:bg-blue-900/40 flex items-center justify-center", children: /* @__PURE__ */ jsx(ShoppingCart, { className: "h-5 w-5 text-blue-600 dark:text-blue-400" }) }) }),
71
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 min-w-0", children: [
72
+ /* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-2", children: [
73
+ /* @__PURE__ */ jsxs("div", { children: [
74
+ /* @__PURE__ */ jsx("h4", { className: cn("text-sm font-medium", isUnread && "font-semibold"), children: notification.title }),
75
+ orderNumber && /* @__PURE__ */ jsx("div", { className: "flex items-center gap-1 mt-0.5", children: /* @__PURE__ */ jsxs("span", { className: "text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded", children: [
76
+ "#",
77
+ orderNumber
78
+ ] }) })
79
+ ] }),
80
+ /* @__PURE__ */ jsxs("span", { className: "flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1", children: [
81
+ /* @__PURE__ */ jsx(Calendar, { className: "h-3 w-3" }),
82
+ formatTimeAgo(notification.createdAt, t)
83
+ ] })
84
+ ] }),
85
+ /* @__PURE__ */ jsxs("div", { className: "mt-2 flex items-center gap-4 text-xs text-muted-foreground", children: [
86
+ currentTotal && /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
87
+ /* @__PURE__ */ jsx(DollarSign, { className: "h-3 w-3" }),
88
+ /* @__PURE__ */ jsx("span", { className: "font-medium text-foreground", children: currentTotal })
89
+ ] }),
90
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
91
+ /* @__PURE__ */ jsx(User, { className: "h-3 w-3" }),
92
+ /* @__PURE__ */ jsx("span", { children: t("sales.notifications.renderer.assignedToYou", "Assigned to you") })
93
+ ] })
94
+ ] }),
95
+ /* @__PURE__ */ jsxs("div", { className: "mt-3 flex gap-2", children: [
96
+ /* @__PURE__ */ jsxs(
97
+ Button,
98
+ {
99
+ variant: "default",
100
+ size: "sm",
101
+ onClick: (e) => {
102
+ e.stopPropagation();
103
+ handleView();
104
+ },
105
+ disabled: executing || !viewAction && !notification.linkHref,
106
+ className: "gap-1",
107
+ children: [
108
+ /* @__PURE__ */ jsx(ExternalLink, { className: "h-3 w-3" }),
109
+ t("sales.notifications.renderer.viewOrder", "View Order")
110
+ ]
111
+ }
112
+ ),
113
+ /* @__PURE__ */ jsx(
114
+ Button,
115
+ {
116
+ variant: "ghost",
117
+ size: "sm",
118
+ onClick: (e) => {
119
+ e.stopPropagation();
120
+ onDismiss();
121
+ },
122
+ children: t("notifications.actions.dismiss", "Dismiss")
123
+ }
124
+ )
125
+ ] })
126
+ ] })
127
+ ] })
128
+ ]
129
+ }
130
+ );
131
+ }
132
+ var SalesOrderCreatedRenderer_default = SalesOrderCreatedRenderer;
133
+ export {
134
+ SalesOrderCreatedRenderer,
135
+ SalesOrderCreatedRenderer_default as default
136
+ };
137
+ //# sourceMappingURL=SalesOrderCreatedRenderer.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../../src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx"],
4
+ "sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport { ShoppingCart, ExternalLink, DollarSign, User, Calendar } from 'lucide-react'\nimport { useRouter } from 'next/navigation'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport type { NotificationRendererProps } from '@open-mercato/shared/modules/notifications/types'\nimport { formatMoney } from '../../components/documents/lineItemUtils'\nimport { useSalesDocumentTotals } from './useSalesDocumentTotals'\n\nfunction formatTimeAgo(dateString: string, t: (key: string, fallback?: string) => string): string {\n const date = new Date(dateString)\n const now = new Date()\n const diffMs = now.getTime() - date.getTime()\n const diffMins = Math.floor(diffMs / 60000)\n const diffHours = Math.floor(diffMs / 3600000)\n const diffDays = Math.floor(diffMs / 86400000)\n\n if (diffMins < 1) return t('common.time.justNow', 'just now')\n if (diffMins < 60) return t('common.time.minutesAgo', '{count}m ago').replace('{count}', String(diffMins))\n if (diffHours < 24) return t('common.time.hoursAgo', '{count}h ago').replace('{count}', String(diffHours))\n if (diffDays < 7) return t('common.time.daysAgo', '{count}d ago').replace('{count}', String(diffDays))\n return date.toLocaleDateString()\n}\n\nfunction normalizeTotal(value?: string | null): string | null {\n if (!value) return null\n let trimmed = value.trim()\n if (trimmed.startsWith('(') && trimmed.endsWith(')')) {\n trimmed = trimmed.slice(1, -1).trim()\n }\n return trimmed.length ? trimmed : null\n}\n\nexport function SalesOrderCreatedRenderer({\n notification,\n onAction,\n onDismiss,\n actions = [],\n}: NotificationRendererProps) {\n const t = useT()\n const router = useRouter()\n const [executing, setExecuting] = React.useState(false)\n const isUnread = notification.status === 'unread'\n const orderNumber = notification.bodyVariables?.orderNumber ?? notification.titleVariables?.orderNumber\n const fallbackTotal =\n normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ??\n normalizeTotal(notification.bodyVariables?.total ?? null)\n const { totals } = useSalesDocumentTotals('order', notification.sourceEntityId)\n\n const currentTotal =\n totals && typeof totals.grandTotalGrossAmount === 'number'\n ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode)\n : fallbackTotal\n\n const viewAction = actions.find((action) => action.id === 'view') ?? actions[0] ?? null\n\n const handleView = async () => {\n if (!viewAction) {\n if (notification.linkHref) router.push(notification.linkHref)\n return\n }\n setExecuting(true)\n try {\n await onAction(viewAction.id)\n } finally {\n setExecuting(false)\n }\n }\n\n return (\n <div\n className={cn(\n 'group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-blue-500',\n isUnread && 'bg-blue-50/50 dark:bg-blue-950/20'\n )}\n onClick={handleView}\n >\n {isUnread && (\n <div className=\"absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary\" />\n )}\n\n <div className=\"flex gap-3\">\n <div className=\"flex-shrink-0 mt-0.5\">\n <div className=\"h-10 w-10 rounded-lg bg-blue-100 dark:bg-blue-900/40 flex items-center justify-center\">\n <ShoppingCart className=\"h-5 w-5 text-blue-600 dark:text-blue-400\" />\n </div>\n </div>\n\n <div className=\"flex-1 min-w-0\">\n <div className=\"flex items-start justify-between gap-2\">\n <div>\n <h4 className={cn('text-sm font-medium', isUnread && 'font-semibold')}>\n {notification.title}\n </h4>\n {orderNumber && (\n <div className=\"flex items-center gap-1 mt-0.5\">\n <span className=\"text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded\">\n #{orderNumber}\n </span>\n </div>\n )}\n </div>\n <span className=\"flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1\">\n <Calendar className=\"h-3 w-3\" />\n {formatTimeAgo(notification.createdAt, t)}\n </span>\n </div>\n\n <div className=\"mt-2 flex items-center gap-4 text-xs text-muted-foreground\">\n {currentTotal && (\n <div className=\"flex items-center gap-1\">\n <DollarSign className=\"h-3 w-3\" />\n <span className=\"font-medium text-foreground\">{currentTotal}</span>\n </div>\n )}\n <div className=\"flex items-center gap-1\">\n <User className=\"h-3 w-3\" />\n <span>{t('sales.notifications.renderer.assignedToYou', 'Assigned to you')}</span>\n </div>\n </div>\n\n <div className=\"mt-3 flex gap-2\">\n <Button\n variant=\"default\"\n size=\"sm\"\n onClick={(e) => {\n e.stopPropagation()\n handleView()\n }}\n disabled={executing || (!viewAction && !notification.linkHref)}\n className=\"gap-1\"\n >\n <ExternalLink className=\"h-3 w-3\" />\n {t('sales.notifications.renderer.viewOrder', 'View Order')}\n </Button>\n <Button\n variant=\"ghost\"\n size=\"sm\"\n onClick={(e) => {\n e.stopPropagation()\n onDismiss()\n }}\n >\n {t('notifications.actions.dismiss', 'Dismiss')}\n </Button>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default SalesOrderCreatedRenderer\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -0,0 +1,137 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
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 { formatMoney } from "../../components/documents/lineItemUtils.js";
10
+ import { useSalesDocumentTotals } from "./useSalesDocumentTotals.js";
11
+ function formatTimeAgo(dateString, t) {
12
+ const date = new Date(dateString);
13
+ const now = /* @__PURE__ */ new Date();
14
+ const diffMs = now.getTime() - date.getTime();
15
+ const diffMins = Math.floor(diffMs / 6e4);
16
+ const diffHours = Math.floor(diffMs / 36e5);
17
+ const diffDays = Math.floor(diffMs / 864e5);
18
+ if (diffMins < 1) return t("common.time.justNow", "just now");
19
+ if (diffMins < 60) return t("common.time.minutesAgo", "{count}m ago").replace("{count}", String(diffMins));
20
+ if (diffHours < 24) return t("common.time.hoursAgo", "{count}h ago").replace("{count}", String(diffHours));
21
+ if (diffDays < 7) return t("common.time.daysAgo", "{count}d ago").replace("{count}", String(diffDays));
22
+ return date.toLocaleDateString();
23
+ }
24
+ function normalizeTotal(value) {
25
+ if (!value) return null;
26
+ let trimmed = value.trim();
27
+ if (trimmed.startsWith("(") && trimmed.endsWith(")")) {
28
+ trimmed = trimmed.slice(1, -1).trim();
29
+ }
30
+ return trimmed.length ? trimmed : null;
31
+ }
32
+ function SalesQuoteCreatedRenderer({
33
+ notification,
34
+ onAction,
35
+ onDismiss,
36
+ actions = []
37
+ }) {
38
+ const t = useT();
39
+ const router = useRouter();
40
+ const [executing, setExecuting] = React.useState(false);
41
+ const isUnread = notification.status === "unread";
42
+ const quoteNumber = notification.bodyVariables?.quoteNumber ?? notification.titleVariables?.quoteNumber;
43
+ const fallbackTotal = normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ?? normalizeTotal(notification.bodyVariables?.total ?? null);
44
+ const { totals } = useSalesDocumentTotals("quote", notification.sourceEntityId);
45
+ const currentTotal = totals && typeof totals.grandTotalGrossAmount === "number" ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode) : fallbackTotal;
46
+ const viewAction = actions.find((action) => action.id === "view") ?? actions[0] ?? null;
47
+ const handleView = async () => {
48
+ if (!viewAction) {
49
+ if (notification.linkHref) router.push(notification.linkHref);
50
+ return;
51
+ }
52
+ setExecuting(true);
53
+ try {
54
+ await onAction(viewAction.id);
55
+ } finally {
56
+ setExecuting(false);
57
+ }
58
+ };
59
+ return /* @__PURE__ */ jsxs(
60
+ "div",
61
+ {
62
+ className: cn(
63
+ "group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-amber-500",
64
+ isUnread && "bg-amber-50/50 dark:bg-amber-950/20"
65
+ ),
66
+ onClick: handleView,
67
+ children: [
68
+ isUnread && /* @__PURE__ */ jsx("div", { className: "absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary" }),
69
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-3", children: [
70
+ /* @__PURE__ */ jsx("div", { className: "flex-shrink-0 mt-0.5", children: /* @__PURE__ */ jsx("div", { className: "h-10 w-10 rounded-lg bg-amber-100 dark:bg-amber-900/40 flex items-center justify-center", children: /* @__PURE__ */ jsx(FileText, { className: "h-5 w-5 text-amber-600 dark:text-amber-400" }) }) }),
71
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 min-w-0", children: [
72
+ /* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-2", children: [
73
+ /* @__PURE__ */ jsxs("div", { children: [
74
+ /* @__PURE__ */ jsx("h4", { className: cn("text-sm font-medium", isUnread && "font-semibold"), children: notification.title }),
75
+ quoteNumber && /* @__PURE__ */ jsx("div", { className: "flex items-center gap-1 mt-0.5", children: /* @__PURE__ */ jsxs("span", { className: "text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded", children: [
76
+ "#",
77
+ quoteNumber
78
+ ] }) })
79
+ ] }),
80
+ /* @__PURE__ */ jsxs("span", { className: "flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1", children: [
81
+ /* @__PURE__ */ jsx(Calendar, { className: "h-3 w-3" }),
82
+ formatTimeAgo(notification.createdAt, t)
83
+ ] })
84
+ ] }),
85
+ /* @__PURE__ */ jsxs("div", { className: "mt-2 flex items-center gap-4 text-xs text-muted-foreground", children: [
86
+ currentTotal && /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
87
+ /* @__PURE__ */ jsx(DollarSign, { className: "h-3 w-3" }),
88
+ /* @__PURE__ */ jsx("span", { className: "font-medium text-foreground", children: currentTotal })
89
+ ] }),
90
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
91
+ /* @__PURE__ */ jsx(User, { className: "h-3 w-3" }),
92
+ /* @__PURE__ */ jsx("span", { children: t("sales.notifications.renderer.pendingReview", "Pending review") })
93
+ ] })
94
+ ] }),
95
+ /* @__PURE__ */ jsxs("div", { className: "mt-3 flex gap-2", children: [
96
+ /* @__PURE__ */ jsxs(
97
+ Button,
98
+ {
99
+ variant: "default",
100
+ size: "sm",
101
+ onClick: (e) => {
102
+ e.stopPropagation();
103
+ handleView();
104
+ },
105
+ disabled: executing || !viewAction && !notification.linkHref,
106
+ className: "gap-1",
107
+ children: [
108
+ /* @__PURE__ */ jsx(ExternalLink, { className: "h-3 w-3" }),
109
+ t("sales.notifications.renderer.viewQuote", "View Quote")
110
+ ]
111
+ }
112
+ ),
113
+ /* @__PURE__ */ jsx(
114
+ Button,
115
+ {
116
+ variant: "ghost",
117
+ size: "sm",
118
+ onClick: (e) => {
119
+ e.stopPropagation();
120
+ onDismiss();
121
+ },
122
+ children: t("notifications.actions.dismiss", "Dismiss")
123
+ }
124
+ )
125
+ ] })
126
+ ] })
127
+ ] })
128
+ ]
129
+ }
130
+ );
131
+ }
132
+ var SalesQuoteCreatedRenderer_default = SalesQuoteCreatedRenderer;
133
+ export {
134
+ SalesQuoteCreatedRenderer,
135
+ SalesQuoteCreatedRenderer_default as default
136
+ };
137
+ //# sourceMappingURL=SalesQuoteCreatedRenderer.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../../src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx"],
4
+ "sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport { FileText, ExternalLink, DollarSign, User, Calendar } from 'lucide-react'\nimport { useRouter } from 'next/navigation'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport type { NotificationRendererProps } from '@open-mercato/shared/modules/notifications/types'\nimport { formatMoney } from '../../components/documents/lineItemUtils'\nimport { useSalesDocumentTotals } from './useSalesDocumentTotals'\n\nfunction formatTimeAgo(dateString: string, t: (key: string, fallback?: string) => string): string {\n const date = new Date(dateString)\n const now = new Date()\n const diffMs = now.getTime() - date.getTime()\n const diffMins = Math.floor(diffMs / 60000)\n const diffHours = Math.floor(diffMs / 3600000)\n const diffDays = Math.floor(diffMs / 86400000)\n\n if (diffMins < 1) return t('common.time.justNow', 'just now')\n if (diffMins < 60) return t('common.time.minutesAgo', '{count}m ago').replace('{count}', String(diffMins))\n if (diffHours < 24) return t('common.time.hoursAgo', '{count}h ago').replace('{count}', String(diffHours))\n if (diffDays < 7) return t('common.time.daysAgo', '{count}d ago').replace('{count}', String(diffDays))\n return date.toLocaleDateString()\n}\n\nfunction normalizeTotal(value?: string | null): string | null {\n if (!value) return null\n let trimmed = value.trim()\n if (trimmed.startsWith('(') && trimmed.endsWith(')')) {\n trimmed = trimmed.slice(1, -1).trim()\n }\n return trimmed.length ? trimmed : null\n}\n\nexport function SalesQuoteCreatedRenderer({\n notification,\n onAction,\n onDismiss,\n actions = [],\n}: NotificationRendererProps) {\n const t = useT()\n const router = useRouter()\n const [executing, setExecuting] = React.useState(false)\n const isUnread = notification.status === 'unread'\n const quoteNumber = notification.bodyVariables?.quoteNumber ?? notification.titleVariables?.quoteNumber\n const fallbackTotal =\n normalizeTotal(notification.bodyVariables?.totalAmount ?? null) ??\n normalizeTotal(notification.bodyVariables?.total ?? null)\n const { totals } = useSalesDocumentTotals('quote', notification.sourceEntityId)\n\n const currentTotal =\n totals && typeof totals.grandTotalGrossAmount === 'number'\n ? formatMoney(totals.grandTotalGrossAmount, totals.currencyCode)\n : fallbackTotal\n\n const viewAction = actions.find((action) => action.id === 'view') ?? actions[0] ?? null\n\n const handleView = async () => {\n if (!viewAction) {\n if (notification.linkHref) router.push(notification.linkHref)\n return\n }\n setExecuting(true)\n try {\n await onAction(viewAction.id)\n } finally {\n setExecuting(false)\n }\n }\n\n return (\n <div\n className={cn(\n 'group relative px-4 py-3 hover:bg-muted/50 cursor-pointer transition-colors border-l-4 border-l-amber-500',\n isUnread && 'bg-amber-50/50 dark:bg-amber-950/20'\n )}\n onClick={handleView}\n >\n {isUnread && (\n <div className=\"absolute left-1.5 top-1/2 -translate-y-1/2 h-2 w-2 rounded-full bg-primary\" />\n )}\n\n <div className=\"flex gap-3\">\n <div className=\"flex-shrink-0 mt-0.5\">\n <div className=\"h-10 w-10 rounded-lg bg-amber-100 dark:bg-amber-900/40 flex items-center justify-center\">\n <FileText className=\"h-5 w-5 text-amber-600 dark:text-amber-400\" />\n </div>\n </div>\n\n <div className=\"flex-1 min-w-0\">\n <div className=\"flex items-start justify-between gap-2\">\n <div>\n <h4 className={cn('text-sm font-medium', isUnread && 'font-semibold')}>\n {notification.title}\n </h4>\n {quoteNumber && (\n <div className=\"flex items-center gap-1 mt-0.5\">\n <span className=\"text-xs font-mono text-muted-foreground bg-muted px-1.5 py-0.5 rounded\">\n #{quoteNumber}\n </span>\n </div>\n )}\n </div>\n <span className=\"flex-shrink-0 text-xs text-muted-foreground flex items-center gap-1\">\n <Calendar className=\"h-3 w-3\" />\n {formatTimeAgo(notification.createdAt, t)}\n </span>\n </div>\n\n <div className=\"mt-2 flex items-center gap-4 text-xs text-muted-foreground\">\n {currentTotal && (\n <div className=\"flex items-center gap-1\">\n <DollarSign className=\"h-3 w-3\" />\n <span className=\"font-medium text-foreground\">{currentTotal}</span>\n </div>\n )}\n <div className=\"flex items-center gap-1\">\n <User className=\"h-3 w-3\" />\n <span>{t('sales.notifications.renderer.pendingReview', 'Pending review')}</span>\n </div>\n </div>\n\n <div className=\"mt-3 flex gap-2\">\n <Button\n variant=\"default\"\n size=\"sm\"\n onClick={(e) => {\n e.stopPropagation()\n handleView()\n }}\n disabled={executing || (!viewAction && !notification.linkHref)}\n className=\"gap-1\"\n >\n <ExternalLink className=\"h-3 w-3\" />\n {t('sales.notifications.renderer.viewQuote', 'View Quote')}\n </Button>\n <Button\n variant=\"ghost\"\n size=\"sm\"\n onClick={(e) => {\n e.stopPropagation()\n onDismiss()\n }}\n >\n {t('notifications.actions.dismiss', 'Dismiss')}\n </Button>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default SalesQuoteCreatedRenderer\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -0,0 +1,7 @@
1
+ import { SalesOrderCreatedRenderer } from "./SalesOrderCreatedRenderer.js";
2
+ import { SalesQuoteCreatedRenderer } from "./SalesQuoteCreatedRenderer.js";
3
+ export {
4
+ SalesOrderCreatedRenderer,
5
+ SalesQuoteCreatedRenderer
6
+ };
7
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../../src/modules/sales/widgets/notifications/index.ts"],
4
+ "sourcesContent": ["export { SalesOrderCreatedRenderer } from './SalesOrderCreatedRenderer'\nexport { SalesQuoteCreatedRenderer } from './SalesQuoteCreatedRenderer'\n"],
5
+ "mappings": "AAAA,SAAS,iCAAiC;AAC1C,SAAS,iCAAiC;",
6
+ "names": []
7
+ }
@@ -0,0 +1,60 @@
1
+ "use client";
2
+ import * as React from "react";
3
+ import { apiCall } from "@open-mercato/ui/backend/utils/apiCall";
4
+ const REFRESH_INTERVAL_MS = 3e4;
5
+ function buildDocumentTotalsUrl(kind, documentId) {
6
+ const params = new URLSearchParams({ id: documentId, page: "1", pageSize: "1" });
7
+ const collection = kind === "order" ? "orders" : "quotes";
8
+ return `/api/sales/${collection}?${params.toString()}`;
9
+ }
10
+ function extractTotals(payload) {
11
+ const item = payload?.items?.[0];
12
+ if (!item) return null;
13
+ const rawAmount = item.grandTotalGrossAmount;
14
+ let grandTotalGrossAmount = null;
15
+ if (typeof rawAmount === "number") {
16
+ grandTotalGrossAmount = Number.isNaN(rawAmount) ? null : rawAmount;
17
+ } else if (typeof rawAmount === "string" && rawAmount.trim().length) {
18
+ const parsed = Number(rawAmount);
19
+ grandTotalGrossAmount = Number.isNaN(parsed) ? null : parsed;
20
+ }
21
+ return {
22
+ grandTotalGrossAmount,
23
+ currencyCode: typeof item.currencyCode === "string" ? item.currencyCode : null
24
+ };
25
+ }
26
+ function useSalesDocumentTotals(kind, documentId) {
27
+ const [totals, setTotals] = React.useState(null);
28
+ React.useEffect(() => {
29
+ if (!documentId) {
30
+ setTotals(null);
31
+ return;
32
+ }
33
+ let active = true;
34
+ const loadTotals = async () => {
35
+ try {
36
+ const call = await apiCall(buildDocumentTotalsUrl(kind, documentId));
37
+ if (!active) return;
38
+ if (call.ok) {
39
+ const nextTotals = extractTotals(call.result ?? null);
40
+ setTotals(nextTotals);
41
+ }
42
+ } catch {
43
+ if (active) {
44
+ setTotals(null);
45
+ }
46
+ }
47
+ };
48
+ loadTotals();
49
+ const interval = setInterval(loadTotals, REFRESH_INTERVAL_MS);
50
+ return () => {
51
+ active = false;
52
+ clearInterval(interval);
53
+ };
54
+ }, [kind, documentId]);
55
+ return { totals };
56
+ }
57
+ export {
58
+ useSalesDocumentTotals
59
+ };
60
+ //# sourceMappingURL=useSalesDocumentTotals.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../../src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts"],
4
+ "sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\n\ntype DocumentKind = 'order' | 'quote'\n\ntype DocumentTotals = {\n grandTotalGrossAmount: number | null\n currencyCode: string | null\n}\n\ntype DocumentListResponse = {\n items?: Array<{\n grandTotalGrossAmount?: number | string | null\n currencyCode?: string | null\n }>\n}\n\nconst REFRESH_INTERVAL_MS = 30000\n\nfunction buildDocumentTotalsUrl(kind: DocumentKind, documentId: string) {\n const params = new URLSearchParams({ id: documentId, page: '1', pageSize: '1' })\n const collection = kind === 'order' ? 'orders' : 'quotes'\n return `/api/sales/${collection}?${params.toString()}`\n}\n\nfunction extractTotals(payload: DocumentListResponse | null): DocumentTotals | null {\n const item = payload?.items?.[0]\n if (!item) return null\n const rawAmount = item.grandTotalGrossAmount\n let grandTotalGrossAmount: number | null = null\n if (typeof rawAmount === 'number') {\n grandTotalGrossAmount = Number.isNaN(rawAmount) ? null : rawAmount\n } else if (typeof rawAmount === 'string' && rawAmount.trim().length) {\n const parsed = Number(rawAmount)\n grandTotalGrossAmount = Number.isNaN(parsed) ? null : parsed\n }\n return {\n grandTotalGrossAmount,\n currencyCode: typeof item.currencyCode === 'string' ? item.currencyCode : null,\n }\n}\n\nexport function useSalesDocumentTotals(kind: DocumentKind, documentId?: string | null) {\n const [totals, setTotals] = React.useState<DocumentTotals | null>(null)\n\n React.useEffect(() => {\n if (!documentId) {\n setTotals(null)\n return\n }\n\n let active = true\n\n const loadTotals = async () => {\n try {\n const call = await apiCall<DocumentListResponse>(buildDocumentTotalsUrl(kind, documentId))\n if (!active) return\n if (call.ok) {\n const nextTotals = extractTotals(call.result ?? null)\n setTotals(nextTotals)\n }\n } catch {\n if (active) {\n setTotals(null)\n }\n }\n }\n\n loadTotals()\n const interval = setInterval(loadTotals, REFRESH_INTERVAL_MS)\n\n return () => {\n active = false\n clearInterval(interval)\n }\n }, [kind, documentId])\n\n return { totals }\n}\n"],
5
+ "mappings": ";AAEA,YAAY,WAAW;AACvB,SAAS,eAAe;AAgBxB,MAAM,sBAAsB;AAE5B,SAAS,uBAAuB,MAAoB,YAAoB;AACtE,QAAM,SAAS,IAAI,gBAAgB,EAAE,IAAI,YAAY,MAAM,KAAK,UAAU,IAAI,CAAC;AAC/E,QAAM,aAAa,SAAS,UAAU,WAAW;AACjD,SAAO,cAAc,UAAU,IAAI,OAAO,SAAS,CAAC;AACtD;AAEA,SAAS,cAAc,SAA6D;AAClF,QAAM,OAAO,SAAS,QAAQ,CAAC;AAC/B,MAAI,CAAC,KAAM,QAAO;AAClB,QAAM,YAAY,KAAK;AACvB,MAAI,wBAAuC;AAC3C,MAAI,OAAO,cAAc,UAAU;AACjC,4BAAwB,OAAO,MAAM,SAAS,IAAI,OAAO;AAAA,EAC3D,WAAW,OAAO,cAAc,YAAY,UAAU,KAAK,EAAE,QAAQ;AACnE,UAAM,SAAS,OAAO,SAAS;AAC/B,4BAAwB,OAAO,MAAM,MAAM,IAAI,OAAO;AAAA,EACxD;AACA,SAAO;AAAA,IACL;AAAA,IACA,cAAc,OAAO,KAAK,iBAAiB,WAAW,KAAK,eAAe;AAAA,EAC5E;AACF;AAEO,SAAS,uBAAuB,MAAoB,YAA4B;AACrF,QAAM,CAAC,QAAQ,SAAS,IAAI,MAAM,SAAgC,IAAI;AAEtE,QAAM,UAAU,MAAM;AACpB,QAAI,CAAC,YAAY;AACf,gBAAU,IAAI;AACd;AAAA,IACF;AAEA,QAAI,SAAS;AAEb,UAAM,aAAa,YAAY;AAC7B,UAAI;AACF,cAAM,OAAO,MAAM,QAA8B,uBAAuB,MAAM,UAAU,CAAC;AACzF,YAAI,CAAC,OAAQ;AACb,YAAI,KAAK,IAAI;AACX,gBAAM,aAAa,cAAc,KAAK,UAAU,IAAI;AACpD,oBAAU,UAAU;AAAA,QACtB;AAAA,MACF,QAAQ;AACN,YAAI,QAAQ;AACV,oBAAU,IAAI;AAAA,QAChB;AAAA,MACF;AAAA,IACF;AAEA,eAAW;AACX,UAAM,WAAW,YAAY,YAAY,mBAAmB;AAE5D,WAAO,MAAM;AACX,eAAS;AACT,oBAAc,QAAQ;AAAA,IACxB;AAAA,EACF,GAAG,CAAC,MAAM,UAAU,CAAC;AAErB,SAAO,EAAE,OAAO;AAClB;",
6
+ "names": []
7
+ }
@@ -14,6 +14,9 @@ import {
14
14
  } from "../data/validators.js";
15
15
  import { ensureOrganizationScope, ensureTenantScope, extractUndoPayload, requireTeamMember } from "./shared.js";
16
16
  import { E } from "../../../generated/entities.ids.generated.js";
17
+ import { resolveNotificationService } from "../../notifications/lib/notificationService.js";
18
+ import { buildFeatureNotificationFromType, buildNotificationFromType } from "../../notifications/lib/notificationBuilder.js";
19
+ import { notificationTypes } from "../notifications.js";
17
20
  const leaveRequestCrudIndexer = {
18
21
  entityType: E.staff.staff_leave_request
19
22
  };
@@ -192,6 +195,31 @@ const createLeaveRequestCommand = {
192
195
  },
193
196
  indexer: leaveRequestCrudIndexer
194
197
  });
198
+ try {
199
+ const notificationService = resolveNotificationService(ctx.container);
200
+ const typeDef = notificationTypes.find((type) => type.type === "staff.leave_request.pending");
201
+ if (typeDef) {
202
+ const memberName = member.displayName || "Team member";
203
+ const startDateStr = request.startDate.toLocaleDateString();
204
+ const endDateStr = request.endDate.toLocaleDateString();
205
+ const notificationInput = buildFeatureNotificationFromType(typeDef, {
206
+ requiredFeature: "staff.leave_requests.manage",
207
+ bodyVariables: {
208
+ memberName,
209
+ startDate: startDateStr,
210
+ endDate: endDateStr
211
+ },
212
+ sourceEntityType: "staff:leave_request",
213
+ sourceEntityId: request.id,
214
+ linkHref: `/backend/staff/leave-requests/${request.id}`
215
+ });
216
+ await notificationService.createForFeature(notificationInput, {
217
+ tenantId: request.tenantId,
218
+ organizationId: request.organizationId
219
+ });
220
+ }
221
+ } catch {
222
+ }
195
223
  return { requestId: request.id };
196
224
  },
197
225
  captureAfter: async (_input, result, ctx) => {
@@ -490,6 +518,31 @@ const acceptLeaveRequestCommand = {
490
518
  organizationId: request.organizationId,
491
519
  ruleIds: createdRuleIds
492
520
  });
521
+ if (request.submittedByUserId) {
522
+ try {
523
+ const notificationService = resolveNotificationService(ctx.container);
524
+ const typeDef = notificationTypes.find((type) => type.type === "staff.leave_request.approved");
525
+ if (typeDef) {
526
+ const startDateStr = request.startDate.toLocaleDateString();
527
+ const endDateStr = request.endDate.toLocaleDateString();
528
+ const notificationInput = buildNotificationFromType(typeDef, {
529
+ recipientUserId: request.submittedByUserId,
530
+ bodyVariables: {
531
+ startDate: startDateStr,
532
+ endDate: endDateStr
533
+ },
534
+ sourceEntityType: "staff:leave_request",
535
+ sourceEntityId: request.id,
536
+ linkHref: `/backend/staff/leave-requests/${request.id}`
537
+ });
538
+ await notificationService.create(notificationInput, {
539
+ tenantId: request.tenantId,
540
+ organizationId: request.organizationId
541
+ });
542
+ }
543
+ } catch {
544
+ }
545
+ }
493
546
  return { requestId: request.id, ruleIds: createdRuleIds };
494
547
  },
495
548
  buildLog: async ({ result, ctx, snapshots }) => {
@@ -604,6 +657,32 @@ const rejectLeaveRequestCommand = {
604
657
  },
605
658
  indexer: leaveRequestCrudIndexer
606
659
  });
660
+ if (request.submittedByUserId) {
661
+ try {
662
+ const notificationService = resolveNotificationService(ctx.container);
663
+ const typeDef = notificationTypes.find((type) => type.type === "staff.leave_request.rejected");
664
+ if (typeDef) {
665
+ const startDateStr = request.startDate.toLocaleDateString();
666
+ const endDateStr = request.endDate.toLocaleDateString();
667
+ const notificationInput = buildNotificationFromType(typeDef, {
668
+ recipientUserId: request.submittedByUserId,
669
+ bodyVariables: {
670
+ startDate: startDateStr,
671
+ endDate: endDateStr,
672
+ reason: request.decisionComment ?? ""
673
+ },
674
+ sourceEntityType: "staff:leave_request",
675
+ sourceEntityId: request.id,
676
+ linkHref: `/backend/staff/leave-requests/${request.id}`
677
+ });
678
+ await notificationService.create(notificationInput, {
679
+ tenantId: request.tenantId,
680
+ organizationId: request.organizationId
681
+ });
682
+ }
683
+ } catch {
684
+ }
685
+ }
607
686
  return { requestId: request.id };
608
687
  },
609
688
  async prepare(rawInput, ctx) {