@open-mercato/core 0.4.2-canary-51881f6bf3 → 0.4.2-canary-5f415b8a44

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 (155) hide show
  1. package/dist/generated/entities/workflow_event_trigger/index.js +33 -0
  2. package/dist/generated/entities/workflow_event_trigger/index.js.map +7 -0
  3. package/dist/generated/entities.ids.generated.js +59 -58
  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/auth/events.js +30 -0
  8. package/dist/modules/auth/events.js.map +7 -0
  9. package/dist/modules/business_rules/api/execute/[ruleId]/route.js +145 -0
  10. package/dist/modules/business_rules/api/execute/[ruleId]/route.js.map +7 -0
  11. package/dist/modules/business_rules/data/validators.js +34 -0
  12. package/dist/modules/business_rules/data/validators.js.map +2 -2
  13. package/dist/modules/business_rules/index.js +21 -1
  14. package/dist/modules/business_rules/index.js.map +2 -2
  15. package/dist/modules/business_rules/lib/rule-engine.js +182 -1
  16. package/dist/modules/business_rules/lib/rule-engine.js.map +2 -2
  17. package/dist/modules/catalog/events.js +34 -0
  18. package/dist/modules/catalog/events.js.map +7 -0
  19. package/dist/modules/customers/events.js +49 -0
  20. package/dist/modules/customers/events.js.map +7 -0
  21. package/dist/modules/directory/events.js +23 -0
  22. package/dist/modules/directory/events.js.map +7 -0
  23. package/dist/modules/sales/acl.js +1 -0
  24. package/dist/modules/sales/acl.js.map +2 -2
  25. package/dist/modules/sales/backend/sales/documents/[id]/page.js +12 -0
  26. package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
  27. package/dist/modules/sales/commands/documents.js +62 -0
  28. package/dist/modules/sales/commands/documents.js.map +2 -2
  29. package/dist/modules/sales/events.js +63 -0
  30. package/dist/modules/sales/events.js.map +7 -0
  31. package/dist/modules/sales/lib/dictionaries.js +3 -0
  32. package/dist/modules/sales/lib/dictionaries.js.map +2 -2
  33. package/dist/modules/sales/lib/frontend/documentDataEvents.js +25 -0
  34. package/dist/modules/sales/lib/frontend/documentDataEvents.js.map +7 -0
  35. package/dist/modules/workflows/acl.js +2 -0
  36. package/dist/modules/workflows/acl.js.map +2 -2
  37. package/dist/modules/workflows/api/instances/route.js +18 -6
  38. package/dist/modules/workflows/api/instances/route.js.map +2 -2
  39. package/dist/modules/workflows/api/tasks/route.js +6 -1
  40. package/dist/modules/workflows/api/tasks/route.js.map +2 -2
  41. package/dist/modules/workflows/backend/definitions/[id]/page.js +9 -1
  42. package/dist/modules/workflows/backend/definitions/[id]/page.js.map +2 -2
  43. package/dist/modules/workflows/backend/definitions/[id]/page.meta.js +1 -1
  44. package/dist/modules/workflows/backend/definitions/[id]/page.meta.js.map +2 -2
  45. package/dist/modules/workflows/backend/definitions/create/page.js +24 -15
  46. package/dist/modules/workflows/backend/definitions/create/page.js.map +2 -2
  47. package/dist/modules/workflows/backend/definitions/create/page.meta.js +1 -1
  48. package/dist/modules/workflows/backend/definitions/create/page.meta.js.map +2 -2
  49. package/dist/modules/workflows/backend/definitions/visual-editor/page.js +150 -132
  50. package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
  51. package/dist/modules/workflows/backend/definitions/visual-editor/page.meta.js +1 -1
  52. package/dist/modules/workflows/backend/definitions/visual-editor/page.meta.js.map +2 -2
  53. package/dist/modules/workflows/backend/events/[id]/page.js +1 -1
  54. package/dist/modules/workflows/backend/events/[id]/page.js.map +2 -2
  55. package/dist/modules/workflows/backend/events/[id]/page.meta.js +2 -2
  56. package/dist/modules/workflows/backend/events/[id]/page.meta.js.map +2 -2
  57. package/dist/modules/workflows/backend/instances/[id]/page.meta.js +2 -2
  58. package/dist/modules/workflows/backend/instances/[id]/page.meta.js.map +2 -2
  59. package/dist/modules/workflows/backend/tasks/[id]/page.js +1 -1
  60. package/dist/modules/workflows/backend/tasks/[id]/page.js.map +2 -2
  61. package/dist/modules/workflows/backend/tasks/[id]/page.meta.js +2 -2
  62. package/dist/modules/workflows/backend/tasks/[id]/page.meta.js.map +2 -2
  63. package/dist/modules/workflows/backend/tasks/page.js +5 -6
  64. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  65. package/dist/modules/workflows/cli.js +81 -3
  66. package/dist/modules/workflows/cli.js.map +3 -3
  67. package/dist/modules/workflows/components/DefinitionTriggersEditor.js +481 -0
  68. package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +7 -0
  69. package/dist/modules/workflows/components/EventTriggersEditor.js +553 -0
  70. package/dist/modules/workflows/components/EventTriggersEditor.js.map +7 -0
  71. package/dist/modules/workflows/data/entities.js +64 -1
  72. package/dist/modules/workflows/data/entities.js.map +2 -2
  73. package/dist/modules/workflows/data/validators.js +115 -0
  74. package/dist/modules/workflows/data/validators.js.map +2 -2
  75. package/dist/modules/workflows/events.js +38 -0
  76. package/dist/modules/workflows/events.js.map +7 -0
  77. package/dist/modules/workflows/examples/checkout-demo-definition.json +1 -5
  78. package/dist/modules/workflows/examples/order-approval-definition.json +257 -0
  79. package/dist/modules/workflows/examples/order-approval-guard-rules.json +32 -0
  80. package/dist/modules/workflows/lib/activity-executor.js +75 -13
  81. package/dist/modules/workflows/lib/activity-executor.js.map +2 -2
  82. package/dist/modules/workflows/lib/event-trigger-service.js +308 -0
  83. package/dist/modules/workflows/lib/event-trigger-service.js.map +7 -0
  84. package/dist/modules/workflows/lib/graph-utils.js +71 -2
  85. package/dist/modules/workflows/lib/graph-utils.js.map +2 -2
  86. package/dist/modules/workflows/lib/seeds.js +22 -5
  87. package/dist/modules/workflows/lib/seeds.js.map +2 -2
  88. package/dist/modules/workflows/lib/start-validator.js +33 -23
  89. package/dist/modules/workflows/lib/start-validator.js.map +2 -2
  90. package/dist/modules/workflows/lib/transition-handler.js +157 -45
  91. package/dist/modules/workflows/lib/transition-handler.js.map +3 -3
  92. package/dist/modules/workflows/migrations/Migration20260123143500.js +36 -0
  93. package/dist/modules/workflows/migrations/Migration20260123143500.js.map +7 -0
  94. package/dist/modules/workflows/subscribers/event-trigger.js +78 -0
  95. package/dist/modules/workflows/subscribers/event-trigger.js.map +7 -0
  96. package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js +323 -0
  97. package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js.map +7 -0
  98. package/dist/modules/workflows/widgets/injection/order-approval/widget.js +17 -0
  99. package/dist/modules/workflows/widgets/injection/order-approval/widget.js.map +7 -0
  100. package/dist/modules/workflows/widgets/injection-table.js +19 -0
  101. package/dist/modules/workflows/widgets/injection-table.js.map +7 -0
  102. package/generated/entities/workflow_event_trigger/index.ts +15 -0
  103. package/generated/entities.ids.generated.ts +59 -58
  104. package/generated/entity-fields-registry.ts +2 -0
  105. package/package.json +3 -5
  106. package/src/modules/auth/events.ts +39 -0
  107. package/src/modules/business_rules/api/execute/[ruleId]/route.ts +163 -0
  108. package/src/modules/business_rules/data/validators.ts +40 -0
  109. package/src/modules/business_rules/index.ts +25 -0
  110. package/src/modules/business_rules/lib/rule-engine.ts +281 -1
  111. package/src/modules/catalog/events.ts +45 -0
  112. package/src/modules/customers/events.ts +63 -0
  113. package/src/modules/directory/events.ts +31 -0
  114. package/src/modules/sales/acl.ts +1 -0
  115. package/src/modules/sales/backend/sales/documents/[id]/page.tsx +16 -0
  116. package/src/modules/sales/commands/documents.ts +74 -1
  117. package/src/modules/sales/events.ts +82 -0
  118. package/src/modules/sales/lib/dictionaries.ts +3 -0
  119. package/src/modules/sales/lib/frontend/documentDataEvents.ts +28 -0
  120. package/src/modules/workflows/acl.ts +2 -0
  121. package/src/modules/workflows/api/instances/route.ts +21 -7
  122. package/src/modules/workflows/api/tasks/route.ts +7 -1
  123. package/src/modules/workflows/backend/definitions/[id]/page.meta.ts +1 -1
  124. package/src/modules/workflows/backend/definitions/[id]/page.tsx +9 -0
  125. package/src/modules/workflows/backend/definitions/create/page.meta.ts +1 -1
  126. package/src/modules/workflows/backend/definitions/create/page.tsx +9 -0
  127. package/src/modules/workflows/backend/definitions/visual-editor/page.meta.ts +1 -1
  128. package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +21 -3
  129. package/src/modules/workflows/backend/events/[id]/page.meta.ts +2 -2
  130. package/src/modules/workflows/backend/events/[id]/page.tsx +1 -1
  131. package/src/modules/workflows/backend/instances/[id]/page.meta.ts +2 -2
  132. package/src/modules/workflows/backend/tasks/[id]/page.meta.ts +2 -2
  133. package/src/modules/workflows/backend/tasks/[id]/page.tsx +1 -1
  134. package/src/modules/workflows/backend/tasks/page.tsx +5 -6
  135. package/src/modules/workflows/cli.ts +111 -0
  136. package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +581 -0
  137. package/src/modules/workflows/components/EventTriggersEditor.tsx +664 -0
  138. package/src/modules/workflows/data/entities.ts +124 -0
  139. package/src/modules/workflows/data/validators.ts +138 -0
  140. package/src/modules/workflows/events.ts +49 -0
  141. package/src/modules/workflows/examples/checkout-demo-definition.json +1 -5
  142. package/src/modules/workflows/examples/order-approval-definition.json +257 -0
  143. package/src/modules/workflows/examples/order-approval-guard-rules.json +32 -0
  144. package/src/modules/workflows/i18n/en.json +71 -0
  145. package/src/modules/workflows/lib/activity-executor.ts +129 -16
  146. package/src/modules/workflows/lib/event-trigger-service.ts +557 -0
  147. package/src/modules/workflows/lib/graph-utils.ts +117 -2
  148. package/src/modules/workflows/lib/seeds.ts +34 -8
  149. package/src/modules/workflows/lib/start-validator.ts +38 -28
  150. package/src/modules/workflows/lib/transition-handler.ts +208 -55
  151. package/src/modules/workflows/migrations/Migration20260123143500.ts +38 -0
  152. package/src/modules/workflows/subscribers/event-trigger.ts +109 -0
  153. package/src/modules/workflows/widgets/injection/order-approval/widget.client.tsx +446 -0
  154. package/src/modules/workflows/widgets/injection/order-approval/widget.ts +16 -0
  155. package/src/modules/workflows/widgets/injection-table.ts +21 -0
