@open-mercato/core 0.4.5-develop-636d33c995 → 0.4.5-develop-3d8e759e45

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 (211) hide show
  1. package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js +17 -2
  2. package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js.map +2 -2
  3. package/dist/modules/catalog/backend/catalog/products/[id]/page.js +15 -0
  4. package/dist/modules/catalog/backend/catalog/products/[id]/page.js.map +2 -2
  5. package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js +30 -0
  6. package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js.map +2 -2
  7. package/dist/modules/catalog/inbox-actions.js +51 -0
  8. package/dist/modules/catalog/inbox-actions.js.map +7 -0
  9. package/dist/modules/catalog/lib/messageObjectPreviews.js +146 -0
  10. package/dist/modules/catalog/lib/messageObjectPreviews.js.map +7 -0
  11. package/dist/modules/catalog/message-objects.js +95 -0
  12. package/dist/modules/catalog/message-objects.js.map +7 -0
  13. package/dist/modules/currencies/backend/currencies/[id]/page.js +21 -0
  14. package/dist/modules/currencies/backend/currencies/[id]/page.js.map +2 -2
  15. package/dist/modules/currencies/lib/messageObjectPreviews.js +51 -0
  16. package/dist/modules/currencies/lib/messageObjectPreviews.js.map +7 -0
  17. package/dist/modules/currencies/message-objects.js +41 -0
  18. package/dist/modules/currencies/message-objects.js.map +7 -0
  19. package/dist/modules/customers/backend/customers/companies/[id]/page.js +20 -0
  20. package/dist/modules/customers/backend/customers/companies/[id]/page.js.map +2 -2
  21. package/dist/modules/customers/backend/customers/deals/[id]/page.js +12 -1
  22. package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
  23. package/dist/modules/customers/backend/customers/people/[id]/page.js +20 -0
  24. package/dist/modules/customers/backend/customers/people/[id]/page.js.map +2 -2
  25. package/dist/modules/customers/components/detail/CompanyHighlights.js +18 -14
  26. package/dist/modules/customers/components/detail/CompanyHighlights.js.map +2 -2
  27. package/dist/modules/customers/components/detail/PersonHighlights.js +18 -14
  28. package/dist/modules/customers/components/detail/PersonHighlights.js.map +2 -2
  29. package/dist/modules/customers/inbox-actions.js +230 -0
  30. package/dist/modules/customers/inbox-actions.js.map +7 -0
  31. package/dist/modules/customers/lib/messageObjectPreviews.js +41 -5
  32. package/dist/modules/customers/lib/messageObjectPreviews.js.map +2 -2
  33. package/dist/modules/customers/message-objects.js +31 -11
  34. package/dist/modules/customers/message-objects.js.map +2 -2
  35. package/dist/modules/inbox_ops/api/emails/[id]/route.js +40 -1
  36. package/dist/modules/inbox_ops/api/emails/[id]/route.js.map +2 -2
  37. package/dist/modules/inbox_ops/api/extract/route.js +87 -0
  38. package/dist/modules/inbox_ops/api/extract/route.js.map +7 -0
  39. package/dist/modules/inbox_ops/api/proposals/[id]/translate/route.js +6 -1
  40. package/dist/modules/inbox_ops/api/proposals/[id]/translate/route.js.map +2 -2
  41. package/dist/modules/inbox_ops/api/proposals/counts/route.js.map +2 -2
  42. package/dist/modules/inbox_ops/backend/inbox-ops/log/page.js +40 -14
  43. package/dist/modules/inbox_ops/backend/inbox-ops/log/page.js.map +2 -2
  44. package/dist/modules/inbox_ops/backend/inbox-ops/log/page.meta.js +2 -2
  45. package/dist/modules/inbox_ops/backend/inbox-ops/log/page.meta.js.map +2 -2
  46. package/dist/modules/inbox_ops/backend/inbox-ops/page.js +161 -79
  47. package/dist/modules/inbox_ops/backend/inbox-ops/page.js.map +2 -2
  48. package/dist/modules/inbox_ops/backend/inbox-ops/page.meta.js +2 -2
  49. package/dist/modules/inbox_ops/backend/inbox-ops/page.meta.js.map +2 -2
  50. package/dist/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.js +109 -62
  51. package/dist/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.js.map +3 -3
  52. package/dist/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.meta.js +2 -2
  53. package/dist/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.meta.js.map +2 -2
  54. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js +36 -14
  55. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js.map +2 -2
  56. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.meta.js +2 -2
  57. package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.meta.js.map +2 -2
  58. package/dist/modules/inbox_ops/components/proposals/ActionCard.js +65 -10
  59. package/dist/modules/inbox_ops/components/proposals/ActionCard.js.map +2 -2
  60. package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js +58 -10
  61. package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js.map +2 -2
  62. package/dist/modules/inbox_ops/lib/constants.js.map +2 -2
  63. package/dist/modules/inbox_ops/lib/contactValidation.js +40 -0
  64. package/dist/modules/inbox_ops/lib/contactValidation.js.map +7 -0
  65. package/dist/modules/inbox_ops/lib/executionEngine.js +31 -826
  66. package/dist/modules/inbox_ops/lib/executionEngine.js.map +3 -3
  67. package/dist/modules/inbox_ops/lib/executionHelpers.js +368 -0
  68. package/dist/modules/inbox_ops/lib/executionHelpers.js.map +7 -0
  69. package/dist/modules/inbox_ops/lib/extractionPrompt.js +28 -35
  70. package/dist/modules/inbox_ops/lib/extractionPrompt.js.map +3 -3
  71. package/dist/modules/inbox_ops/lib/inbox-actions-generated.d.js +1 -0
  72. package/dist/modules/inbox_ops/lib/inbox-actions-generated.d.js.map +7 -0
  73. package/dist/modules/inbox_ops/lib/translationProvider.js +15 -10
  74. package/dist/modules/inbox_ops/lib/translationProvider.js.map +2 -2
  75. package/dist/modules/inbox_ops/subscribers/extractionWorker.js +16 -16
  76. package/dist/modules/inbox_ops/subscribers/extractionWorker.js.map +2 -2
  77. package/dist/modules/messages/commands/messages.js +3 -0
  78. package/dist/modules/messages/commands/messages.js.map +2 -2
  79. package/dist/modules/messages/components/message-detail/panels/objects-panel.js +6 -1
  80. package/dist/modules/messages/components/message-detail/panels/objects-panel.js.map +2 -2
  81. package/dist/modules/messages/components/message-detail/panels/thread-panel.js +4 -1
  82. package/dist/modules/messages/components/message-detail/panels/thread-panel.js.map +2 -2
  83. package/dist/modules/messages/frontend/messages/view/[token]/page.js +1 -0
  84. package/dist/modules/messages/frontend/messages/view/[token]/page.js.map +2 -2
  85. package/dist/modules/resources/backend/resources/resources/[id]/page.js +24 -7
  86. package/dist/modules/resources/backend/resources/resources/[id]/page.js.map +2 -2
  87. package/dist/modules/resources/lib/messageObjectPreviews.js +43 -0
  88. package/dist/modules/resources/lib/messageObjectPreviews.js.map +7 -0
  89. package/dist/modules/resources/message-objects.js +37 -0
  90. package/dist/modules/resources/message-objects.js.map +7 -0
  91. package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js +19 -0
  92. package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js.map +2 -2
  93. package/dist/modules/sales/backend/sales/documents/[id]/page.js +23 -2
  94. package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
  95. package/dist/modules/sales/backend/sales/quotes/[id]/page.js +1 -1
  96. package/dist/modules/sales/backend/sales/quotes/[id]/page.js.map +2 -2
  97. package/dist/modules/sales/inbox-actions.js +278 -0
  98. package/dist/modules/sales/inbox-actions.js.map +7 -0
  99. package/dist/modules/sales/lib/messageObjectPreviews.js +49 -4
  100. package/dist/modules/sales/lib/messageObjectPreviews.js.map +2 -2
  101. package/dist/modules/sales/message-objects.js +44 -2
  102. package/dist/modules/sales/message-objects.js.map +2 -2
  103. package/dist/modules/sales/widgets/messages/SalesDocumentMessageDetail.js +59 -30
  104. package/dist/modules/sales/widgets/messages/SalesDocumentMessageDetail.js.map +2 -2
  105. package/dist/modules/sales/widgets/messages/SalesDocumentMessagePreview.js +1 -1
  106. package/dist/modules/sales/widgets/messages/SalesDocumentMessagePreview.js.map +1 -1
  107. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js +8 -30
  108. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js.map +2 -2
  109. package/dist/modules/staff/backend/staff/my-availability/page.js +13 -0
  110. package/dist/modules/staff/backend/staff/my-availability/page.js.map +2 -2
  111. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js +8 -31
  112. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js.map +2 -2
  113. package/dist/modules/staff/backend/staff/team-members/[id]/page.js +32 -10
  114. package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
  115. package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js +14 -1
  116. package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js.map +2 -2
  117. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +14 -1
  118. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  119. package/dist/modules/staff/components/TeamForm.js +4 -2
  120. package/dist/modules/staff/components/TeamForm.js.map +2 -2
  121. package/dist/modules/staff/components/TeamRoleForm.js +4 -2
  122. package/dist/modules/staff/components/TeamRoleForm.js.map +2 -2
  123. package/dist/modules/staff/lib/messageObjectPreviews.js +111 -2
  124. package/dist/modules/staff/lib/messageObjectPreviews.js.map +2 -2
  125. package/dist/modules/staff/message-objects.js +79 -8
  126. package/dist/modules/staff/message-objects.js.map +2 -2
  127. package/jest.config.cjs +1 -0
  128. package/jest.mocks/inbox-actions.generated.js +5 -0
  129. package/package.json +2 -2
  130. package/src/modules/catalog/backend/catalog/categories/[id]/edit/page.tsx +19 -5
  131. package/src/modules/catalog/backend/catalog/products/[id]/page.tsx +14 -0
  132. package/src/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.tsx +40 -0
  133. package/src/modules/catalog/inbox-actions.ts +60 -0
  134. package/src/modules/catalog/lib/messageObjectPreviews.ts +176 -0
  135. package/src/modules/catalog/message-objects.ts +102 -0
  136. package/src/modules/currencies/backend/currencies/[id]/page.tsx +20 -0
  137. package/src/modules/currencies/lib/messageObjectPreviews.ts +65 -0
  138. package/src/modules/currencies/message-objects.ts +40 -0
  139. package/src/modules/customers/backend/customers/companies/[id]/page.tsx +19 -0
  140. package/src/modules/customers/backend/customers/deals/[id]/page.tsx +13 -0
  141. package/src/modules/customers/backend/customers/people/[id]/page.tsx +19 -0
  142. package/src/modules/customers/components/detail/CompanyHighlights.tsx +14 -9
  143. package/src/modules/customers/components/detail/PersonHighlights.tsx +14 -9
  144. package/src/modules/customers/inbox-actions.ts +285 -0
  145. package/src/modules/customers/lib/messageObjectPreviews.ts +43 -3
  146. package/src/modules/customers/message-objects.ts +31 -11
  147. package/src/modules/inbox_ops/api/emails/[id]/route.ts +44 -0
  148. package/src/modules/inbox_ops/api/extract/route.ts +94 -0
  149. package/src/modules/inbox_ops/api/proposals/[id]/translate/route.ts +6 -1
  150. package/src/modules/inbox_ops/api/proposals/counts/route.ts +2 -0
  151. package/src/modules/inbox_ops/backend/inbox-ops/log/page.meta.ts +2 -2
  152. package/src/modules/inbox_ops/backend/inbox-ops/log/page.tsx +43 -13
  153. package/src/modules/inbox_ops/backend/inbox-ops/page.meta.ts +2 -2
  154. package/src/modules/inbox_ops/backend/inbox-ops/page.tsx +176 -81
  155. package/src/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.meta.ts +2 -2
  156. package/src/modules/inbox_ops/backend/inbox-ops/proposals/[id]/page.tsx +122 -68
  157. package/src/modules/inbox_ops/backend/inbox-ops/settings/page.meta.ts +2 -2
  158. package/src/modules/inbox_ops/backend/inbox-ops/settings/page.tsx +36 -14
  159. package/src/modules/inbox_ops/components/proposals/ActionCard.tsx +91 -7
  160. package/src/modules/inbox_ops/components/proposals/EditActionDialog.tsx +64 -12
  161. package/src/modules/inbox_ops/lib/constants.ts +9 -0
  162. package/src/modules/inbox_ops/lib/contactValidation.ts +54 -0
  163. package/src/modules/inbox_ops/lib/executionEngine.ts +47 -1060
  164. package/src/modules/inbox_ops/lib/executionHelpers.ts +527 -0
  165. package/src/modules/inbox_ops/lib/extractionPrompt.ts +45 -34
  166. package/src/modules/inbox_ops/lib/inbox-actions-generated.d.ts +11 -0
  167. package/src/modules/inbox_ops/lib/translationProvider.ts +16 -10
  168. package/src/modules/inbox_ops/subscribers/extractionWorker.ts +16 -18
  169. package/src/modules/messages/commands/messages.ts +4 -0
  170. package/src/modules/messages/components/message-detail/panels/objects-panel.tsx +8 -1
  171. package/src/modules/messages/components/message-detail/panels/thread-panel.tsx +3 -0
  172. package/src/modules/messages/frontend/messages/view/[token]/page.tsx +1 -0
  173. package/src/modules/resources/backend/resources/resources/[id]/page.tsx +20 -4
  174. package/src/modules/resources/lib/messageObjectPreviews.ts +55 -0
  175. package/src/modules/resources/message-objects.ts +36 -0
  176. package/src/modules/sales/backend/sales/channels/[channelId]/edit/page.tsx +18 -0
  177. package/src/modules/sales/backend/sales/documents/[id]/page.tsx +23 -0
  178. package/src/modules/sales/backend/sales/quotes/[id]/page.tsx +1 -1
  179. package/src/modules/sales/inbox-actions.ts +359 -0
  180. package/src/modules/sales/lib/messageObjectPreviews.ts +54 -4
  181. package/src/modules/sales/message-objects.ts +44 -2
  182. package/src/modules/sales/widgets/messages/SalesDocumentMessageDetail.tsx +72 -34
  183. package/src/modules/sales/widgets/messages/SalesDocumentMessagePreview.tsx +1 -1
  184. package/src/modules/staff/backend/staff/leave-requests/[id]/page.tsx +7 -29
  185. package/src/modules/staff/backend/staff/my-availability/page.tsx +14 -0
  186. package/src/modules/staff/backend/staff/my-leave-requests/[id]/page.tsx +8 -30
  187. package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +28 -7
  188. package/src/modules/staff/backend/staff/team-roles/[id]/edit/page.tsx +12 -0
  189. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +12 -0
  190. package/src/modules/staff/components/TeamForm.tsx +3 -0
  191. package/src/modules/staff/components/TeamRoleForm.tsx +3 -0
  192. package/src/modules/staff/lib/messageObjectPreviews.ts +133 -2
  193. package/src/modules/staff/message-objects.ts +79 -8
  194. package/dist/modules/customers/widgets/messages/CustomerMessageObjectDetail.js +0 -51
  195. package/dist/modules/customers/widgets/messages/CustomerMessageObjectDetail.js.map +0 -7
  196. package/dist/modules/customers/widgets/messages/CustomerMessageObjectPreview.js +0 -35
  197. package/dist/modules/customers/widgets/messages/CustomerMessageObjectPreview.js.map +0 -7
  198. package/dist/modules/customers/widgets/messages/index.js +0 -7
  199. package/dist/modules/customers/widgets/messages/index.js.map +0 -7
  200. package/dist/modules/staff/widgets/messages/StaffMessageObjectDetail.js +0 -51
  201. package/dist/modules/staff/widgets/messages/StaffMessageObjectDetail.js.map +0 -7
  202. package/dist/modules/staff/widgets/messages/StaffMessageObjectPreview.js +0 -34
  203. package/dist/modules/staff/widgets/messages/StaffMessageObjectPreview.js.map +0 -7
  204. package/dist/modules/staff/widgets/messages/index.js +0 -7
  205. package/dist/modules/staff/widgets/messages/index.js.map +0 -7
  206. package/src/modules/customers/widgets/messages/CustomerMessageObjectDetail.tsx +0 -57
  207. package/src/modules/customers/widgets/messages/CustomerMessageObjectPreview.tsx +0 -49
  208. package/src/modules/customers/widgets/messages/index.ts +0 -2
  209. package/src/modules/staff/widgets/messages/StaffMessageObjectDetail.tsx +0 -57
  210. package/src/modules/staff/widgets/messages/StaffMessageObjectPreview.tsx +0 -44
  211. package/src/modules/staff/widgets/messages/index.ts +0 -2
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/customers/components/detail/CompanyHighlights.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { FormHeader } from '@open-mercato/ui/backend/forms'\nimport { VersionHistoryAction } from '@open-mercato/ui/backend/version-history'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport {\n InlineTextEditor,\n InlineDictionaryEditor,\n InlineNextInteractionEditor,\n type InlineFieldProps,\n type NextInteractionPayload,\n} from './InlineEditors'\n\ntype CompanyHighlightsCompany = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionRefId?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n}\n\ntype CompanyHighlightsProfile = {\n id?: string\n brandName?: string | null\n legalName?: string | null\n websiteUrl?: string | null\n industry?: string | null\n annualRevenue?: string | null\n} | null\n\ntype CompanyHighlightsValidators = {\n email: NonNullable<InlineFieldProps['validator']>\n phone: NonNullable<InlineFieldProps['validator']>\n displayName: NonNullable<InlineFieldProps['validator']>\n}\n\nexport type CompanyHighlightsProps = {\n company: CompanyHighlightsCompany\n profile?: CompanyHighlightsProfile\n validators: CompanyHighlightsValidators\n onDisplayNameSave: (value: string | null) => Promise<void>\n onPrimaryEmailSave: (value: string | null) => Promise<void>\n onPrimaryPhoneSave: (value: string | null) => Promise<void>\n onStatusSave: (value: string | null) => Promise<void>\n onNextInteractionSave: (payload: NextInteractionPayload | null) => Promise<void>\n onDelete: () => void\n isDeleting: boolean\n}\n\nexport function CompanyHighlights({\n company,\n profile,\n validators,\n onDisplayNameSave,\n onPrimaryEmailSave,\n onPrimaryPhoneSave,\n onStatusSave,\n onNextInteractionSave,\n onDelete,\n isDeleting,\n}: CompanyHighlightsProps) {\n const t = useT()\n const historyFallbackId =\n profile?.id && profile.id !== company.id ? profile.id : undefined\n\n return (\n <div className=\"space-y-6\">\n <FormHeader\n mode=\"detail\"\n backHref=\"/backend/customers/companies\"\n backLabel={t('customers.companies.detail.actions.backToList', 'Back to companies')}\n utilityActions={(\n <VersionHistoryAction\n config={{\n resourceKind: 'customers.company',\n resourceId: company.id,\n resourceIdFallback: historyFallbackId,\n organizationId: company.organizationId ?? undefined,\n }}\n t={t}\n />\n )}\n title={\n <InlineTextEditor\n label={t('customers.companies.form.displayName.label', 'Display name')}\n value={company.displayName}\n placeholder={t('customers.companies.form.displayName.placeholder', 'Enter company name')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n validator={validators.displayName}\n onSave={onDisplayNameSave}\n hideLabel\n variant=\"plain\"\n activateOnClick\n triggerClassName=\"opacity-0 transition-opacity duration-150 group-hover:opacity-100 group-focus-within:opacity-100 focus-visible:opacity-100\"\n containerClassName=\"max-w-full\"\n />\n }\n onDelete={() => {\n onDelete()\n }}\n isDeleting={isDeleting}\n deleteLabel={t('customers.companies.detail.actions.delete', 'Delete company')}\n />\n\n <div className=\"grid gap-4 md:grid-cols-2 xl:grid-cols-4\">\n <InlineTextEditor\n label={t('customers.companies.detail.highlights.primaryEmail', 'Primary email')}\n value={company.primaryEmail || ''}\n placeholder={t('customers.companies.form.primaryEmail', 'Add email')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n type=\"email\"\n validator={validators.email}\n recordId={company.id}\n activateOnClick\n onSave={onPrimaryEmailSave}\n />\n <InlineTextEditor\n label={t('customers.companies.detail.highlights.primaryPhone', 'Primary phone')}\n value={company.primaryPhone || ''}\n placeholder={t('customers.companies.form.primaryPhone', 'Add phone')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n type=\"tel\"\n validator={validators.phone}\n recordId={company.id}\n activateOnClick\n onSave={onPrimaryPhoneSave}\n />\n <InlineDictionaryEditor\n label={t('customers.companies.detail.highlights.status', 'Status')}\n value={company.status ?? null}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n activateOnClick\n onSave={onStatusSave}\n kind=\"statuses\"\n />\n <InlineNextInteractionEditor\n label={t('customers.companies.detail.highlights.nextInteraction', 'Next interaction')}\n valueAt={company.nextInteractionAt || null}\n valueName={company.nextInteractionName || null}\n valueRefId={company.nextInteractionRefId || null}\n valueIcon={company.nextInteractionIcon || null}\n valueColor={company.nextInteractionColor || null}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n onSave={onNextInteractionSave}\n activateOnClick\n />\n </div>\n </div>\n )\n}\n\nexport default CompanyHighlights\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { FormHeader } from '@open-mercato/ui/backend/forms'\nimport { VersionHistoryAction } from '@open-mercato/ui/backend/version-history'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport {\n InlineTextEditor,\n InlineDictionaryEditor,\n InlineNextInteractionEditor,\n type InlineFieldProps,\n type NextInteractionPayload,\n} from './InlineEditors'\n\ntype CompanyHighlightsCompany = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionRefId?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n}\n\ntype CompanyHighlightsProfile = {\n id?: string\n brandName?: string | null\n legalName?: string | null\n websiteUrl?: string | null\n industry?: string | null\n annualRevenue?: string | null\n} | null\n\ntype CompanyHighlightsValidators = {\n email: NonNullable<InlineFieldProps['validator']>\n phone: NonNullable<InlineFieldProps['validator']>\n displayName: NonNullable<InlineFieldProps['validator']>\n}\n\nexport type CompanyHighlightsProps = {\n company: CompanyHighlightsCompany\n profile?: CompanyHighlightsProfile\n validators: CompanyHighlightsValidators\n onDisplayNameSave: (value: string | null) => Promise<void>\n onPrimaryEmailSave: (value: string | null) => Promise<void>\n onPrimaryPhoneSave: (value: string | null) => Promise<void>\n onStatusSave: (value: string | null) => Promise<void>\n onNextInteractionSave: (payload: NextInteractionPayload | null) => Promise<void>\n onDelete: () => void\n isDeleting: boolean\n utilityActions?: React.ReactNode\n}\n\nexport function CompanyHighlights({\n company,\n profile,\n validators,\n onDisplayNameSave,\n onPrimaryEmailSave,\n onPrimaryPhoneSave,\n onStatusSave,\n onNextInteractionSave,\n onDelete,\n isDeleting,\n utilityActions,\n}: CompanyHighlightsProps) {\n const t = useT()\n const historyFallbackId =\n profile?.id && profile.id !== company.id ? profile.id : undefined\n\n return (\n <div className=\"space-y-6\">\n <FormHeader\n mode=\"detail\"\n backHref=\"/backend/customers/companies\"\n backLabel={t('customers.companies.detail.actions.backToList', 'Back to companies')}\n utilityActions={(\n <>\n {utilityActions}\n <VersionHistoryAction\n config={{\n resourceKind: 'customers.company',\n resourceId: company.id,\n resourceIdFallback: historyFallbackId,\n organizationId: company.organizationId ?? undefined,\n }}\n t={t}\n />\n </>\n )}\n title={\n <InlineTextEditor\n label={t('customers.companies.form.displayName.label', 'Display name')}\n value={company.displayName}\n placeholder={t('customers.companies.form.displayName.placeholder', 'Enter company name')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n validator={validators.displayName}\n onSave={onDisplayNameSave}\n hideLabel\n variant=\"plain\"\n activateOnClick\n triggerClassName=\"opacity-0 transition-opacity duration-150 group-hover:opacity-100 group-focus-within:opacity-100 focus-visible:opacity-100\"\n containerClassName=\"max-w-full\"\n />\n }\n onDelete={() => {\n onDelete()\n }}\n isDeleting={isDeleting}\n deleteLabel={t('customers.companies.detail.actions.delete', 'Delete company')}\n />\n\n <div className=\"grid gap-4 md:grid-cols-2 xl:grid-cols-4\">\n <InlineTextEditor\n label={t('customers.companies.detail.highlights.primaryEmail', 'Primary email')}\n value={company.primaryEmail || ''}\n placeholder={t('customers.companies.form.primaryEmail', 'Add email')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n type=\"email\"\n validator={validators.email}\n recordId={company.id}\n activateOnClick\n onSave={onPrimaryEmailSave}\n />\n <InlineTextEditor\n label={t('customers.companies.detail.highlights.primaryPhone', 'Primary phone')}\n value={company.primaryPhone || ''}\n placeholder={t('customers.companies.form.primaryPhone', 'Add phone')}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n type=\"tel\"\n validator={validators.phone}\n recordId={company.id}\n activateOnClick\n onSave={onPrimaryPhoneSave}\n />\n <InlineDictionaryEditor\n label={t('customers.companies.detail.highlights.status', 'Status')}\n value={company.status ?? null}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n activateOnClick\n onSave={onStatusSave}\n kind=\"statuses\"\n />\n <InlineNextInteractionEditor\n label={t('customers.companies.detail.highlights.nextInteraction', 'Next interaction')}\n valueAt={company.nextInteractionAt || null}\n valueName={company.nextInteractionName || null}\n valueRefId={company.nextInteractionRefId || null}\n valueIcon={company.nextInteractionIcon || null}\n valueColor={company.nextInteractionColor || null}\n emptyLabel={t('customers.companies.detail.noValue', 'Not provided')}\n onSave={onNextInteractionSave}\n activateOnClick\n />\n </div>\n </div>\n )\n}\n\nexport default CompanyHighlights\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,5 +1,5 @@
1
1
  "use client";
2
- import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
3
3
  import * as React from "react";
4
4
  import { useRouter } from "next/navigation";
5
5
  import { Building2, Loader2, Pencil, X } from "lucide-react";
@@ -26,7 +26,8 @@ function PersonHighlights({
26
26
  onNextInteractionSave,
27
27
  onDelete,
28
28
  isDeleting,
29
- onCompanySave
29
+ onCompanySave,
30
+ utilityActions
30
31
  }) {
31
32
  const router = useRouter();
32
33
  const t = useT();
@@ -260,18 +261,21 @@ function PersonHighlights({
260
261
  mode: "detail",
261
262
  backHref: "/backend/customers/people",
262
263
  backLabel: t("customers.people.detail.actions.backToList"),
263
- utilityActions: /* @__PURE__ */ jsx(
264
- VersionHistoryAction,
265
- {
266
- config: {
267
- resourceKind: "customers.person",
268
- resourceId: person.id,
269
- resourceIdFallback: historyFallbackId,
270
- organizationId: person.organizationId ?? void 0
271
- },
272
- t
273
- }
274
- ),
264
+ utilityActions: /* @__PURE__ */ jsxs(Fragment, { children: [
265
+ utilityActions,
266
+ /* @__PURE__ */ jsx(
267
+ VersionHistoryAction,
268
+ {
269
+ config: {
270
+ resourceKind: "customers.person",
271
+ resourceId: person.id,
272
+ resourceIdFallback: historyFallbackId,
273
+ organizationId: person.organizationId ?? void 0
274
+ },
275
+ t
276
+ }
277
+ )
278
+ ] }),
275
279
  title: /* @__PURE__ */ jsx(
276
280
  InlineTextEditor,
277
281
  {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/customers/components/detail/PersonHighlights.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { useRouter } from 'next/navigation'\nimport { Building2, Loader2, Pencil, X } from 'lucide-react'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { FormHeader } from '@open-mercato/ui/backend/forms'\nimport { VersionHistoryAction } from '@open-mercato/ui/backend/version-history'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { CompanySelectField } from '../formConfig'\nimport {\n InlineTextEditor,\n InlineDictionaryEditor,\n InlineNextInteractionEditor,\n type InlineFieldProps,\n type NextInteractionPayload,\n} from './InlineEditors'\n\ntype PersonHighlightsPerson = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionRefId?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n}\n\ntype PersonHighlightsProfile = {\n id?: string\n companyEntityId?: string | null\n} | null\n\ntype PersonHighlightsValidators = {\n email: NonNullable<InlineFieldProps['validator']>\n phone: NonNullable<InlineFieldProps['validator']>\n displayName: NonNullable<InlineFieldProps['validator']>\n}\n\nexport type PersonHighlightsProps = {\n person: PersonHighlightsPerson\n profile: PersonHighlightsProfile\n validators: PersonHighlightsValidators\n onDisplayNameSave: (value: string | null) => Promise<void>\n onPrimaryEmailSave: (value: string | null) => Promise<void>\n onPrimaryPhoneSave: (value: string | null) => Promise<void>\n onStatusSave: (value: string | null) => Promise<void>\n onNextInteractionSave: (value: NextInteractionPayload | null) => Promise<void>\n onDelete: () => void\n isDeleting: boolean\n onCompanySave: (companyId: string | null) => Promise<void>\n}\n\ntype CompanyInfo = { id: string; name: string }\n\nexport function PersonHighlights({\n person,\n profile,\n validators,\n onDisplayNameSave,\n onPrimaryEmailSave,\n onPrimaryPhoneSave,\n onStatusSave,\n onNextInteractionSave,\n onDelete,\n isDeleting,\n onCompanySave,\n}: PersonHighlightsProps) {\n const router = useRouter()\n const t = useT()\n const runMutation = React.useCallback(async (operation: () => Promise<void>) => operation(), [])\n const [editingCompany, setEditingCompany] = React.useState(false)\n const [companyDraftId, setCompanyDraftId] = React.useState<string>('')\n const [company, setCompany] = React.useState<CompanyInfo | null>(null)\n const [companyLoading, setCompanyLoading] = React.useState(false)\n const [companyError, setCompanyError] = React.useState<string | null>(null)\n const [companySaving, setCompanySaving] = React.useState(false)\n const companyHref = React.useMemo(\n () => (company ? `/backend/customers/companies/${encodeURIComponent(company.id)}` : null),\n [company],\n )\n const isCompanyInteractive = !editingCompany && !companyLoading && !companyError && Boolean(companyHref)\n\n const navigateToCompany = React.useCallback(() => {\n if (!companyHref) return\n router.push(companyHref)\n }, [companyHref])\n\n const handleCompanyClick = React.useCallback(\n (event: React.MouseEvent<HTMLDivElement>) => {\n if (!isCompanyInteractive || !companyHref) return\n const target = event.target as HTMLElement\n if (target.closest('button')) return\n if (event.defaultPrevented) return\n if (event.button !== 0) return\n if (event.metaKey || event.ctrlKey || event.shiftKey || event.altKey) {\n if (typeof window !== 'undefined') {\n window.open(companyHref, '_blank', 'noopener,noreferrer')\n }\n return\n }\n navigateToCompany()\n },\n [companyHref, isCompanyInteractive, navigateToCompany],\n )\n\n const handleCompanyKeyDown = React.useCallback(\n (event: React.KeyboardEvent<HTMLDivElement>) => {\n if (!isCompanyInteractive) return\n if (event.key === 'Enter' || event.key === ' ') {\n event.preventDefault()\n navigateToCompany()\n }\n },\n [isCompanyInteractive, navigateToCompany],\n )\n\n const activeCompanyId = profile?.companyEntityId ?? null\n const historyFallbackId =\n profile?.id && profile.id !== person.id ? profile.id : undefined\n\n const loadCompany = React.useCallback(async (companyId: string | null) => {\n if (!companyId) {\n setCompany(null)\n return\n }\n setCompanyLoading(true)\n setCompanyError(null)\n try {\n const payload = await readApiResultOrThrow<Record<string, unknown>>(\n `/api/customers/companies?id=${encodeURIComponent(companyId)}`,\n undefined,\n { errorMessage: t('customers.people.detail.company.loadError', 'Unable to load company information.') },\n )\n const items = Array.isArray(payload?.items) ? (payload.items as Array<Record<string, unknown>>) : []\n const item = items.find((entry) => {\n if (!entry) return false\n const rawId = (entry as Record<string, unknown>).id\n if (typeof rawId === 'string') return rawId === companyId\n if (typeof rawId === 'number') return String(rawId) === companyId\n return false\n }) as Record<string, unknown> | undefined\n if (item && typeof item.display_name === 'string') {\n setCompany({ id: companyId, name: item.display_name })\n } else if (item && typeof item.displayName === 'string') {\n setCompany({ id: companyId, name: item.displayName })\n } else {\n setCompany({ id: companyId, name: t('customers.people.detail.company.unknown', 'Unknown company') })\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : t('customers.people.detail.company.loadError', 'Unable to load company.')\n setCompanyError(message)\n setCompany(null)\n } finally {\n setCompanyLoading(false)\n }\n }, [t])\n\n React.useEffect(() => {\n setCompanyDraftId(activeCompanyId ?? '')\n loadCompany(activeCompanyId).catch(() => {})\n }, [activeCompanyId, loadCompany])\n\n const handleCompanySave = React.useCallback(async () => {\n if (companySaving) return\n setCompanySaving(true)\n setCompanyError(null)\n const nextId = companyDraftId.trim()\n try {\n await runMutation(async () => {\n await onCompanySave(nextId.length ? nextId : null)\n })\n await loadCompany(nextId.length ? nextId : null)\n setEditingCompany(false)\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : t('customers.people.detail.company.saveError', 'Unable to update company.')\n setCompanyError(message)\n } finally {\n setCompanySaving(false)\n }\n }, [companyDraftId, companySaving, loadCompany, onCompanySave, runMutation, t])\n\n const handleCompanyClear = React.useCallback(async () => {\n if (companySaving) return\n setCompanySaving(true)\n setCompanyError(null)\n try {\n await runMutation(async () => {\n await onCompanySave(null)\n })\n await loadCompany(null)\n setCompanyDraftId('')\n setEditingCompany(false)\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : t('customers.people.detail.company.saveError', 'Unable to update company.')\n setCompanyError(message)\n } finally {\n setCompanySaving(false)\n }\n }, [companySaving, loadCompany, onCompanySave, runMutation, t])\n\n const companyPanel = (\n <div\n className={cn(\n 'group rounded-lg border bg-muted/30 p-3',\n isCompanyInteractive\n ? 'cursor-pointer transition-colors hover:bg-muted/50 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring'\n : null,\n )}\n role={isCompanyInteractive ? 'link' : undefined}\n tabIndex={isCompanyInteractive ? 0 : undefined}\n onClick={handleCompanyClick}\n onKeyDown={handleCompanyKeyDown}\n >\n <div className=\"flex items-start justify-between gap-3\">\n <div className=\"flex-1 space-y-1\">\n <p className=\"flex items-center gap-2 text-xs uppercase tracking-wide text-muted-foreground\">\n <Building2 aria-hidden className=\"h-3.5 w-3.5\" />\n {t('customers.people.detail.company.label', 'Company')}\n </p>\n {editingCompany ? (\n <div\n className=\"space-y-3\"\n onKeyDown={(event) => {\n if (event.key === 'Escape') {\n event.preventDefault()\n if (!companySaving) {\n setEditingCompany(false)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }\n }\n if (event.key === 'Enter' && (event.metaKey || event.ctrlKey)) {\n event.preventDefault()\n if (!companySaving) {\n void handleCompanySave()\n }\n }\n }}\n >\n <CompanySelectField\n value={companyDraftId || undefined}\n onChange={(next) => setCompanyDraftId(next ?? '')}\n labels={{\n placeholder: t('customers.people.form.company.placeholder'),\n addLabel: t('customers.people.form.company.add'),\n addPrompt: t('customers.people.form.company.prompt'),\n dialogTitle: t('customers.people.form.company.dialogTitle'),\n inputLabel: t('customers.people.form.company.inputLabel'),\n inputPlaceholder: t('customers.people.form.company.inputPlaceholder'),\n emptyError: t('customers.people.form.dictionary.errorRequired'),\n cancelLabel: t('customers.people.form.dictionary.cancel'),\n saveLabel: t('customers.people.form.dictionary.save'),\n errorLoad: t('customers.people.form.dictionary.errorLoad'),\n errorSave: t('customers.people.form.dictionary.error'),\n loadingLabel: t('customers.people.form.company.loading'),\n }}\n />\n {companyError ? <p className=\"text-xs text-red-600\">{companyError}</p> : null}\n <div className=\"flex flex-wrap items-center gap-2\">\n <Button type=\"button\" size=\"sm\" onClick={handleCompanySave} disabled={companySaving}>\n {companySaving ? <Loader2 className=\"mr-2 h-3.5 w-3.5 animate-spin\" /> : null}\n {t('ui.forms.actions.save')}\n </Button>\n <Button type=\"button\" size=\"sm\" variant=\"ghost\" onClick={() => {\n setEditingCompany(false)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }} disabled={companySaving}>\n {t('ui.forms.actions.cancel')}\n </Button>\n <Button type=\"button\" size=\"sm\" variant=\"secondary\" onClick={handleCompanyClear} disabled={companySaving}>\n {t('ui.forms.actions.clear')}\n </Button>\n </div>\n </div>\n ) : (\n <div className=\"flex items-center gap-2 text-sm\">\n {companyLoading ? (\n <span className=\"inline-flex items-center gap-2 text-muted-foreground\">\n <Loader2 className=\"h-4 w-4 animate-spin\" />\n {t('customers.people.detail.company.loading', 'Loading company\u2026')}\n </span>\n ) : company ? (\n <span className=\"text-primary transition group-hover:underline\">\n {t('customers.people.detail.company.current', undefined, { company: company.name })}\n </span>\n ) : companyError ? (\n <span className=\"text-xs text-red-600\">{companyError}</span>\n ) : (\n <span className=\"text-muted-foreground\">\n {t('customers.people.detail.company.empty', 'No company assigned')}\n </span>\n )}\n </div>\n )}\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => {\n if (companySaving) return\n setEditingCompany((prev) => !prev)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }}\n className={\n editingCompany\n ? 'opacity-100'\n : 'opacity-0 transition-opacity duration-150 group-hover:opacity-100 hover:opacity-100 focus-visible:opacity-100'\n }\n >\n {editingCompany ? <X className=\"h-4 w-4\" /> : <Pencil className=\"h-4 w-4\" />}\n <span className=\"sr-only\">\n {editingCompany ? t('ui.forms.actions.cancel') : t('ui.forms.actions.edit')}\n </span>\n </Button>\n </div>\n </div>\n )\n\n return (\n <div className=\"space-y-6\">\n <FormHeader\n mode=\"detail\"\n backHref=\"/backend/customers/people\"\n backLabel={t('customers.people.detail.actions.backToList')}\n utilityActions={(\n <VersionHistoryAction\n config={{\n resourceKind: 'customers.person',\n resourceId: person.id,\n resourceIdFallback: historyFallbackId,\n organizationId: person.organizationId ?? undefined,\n }}\n t={t}\n />\n )}\n title={\n <InlineTextEditor\n label={t('customers.people.form.displayName.label')}\n value={person.displayName}\n placeholder={t('customers.people.form.displayName.placeholder')}\n emptyLabel={t('customers.people.detail.noValue')}\n validator={validators.displayName}\n onSave={onDisplayNameSave}\n hideLabel\n variant=\"plain\"\n activateOnClick\n triggerClassName=\"opacity-0 transition-opacity duration-150 group-hover:opacity-100 group-focus-within:opacity-100 focus-visible:opacity-100\"\n containerClassName=\"max-w-full\"\n />\n }\n onDelete={() => {\n onDelete()\n }}\n isDeleting={isDeleting}\n deleteLabel={t('customers.people.list.actions.delete')}\n />\n\n {companyPanel}\n\n <div className=\"grid gap-4 md:grid-cols-2 xl:grid-cols-4\">\n <InlineTextEditor\n label={t('customers.people.detail.highlights.primaryEmail')}\n value={person.primaryEmail || ''}\n placeholder={t('customers.people.form.primaryEmail')}\n emptyLabel={t('customers.people.detail.noValue')}\n type=\"email\"\n validator={validators.email}\n recordId={person.id}\n activateOnClick\n onSave={onPrimaryEmailSave}\n />\n <InlineTextEditor\n label={t('customers.people.detail.highlights.primaryPhone')}\n value={person.primaryPhone || ''}\n placeholder={t('customers.people.form.primaryPhone')}\n emptyLabel={t('customers.people.detail.noValue')}\n type=\"tel\"\n validator={validators.phone}\n recordId={person.id}\n activateOnClick\n onSave={onPrimaryPhoneSave}\n />\n <InlineDictionaryEditor\n label={t('customers.people.detail.highlights.status')}\n value={person.status ?? null}\n emptyLabel={t('customers.people.detail.noValue')}\n activateOnClick\n onSave={onStatusSave}\n kind=\"statuses\"\n />\n <InlineNextInteractionEditor\n label={t('customers.people.detail.highlights.nextInteraction')}\n valueAt={person.nextInteractionAt || null}\n valueName={person.nextInteractionName || null}\n valueRefId={person.nextInteractionRefId || null}\n valueIcon={person.nextInteractionIcon || null}\n valueColor={person.nextInteractionColor || null}\n emptyLabel={t('customers.people.detail.noValue')}\n onSave={onNextInteractionSave}\n activateOnClick\n />\n </div>\n </div>\n )\n}\n\nexport default PersonHighlights\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { useRouter } from 'next/navigation'\nimport { Building2, Loader2, Pencil, X } from 'lucide-react'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { FormHeader } from '@open-mercato/ui/backend/forms'\nimport { VersionHistoryAction } from '@open-mercato/ui/backend/version-history'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { CompanySelectField } from '../formConfig'\nimport {\n InlineTextEditor,\n InlineDictionaryEditor,\n InlineNextInteractionEditor,\n type InlineFieldProps,\n type NextInteractionPayload,\n} from './InlineEditors'\n\ntype PersonHighlightsPerson = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionRefId?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n}\n\ntype PersonHighlightsProfile = {\n id?: string\n companyEntityId?: string | null\n} | null\n\ntype PersonHighlightsValidators = {\n email: NonNullable<InlineFieldProps['validator']>\n phone: NonNullable<InlineFieldProps['validator']>\n displayName: NonNullable<InlineFieldProps['validator']>\n}\n\nexport type PersonHighlightsProps = {\n person: PersonHighlightsPerson\n profile: PersonHighlightsProfile\n validators: PersonHighlightsValidators\n onDisplayNameSave: (value: string | null) => Promise<void>\n onPrimaryEmailSave: (value: string | null) => Promise<void>\n onPrimaryPhoneSave: (value: string | null) => Promise<void>\n onStatusSave: (value: string | null) => Promise<void>\n onNextInteractionSave: (value: NextInteractionPayload | null) => Promise<void>\n onDelete: () => void\n isDeleting: boolean\n onCompanySave: (companyId: string | null) => Promise<void>\n utilityActions?: React.ReactNode\n}\n\ntype CompanyInfo = { id: string; name: string }\n\nexport function PersonHighlights({\n person,\n profile,\n validators,\n onDisplayNameSave,\n onPrimaryEmailSave,\n onPrimaryPhoneSave,\n onStatusSave,\n onNextInteractionSave,\n onDelete,\n isDeleting,\n onCompanySave,\n utilityActions,\n}: PersonHighlightsProps) {\n const router = useRouter()\n const t = useT()\n const runMutation = React.useCallback(async (operation: () => Promise<void>) => operation(), [])\n const [editingCompany, setEditingCompany] = React.useState(false)\n const [companyDraftId, setCompanyDraftId] = React.useState<string>('')\n const [company, setCompany] = React.useState<CompanyInfo | null>(null)\n const [companyLoading, setCompanyLoading] = React.useState(false)\n const [companyError, setCompanyError] = React.useState<string | null>(null)\n const [companySaving, setCompanySaving] = React.useState(false)\n const companyHref = React.useMemo(\n () => (company ? `/backend/customers/companies/${encodeURIComponent(company.id)}` : null),\n [company],\n )\n const isCompanyInteractive = !editingCompany && !companyLoading && !companyError && Boolean(companyHref)\n\n const navigateToCompany = React.useCallback(() => {\n if (!companyHref) return\n router.push(companyHref)\n }, [companyHref])\n\n const handleCompanyClick = React.useCallback(\n (event: React.MouseEvent<HTMLDivElement>) => {\n if (!isCompanyInteractive || !companyHref) return\n const target = event.target as HTMLElement\n if (target.closest('button')) return\n if (event.defaultPrevented) return\n if (event.button !== 0) return\n if (event.metaKey || event.ctrlKey || event.shiftKey || event.altKey) {\n if (typeof window !== 'undefined') {\n window.open(companyHref, '_blank', 'noopener,noreferrer')\n }\n return\n }\n navigateToCompany()\n },\n [companyHref, isCompanyInteractive, navigateToCompany],\n )\n\n const handleCompanyKeyDown = React.useCallback(\n (event: React.KeyboardEvent<HTMLDivElement>) => {\n if (!isCompanyInteractive) return\n if (event.key === 'Enter' || event.key === ' ') {\n event.preventDefault()\n navigateToCompany()\n }\n },\n [isCompanyInteractive, navigateToCompany],\n )\n\n const activeCompanyId = profile?.companyEntityId ?? null\n const historyFallbackId =\n profile?.id && profile.id !== person.id ? profile.id : undefined\n\n const loadCompany = React.useCallback(async (companyId: string | null) => {\n if (!companyId) {\n setCompany(null)\n return\n }\n setCompanyLoading(true)\n setCompanyError(null)\n try {\n const payload = await readApiResultOrThrow<Record<string, unknown>>(\n `/api/customers/companies?id=${encodeURIComponent(companyId)}`,\n undefined,\n { errorMessage: t('customers.people.detail.company.loadError', 'Unable to load company information.') },\n )\n const items = Array.isArray(payload?.items) ? (payload.items as Array<Record<string, unknown>>) : []\n const item = items.find((entry) => {\n if (!entry) return false\n const rawId = (entry as Record<string, unknown>).id\n if (typeof rawId === 'string') return rawId === companyId\n if (typeof rawId === 'number') return String(rawId) === companyId\n return false\n }) as Record<string, unknown> | undefined\n if (item && typeof item.display_name === 'string') {\n setCompany({ id: companyId, name: item.display_name })\n } else if (item && typeof item.displayName === 'string') {\n setCompany({ id: companyId, name: item.displayName })\n } else {\n setCompany({ id: companyId, name: t('customers.people.detail.company.unknown', 'Unknown company') })\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : t('customers.people.detail.company.loadError', 'Unable to load company.')\n setCompanyError(message)\n setCompany(null)\n } finally {\n setCompanyLoading(false)\n }\n }, [t])\n\n React.useEffect(() => {\n setCompanyDraftId(activeCompanyId ?? '')\n loadCompany(activeCompanyId).catch(() => {})\n }, [activeCompanyId, loadCompany])\n\n const handleCompanySave = React.useCallback(async () => {\n if (companySaving) return\n setCompanySaving(true)\n setCompanyError(null)\n const nextId = companyDraftId.trim()\n try {\n await runMutation(async () => {\n await onCompanySave(nextId.length ? nextId : null)\n })\n await loadCompany(nextId.length ? nextId : null)\n setEditingCompany(false)\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : t('customers.people.detail.company.saveError', 'Unable to update company.')\n setCompanyError(message)\n } finally {\n setCompanySaving(false)\n }\n }, [companyDraftId, companySaving, loadCompany, onCompanySave, runMutation, t])\n\n const handleCompanyClear = React.useCallback(async () => {\n if (companySaving) return\n setCompanySaving(true)\n setCompanyError(null)\n try {\n await runMutation(async () => {\n await onCompanySave(null)\n })\n await loadCompany(null)\n setCompanyDraftId('')\n setEditingCompany(false)\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : t('customers.people.detail.company.saveError', 'Unable to update company.')\n setCompanyError(message)\n } finally {\n setCompanySaving(false)\n }\n }, [companySaving, loadCompany, onCompanySave, runMutation, t])\n\n const companyPanel = (\n <div\n className={cn(\n 'group rounded-lg border bg-muted/30 p-3',\n isCompanyInteractive\n ? 'cursor-pointer transition-colors hover:bg-muted/50 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring'\n : null,\n )}\n role={isCompanyInteractive ? 'link' : undefined}\n tabIndex={isCompanyInteractive ? 0 : undefined}\n onClick={handleCompanyClick}\n onKeyDown={handleCompanyKeyDown}\n >\n <div className=\"flex items-start justify-between gap-3\">\n <div className=\"flex-1 space-y-1\">\n <p className=\"flex items-center gap-2 text-xs uppercase tracking-wide text-muted-foreground\">\n <Building2 aria-hidden className=\"h-3.5 w-3.5\" />\n {t('customers.people.detail.company.label', 'Company')}\n </p>\n {editingCompany ? (\n <div\n className=\"space-y-3\"\n onKeyDown={(event) => {\n if (event.key === 'Escape') {\n event.preventDefault()\n if (!companySaving) {\n setEditingCompany(false)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }\n }\n if (event.key === 'Enter' && (event.metaKey || event.ctrlKey)) {\n event.preventDefault()\n if (!companySaving) {\n void handleCompanySave()\n }\n }\n }}\n >\n <CompanySelectField\n value={companyDraftId || undefined}\n onChange={(next) => setCompanyDraftId(next ?? '')}\n labels={{\n placeholder: t('customers.people.form.company.placeholder'),\n addLabel: t('customers.people.form.company.add'),\n addPrompt: t('customers.people.form.company.prompt'),\n dialogTitle: t('customers.people.form.company.dialogTitle'),\n inputLabel: t('customers.people.form.company.inputLabel'),\n inputPlaceholder: t('customers.people.form.company.inputPlaceholder'),\n emptyError: t('customers.people.form.dictionary.errorRequired'),\n cancelLabel: t('customers.people.form.dictionary.cancel'),\n saveLabel: t('customers.people.form.dictionary.save'),\n errorLoad: t('customers.people.form.dictionary.errorLoad'),\n errorSave: t('customers.people.form.dictionary.error'),\n loadingLabel: t('customers.people.form.company.loading'),\n }}\n />\n {companyError ? <p className=\"text-xs text-red-600\">{companyError}</p> : null}\n <div className=\"flex flex-wrap items-center gap-2\">\n <Button type=\"button\" size=\"sm\" onClick={handleCompanySave} disabled={companySaving}>\n {companySaving ? <Loader2 className=\"mr-2 h-3.5 w-3.5 animate-spin\" /> : null}\n {t('ui.forms.actions.save')}\n </Button>\n <Button type=\"button\" size=\"sm\" variant=\"ghost\" onClick={() => {\n setEditingCompany(false)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }} disabled={companySaving}>\n {t('ui.forms.actions.cancel')}\n </Button>\n <Button type=\"button\" size=\"sm\" variant=\"secondary\" onClick={handleCompanyClear} disabled={companySaving}>\n {t('ui.forms.actions.clear')}\n </Button>\n </div>\n </div>\n ) : (\n <div className=\"flex items-center gap-2 text-sm\">\n {companyLoading ? (\n <span className=\"inline-flex items-center gap-2 text-muted-foreground\">\n <Loader2 className=\"h-4 w-4 animate-spin\" />\n {t('customers.people.detail.company.loading', 'Loading company\u2026')}\n </span>\n ) : company ? (\n <span className=\"text-primary transition group-hover:underline\">\n {t('customers.people.detail.company.current', undefined, { company: company.name })}\n </span>\n ) : companyError ? (\n <span className=\"text-xs text-red-600\">{companyError}</span>\n ) : (\n <span className=\"text-muted-foreground\">\n {t('customers.people.detail.company.empty', 'No company assigned')}\n </span>\n )}\n </div>\n )}\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => {\n if (companySaving) return\n setEditingCompany((prev) => !prev)\n setCompanyError(null)\n setCompanyDraftId(activeCompanyId ?? '')\n }}\n className={\n editingCompany\n ? 'opacity-100'\n : 'opacity-0 transition-opacity duration-150 group-hover:opacity-100 hover:opacity-100 focus-visible:opacity-100'\n }\n >\n {editingCompany ? <X className=\"h-4 w-4\" /> : <Pencil className=\"h-4 w-4\" />}\n <span className=\"sr-only\">\n {editingCompany ? t('ui.forms.actions.cancel') : t('ui.forms.actions.edit')}\n </span>\n </Button>\n </div>\n </div>\n )\n\n return (\n <div className=\"space-y-6\">\n <FormHeader\n mode=\"detail\"\n backHref=\"/backend/customers/people\"\n backLabel={t('customers.people.detail.actions.backToList')}\n utilityActions={(\n <>\n {utilityActions}\n <VersionHistoryAction\n config={{\n resourceKind: 'customers.person',\n resourceId: person.id,\n resourceIdFallback: historyFallbackId,\n organizationId: person.organizationId ?? undefined,\n }}\n t={t}\n />\n </>\n )}\n title={\n <InlineTextEditor\n label={t('customers.people.form.displayName.label')}\n value={person.displayName}\n placeholder={t('customers.people.form.displayName.placeholder')}\n emptyLabel={t('customers.people.detail.noValue')}\n validator={validators.displayName}\n onSave={onDisplayNameSave}\n hideLabel\n variant=\"plain\"\n activateOnClick\n triggerClassName=\"opacity-0 transition-opacity duration-150 group-hover:opacity-100 group-focus-within:opacity-100 focus-visible:opacity-100\"\n containerClassName=\"max-w-full\"\n />\n }\n onDelete={() => {\n onDelete()\n }}\n isDeleting={isDeleting}\n deleteLabel={t('customers.people.list.actions.delete')}\n />\n\n {companyPanel}\n\n <div className=\"grid gap-4 md:grid-cols-2 xl:grid-cols-4\">\n <InlineTextEditor\n label={t('customers.people.detail.highlights.primaryEmail')}\n value={person.primaryEmail || ''}\n placeholder={t('customers.people.form.primaryEmail')}\n emptyLabel={t('customers.people.detail.noValue')}\n type=\"email\"\n validator={validators.email}\n recordId={person.id}\n activateOnClick\n onSave={onPrimaryEmailSave}\n />\n <InlineTextEditor\n label={t('customers.people.detail.highlights.primaryPhone')}\n value={person.primaryPhone || ''}\n placeholder={t('customers.people.form.primaryPhone')}\n emptyLabel={t('customers.people.detail.noValue')}\n type=\"tel\"\n validator={validators.phone}\n recordId={person.id}\n activateOnClick\n onSave={onPrimaryPhoneSave}\n />\n <InlineDictionaryEditor\n label={t('customers.people.detail.highlights.status')}\n value={person.status ?? null}\n emptyLabel={t('customers.people.detail.noValue')}\n activateOnClick\n onSave={onStatusSave}\n kind=\"statuses\"\n />\n <InlineNextInteractionEditor\n label={t('customers.people.detail.highlights.nextInteraction')}\n valueAt={person.nextInteractionAt || null}\n valueName={person.nextInteractionName || null}\n valueRefId={person.nextInteractionRefId || null}\n valueIcon={person.nextInteractionIcon || null}\n valueColor={person.nextInteractionColor || null}\n emptyLabel={t('customers.people.detail.noValue')}\n onSave={onNextInteractionSave}\n activateOnClick\n />\n </div>\n </div>\n )\n}\n\nexport default PersonHighlights\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -0,0 +1,230 @@
1
+ import {
2
+ createContactPayloadSchema,
3
+ linkContactPayloadSchema,
4
+ logActivityPayloadSchema,
5
+ draftReplyPayloadSchema
6
+ } from "../inbox_ops/data/validators.js";
7
+ import {
8
+ asHelperContext,
9
+ ExecutionError,
10
+ executeCommand,
11
+ resolveEntityClass,
12
+ resolveCustomerEntityIdByEmail,
13
+ resolveContactIdByNameAndType
14
+ } from "../inbox_ops/lib/executionHelpers.js";
15
+ import { splitPersonName } from "../inbox_ops/lib/contactValidation.js";
16
+ import { findOneWithDecryption, findWithDecryption } from "@open-mercato/shared/lib/encryption/find";
17
+ async function executeCreateContactAction(action, ctx) {
18
+ const hCtx = asHelperContext(ctx);
19
+ const payload = action.payload;
20
+ const CustomerEntityClass = resolveEntityClass(hCtx, "CustomerEntity");
21
+ if (payload.email && CustomerEntityClass) {
22
+ const emailLower = payload.email.trim().toLowerCase();
23
+ let existingContact = await findOneWithDecryption(
24
+ hCtx.em,
25
+ CustomerEntityClass,
26
+ {
27
+ primaryEmail: emailLower,
28
+ tenantId: hCtx.tenantId,
29
+ organizationId: hCtx.organizationId,
30
+ deletedAt: null
31
+ },
32
+ void 0,
33
+ { tenantId: hCtx.tenantId, organizationId: hCtx.organizationId }
34
+ );
35
+ if (!existingContact) {
36
+ const candidates = await findWithDecryption(
37
+ hCtx.em,
38
+ CustomerEntityClass,
39
+ {
40
+ tenantId: hCtx.tenantId,
41
+ organizationId: hCtx.organizationId,
42
+ deletedAt: null
43
+ },
44
+ { limit: 100, orderBy: { createdAt: "DESC" } },
45
+ { tenantId: hCtx.tenantId, organizationId: hCtx.organizationId }
46
+ );
47
+ existingContact = candidates.find(
48
+ (e) => e.primaryEmail && e.primaryEmail.toLowerCase() === emailLower
49
+ ) ?? null;
50
+ }
51
+ if (existingContact) {
52
+ const isCompany = existingContact.kind === "company";
53
+ return {
54
+ createdEntityId: existingContact.id,
55
+ createdEntityType: isCompany ? "customer_company" : "customer_person",
56
+ matchedEntityId: existingContact.id,
57
+ matchedEntityType: isCompany ? "company" : "person"
58
+ };
59
+ }
60
+ }
61
+ if (payload.type === "company") {
62
+ const result2 = await executeCommand(
63
+ hCtx,
64
+ "customers.companies.create",
65
+ {
66
+ organizationId: hCtx.organizationId,
67
+ tenantId: hCtx.tenantId,
68
+ displayName: payload.name,
69
+ legalName: payload.companyName ?? payload.name,
70
+ primaryEmail: payload.email,
71
+ primaryPhone: payload.phone,
72
+ source: payload.source
73
+ }
74
+ );
75
+ if (!result2.entityId) {
76
+ throw new ExecutionError("Company creation did not return an entity ID", 500);
77
+ }
78
+ return { createdEntityId: result2.entityId, createdEntityType: "customer_company" };
79
+ }
80
+ const { firstName, lastName } = splitPersonName(payload.name, payload.email);
81
+ const result = await executeCommand(
82
+ hCtx,
83
+ "customers.people.create",
84
+ {
85
+ organizationId: hCtx.organizationId,
86
+ tenantId: hCtx.tenantId,
87
+ displayName: payload.name,
88
+ firstName,
89
+ lastName,
90
+ primaryEmail: payload.email,
91
+ primaryPhone: payload.phone,
92
+ jobTitle: payload.role,
93
+ source: payload.source
94
+ }
95
+ );
96
+ if (!result.entityId) {
97
+ throw new ExecutionError("Person creation did not return an entity ID", 500);
98
+ }
99
+ return { createdEntityId: result.entityId, createdEntityType: "customer_person" };
100
+ }
101
+ function executeLinkContactAction(action) {
102
+ const payload = action.payload;
103
+ return {
104
+ createdEntityId: payload.contactId,
105
+ createdEntityType: payload.contactType === "company" ? "customer_company" : "customer_person",
106
+ matchedEntityId: payload.contactId,
107
+ matchedEntityType: payload.contactType
108
+ };
109
+ }
110
+ async function executeLogActivityAction(action, ctx) {
111
+ const hCtx = asHelperContext(ctx);
112
+ let payload = action.payload;
113
+ if (!payload.contactId) {
114
+ const resolved = await resolveContactIdByNameAndType(hCtx, payload.contactName, payload.contactType);
115
+ if (resolved) {
116
+ payload = { ...payload, contactId: resolved };
117
+ } else {
118
+ throw new ExecutionError(
119
+ `log_activity requires contactId \u2014 could not resolve contact "${payload.contactName}" (${payload.contactType})`,
120
+ 400
121
+ );
122
+ }
123
+ }
124
+ const result = await executeCommand(
125
+ hCtx,
126
+ "customers.activities.create",
127
+ {
128
+ organizationId: hCtx.organizationId,
129
+ tenantId: hCtx.tenantId,
130
+ entityId: payload.contactId,
131
+ activityType: payload.activityType,
132
+ subject: payload.subject,
133
+ body: payload.body,
134
+ authorUserId: hCtx.userId
135
+ }
136
+ );
137
+ if (!result.activityId) {
138
+ throw new ExecutionError("Activity creation did not return an activity ID", 500);
139
+ }
140
+ return { createdEntityId: result.activityId, createdEntityType: "customer_activity" };
141
+ }
142
+ async function executeDraftReplyAction(action, ctx) {
143
+ const hCtx = asHelperContext(ctx);
144
+ const payload = action.payload;
145
+ const payloadRecord = action.payload;
146
+ const explicitContactId = typeof payloadRecord.contactId === "string" ? payloadRecord.contactId : null;
147
+ const contactId = explicitContactId ?? await resolveCustomerEntityIdByEmail(hCtx, payload.to);
148
+ if (!contactId) {
149
+ throw new ExecutionError(
150
+ `No matching contact found for "${payload.to}". Create the contact first or link an existing one.`,
151
+ 400
152
+ );
153
+ }
154
+ const details = [
155
+ payload.body.trim(),
156
+ "",
157
+ "---",
158
+ `Draft reply target: ${payload.to}`,
159
+ `Subject: ${payload.subject}`,
160
+ payload.context ? `Context: ${payload.context}` : null,
161
+ `InboxOps Proposal: ${action.proposalId}`,
162
+ `InboxOps Action: ${action.id}`
163
+ ].filter((line) => typeof line === "string" && line.length > 0).join("\n");
164
+ const result = await executeCommand(
165
+ hCtx,
166
+ "customers.activities.create",
167
+ {
168
+ organizationId: hCtx.organizationId,
169
+ tenantId: hCtx.tenantId,
170
+ entityId: contactId,
171
+ activityType: "email",
172
+ subject: payload.subject,
173
+ body: details,
174
+ authorUserId: hCtx.userId
175
+ }
176
+ );
177
+ if (!result.activityId) {
178
+ throw new ExecutionError("Draft reply activity did not return an activity ID", 500);
179
+ }
180
+ return { createdEntityId: result.activityId, createdEntityType: "customer_activity" };
181
+ }
182
+ const inboxActions = [
183
+ {
184
+ type: "create_contact",
185
+ requiredFeature: "customers.people.manage",
186
+ payloadSchema: createContactPayloadSchema,
187
+ label: "Create Contact",
188
+ promptSchema: `create_contact payload:
189
+ { type: "person"|"company", name: string, email?: string, phone?: string, companyName?: string, role?: string, source: "inbox_ops" }`,
190
+ promptRules: [
191
+ 'For create_contact: always include email when available from the thread. Set source to "inbox_ops", type must be lowercase "person" or "company".',
192
+ `For create_contact with type "person": if the sender's display name is not available in the email header or signature, attempt to derive a human-readable name from the email address (e.g., john.doe@company.com -> "John Doe", m.smith@corp.net -> "M Smith"). If the email address does not contain a derivable name (e.g., info@, noreply@), use the full email address as the name. Always aim to provide both a first and last name when possible.`
193
+ ],
194
+ execute: executeCreateContactAction
195
+ },
196
+ {
197
+ type: "link_contact",
198
+ requiredFeature: "customers.people.manage",
199
+ payloadSchema: linkContactPayloadSchema,
200
+ label: "Link Contact",
201
+ promptSchema: `link_contact payload:
202
+ { emailAddress: string (email), contactId: uuid, contactType: "person"|"company", contactName: string }`,
203
+ execute: (action) => Promise.resolve(executeLinkContactAction(action))
204
+ },
205
+ {
206
+ type: "log_activity",
207
+ requiredFeature: "customers.activities.manage",
208
+ payloadSchema: logActivityPayloadSchema,
209
+ label: "Log Activity",
210
+ promptSchema: `log_activity payload:
211
+ { contactId?: uuid, contactType: "person"|"company", contactName: string, activityType: "email"|"call"|"meeting"|"note", subject: string, body: string }`,
212
+ execute: executeLogActivityAction
213
+ },
214
+ {
215
+ type: "draft_reply",
216
+ requiredFeature: "inbox_ops.replies.send",
217
+ payloadSchema: draftReplyPayloadSchema,
218
+ label: "Draft Reply",
219
+ promptSchema: `draft_reply payload:
220
+ { to: string (email), toName?: string, subject: string, body: string, context?: string }`,
221
+ promptRules: ["For draft_reply: include ERP context when available."],
222
+ execute: executeDraftReplyAction
223
+ }
224
+ ];
225
+ var inbox_actions_default = inboxActions;
226
+ export {
227
+ inbox_actions_default as default,
228
+ inboxActions
229
+ };
230
+ //# sourceMappingURL=inbox-actions.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/modules/customers/inbox-actions.ts"],
4
+ "sourcesContent": ["import type { InboxActionDefinition, InboxActionExecutionContext } from '@open-mercato/shared/modules/inbox-actions'\nimport {\n createContactPayloadSchema,\n linkContactPayloadSchema,\n logActivityPayloadSchema,\n draftReplyPayloadSchema,\n} from '../inbox_ops/data/validators'\nimport type {\n CreateContactPayload,\n LinkContactPayload,\n LogActivityPayload,\n DraftReplyPayload,\n} from '../inbox_ops/data/validators'\nimport {\n asHelperContext,\n ExecutionError,\n executeCommand,\n resolveEntityClass,\n resolveCustomerEntityIdByEmail,\n resolveContactIdByNameAndType,\n} from '../inbox_ops/lib/executionHelpers'\nimport { splitPersonName } from '../inbox_ops/lib/contactValidation'\nimport { findOneWithDecryption, findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\n\n// ---------------------------------------------------------------------------\n// create_contact\n// ---------------------------------------------------------------------------\n\nasync function executeCreateContactAction(\n action: { id: string; proposalId: string; payload: unknown },\n ctx: InboxActionExecutionContext,\n): Promise<{ createdEntityId?: string | null; createdEntityType?: string | null; matchedEntityId?: string | null; matchedEntityType?: string | null }> {\n const hCtx = asHelperContext(ctx)\n const payload = action.payload as CreateContactPayload\n\n const CustomerEntityClass = resolveEntityClass(hCtx, 'CustomerEntity')\n if (payload.email && CustomerEntityClass) {\n const emailLower = payload.email.trim().toLowerCase()\n let existingContact = await findOneWithDecryption(\n hCtx.em,\n CustomerEntityClass,\n {\n primaryEmail: emailLower,\n tenantId: hCtx.tenantId,\n organizationId: hCtx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: hCtx.tenantId, organizationId: hCtx.organizationId },\n )\n if (!existingContact) {\n const candidates = await findWithDecryption(\n hCtx.em,\n CustomerEntityClass,\n {\n tenantId: hCtx.tenantId,\n organizationId: hCtx.organizationId,\n deletedAt: null,\n },\n { limit: 100, orderBy: { createdAt: 'DESC' } },\n { tenantId: hCtx.tenantId, organizationId: hCtx.organizationId },\n )\n existingContact = candidates.find(\n (e) => e.primaryEmail && e.primaryEmail.toLowerCase() === emailLower,\n ) ?? null\n }\n if (existingContact) {\n const isCompany = existingContact.kind === 'company'\n return {\n createdEntityId: existingContact.id,\n createdEntityType: isCompany ? 'customer_company' : 'customer_person',\n matchedEntityId: existingContact.id,\n matchedEntityType: isCompany ? 'company' : 'person',\n }\n }\n }\n\n if (payload.type === 'company') {\n const result = await executeCommand<Record<string, unknown>, { entityId?: string }>(\n hCtx,\n 'customers.companies.create',\n {\n organizationId: hCtx.organizationId,\n tenantId: hCtx.tenantId,\n displayName: payload.name,\n legalName: payload.companyName ?? payload.name,\n primaryEmail: payload.email,\n primaryPhone: payload.phone,\n source: payload.source,\n },\n )\n if (!result.entityId) {\n throw new ExecutionError('Company creation did not return an entity ID', 500)\n }\n return { createdEntityId: result.entityId, createdEntityType: 'customer_company' }\n }\n\n const { firstName, lastName } = splitPersonName(payload.name, payload.email)\n const result = await executeCommand<Record<string, unknown>, { entityId?: string }>(\n hCtx,\n 'customers.people.create',\n {\n organizationId: hCtx.organizationId,\n tenantId: hCtx.tenantId,\n displayName: payload.name,\n firstName,\n lastName,\n primaryEmail: payload.email,\n primaryPhone: payload.phone,\n jobTitle: payload.role,\n source: payload.source,\n },\n )\n\n if (!result.entityId) {\n throw new ExecutionError('Person creation did not return an entity ID', 500)\n }\n\n return { createdEntityId: result.entityId, createdEntityType: 'customer_person' }\n}\n\n// ---------------------------------------------------------------------------\n// link_contact\n// ---------------------------------------------------------------------------\n\nfunction executeLinkContactAction(\n action: { id: string; proposalId: string; payload: unknown },\n): { createdEntityId?: string | null; createdEntityType?: string | null; matchedEntityId?: string | null; matchedEntityType?: string | null } {\n const payload = action.payload as LinkContactPayload\n return {\n createdEntityId: payload.contactId,\n createdEntityType: payload.contactType === 'company' ? 'customer_company' : 'customer_person',\n matchedEntityId: payload.contactId,\n matchedEntityType: payload.contactType,\n }\n}\n\n// ---------------------------------------------------------------------------\n// log_activity\n// ---------------------------------------------------------------------------\n\nasync function executeLogActivityAction(\n action: { id: string; proposalId: string; payload: unknown },\n ctx: InboxActionExecutionContext,\n): Promise<{ createdEntityId?: string | null; createdEntityType?: string | null }> {\n const hCtx = asHelperContext(ctx)\n let payload = action.payload as LogActivityPayload\n\n if (!payload.contactId) {\n const resolved = await resolveContactIdByNameAndType(hCtx, payload.contactName, payload.contactType)\n if (resolved) {\n payload = { ...payload, contactId: resolved }\n } else {\n throw new ExecutionError(\n `log_activity requires contactId \u2014 could not resolve contact \"${payload.contactName}\" (${payload.contactType})`,\n 400,\n )\n }\n }\n\n const result = await executeCommand<Record<string, unknown>, { activityId?: string }>(\n hCtx,\n 'customers.activities.create',\n {\n organizationId: hCtx.organizationId,\n tenantId: hCtx.tenantId,\n entityId: payload.contactId,\n activityType: payload.activityType,\n subject: payload.subject,\n body: payload.body,\n authorUserId: hCtx.userId,\n },\n )\n\n if (!result.activityId) {\n throw new ExecutionError('Activity creation did not return an activity ID', 500)\n }\n\n return { createdEntityId: result.activityId, createdEntityType: 'customer_activity' }\n}\n\n// ---------------------------------------------------------------------------\n// draft_reply\n// ---------------------------------------------------------------------------\n\nasync function executeDraftReplyAction(\n action: { id: string; proposalId: string; payload: unknown },\n ctx: InboxActionExecutionContext,\n): Promise<{ createdEntityId?: string | null; createdEntityType?: string | null }> {\n const hCtx = asHelperContext(ctx)\n const payload = action.payload as DraftReplyPayload\n const payloadRecord = action.payload as Record<string, unknown>\n const explicitContactId = typeof payloadRecord.contactId === 'string' ? payloadRecord.contactId : null\n const contactId = explicitContactId ?? (await resolveCustomerEntityIdByEmail(hCtx, payload.to))\n\n if (!contactId) {\n throw new ExecutionError(\n `No matching contact found for \"${payload.to}\". Create the contact first or link an existing one.`,\n 400,\n )\n }\n\n const details = [\n payload.body.trim(),\n '',\n '---',\n `Draft reply target: ${payload.to}`,\n `Subject: ${payload.subject}`,\n payload.context ? `Context: ${payload.context}` : null,\n `InboxOps Proposal: ${action.proposalId}`,\n `InboxOps Action: ${action.id}`,\n ]\n .filter((line) => typeof line === 'string' && line.length > 0)\n .join('\\n')\n\n const result = await executeCommand<Record<string, unknown>, { activityId?: string }>(\n hCtx,\n 'customers.activities.create',\n {\n organizationId: hCtx.organizationId,\n tenantId: hCtx.tenantId,\n entityId: contactId,\n activityType: 'email',\n subject: payload.subject,\n body: details,\n authorUserId: hCtx.userId,\n },\n )\n\n if (!result.activityId) {\n throw new ExecutionError('Draft reply activity did not return an activity ID', 500)\n }\n\n return { createdEntityId: result.activityId, createdEntityType: 'customer_activity' }\n}\n\n// ---------------------------------------------------------------------------\n// Exported action definitions\n// ---------------------------------------------------------------------------\n\nexport const inboxActions: InboxActionDefinition[] = [\n {\n type: 'create_contact',\n requiredFeature: 'customers.people.manage',\n payloadSchema: createContactPayloadSchema,\n label: 'Create Contact',\n promptSchema: `create_contact payload:\n{ type: \"person\"|\"company\", name: string, email?: string, phone?: string, companyName?: string, role?: string, source: \"inbox_ops\" }`,\n promptRules: [\n 'For create_contact: always include email when available from the thread. Set source to \"inbox_ops\", type must be lowercase \"person\" or \"company\".',\n 'For create_contact with type \"person\": if the sender\\'s display name is not available in the email header or signature, attempt to derive a human-readable name from the email address (e.g., john.doe@company.com -> \"John Doe\", m.smith@corp.net -> \"M Smith\"). If the email address does not contain a derivable name (e.g., info@, noreply@), use the full email address as the name. Always aim to provide both a first and last name when possible.',\n ],\n execute: executeCreateContactAction,\n },\n {\n type: 'link_contact',\n requiredFeature: 'customers.people.manage',\n payloadSchema: linkContactPayloadSchema,\n label: 'Link Contact',\n promptSchema: `link_contact payload:\n{ emailAddress: string (email), contactId: uuid, contactType: \"person\"|\"company\", contactName: string }`,\n execute: (action) => Promise.resolve(executeLinkContactAction(action)),\n },\n {\n type: 'log_activity',\n requiredFeature: 'customers.activities.manage',\n payloadSchema: logActivityPayloadSchema,\n label: 'Log Activity',\n promptSchema: `log_activity payload:\n{ contactId?: uuid, contactType: \"person\"|\"company\", contactName: string, activityType: \"email\"|\"call\"|\"meeting\"|\"note\", subject: string, body: string }`,\n execute: executeLogActivityAction,\n },\n {\n type: 'draft_reply',\n requiredFeature: 'inbox_ops.replies.send',\n payloadSchema: draftReplyPayloadSchema,\n label: 'Draft Reply',\n promptSchema: `draft_reply payload:\n{ to: string (email), toName?: string, subject: string, body: string, context?: string }`,\n promptRules: ['For draft_reply: include ERP context when available.'],\n execute: executeDraftReplyAction,\n },\n]\n\nexport default inboxActions\n"],
5
+ "mappings": "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",
6
+ "names": ["result"]
7
+ }
@@ -1,7 +1,7 @@
1
1
  import { createRequestContainer } from "@open-mercato/shared/lib/di/container";
2
2
  import { findOneWithDecryption } from "@open-mercato/shared/lib/encryption/find";
3
3
  import { resolveTranslations } from "@open-mercato/shared/lib/i18n/server";
4
- import { CustomerDeal, CustomerEntity } from "../data/entities.js";
4
+ import { CustomerCompanyProfile, CustomerDeal, CustomerEntity, CustomerPersonProfile } from "../data/entities.js";
5
5
  function formatCurrency(amount, currency) {
6
6
  if (!amount) return null;
7
7
  const value = Number(amount);
@@ -48,12 +48,29 @@ async function loadCustomerPersonPreview(entityId, ctx) {
48
48
  if (!entity) {
49
49
  return { title: defaultTitle, subtitle: entityId, status: t("customers.messageObjects.notFound"), statusColor: "gray" };
50
50
  }
51
+ const profile = await findOneWithDecryption(
52
+ em,
53
+ CustomerPersonProfile,
54
+ {
55
+ entity,
56
+ tenantId: ctx.tenantId,
57
+ organizationId: ctx.organizationId
58
+ },
59
+ void 0,
60
+ { tenantId: ctx.tenantId, organizationId: ctx.organizationId }
61
+ );
51
62
  const subtitleParts = [entity.primaryEmail, entity.primaryPhone].filter((part) => Boolean(part && part.trim().length > 0));
63
+ const metadata = {};
64
+ const phoneLabel = t("customers.people.detail.highlights.primaryPhone");
65
+ const jobTitleLabel = t("customers.people.detail.fields.jobTitle");
66
+ if (entity.primaryPhone && entity.primaryPhone.trim().length > 0) metadata[phoneLabel] = entity.primaryPhone;
67
+ if (profile?.jobTitle && profile.jobTitle.trim().length > 0) metadata[jobTitleLabel] = profile.jobTitle;
52
68
  return {
53
69
  title: entity.displayName,
54
70
  subtitle: subtitleParts.join(" \u2022 ") || entityId,
55
71
  status: entity.status ?? void 0,
56
- statusColor: statusColor(entity.status)
72
+ statusColor: statusColor(entity.status),
73
+ metadata: Object.keys(metadata).length > 0 ? metadata : void 0
57
74
  };
58
75
  }
59
76
  async function loadCustomerCompanyPreview(entityId, ctx) {
@@ -79,12 +96,29 @@ async function loadCustomerCompanyPreview(entityId, ctx) {
79
96
  if (!entity) {
80
97
  return { title: defaultTitle, subtitle: entityId, status: t("customers.messageObjects.notFound"), statusColor: "gray" };
81
98
  }
99
+ const profile = await findOneWithDecryption(
100
+ em,
101
+ CustomerCompanyProfile,
102
+ {
103
+ entity,
104
+ tenantId: ctx.tenantId,
105
+ organizationId: ctx.organizationId
106
+ },
107
+ void 0,
108
+ { tenantId: ctx.tenantId, organizationId: ctx.organizationId }
109
+ );
82
110
  const subtitleParts = [entity.primaryEmail, entity.primaryPhone].filter((part) => Boolean(part && part.trim().length > 0));
111
+ const metadata = {};
112
+ const phoneLabel = t("customers.companies.detail.highlights.primaryPhone");
113
+ const industryLabel = t("customers.companies.detail.fields.industry");
114
+ if (entity.primaryPhone && entity.primaryPhone.trim().length > 0) metadata[phoneLabel] = entity.primaryPhone;
115
+ if (profile?.industry && profile.industry.trim().length > 0) metadata[industryLabel] = profile.industry;
83
116
  return {
84
117
  title: entity.displayName,
85
118
  subtitle: subtitleParts.join(" \u2022 ") || entityId,
86
119
  status: entity.status ?? void 0,
87
- statusColor: statusColor(entity.status)
120
+ statusColor: statusColor(entity.status),
121
+ metadata: Object.keys(metadata).length > 0 ? metadata : void 0
88
122
  };
89
123
  }
90
124
  async function loadCustomerDealPreview(entityId, ctx) {
@@ -113,8 +147,10 @@ async function loadCustomerDealPreview(entityId, ctx) {
113
147
  const probability = typeof deal.probability === "number" ? `${deal.probability}%` : null;
114
148
  const subtitle = [amount, probability].filter((part) => Boolean(part && part.length > 0)).join(" \u2022 ");
115
149
  const metadata = {};
116
- if (amount) metadata.Value = amount;
117
- if (probability) metadata.Probability = probability;
150
+ const valueLabel = t("customers.deals.detail.fields.value");
151
+ const probabilityLabel = t("customers.deals.detail.fields.probability");
152
+ if (amount) metadata[valueLabel] = amount;
153
+ if (probability) metadata[probabilityLabel] = probability;
118
154
  return {
119
155
  title: deal.title,
120
156
  subtitle: subtitle || entityId,