@@ -0,0 +1,446 @@
1
+ "use client"
2
+
3
+ import * as React from 'react'
4
+ import type { InjectionWidgetComponentProps } from '@open-mercato/shared/modules/widgets/injection'
5
+ import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'
6
+ import { apiCall } from '@open-mercato/ui/backend/utils/apiCall'
7
+ import { useT } from '@open-mercato/shared/lib/i18n/context'
8
+ import { Button } from '@open-mercato/ui/primitives/button'
9
+ import { Spinner } from '@open-mercato/ui/primitives/spinner'
10
+ import { Textarea } from '@open-mercato/ui/primitives/textarea'
11
+ import { Badge } from '@open-mercato/ui/primitives/badge'
12
+ import { emitSalesDocumentDataRefresh } from '@open-mercato/core/modules/sales/lib/frontend/documentDataEvents'
13
+
14
+ type OrderRecord = {
15
+ id: string
16
+ orderNumber?: string
17
+ status?: string
18
+ statusEntryId?: string
19
+ }
20
+
21
+ type WorkflowInstance = {
22
+ id: string
23
+ workflowId: string
24
+ status: 'RUNNING' | 'PAUSED' | 'COMPLETED' | 'FAILED' | 'CANCELLED' | 'WAITING_FOR_ACTIVITIES'
25
+ currentStepId: string
26
+ context: Record<string, any>
27
+ }
28
+
29
+ type UserTask = {
30
+ id: string
31
+ taskName: string
32
+ status: 'PENDING' | 'IN_PROGRESS' | 'COMPLETED' | 'CANCELLED'
33
+ workflowInstanceId: string
34
+ claimedBy: string | null
35
+ }
36
+
37
+ type DictionaryEntry = {
38
+ id: string
39
+ value: string
40
+ label: string
41
+ color?: string
42
+ icon?: string
43
+ }
44
+
45
+ const WORKFLOW_ID = 'sales_order_approval_v1'
46
+
47
+ export default function OrderApprovalWidget({ data }: InjectionWidgetComponentProps<unknown, OrderRecord>) {
48
+ const t = useT()
49
+ const queryClient = useQueryClient()
50
+ const orderId = data?.id
51
+
52
+ const [decision, setDecision] = React.useState<'approve' | 'reject' | ''>('')
53
+ const [comments, setComments] = React.useState('')
54
+ const [error, setError] = React.useState<string | null>(null)
55
+
56
+ // Track if we're waiting for workflow to process (for polling)
57
+ const [isWaitingForProcessing, setIsWaitingForProcessing] = React.useState(false)
58
+
59
+ // Fetch current order status directly (to detect when workflow updates it)
60
+ // Use list endpoint with ID filter since single-item endpoint doesn't exist
61
+ const { data: orderData } = useQuery({
62
+ queryKey: ['order-status', orderId],
63
+ queryFn: async () => {
64
+ if (!orderId) return null
65
+ const result = await apiCall<{ items: Array<{ id: string; status?: string | null }> }>(
66
+ `/api/sales/orders?id=${orderId}&pageSize=1`
67
+ )
68
+ return result.ok && result.result?.items?.[0] ? result.result.items[0] : null
69
+ },
70
+ enabled: Boolean(orderId),
71
+ staleTime: 5_000,
72
+ // Poll when waiting for processing to detect status change
73
+ refetchInterval: isWaitingForProcessing ? 2_000 : false,
74
+ })
75
+
76
+ // Use fresh order status from query, fallback to prop data
77
+ // The API returns status as a string value (e.g., "approved", "pending_approval")
78
+ const currentOrderStatus = orderData?.status || data?.status
79
+
80
+ // Fetch active workflow instances for this order
81
+ const { data: instancesData, isLoading: instancesLoading } = useQuery({
82
+ queryKey: ['workflow-instances', orderId],
83
+ queryFn: async () => {
84
+ if (!orderId) return { data: [] }
85
+ const result = await apiCall<{ data: WorkflowInstance[] }>(
86
+ `/api/workflows/instances?entityId=${orderId}&status=RUNNING,PAUSED,WAITING_FOR_ACTIVITIES`
87
+ )
88
+ return result.ok ? result.result : { data: [] }
89
+ },
90
+ enabled: Boolean(orderId),
91
+ staleTime: 5_000,
92
+ // Poll every 2 seconds when waiting for processing
93
+ refetchInterval: isWaitingForProcessing ? 2_000 : false,
94
+ })
95
+
96
+ const activeInstance = instancesData?.data?.find(
97
+ (inst) => inst.workflowId === WORKFLOW_ID && !['COMPLETED', 'FAILED', 'CANCELLED'].includes(inst.status)
98
+ )
99
+
100
+ // Fetch pending user tasks for active instance
101
+ const { data: tasksData, isLoading: tasksLoading } = useQuery({
102
+ queryKey: ['workflow-tasks', activeInstance?.id],
103
+ queryFn: async () => {
104
+ if (!activeInstance?.id) return { data: [] }
105
+ const result = await apiCall<{ data: UserTask[] }>(
106
+ `/api/workflows/tasks?workflowInstanceId=${activeInstance.id}&status=PENDING,IN_PROGRESS`
107
+ )
108
+ return result.ok ? result.result : { data: [] }
109
+ },
110
+ enabled: Boolean(activeInstance?.id),
111
+ staleTime: 5_000,
112
+ // Poll every 2 seconds when waiting for processing
113
+ refetchInterval: isWaitingForProcessing ? 2_000 : false,
114
+ })
115
+
116
+ const pendingTask = tasksData?.data?.[0]
117
+
118
+ // Auto-detect when workflow is processing (active instance but no pending task)
119
+ const isProcessing = activeInstance && !pendingTask && !['COMPLETED', 'FAILED', 'CANCELLED'].includes(activeInstance.status)
120
+
121
+ // Track previous status to detect changes
122
+ const prevStatusRef = React.useRef<string | undefined>(data?.status)
123
+
124
+ // Update polling state when processing state changes
125
+ React.useEffect(() => {
126
+ if (isProcessing) {
127
+ setIsWaitingForProcessing(true)
128
+ } else if (!activeInstance || activeInstance.status === 'COMPLETED') {
129
+ setIsWaitingForProcessing(false)
130
+ }
131
+ }, [isProcessing, activeInstance])
132
+
133
+ // When order status changes, emit refresh event to update the page
134
+ React.useEffect(() => {
135
+ const newStatus = orderData?.status
136
+ const oldStatus = prevStatusRef.current
137
+
138
+ if (newStatus && oldStatus && newStatus !== oldStatus) {
139
+ // Status changed - emit document refresh event to reload the page data
140
+ if (orderId) {
141
+ emitSalesDocumentDataRefresh({ documentId: orderId, kind: 'order' })
142
+ }
143
+ }
144
+
145
+ prevStatusRef.current = newStatus || oldStatus
146
+ }, [orderData?.status, orderId])
147
+
148
+ // First fetch dictionaries to find the order_status dictionary ID
149
+ const { data: dictionariesData } = useQuery({
150
+ queryKey: ['dictionaries'],
151
+ queryFn: async () => {
152
+ const result = await apiCall<{ items: Array<{ id: string; key: string }> }>(
153
+ '/api/dictionaries'
154
+ )
155
+ return result.ok ? result.result : { items: [] }
156
+ },
157
+ staleTime: 60_000,
158
+ })
159
+
160
+ const orderStatusDictionaryId = React.useMemo(() => {
161
+ const dict = dictionariesData?.items?.find(d => d.key === 'sales.order_status')
162
+ return dict?.id
163
+ }, [dictionariesData])
164
+
165
+ // Fetch order status dictionary entries using the dictionary ID
166
+ const { data: statusEntriesData } = useQuery({
167
+ queryKey: ['dictionary-entries', orderStatusDictionaryId],
168
+ queryFn: async () => {
169
+ if (!orderStatusDictionaryId) return { items: [] }
170
+ const result = await apiCall<{ items: DictionaryEntry[] }>(
171
+ `/api/dictionaries/${orderStatusDictionaryId}/entries`
172
+ )
173
+ return result.ok ? result.result : { items: [] }
174
+ },
175
+ enabled: Boolean(orderStatusDictionaryId),
176
+ staleTime: 60_000,
177
+ })
178
+
179
+ const findStatusId = React.useCallback((code: string) => {
180
+ const entries = statusEntriesData?.items || []
181
+ // The API returns 'value' as the code/key for dictionary entries
182
+ const entry = entries.find(
183
+ (e) => e.value === code || e.label?.toLowerCase() === code.toLowerCase()
184
+ )
185
+ return entry?.id
186
+ }, [statusEntriesData])
187
+
188
+ // Start workflow mutation
189
+ const startWorkflowMutation = useMutation({
190
+ mutationFn: async () => {
191
+ const pendingApprovalStatusId = findStatusId('pending_approval')
192
+ const approvedStatusId = findStatusId('approved')
193
+ const rejectedStatusId = findStatusId('rejected')
194
+
195
+ if (!pendingApprovalStatusId || !approvedStatusId || !rejectedStatusId) {
196
+ throw new Error(t('workflows.orderApproval.missingStatuses', 'Missing order status entries. Please ensure pending_approval, approved, and rejected statuses exist in the sales.order_status dictionary.'))
197
+ }
198
+
199
+ const result = await apiCall<{ data: WorkflowInstance }>('/api/workflows/instances', {
200
+ method: 'POST',
201
+ body: JSON.stringify({
202
+ workflowId: WORKFLOW_ID,
203
+ initialContext: {
204
+ orderId,
205
+ pendingApprovalStatusId,
206
+ approvedStatusId,
207
+ rejectedStatusId,
208
+ },
209
+ metadata: {
210
+ entityType: 'SalesOrder',
211
+ entityId: orderId,
212
+ },
213
+ }),
214
+ })
215
+
216
+ if (!result.ok) {
217
+ const errorResult = result.result as { error?: string } | null
218
+ throw new Error(errorResult?.error || t('workflows.orderApproval.startError', 'Failed to start approval workflow'))
219
+ }
220
+
221
+ return result.result
222
+ },
223
+ onSuccess: () => {
224
+ setError(null)
225
+ queryClient.invalidateQueries({ queryKey: ['workflow-instances', orderId] })
226
+ },
227
+ onError: (err: Error) => {
228
+ setError(err.message)
229
+ },
230
+ })
231
+
232
+ // Complete task mutation
233
+ const completeTaskMutation = useMutation({
234
+ mutationFn: async ({ taskId, formData }: { taskId: string; formData: { decision: string; comments?: string } }) => {
235
+ const result = await apiCall(`/api/workflows/tasks/${taskId}/complete`, {
236
+ method: 'POST',
237
+ body: JSON.stringify({ formData }),
238
+ })
239
+
240
+ if (!result.ok) {
241
+ const errorResult = result.result as { error?: string } | null
242
+ throw new Error(errorResult?.error || t('workflows.orderApproval.completeError', 'Failed to complete approval task'))
243
+ }
244
+
245
+ return result.result
246
+ },
247
+ onSuccess: () => {
248
+ setError(null)
249
+ setDecision('')
250
+ setComments('')
251
+ // Start polling immediately after submitting decision
252
+ setIsWaitingForProcessing(true)
253
+ queryClient.invalidateQueries({ queryKey: ['workflow-instances', orderId] })
254
+ queryClient.invalidateQueries({ queryKey: ['workflow-tasks', activeInstance?.id] })
255
+ // Also refresh the order data
256
+ queryClient.invalidateQueries({ queryKey: ['sales-order', orderId] })
257
+ },
258
+ onError: (err: Error) => {
259
+ setError(err.message)
260
+ },
261
+ })
262
+
263
+ const handleStartWorkflow = () => {
264
+ // Start polling after starting workflow
265
+ setIsWaitingForProcessing(true)
266
+ startWorkflowMutation.mutate()
267
+ }
268
+
269
+ const handleCompleteTask = () => {
270
+ if (!pendingTask || !decision) return
271
+ completeTaskMutation.mutate({
272
+ taskId: pendingTask.id,
273
+ formData: { decision, comments },
274
+ })
275
+ }
276
+
277
+ // Handle keyboard shortcuts (Cmd/Ctrl+Enter to submit)
278
+ const handleKeyDown = (e: React.KeyboardEvent) => {
279
+ if ((e.metaKey || e.ctrlKey) && e.key === 'Enter') {
280
+ e.preventDefault()
281
+ if (pendingTask && decision && !isSubmitting) {
282
+ handleCompleteTask()
283
+ }
284
+ }
285
+ }
286
+
287
+ const isLoading = instancesLoading || tasksLoading
288
+ const isSubmitting = startWorkflowMutation.isPending || completeTaskMutation.isPending
289
+
290
+ // Don't render if no orderId
291
+ if (!orderId) return null
292
+
293
+ // Only show widget when order status is pending_approval
294
+ const orderStatus = currentOrderStatus?.toLowerCase()
295
+ if (orderStatus !== 'pending_approval') {
296
+ return null
297
+ }
298
+
299
+ // Loading state
300
+ if (isLoading) {
301
+ return (
302
+ <div className="flex items-center justify-center p-4">
303
+ <Spinner size="sm" />
304
+ <span className="ml-2 text-sm text-muted-foreground">{t('common.loading', 'Loading...')}</span>
305
+ </div>
306
+ )
307
+ }
308
+
309
+ // Show workflow status badge
310
+ const getStatusBadge = () => {
311
+ if (!activeInstance) return null
312
+
313
+ const statusVariants: Record<string, 'default' | 'secondary' | 'destructive' | 'outline'> = {
314
+ RUNNING: 'default',
315
+ PAUSED: 'secondary',
316
+ WAITING_FOR_ACTIVITIES: 'secondary',
317
+ COMPLETED: 'default',
318
+ FAILED: 'destructive',
319
+ CANCELLED: 'outline',
320
+ }
321
+
322
+ return (
323
+ <Badge variant={statusVariants[activeInstance.status] || 'outline'}>
324
+ {t(`workflows.instances.statuses.${activeInstance.status}`, activeInstance.status)}
325
+ </Badge>
326
+ )
327
+ }
328
+
329
+ return (
330
+ <div className="space-y-3 rounded-lg border bg-card p-4 shadow-sm">
331
+ <div className="flex items-start justify-between gap-3">
332
+ <div>
333
+ <div className="text-sm font-semibold text-foreground">
334
+ {t('workflows.orderApproval.groupLabel', 'Order Approval')}
335
+ </div>
336
+ <p className="text-xs text-muted-foreground">
337
+ {t('workflows.orderApproval.groupDescription', 'Review and approve or reject this order')}
338
+ </p>
339
+ </div>
340
+ {getStatusBadge()}
341
+ </div>
342
+
343
+ {error && (
344
+ <div className="rounded-md border border-red-200 bg-red-50 p-3 text-xs text-red-800">
345
+ {error}
346
+ </div>
347
+ )}
348
+
349
+ {/* No active workflow - show request approval button */}
350
+ {!activeInstance && (
351
+ <div className="space-y-3">
352
+ <p className="text-sm text-muted-foreground">
353
+ {t('workflows.orderApproval.noWorkflowActive', 'No approval workflow is active for this order.')}
354
+ </p>
355
+ <Button
356
+ onClick={handleStartWorkflow}
357
+ disabled={isSubmitting}
358
+ variant="default"
359
+ size="sm"
360
+ >
361
+ {isSubmitting && <Spinner size="sm" className="mr-2" />}
362
+ {t('workflows.orderApproval.requestApproval', 'Request Approval')}
363
+ </Button>
364
+ </div>
365
+ )}
366
+
367
+ {/* Active workflow with pending task - show approve/reject UI */}
368
+ {activeInstance && pendingTask && (
369
+ <div className="space-y-3" onKeyDown={handleKeyDown}>
370
+ <div className="rounded-md border border-amber-200 bg-amber-50 p-3">
371
+ <p className="text-sm font-medium text-amber-800">
372
+ {t('workflows.orderApproval.pendingTitle', 'Pending Approval')}
373
+ </p>
374
+ <p className="text-xs text-amber-700 mt-1">
375
+ {t('workflows.orderApproval.pendingDescription', 'This order requires approval before processing.')}
376
+ </p>
377
+ </div>
378
+
379
+ <div className="space-y-2">
380
+ <label className="text-sm font-medium">
381
+ {t('workflows.orderApproval.decisionLabel', 'Decision')}
382
+ </label>
383
+ <div className="flex gap-2">
384
+ <Button
385
+ variant={decision === 'approve' ? 'default' : 'outline'}
386
+ size="sm"
387
+ onClick={() => setDecision('approve')}
388
+ className={decision === 'approve' ? 'bg-emerald-600 hover:bg-emerald-700' : ''}
389
+ >
390
+ {t('workflows.orderApproval.approveButton', 'Approve')}
391
+ </Button>
392
+ <Button
393
+ variant={decision === 'reject' ? 'destructive' : 'outline'}
394
+ size="sm"
395
+ onClick={() => setDecision('reject')}
396
+ >
397
+ {t('workflows.orderApproval.rejectButton', 'Reject')}
398
+ </Button>
399
+ </div>
400
+ </div>
401
+
402
+ <div className="space-y-2">
403
+ <label className="text-sm font-medium">
404
+ {t('workflows.orderApproval.commentsLabel', 'Comments')} <span className="text-muted-foreground font-normal">({t('common.optional', 'optional')})</span>
405
+ </label>
406
+ <Textarea
407
+ value={comments}
408
+ onChange={(e) => setComments(e.target.value)}
409
+ placeholder={t('workflows.orderApproval.commentsPlaceholder', 'Add optional comments...')}
410
+ rows={2}
411
+ />
412
+ </div>
413
+
414
+ <Button
415
+ onClick={handleCompleteTask}
416
+ disabled={!decision || isSubmitting}
417
+ variant="default"
418
+ size="sm"
419
+ className="w-full"
420
+ >
421
+ {isSubmitting && <Spinner size="sm" className="mr-2" />}
422
+ {t('workflows.orderApproval.submitDecision', 'Submit Decision')}
423
+ </Button>
424
+ </div>
425
+ )}
426
+
427
+ {/* Active workflow but no pending task (processing) */}
428
+ {activeInstance && !pendingTask && activeInstance.status !== 'COMPLETED' && (
429
+ <div className="rounded-md border border-blue-200 bg-blue-50 p-3">
430
+ <p className="text-sm text-blue-800">
431
+ {t('workflows.orderApproval.processing', 'Workflow is processing...')}
432
+ </p>
433
+ </div>
434
+ )}
435
+
436
+ {/* Completed workflow */}
437
+ {activeInstance && activeInstance.status === 'COMPLETED' && (
438
+ <div className="rounded-md border border-emerald-200 bg-emerald-50 p-3">
439
+ <p className="text-sm text-emerald-800">
440
+ {t('workflows.orderApproval.completed', 'Approval workflow completed.')}
441
+ </p>
442
+ </div>
443
+ )}
444
+ </div>
445
+ )
446
+ }
@@ -0,0 +1,16 @@
1
+ import type { InjectionWidgetModule } from '@open-mercato/shared/modules/widgets/injection'
2
+ import OrderApprovalWidget from './widget.client'
3
+
4
+ const widget: InjectionWidgetModule<any, any> = {
5
+ metadata: {
6
+ id: 'workflows.injection.order-approval',
7
+ title: 'Order Approval',
8
+ description: 'Approve or reject orders requiring authorization',
9
+ features: ['sales.orders.approve'],
10
+ priority: 100,
11
+ enabled: true,
12
+ },
13
+ Widget: OrderApprovalWidget,
14
+ }
15
+
16
+ export default widget
@@ -0,0 +1,21 @@
1
+ import type { ModuleInjectionTable } from '@open-mercato/shared/modules/widgets/injection'
2
+
3
+ /**
4
+ * Workflows module injection table
5
+ * Maps injection spot IDs to widget IDs for automatic widget injection
6
+ */
7
+ export const injectionTable: ModuleInjectionTable = {
8
+ // Inject the order approval widget into the sales order detail page
9
+ 'sales.document.detail.order:details': [
10
+ {
11
+ widgetId: 'workflows.injection.order-approval',
12
+ kind: 'group',
13
+ column: 2,
14
+ groupLabel: 'workflows.orderApproval.groupLabel',
15
+ groupDescription: 'workflows.orderApproval.groupDescription',
16
+ priority: 200,
17
+ },
18
+ ],
19
+ }
20
+
21
+ export default injectionTable