@open-mercato/core 0.5.1-develop.2663.2c29774b5b → 0.5.1-develop.2681.c559bb2bc3

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 (687) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/dist/generated/entities/action_log/index.js +8 -0
  3. package/dist/generated/entities/action_log/index.js.map +2 -2
  4. package/dist/generated/entities/customer_company_billing/index.js +23 -0
  5. package/dist/generated/entities/customer_company_billing/index.js.map +7 -0
  6. package/dist/generated/entities/customer_deal/index.js +8 -0
  7. package/dist/generated/entities/customer_deal/index.js.map +2 -2
  8. package/dist/generated/entities/customer_deal_stage_transition/index.js +31 -0
  9. package/dist/generated/entities/customer_deal_stage_transition/index.js.map +7 -0
  10. package/dist/generated/entities/customer_dictionary_kind_setting/index.js +21 -0
  11. package/dist/generated/entities/customer_dictionary_kind_setting/index.js.map +7 -0
  12. package/dist/generated/entities/customer_entity/index.js +8 -0
  13. package/dist/generated/entities/customer_entity/index.js.map +2 -2
  14. package/dist/generated/entities/customer_entity_role/index.js +23 -0
  15. package/dist/generated/entities/customer_entity_role/index.js.map +7 -0
  16. package/dist/generated/entities/customer_interaction/index.js +23 -1
  17. package/dist/generated/entities/customer_interaction/index.js.map +2 -2
  18. package/dist/generated/entities/customer_label/index.js +19 -0
  19. package/dist/generated/entities/customer_label/index.js.map +7 -0
  20. package/dist/generated/entities/customer_label_assignment/index.js +17 -0
  21. package/dist/generated/entities/customer_label_assignment/index.js.map +7 -0
  22. package/dist/generated/entities/customer_person_company_link/index.js +21 -0
  23. package/dist/generated/entities/customer_person_company_link/index.js.map +7 -0
  24. package/dist/generated/entities/customer_person_company_role/index.js +17 -0
  25. package/dist/generated/entities/customer_person_company_role/index.js.map +7 -0
  26. package/dist/generated/entities/dictionary_entry/index.js +4 -0
  27. package/dist/generated/entities/dictionary_entry/index.js.map +2 -2
  28. package/dist/generated/entities.ids.generated.js +9 -1
  29. package/dist/generated/entities.ids.generated.js.map +2 -2
  30. package/dist/generated/entity-fields-registry.js +116 -1
  31. package/dist/generated/entity-fields-registry.js.map +2 -2
  32. package/dist/modules/attachments/api/route.js +46 -8
  33. package/dist/modules/attachments/api/route.js.map +2 -2
  34. package/dist/modules/audit_logs/api/audit-logs/actions/export/route.js +208 -0
  35. package/dist/modules/audit_logs/api/audit-logs/actions/export/route.js.map +7 -0
  36. package/dist/modules/audit_logs/api/audit-logs/actions/route.js +52 -6
  37. package/dist/modules/audit_logs/api/audit-logs/actions/route.js.map +2 -2
  38. package/dist/modules/audit_logs/cli.js +62 -0
  39. package/dist/modules/audit_logs/cli.js.map +7 -0
  40. package/dist/modules/audit_logs/data/entities.js +21 -1
  41. package/dist/modules/audit_logs/data/entities.js.map +2 -2
  42. package/dist/modules/audit_logs/data/validators.js +9 -1
  43. package/dist/modules/audit_logs/data/validators.js.map +2 -2
  44. package/dist/modules/audit_logs/lib/changeRows.js +34 -0
  45. package/dist/modules/audit_logs/lib/changeRows.js.map +7 -0
  46. package/dist/modules/audit_logs/lib/display-helpers.js +2 -20
  47. package/dist/modules/audit_logs/lib/display-helpers.js.map +3 -3
  48. package/dist/modules/audit_logs/lib/projections.js +58 -0
  49. package/dist/modules/audit_logs/lib/projections.js.map +7 -0
  50. package/dist/modules/audit_logs/migrations/Migration20260412160533.js +21 -0
  51. package/dist/modules/audit_logs/migrations/Migration20260412160533.js.map +7 -0
  52. package/dist/modules/audit_logs/services/actionLogService.js +313 -79
  53. package/dist/modules/audit_logs/services/actionLogService.js.map +2 -2
  54. package/dist/modules/customers/acl.js +3 -1
  55. package/dist/modules/customers/acl.js.map +2 -2
  56. package/dist/modules/customers/api/activities/route.js +4 -0
  57. package/dist/modules/customers/api/activities/route.js.map +2 -2
  58. package/dist/modules/customers/api/assignable-staff/route.js +208 -0
  59. package/dist/modules/customers/api/assignable-staff/route.js.map +7 -0
  60. package/dist/modules/customers/api/companies/[id]/people/route.js +205 -0
  61. package/dist/modules/customers/api/companies/[id]/people/route.js.map +7 -0
  62. package/dist/modules/customers/api/companies/[id]/roles/route.js +22 -0
  63. package/dist/modules/customers/api/companies/[id]/roles/route.js.map +7 -0
  64. package/dist/modules/customers/api/companies/[id]/route.js +374 -32
  65. package/dist/modules/customers/api/companies/[id]/route.js.map +2 -2
  66. package/dist/modules/customers/api/companies/route.js +82 -7
  67. package/dist/modules/customers/api/companies/route.js.map +2 -2
  68. package/dist/modules/customers/api/deals/[id]/companies/route.js +172 -0
  69. package/dist/modules/customers/api/deals/[id]/companies/route.js.map +7 -0
  70. package/dist/modules/customers/api/deals/[id]/people/route.js +156 -0
  71. package/dist/modules/customers/api/deals/[id]/people/route.js.map +7 -0
  72. package/dist/modules/customers/api/deals/[id]/route.js +459 -53
  73. package/dist/modules/customers/api/deals/[id]/route.js.map +2 -2
  74. package/dist/modules/customers/api/deals/[id]/stats/route.js +195 -0
  75. package/dist/modules/customers/api/deals/[id]/stats/route.js.map +7 -0
  76. package/dist/modules/customers/api/deals/route.js +20 -10
  77. package/dist/modules/customers/api/deals/route.js.map +3 -3
  78. package/dist/modules/customers/api/dictionaries/[kind]/[id]/route.js +105 -4
  79. package/dist/modules/customers/api/dictionaries/[kind]/[id]/route.js.map +2 -2
  80. package/dist/modules/customers/api/dictionaries/[kind]/route.js +118 -42
  81. package/dist/modules/customers/api/dictionaries/[kind]/route.js.map +2 -2
  82. package/dist/modules/customers/api/dictionaries/context.js +30 -6
  83. package/dist/modules/customers/api/dictionaries/context.js.map +2 -2
  84. package/dist/modules/customers/api/dictionaries/kind-settings/route.js +207 -0
  85. package/dist/modules/customers/api/dictionaries/kind-settings/route.js.map +7 -0
  86. package/dist/modules/customers/api/entity-roles-factory.js +471 -0
  87. package/dist/modules/customers/api/entity-roles-factory.js.map +7 -0
  88. package/dist/modules/customers/api/interactions/conflicts/route.js +158 -0
  89. package/dist/modules/customers/api/interactions/conflicts/route.js.map +7 -0
  90. package/dist/modules/customers/api/interactions/counts/route.js +92 -0
  91. package/dist/modules/customers/api/interactions/counts/route.js.map +7 -0
  92. package/dist/modules/customers/api/interactions/route.js +83 -4
  93. package/dist/modules/customers/api/interactions/route.js.map +2 -2
  94. package/dist/modules/customers/api/labels/assign/route.js +189 -0
  95. package/dist/modules/customers/api/labels/assign/route.js.map +7 -0
  96. package/dist/modules/customers/api/labels/auth.js +17 -0
  97. package/dist/modules/customers/api/labels/auth.js.map +7 -0
  98. package/dist/modules/customers/api/labels/route.js +281 -0
  99. package/dist/modules/customers/api/labels/route.js.map +7 -0
  100. package/dist/modules/customers/api/labels/table-errors.js +38 -0
  101. package/dist/modules/customers/api/labels/table-errors.js.map +7 -0
  102. package/dist/modules/customers/api/labels/unassign/route.js +184 -0
  103. package/dist/modules/customers/api/labels/unassign/route.js.map +7 -0
  104. package/dist/modules/customers/api/people/[id]/companies/[linkId]/route.js +292 -0
  105. package/dist/modules/customers/api/people/[id]/companies/[linkId]/route.js.map +7 -0
  106. package/dist/modules/customers/api/people/[id]/companies/context.js +66 -0
  107. package/dist/modules/customers/api/people/[id]/companies/context.js.map +7 -0
  108. package/dist/modules/customers/api/people/[id]/companies/enriched/route.js +334 -0
  109. package/dist/modules/customers/api/people/[id]/companies/enriched/route.js.map +7 -0
  110. package/dist/modules/customers/api/people/[id]/companies/route.js +205 -0
  111. package/dist/modules/customers/api/people/[id]/companies/route.js.map +7 -0
  112. package/dist/modules/customers/api/people/[id]/roles/route.js +22 -0
  113. package/dist/modules/customers/api/people/[id]/roles/route.js.map +7 -0
  114. package/dist/modules/customers/api/people/[id]/route.js +134 -21
  115. package/dist/modules/customers/api/people/[id]/route.js.map +2 -2
  116. package/dist/modules/customers/api/people/route.js +122 -23
  117. package/dist/modules/customers/api/people/route.js.map +2 -2
  118. package/dist/modules/customers/api/todos/route.js +4 -0
  119. package/dist/modules/customers/api/todos/route.js.map +2 -2
  120. package/dist/modules/customers/api/utils.js +22 -0
  121. package/dist/modules/customers/api/utils.js.map +2 -2
  122. package/dist/modules/customers/backend/config/customers/page.js +2 -6
  123. package/dist/modules/customers/backend/config/customers/page.js.map +2 -2
  124. package/dist/modules/customers/backend/customers/companies/page.js +37 -26
  125. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  126. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js +265 -262
  127. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js.map +3 -3
  128. package/dist/modules/customers/backend/customers/deals/[id]/hooks/formatters.js +23 -0
  129. package/dist/modules/customers/backend/customers/deals/[id]/hooks/formatters.js.map +7 -0
  130. package/dist/modules/customers/backend/customers/deals/[id]/hooks/types.js +1 -0
  131. package/dist/modules/customers/backend/customers/deals/[id]/hooks/types.js.map +7 -0
  132. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js +43 -0
  133. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js.map +7 -0
  134. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.js +264 -0
  135. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.js.map +7 -0
  136. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.js +88 -0
  137. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.js.map +7 -0
  138. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js +41 -0
  139. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js.map +7 -0
  140. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js +66 -0
  141. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js.map +7 -0
  142. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.js +39 -0
  143. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.js.map +7 -0
  144. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.js +49 -0
  145. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.js.map +7 -0
  146. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.js +43 -0
  147. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.js.map +7 -0
  148. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.js +28 -0
  149. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.js.map +7 -0
  150. package/dist/modules/customers/backend/customers/deals/[id]/page.js +556 -503
  151. package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +3 -3
  152. package/dist/modules/customers/backend/customers/deals/page.js +66 -21
  153. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  154. package/dist/modules/customers/backend/customers/people/page.js +36 -28
  155. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  156. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +318 -203
  157. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +3 -3
  158. package/dist/modules/customers/cli.js +105 -13
  159. package/dist/modules/customers/cli.js.map +2 -2
  160. package/dist/modules/customers/commands/activities.js +6 -0
  161. package/dist/modules/customers/commands/activities.js.map +2 -2
  162. package/dist/modules/customers/commands/deals.js +315 -107
  163. package/dist/modules/customers/commands/deals.js.map +2 -2
  164. package/dist/modules/customers/commands/dictionaries.js +166 -32
  165. package/dist/modules/customers/commands/dictionaries.js.map +2 -2
  166. package/dist/modules/customers/commands/dictionaryKindSettings.js +208 -0
  167. package/dist/modules/customers/commands/dictionaryKindSettings.js.map +7 -0
  168. package/dist/modules/customers/commands/entity-roles.js +415 -0
  169. package/dist/modules/customers/commands/entity-roles.js.map +7 -0
  170. package/dist/modules/customers/commands/index.js +4 -0
  171. package/dist/modules/customers/commands/index.js.map +2 -2
  172. package/dist/modules/customers/commands/interactions.js +108 -21
  173. package/dist/modules/customers/commands/interactions.js.map +2 -2
  174. package/dist/modules/customers/commands/labels.js +539 -0
  175. package/dist/modules/customers/commands/labels.js.map +7 -0
  176. package/dist/modules/customers/commands/people.js +560 -463
  177. package/dist/modules/customers/commands/people.js.map +3 -3
  178. package/dist/modules/customers/commands/personCompanyLinks.js +568 -0
  179. package/dist/modules/customers/commands/personCompanyLinks.js.map +7 -0
  180. package/dist/modules/customers/commands/shared.js +12 -4
  181. package/dist/modules/customers/commands/shared.js.map +2 -2
  182. package/dist/modules/customers/commands/todos.js +10 -1
  183. package/dist/modules/customers/commands/todos.js.map +2 -2
  184. package/dist/modules/customers/components/AddressEditor.js +1 -1
  185. package/dist/modules/customers/components/AddressEditor.js.map +2 -2
  186. package/dist/modules/customers/components/CustomersConfigurationSections.js +31 -0
  187. package/dist/modules/customers/components/CustomersConfigurationSections.js.map +7 -0
  188. package/dist/modules/customers/components/DictionarySettings.js +37 -2
  189. package/dist/modules/customers/components/DictionarySettings.js.map +2 -2
  190. package/dist/modules/customers/components/detail/ActiveDealCard.js +121 -0
  191. package/dist/modules/customers/components/detail/ActiveDealCard.js.map +7 -0
  192. package/dist/modules/customers/components/detail/ActivitiesSection.js +222 -331
  193. package/dist/modules/customers/components/detail/ActivitiesSection.js.map +3 -3
  194. package/dist/modules/customers/components/detail/ActivityAiActions.js +36 -0
  195. package/dist/modules/customers/components/detail/ActivityAiActions.js.map +7 -0
  196. package/dist/modules/customers/components/detail/ActivityCard.js +126 -0
  197. package/dist/modules/customers/components/detail/ActivityCard.js.map +7 -0
  198. package/dist/modules/customers/components/detail/ActivityHistorySection.js +340 -0
  199. package/dist/modules/customers/components/detail/ActivityHistorySection.js.map +7 -0
  200. package/dist/modules/customers/components/detail/ActivityLogTab.js +56 -0
  201. package/dist/modules/customers/components/detail/ActivityLogTab.js.map +7 -0
  202. package/dist/modules/customers/components/detail/ActivityTimeline.js +108 -0
  203. package/dist/modules/customers/components/detail/ActivityTimeline.js.map +7 -0
  204. package/dist/modules/customers/components/detail/ActivityTimelineFilters.js +139 -0
  205. package/dist/modules/customers/components/detail/ActivityTimelineFilters.js.map +7 -0
  206. package/dist/modules/customers/components/detail/ActivityTypeSelector.js +42 -0
  207. package/dist/modules/customers/components/detail/ActivityTypeSelector.js.map +7 -0
  208. package/dist/modules/customers/components/detail/AiActionChips.js +38 -0
  209. package/dist/modules/customers/components/detail/AiActionChips.js.map +7 -0
  210. package/dist/modules/customers/components/detail/AssignRoleDialog.js +534 -0
  211. package/dist/modules/customers/components/detail/AssignRoleDialog.js.map +7 -0
  212. package/dist/modules/customers/components/detail/ChangelogEntryRow.js +79 -0
  213. package/dist/modules/customers/components/detail/ChangelogEntryRow.js.map +7 -0
  214. package/dist/modules/customers/components/detail/ChangelogFilters.js +176 -0
  215. package/dist/modules/customers/components/detail/ChangelogFilters.js.map +7 -0
  216. package/dist/modules/customers/components/detail/ChangelogKpiCards.js +88 -0
  217. package/dist/modules/customers/components/detail/ChangelogKpiCards.js.map +7 -0
  218. package/dist/modules/customers/components/detail/ChangelogTab.js +470 -0
  219. package/dist/modules/customers/components/detail/ChangelogTab.js.map +7 -0
  220. package/dist/modules/customers/components/detail/ComingSoonPlaceholder.js +16 -0
  221. package/dist/modules/customers/components/detail/ComingSoonPlaceholder.js.map +7 -0
  222. package/dist/modules/customers/components/detail/CompanyCard.js +283 -0
  223. package/dist/modules/customers/components/detail/CompanyCard.js.map +7 -0
  224. package/dist/modules/customers/components/detail/CompanyDashboardTab.js +133 -0
  225. package/dist/modules/customers/components/detail/CompanyDashboardTab.js.map +7 -0
  226. package/dist/modules/customers/components/detail/CompanyDetailHeader.js +191 -0
  227. package/dist/modules/customers/components/detail/CompanyDetailHeader.js.map +7 -0
  228. package/dist/modules/customers/components/detail/CompanyDetailTabs.js +123 -0
  229. package/dist/modules/customers/components/detail/CompanyDetailTabs.js.map +7 -0
  230. package/dist/modules/customers/components/detail/CompanyKpiBar.js +174 -0
  231. package/dist/modules/customers/components/detail/CompanyKpiBar.js.map +7 -0
  232. package/dist/modules/customers/components/detail/CompanyPeopleSection.js +514 -230
  233. package/dist/modules/customers/components/detail/CompanyPeopleSection.js.map +2 -2
  234. package/dist/modules/customers/components/detail/CompanyTagsDialog.js +22 -0
  235. package/dist/modules/customers/components/detail/CompanyTagsDialog.js.map +7 -0
  236. package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js +159 -0
  237. package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js.map +7 -0
  238. package/dist/modules/customers/components/detail/CreatePersonDialog.js +135 -0
  239. package/dist/modules/customers/components/detail/CreatePersonDialog.js.map +7 -0
  240. package/dist/modules/customers/components/detail/DealClosureActionBar.js +59 -0
  241. package/dist/modules/customers/components/detail/DealClosureActionBar.js.map +7 -0
  242. package/dist/modules/customers/components/detail/DealDetailHeader.js +237 -0
  243. package/dist/modules/customers/components/detail/DealDetailHeader.js.map +7 -0
  244. package/dist/modules/customers/components/detail/DealDetailTabs.js +109 -0
  245. package/dist/modules/customers/components/detail/DealDetailTabs.js.map +7 -0
  246. package/dist/modules/customers/components/detail/DealForm.js +219 -92
  247. package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
  248. package/dist/modules/customers/components/detail/DealLinkedEntitiesTab.js +295 -0
  249. package/dist/modules/customers/components/detail/DealLinkedEntitiesTab.js.map +7 -0
  250. package/dist/modules/customers/components/detail/DealLostSummaryDialog.js +107 -0
  251. package/dist/modules/customers/components/detail/DealLostSummaryDialog.js.map +7 -0
  252. package/dist/modules/customers/components/detail/DealWonPopup.js +113 -0
  253. package/dist/modules/customers/components/detail/DealWonPopup.js.map +7 -0
  254. package/dist/modules/customers/components/detail/DealsSection.js +206 -193
  255. package/dist/modules/customers/components/detail/DealsSection.js.map +2 -2
  256. package/dist/modules/customers/components/detail/DecisionMakersFooter.js +39 -0
  257. package/dist/modules/customers/components/detail/DecisionMakersFooter.js.map +7 -0
  258. package/dist/modules/customers/components/detail/EntityTagsDialog.js +1096 -0
  259. package/dist/modules/customers/components/detail/EntityTagsDialog.js.map +7 -0
  260. package/dist/modules/customers/components/detail/InlineActivityComposer.js +197 -0
  261. package/dist/modules/customers/components/detail/InlineActivityComposer.js.map +7 -0
  262. package/dist/modules/customers/components/detail/ManageTagsDialog.js +1091 -0
  263. package/dist/modules/customers/components/detail/ManageTagsDialog.js.map +7 -0
  264. package/dist/modules/customers/components/detail/MiniWeekCalendar.js +272 -0
  265. package/dist/modules/customers/components/detail/MiniWeekCalendar.js.map +7 -0
  266. package/dist/modules/customers/components/detail/MobilePersonDetail.js +106 -0
  267. package/dist/modules/customers/components/detail/MobilePersonDetail.js.map +7 -0
  268. package/dist/modules/customers/components/detail/NextStepCard.js +72 -0
  269. package/dist/modules/customers/components/detail/NextStepCard.js.map +7 -0
  270. package/dist/modules/customers/components/detail/PersonCard.js +192 -0
  271. package/dist/modules/customers/components/detail/PersonCard.js.map +7 -0
  272. package/dist/modules/customers/components/detail/PersonCompaniesSection.js +345 -0
  273. package/dist/modules/customers/components/detail/PersonCompaniesSection.js.map +7 -0
  274. package/dist/modules/customers/components/detail/PersonDetailHeader.js +220 -0
  275. package/dist/modules/customers/components/detail/PersonDetailHeader.js.map +7 -0
  276. package/dist/modules/customers/components/detail/PersonDetailTabs.js +122 -0
  277. package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +7 -0
  278. package/dist/modules/customers/components/detail/PersonTagsDialog.js +24 -0
  279. package/dist/modules/customers/components/detail/PersonTagsDialog.js.map +7 -0
  280. package/dist/modules/customers/components/detail/PipelineStepper.js +191 -0
  281. package/dist/modules/customers/components/detail/PipelineStepper.js.map +7 -0
  282. package/dist/modules/customers/components/detail/PlannedActivitiesSection.js +222 -0
  283. package/dist/modules/customers/components/detail/PlannedActivitiesSection.js.map +7 -0
  284. package/dist/modules/customers/components/detail/RelationshipHealthCard.js +49 -0
  285. package/dist/modules/customers/components/detail/RelationshipHealthCard.js.map +7 -0
  286. package/dist/modules/customers/components/detail/RoleAssignmentRow.js +189 -0
  287. package/dist/modules/customers/components/detail/RoleAssignmentRow.js.map +7 -0
  288. package/dist/modules/customers/components/detail/RolesSection.js +234 -0
  289. package/dist/modules/customers/components/detail/RolesSection.js.map +7 -0
  290. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +410 -0
  291. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +7 -0
  292. package/dist/modules/customers/components/detail/aiActionCatalog.js +41 -0
  293. package/dist/modules/customers/components/detail/aiActionCatalog.js.map +7 -0
  294. package/dist/modules/customers/components/detail/assignableStaff.js +48 -0
  295. package/dist/modules/customers/components/detail/assignableStaff.js.map +7 -0
  296. package/dist/modules/customers/components/detail/dashboard/ActiveDealWidget.js +48 -0
  297. package/dist/modules/customers/components/detail/dashboard/ActiveDealWidget.js.map +7 -0
  298. package/dist/modules/customers/components/detail/dashboard/OpenTasksWidget.js +86 -0
  299. package/dist/modules/customers/components/detail/dashboard/OpenTasksWidget.js.map +7 -0
  300. package/dist/modules/customers/components/detail/dashboard/RecentActivityWidget.js +53 -0
  301. package/dist/modules/customers/components/detail/dashboard/RecentActivityWidget.js.map +7 -0
  302. package/dist/modules/customers/components/detail/dashboard/RelationshipHealthWidget.js +30 -0
  303. package/dist/modules/customers/components/detail/dashboard/RelationshipHealthWidget.js.map +7 -0
  304. package/dist/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.js +43 -0
  305. package/dist/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.js.map +7 -0
  306. package/dist/modules/customers/components/detail/dashboard/helpers.js +71 -0
  307. package/dist/modules/customers/components/detail/dashboard/helpers.js.map +7 -0
  308. package/dist/modules/customers/components/detail/healthScoreUtils.js +69 -0
  309. package/dist/modules/customers/components/detail/healthScoreUtils.js.map +7 -0
  310. package/dist/modules/customers/components/detail/hooks/useCurrencyDictionary.js +5 -5
  311. package/dist/modules/customers/components/detail/hooks/useCurrencyDictionary.js.map +2 -2
  312. package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js +9 -8
  313. package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js.map +3 -3
  314. package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js +65 -0
  315. package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js.map +7 -0
  316. package/dist/modules/customers/components/detail/notesAdapter.js +70 -30
  317. package/dist/modules/customers/components/detail/notesAdapter.js.map +2 -2
  318. package/dist/modules/customers/components/detail/pipelineStageUtils.js +26 -0
  319. package/dist/modules/customers/components/detail/pipelineStageUtils.js.map +7 -0
  320. package/dist/modules/customers/components/detail/schedule/DateTimeFields.js +144 -0
  321. package/dist/modules/customers/components/detail/schedule/DateTimeFields.js.map +7 -0
  322. package/dist/modules/customers/components/detail/schedule/FooterFields.js +60 -0
  323. package/dist/modules/customers/components/detail/schedule/FooterFields.js.map +7 -0
  324. package/dist/modules/customers/components/detail/schedule/LinkedEntitiesField.js +216 -0
  325. package/dist/modules/customers/components/detail/schedule/LinkedEntitiesField.js.map +7 -0
  326. package/dist/modules/customers/components/detail/schedule/LocationField.js +34 -0
  327. package/dist/modules/customers/components/detail/schedule/LocationField.js.map +7 -0
  328. package/dist/modules/customers/components/detail/schedule/ParticipantsField.js +226 -0
  329. package/dist/modules/customers/components/detail/schedule/ParticipantsField.js.map +7 -0
  330. package/dist/modules/customers/components/detail/schedule/fieldConfig.js +69 -0
  331. package/dist/modules/customers/components/detail/schedule/fieldConfig.js.map +7 -0
  332. package/dist/modules/customers/components/detail/schedule/index.js +21 -0
  333. package/dist/modules/customers/components/detail/schedule/index.js.map +7 -0
  334. package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js +172 -0
  335. package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js.map +7 -0
  336. package/dist/modules/customers/components/detail/utils.js +23 -0
  337. package/dist/modules/customers/components/detail/utils.js.map +2 -2
  338. package/dist/modules/customers/components/formConfig.js +144 -22
  339. package/dist/modules/customers/components/formConfig.js.map +2 -2
  340. package/dist/modules/customers/components/linking/LinkEntityDialog.js +661 -0
  341. package/dist/modules/customers/components/linking/LinkEntityDialog.js.map +7 -0
  342. package/dist/modules/customers/components/linking/adapters/companyAdapter.js +252 -0
  343. package/dist/modules/customers/components/linking/adapters/companyAdapter.js.map +7 -0
  344. package/dist/modules/customers/components/linking/adapters/dealAdapter.js +384 -0
  345. package/dist/modules/customers/components/linking/adapters/dealAdapter.js.map +7 -0
  346. package/dist/modules/customers/components/linking/adapters/personAdapter.js +324 -0
  347. package/dist/modules/customers/components/linking/adapters/personAdapter.js.map +7 -0
  348. package/dist/modules/customers/components/list/CollectionPreviewCell.js +53 -0
  349. package/dist/modules/customers/components/list/CollectionPreviewCell.js.map +7 -0
  350. package/dist/modules/customers/data/entities.js +407 -1
  351. package/dist/modules/customers/data/entities.js.map +2 -2
  352. package/dist/modules/customers/data/validators.js +139 -21
  353. package/dist/modules/customers/data/validators.js.map +2 -2
  354. package/dist/modules/customers/events.js +19 -1
  355. package/dist/modules/customers/events.js.map +2 -2
  356. package/dist/modules/customers/lib/customerRoleTypes.js +19 -0
  357. package/dist/modules/customers/lib/customerRoleTypes.js.map +7 -0
  358. package/dist/modules/customers/lib/dealClosureNotification.js +39 -0
  359. package/dist/modules/customers/lib/dealClosureNotification.js.map +7 -0
  360. package/dist/modules/customers/lib/dealStageTransitionTable.js +29 -0
  361. package/dist/modules/customers/lib/dealStageTransitionTable.js.map +7 -0
  362. package/dist/modules/customers/lib/dictionaries.js +25 -0
  363. package/dist/modules/customers/lib/dictionaries.js.map +2 -2
  364. package/dist/modules/customers/lib/interactionReadModel.js +10 -0
  365. package/dist/modules/customers/lib/interactionReadModel.js.map +2 -2
  366. package/dist/modules/customers/lib/personCompanies.js +235 -0
  367. package/dist/modules/customers/lib/personCompanies.js.map +7 -0
  368. package/dist/modules/customers/lib/personCompanyLinkTable.js +42 -0
  369. package/dist/modules/customers/lib/personCompanyLinkTable.js.map +7 -0
  370. package/dist/modules/customers/lib/roleTypeUsage.js +104 -0
  371. package/dist/modules/customers/lib/roleTypeUsage.js.map +7 -0
  372. package/dist/modules/customers/migrations/Migration20260406214502.js +18 -0
  373. package/dist/modules/customers/migrations/Migration20260406214502.js.map +7 -0
  374. package/dist/modules/customers/migrations/Migration20260408135736.js +17 -0
  375. package/dist/modules/customers/migrations/Migration20260408135736.js.map +7 -0
  376. package/dist/modules/customers/migrations/Migration20260408225345.js +21 -0
  377. package/dist/modules/customers/migrations/Migration20260408225345.js.map +7 -0
  378. package/dist/modules/customers/migrations/Migration20260411075533.js +27 -0
  379. package/dist/modules/customers/migrations/Migration20260411075533.js.map +7 -0
  380. package/dist/modules/customers/migrations/Migration20260411103551.js +13 -0
  381. package/dist/modules/customers/migrations/Migration20260411103551.js.map +7 -0
  382. package/dist/modules/customers/migrations/Migration20260411130944.js +26 -0
  383. package/dist/modules/customers/migrations/Migration20260411130944.js.map +7 -0
  384. package/dist/modules/customers/migrations/Migration20260415095203.js +13 -0
  385. package/dist/modules/customers/migrations/Migration20260415095203.js.map +7 -0
  386. package/dist/modules/customers/migrations/Migration20260415135056.js +20 -0
  387. package/dist/modules/customers/migrations/Migration20260415135056.js.map +7 -0
  388. package/dist/modules/customers/migrations/Migration20260417140000.js +15 -0
  389. package/dist/modules/customers/migrations/Migration20260417140000.js.map +7 -0
  390. package/dist/modules/customers/migrations/Migration20260417160000.js +17 -0
  391. package/dist/modules/customers/migrations/Migration20260417160000.js.map +7 -0
  392. package/dist/modules/customers/migrations/Migration20260417235407.js +13 -0
  393. package/dist/modules/customers/migrations/Migration20260417235407.js.map +7 -0
  394. package/dist/modules/customers/setup.js +16 -1
  395. package/dist/modules/customers/setup.js.map +2 -2
  396. package/dist/modules/customers/subscribers/deal-closure-notification.js +16 -0
  397. package/dist/modules/customers/subscribers/deal-closure-notification.js.map +7 -0
  398. package/dist/modules/customers/subscribers/deal-lost-notification.js +16 -0
  399. package/dist/modules/customers/subscribers/deal-lost-notification.js.map +7 -0
  400. package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js +2 -0
  401. package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js.map +2 -2
  402. package/dist/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.js +154 -0
  403. package/dist/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.js.map +7 -0
  404. package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js +6 -2
  405. package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js.map +2 -2
  406. package/dist/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.js +154 -0
  407. package/dist/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.js.map +7 -0
  408. package/dist/modules/dictionaries/api/context.js +8 -1
  409. package/dist/modules/dictionaries/api/context.js.map +2 -2
  410. package/dist/modules/dictionaries/api/openapi.js +18 -1
  411. package/dist/modules/dictionaries/api/openapi.js.map +2 -2
  412. package/dist/modules/dictionaries/commands/entry-operations.js +388 -0
  413. package/dist/modules/dictionaries/commands/entry-operations.js.map +7 -0
  414. package/dist/modules/dictionaries/commands/factory.js +24 -3
  415. package/dist/modules/dictionaries/commands/factory.js.map +2 -2
  416. package/dist/modules/dictionaries/commands/index.js +1 -0
  417. package/dist/modules/dictionaries/commands/index.js.map +2 -2
  418. package/dist/modules/dictionaries/components/DictionaryTable.js +6 -3
  419. package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
  420. package/dist/modules/dictionaries/data/entities.js +11 -1
  421. package/dist/modules/dictionaries/data/entities.js.map +2 -2
  422. package/dist/modules/dictionaries/data/validators.js +28 -2
  423. package/dist/modules/dictionaries/data/validators.js.map +2 -2
  424. package/dist/modules/dictionaries/events.js +18 -0
  425. package/dist/modules/dictionaries/events.js.map +7 -0
  426. package/dist/modules/dictionaries/lib/clientEntries.js +43 -0
  427. package/dist/modules/dictionaries/lib/clientEntries.js.map +7 -0
  428. package/dist/modules/dictionaries/migrations/Migration20260410171544.js +45 -0
  429. package/dist/modules/dictionaries/migrations/Migration20260410171544.js.map +7 -0
  430. package/dist/modules/inbox_ops/api/proposals/[id]/route.js +4 -1
  431. package/dist/modules/inbox_ops/api/proposals/[id]/route.js.map +2 -2
  432. package/dist/modules/query_index/lib/engine.js +1 -1
  433. package/dist/modules/query_index/lib/engine.js.map +2 -2
  434. package/dist/modules/sales/components/documents/AddressesSection.js +82 -42
  435. package/dist/modules/sales/components/documents/AddressesSection.js.map +2 -2
  436. package/dist/modules/sales/lib/dictionaries.js +16 -0
  437. package/dist/modules/sales/lib/dictionaries.js.map +2 -2
  438. package/dist/modules/sales/widgets/injection-table.js +5 -1
  439. package/dist/modules/sales/widgets/injection-table.js.map +2 -2
  440. package/generated/entities/action_log/index.ts +4 -0
  441. package/generated/entities/customer_company_billing/index.ts +10 -0
  442. package/generated/entities/customer_deal/index.ts +4 -0
  443. package/generated/entities/customer_deal_stage_transition/index.ts +14 -0
  444. package/generated/entities/customer_dictionary_kind_setting/index.ts +9 -0
  445. package/generated/entities/customer_entity/index.ts +4 -0
  446. package/generated/entities/customer_entity_role/index.ts +10 -0
  447. package/generated/entities/customer_interaction/index.ts +11 -0
  448. package/generated/entities/customer_label/index.ts +8 -0
  449. package/generated/entities/customer_label_assignment/index.ts +7 -0
  450. package/generated/entities/customer_person_company_link/index.ts +9 -0
  451. package/generated/entities/customer_person_company_role/index.ts +7 -0
  452. package/generated/entities/dictionary_entry/index.ts +2 -0
  453. package/generated/entities.ids.generated.ts +9 -1
  454. package/generated/entity-fields-registry.ts +116 -1
  455. package/package.json +3 -3
  456. package/src/modules/attachments/api/route.ts +48 -6
  457. package/src/modules/attachments/i18n/de.json +4 -0
  458. package/src/modules/attachments/i18n/en.json +4 -0
  459. package/src/modules/attachments/i18n/es.json +4 -0
  460. package/src/modules/attachments/i18n/pl.json +4 -0
  461. package/src/modules/audit_logs/api/audit-logs/actions/export/route.ts +260 -0
  462. package/src/modules/audit_logs/api/audit-logs/actions/route.ts +81 -6
  463. package/src/modules/audit_logs/cli.ts +79 -0
  464. package/src/modules/audit_logs/data/entities.ts +17 -0
  465. package/src/modules/audit_logs/data/validators.ts +9 -1
  466. package/src/modules/audit_logs/lib/changeRows.ts +47 -0
  467. package/src/modules/audit_logs/lib/display-helpers.tsx +4 -30
  468. package/src/modules/audit_logs/lib/projections.ts +110 -0
  469. package/src/modules/audit_logs/migrations/.snapshot-open-mercato.json +325 -2
  470. package/src/modules/audit_logs/migrations/Migration20260412160533.ts +21 -0
  471. package/src/modules/audit_logs/services/actionLogService.ts +455 -85
  472. package/src/modules/catalog/i18n/de.json +1 -0
  473. package/src/modules/catalog/i18n/en.json +1 -0
  474. package/src/modules/catalog/i18n/es.json +1 -0
  475. package/src/modules/catalog/i18n/pl.json +1 -0
  476. package/src/modules/customer_accounts/i18n/de.json +2 -0
  477. package/src/modules/customer_accounts/i18n/en.json +2 -0
  478. package/src/modules/customer_accounts/i18n/es.json +2 -0
  479. package/src/modules/customer_accounts/i18n/pl.json +2 -0
  480. package/src/modules/customers/acl.ts +2 -0
  481. package/src/modules/customers/api/activities/route.ts +4 -0
  482. package/src/modules/customers/api/assignable-staff/route.ts +250 -0
  483. package/src/modules/customers/api/companies/[id]/people/route.ts +244 -0
  484. package/src/modules/customers/api/companies/[id]/roles/route.ts +15 -0
  485. package/src/modules/customers/api/companies/[id]/route.ts +458 -40
  486. package/src/modules/customers/api/companies/route.ts +93 -15
  487. package/src/modules/customers/api/deals/[id]/companies/route.ts +203 -0
  488. package/src/modules/customers/api/deals/[id]/people/route.ts +182 -0
  489. package/src/modules/customers/api/deals/[id]/route.ts +554 -57
  490. package/src/modules/customers/api/deals/[id]/stats/route.ts +221 -0
  491. package/src/modules/customers/api/deals/route.ts +35 -46
  492. package/src/modules/customers/api/dictionaries/[kind]/[id]/route.ts +105 -3
  493. package/src/modules/customers/api/dictionaries/[kind]/route.ts +143 -44
  494. package/src/modules/customers/api/dictionaries/context.ts +45 -16
  495. package/src/modules/customers/api/dictionaries/kind-settings/route.ts +232 -0
  496. package/src/modules/customers/api/entity-roles-factory.ts +520 -0
  497. package/src/modules/customers/api/interactions/conflicts/route.ts +196 -0
  498. package/src/modules/customers/api/interactions/counts/route.ts +112 -0
  499. package/src/modules/customers/api/interactions/route.ts +95 -2
  500. package/src/modules/customers/api/labels/assign/route.ts +202 -0
  501. package/src/modules/customers/api/labels/auth.ts +19 -0
  502. package/src/modules/customers/api/labels/route.ts +310 -0
  503. package/src/modules/customers/api/labels/table-errors.ts +36 -0
  504. package/src/modules/customers/api/labels/unassign/route.ts +197 -0
  505. package/src/modules/customers/api/people/[id]/companies/[linkId]/route.ts +331 -0
  506. package/src/modules/customers/api/people/[id]/companies/context.ts +70 -0
  507. package/src/modules/customers/api/people/[id]/companies/enriched/route.ts +384 -0
  508. package/src/modules/customers/api/people/[id]/companies/route.ts +215 -0
  509. package/src/modules/customers/api/people/[id]/roles/route.ts +15 -0
  510. package/src/modules/customers/api/people/[id]/route.ts +153 -26
  511. package/src/modules/customers/api/people/route.ts +134 -31
  512. package/src/modules/customers/api/todos/route.ts +4 -0
  513. package/src/modules/customers/api/utils.ts +36 -0
  514. package/src/modules/customers/backend/config/customers/page.tsx +2 -6
  515. package/src/modules/customers/backend/customers/companies/page.tsx +36 -26
  516. package/src/modules/customers/backend/customers/companies-v2/[id]/page.tsx +277 -262
  517. package/src/modules/customers/backend/customers/deals/[id]/hooks/formatters.ts +19 -0
  518. package/src/modules/customers/backend/customers/deals/[id]/hooks/types.ts +104 -0
  519. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.ts +60 -0
  520. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.ts +362 -0
  521. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.ts +113 -0
  522. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealData.ts +52 -0
  523. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.ts +86 -0
  524. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.tsx +60 -0
  525. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.ts +76 -0
  526. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.ts +56 -0
  527. package/src/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.ts +38 -0
  528. package/src/modules/customers/backend/customers/deals/[id]/page.tsx +587 -624
  529. package/src/modules/customers/backend/customers/deals/page.tsx +71 -28
  530. package/src/modules/customers/backend/customers/people/page.tsx +35 -29
  531. package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +343 -209
  532. package/src/modules/customers/cli.ts +107 -12
  533. package/src/modules/customers/commands/activities.ts +13 -0
  534. package/src/modules/customers/commands/deals.ts +386 -114
  535. package/src/modules/customers/commands/dictionaries.ts +175 -32
  536. package/src/modules/customers/commands/dictionaryKindSettings.ts +268 -0
  537. package/src/modules/customers/commands/entity-roles.ts +494 -0
  538. package/src/modules/customers/commands/index.ts +4 -0
  539. package/src/modules/customers/commands/interactions.ts +125 -21
  540. package/src/modules/customers/commands/labels.ts +626 -0
  541. package/src/modules/customers/commands/people.ts +373 -259
  542. package/src/modules/customers/commands/personCompanyLinks.ts +654 -0
  543. package/src/modules/customers/commands/shared.ts +16 -15
  544. package/src/modules/customers/commands/todos.ts +17 -1
  545. package/src/modules/customers/components/AddressEditor.tsx +1 -1
  546. package/src/modules/customers/components/CustomersConfigurationSections.tsx +36 -0
  547. package/src/modules/customers/components/DictionarySettings.tsx +43 -2
  548. package/src/modules/customers/components/detail/ActiveDealCard.tsx +175 -0
  549. package/src/modules/customers/components/detail/ActivitiesSection.tsx +267 -361
  550. package/src/modules/customers/components/detail/ActivityAiActions.tsx +49 -0
  551. package/src/modules/customers/components/detail/ActivityCard.tsx +154 -0
  552. package/src/modules/customers/components/detail/ActivityHistorySection.tsx +412 -0
  553. package/src/modules/customers/components/detail/ActivityLogTab.tsx +67 -0
  554. package/src/modules/customers/components/detail/ActivityTimeline.tsx +158 -0
  555. package/src/modules/customers/components/detail/ActivityTimelineFilters.tsx +163 -0
  556. package/src/modules/customers/components/detail/ActivityTypeSelector.tsx +53 -0
  557. package/src/modules/customers/components/detail/AiActionChips.tsx +48 -0
  558. package/src/modules/customers/components/detail/AssignRoleDialog.tsx +672 -0
  559. package/src/modules/customers/components/detail/ChangelogEntryRow.tsx +132 -0
  560. package/src/modules/customers/components/detail/ChangelogFilters.tsx +193 -0
  561. package/src/modules/customers/components/detail/ChangelogKpiCards.tsx +107 -0
  562. package/src/modules/customers/components/detail/ChangelogTab.tsx +629 -0
  563. package/src/modules/customers/components/detail/ComingSoonPlaceholder.tsx +21 -0
  564. package/src/modules/customers/components/detail/CompanyCard.tsx +419 -0
  565. package/src/modules/customers/components/detail/CompanyDashboardTab.tsx +161 -0
  566. package/src/modules/customers/components/detail/CompanyDetailHeader.tsx +243 -0
  567. package/src/modules/customers/components/detail/CompanyDetailTabs.tsx +172 -0
  568. package/src/modules/customers/components/detail/CompanyKpiBar.tsx +206 -0
  569. package/src/modules/customers/components/detail/CompanyPeopleSection.tsx +582 -288
  570. package/src/modules/customers/components/detail/CompanyTagsDialog.tsx +23 -0
  571. package/src/modules/customers/components/detail/ConfirmDealLostDialog.tsx +210 -0
  572. package/src/modules/customers/components/detail/CreatePersonDialog.tsx +178 -0
  573. package/src/modules/customers/components/detail/DealClosureActionBar.tsx +63 -0
  574. package/src/modules/customers/components/detail/DealDetailHeader.tsx +335 -0
  575. package/src/modules/customers/components/detail/DealDetailTabs.tsx +154 -0
  576. package/src/modules/customers/components/detail/DealForm.tsx +253 -101
  577. package/src/modules/customers/components/detail/DealLinkedEntitiesTab.tsx +349 -0
  578. package/src/modules/customers/components/detail/DealLostSummaryDialog.tsx +156 -0
  579. package/src/modules/customers/components/detail/DealWonPopup.tsx +164 -0
  580. package/src/modules/customers/components/detail/DealsSection.tsx +276 -221
  581. package/src/modules/customers/components/detail/DecisionMakersFooter.tsx +56 -0
  582. package/src/modules/customers/components/detail/EntityTagsDialog.tsx +1372 -0
  583. package/src/modules/customers/components/detail/InlineActivityComposer.tsx +239 -0
  584. package/src/modules/customers/components/detail/ManageTagsDialog.tsx +1331 -0
  585. package/src/modules/customers/components/detail/MiniWeekCalendar.tsx +338 -0
  586. package/src/modules/customers/components/detail/MobilePersonDetail.tsx +124 -0
  587. package/src/modules/customers/components/detail/NextStepCard.tsx +104 -0
  588. package/src/modules/customers/components/detail/PersonCard.tsx +238 -0
  589. package/src/modules/customers/components/detail/PersonCompaniesSection.tsx +426 -0
  590. package/src/modules/customers/components/detail/PersonDetailHeader.tsx +294 -0
  591. package/src/modules/customers/components/detail/PersonDetailTabs.tsx +172 -0
  592. package/src/modules/customers/components/detail/PersonTagsDialog.tsx +26 -0
  593. package/src/modules/customers/components/detail/PipelineStepper.tsx +245 -0
  594. package/src/modules/customers/components/detail/PlannedActivitiesSection.tsx +255 -0
  595. package/src/modules/customers/components/detail/RelationshipHealthCard.tsx +63 -0
  596. package/src/modules/customers/components/detail/RoleAssignmentRow.tsx +248 -0
  597. package/src/modules/customers/components/detail/RolesSection.tsx +311 -0
  598. package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +481 -0
  599. package/src/modules/customers/components/detail/aiActionCatalog.ts +77 -0
  600. package/src/modules/customers/components/detail/assignableStaff.ts +124 -0
  601. package/src/modules/customers/components/detail/dashboard/ActiveDealWidget.tsx +63 -0
  602. package/src/modules/customers/components/detail/dashboard/OpenTasksWidget.tsx +114 -0
  603. package/src/modules/customers/components/detail/dashboard/RecentActivityWidget.tsx +69 -0
  604. package/src/modules/customers/components/detail/dashboard/RelationshipHealthWidget.tsx +40 -0
  605. package/src/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.tsx +64 -0
  606. package/src/modules/customers/components/detail/dashboard/helpers.ts +78 -0
  607. package/src/modules/customers/components/detail/healthScoreUtils.ts +91 -0
  608. package/src/modules/customers/components/detail/hooks/useCurrencyDictionary.ts +8 -8
  609. package/src/modules/customers/components/detail/hooks/useCustomerDictionary.ts +10 -6
  610. package/src/modules/customers/components/detail/hooks/useInteractionMutations.ts +91 -0
  611. package/src/modules/customers/components/detail/notesAdapter.ts +91 -30
  612. package/src/modules/customers/components/detail/pipelineStageUtils.ts +29 -0
  613. package/src/modules/customers/components/detail/schedule/DateTimeFields.tsx +187 -0
  614. package/src/modules/customers/components/detail/schedule/FooterFields.tsx +79 -0
  615. package/src/modules/customers/components/detail/schedule/LinkedEntitiesField.tsx +277 -0
  616. package/src/modules/customers/components/detail/schedule/LocationField.tsx +42 -0
  617. package/src/modules/customers/components/detail/schedule/ParticipantsField.tsx +255 -0
  618. package/src/modules/customers/components/detail/schedule/fieldConfig.ts +70 -0
  619. package/src/modules/customers/components/detail/schedule/index.ts +9 -0
  620. package/src/modules/customers/components/detail/schedule/useScheduleFormState.ts +221 -0
  621. package/src/modules/customers/components/detail/types.ts +16 -0
  622. package/src/modules/customers/components/detail/utils.ts +25 -0
  623. package/src/modules/customers/components/formConfig.tsx +223 -28
  624. package/src/modules/customers/components/linking/LinkEntityDialog.tsx +920 -0
  625. package/src/modules/customers/components/linking/adapters/companyAdapter.tsx +398 -0
  626. package/src/modules/customers/components/linking/adapters/dealAdapter.tsx +578 -0
  627. package/src/modules/customers/components/linking/adapters/personAdapter.tsx +512 -0
  628. package/src/modules/customers/components/list/CollectionPreviewCell.tsx +66 -0
  629. package/src/modules/customers/data/entities.ts +353 -1
  630. package/src/modules/customers/data/validators.ts +170 -19
  631. package/src/modules/customers/events.ts +22 -0
  632. package/src/modules/customers/i18n/de.json +841 -2
  633. package/src/modules/customers/i18n/en.json +841 -2
  634. package/src/modules/customers/i18n/es.json +840 -1
  635. package/src/modules/customers/i18n/pl.json +841 -2
  636. package/src/modules/customers/lib/customerRoleTypes.ts +24 -0
  637. package/src/modules/customers/lib/dealClosureNotification.ts +64 -0
  638. package/src/modules/customers/lib/dealStageTransitionTable.ts +32 -0
  639. package/src/modules/customers/lib/dictionaries.ts +26 -10
  640. package/src/modules/customers/lib/interactionReadModel.ts +10 -0
  641. package/src/modules/customers/lib/personCompanies.ts +317 -0
  642. package/src/modules/customers/lib/personCompanyLinkTable.ts +58 -0
  643. package/src/modules/customers/lib/roleTypeUsage.ts +146 -0
  644. package/src/modules/customers/migrations/.snapshot-open-mercato.json +2747 -798
  645. package/src/modules/customers/migrations/Migration20260406214502.ts +19 -0
  646. package/src/modules/customers/migrations/Migration20260408135736.ts +15 -0
  647. package/src/modules/customers/migrations/Migration20260408225345.ts +23 -0
  648. package/src/modules/customers/migrations/Migration20260411075533.ts +30 -0
  649. package/src/modules/customers/migrations/Migration20260411103551.ts +13 -0
  650. package/src/modules/customers/migrations/Migration20260411130944.ts +30 -0
  651. package/src/modules/customers/migrations/Migration20260415095203.ts +13 -0
  652. package/src/modules/customers/migrations/Migration20260415135056.ts +22 -0
  653. package/src/modules/customers/migrations/Migration20260417140000.ts +15 -0
  654. package/src/modules/customers/migrations/Migration20260417160000.ts +17 -0
  655. package/src/modules/customers/migrations/Migration20260417235407.ts +13 -0
  656. package/src/modules/customers/setup.ts +15 -0
  657. package/src/modules/customers/subscribers/deal-closure-notification.ts +22 -0
  658. package/src/modules/customers/subscribers/deal-lost-notification.ts +22 -0
  659. package/src/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.ts +2 -0
  660. package/src/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.ts +162 -0
  661. package/src/modules/dictionaries/api/[dictionaryId]/entries/route.ts +6 -2
  662. package/src/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.ts +162 -0
  663. package/src/modules/dictionaries/api/context.ts +9 -0
  664. package/src/modules/dictionaries/api/openapi.ts +17 -0
  665. package/src/modules/dictionaries/commands/entry-operations.ts +457 -0
  666. package/src/modules/dictionaries/commands/factory.ts +31 -3
  667. package/src/modules/dictionaries/commands/index.ts +1 -0
  668. package/src/modules/dictionaries/components/DictionaryTable.tsx +15 -6
  669. package/src/modules/dictionaries/data/entities.ts +9 -0
  670. package/src/modules/dictionaries/data/validators.ts +34 -0
  671. package/src/modules/dictionaries/events.ts +20 -0
  672. package/src/modules/dictionaries/i18n/de.json +2 -0
  673. package/src/modules/dictionaries/i18n/en.json +2 -0
  674. package/src/modules/dictionaries/i18n/es.json +2 -0
  675. package/src/modules/dictionaries/i18n/pl.json +2 -0
  676. package/src/modules/dictionaries/lib/clientEntries.ts +66 -0
  677. package/src/modules/dictionaries/migrations/.snapshot-open-mercato.json +185 -3
  678. package/src/modules/dictionaries/migrations/Migration20260410171544.ts +49 -0
  679. package/src/modules/inbox_ops/api/proposals/[id]/route.ts +4 -1
  680. package/src/modules/query_index/lib/engine.ts +9 -1
  681. package/src/modules/sales/components/documents/AddressesSection.tsx +92 -42
  682. package/src/modules/sales/i18n/de.json +28 -0
  683. package/src/modules/sales/i18n/en.json +28 -0
  684. package/src/modules/sales/i18n/es.json +28 -0
  685. package/src/modules/sales/i18n/pl.json +28 -0
  686. package/src/modules/sales/lib/dictionaries.ts +18 -0
  687. package/src/modules/sales/widgets/injection-table.ts +4 -0
@@ -0,0 +1,19 @@
1
+ export function formatCurrency(amount: string | null, currency: string | null): string | null {
2
+ if (!amount) return null
3
+ const parsed = Number(amount)
4
+ if (!Number.isFinite(parsed)) return currency ? `${amount} ${currency}` : amount
5
+ if (!currency) return parsed.toLocaleString()
6
+ try {
7
+ return new Intl.NumberFormat(undefined, { style: 'currency', currency }).format(parsed)
8
+ } catch {
9
+ return `${parsed.toLocaleString()} ${currency}`
10
+ }
11
+ }
12
+
13
+ export function startOfNextQuarter(baseDate: Date): Date {
14
+ const year = baseDate.getFullYear()
15
+ const currentQuarter = Math.floor(baseDate.getMonth() / 3)
16
+ const nextQuarter = currentQuarter + 1
17
+ if (nextQuarter >= 4) return new Date(year + 1, 0, 1, 10, 0, 0, 0)
18
+ return new Date(year, nextQuarter * 3, 1, 10, 0, 0, 0)
19
+ }
@@ -0,0 +1,104 @@
1
+ export type DealAssociation = {
2
+ id: string
3
+ label: string
4
+ subtitle: string | null
5
+ kind: 'person' | 'company'
6
+ }
7
+
8
+ export type PersonAssociationApiRecord = {
9
+ id?: string
10
+ displayName?: string | null
11
+ display_name?: string | null
12
+ primaryEmail?: string | null
13
+ primary_email?: string | null
14
+ primaryPhone?: string | null
15
+ primary_phone?: string | null
16
+ personProfile?: { jobTitle?: string | null } | null
17
+ person_profile?: { jobTitle?: string | null } | null
18
+ }
19
+
20
+ export type CompanyAssociationApiRecord = {
21
+ id?: string
22
+ displayName?: string | null
23
+ display_name?: string | null
24
+ domain?: string | null
25
+ websiteUrl?: string | null
26
+ website_url?: string | null
27
+ companyProfile?: { domain?: string | null; websiteUrl?: string | null } | null
28
+ company_profile?: { domain?: string | null; websiteUrl?: string | null } | null
29
+ }
30
+
31
+ export type PipelineStageInfo = {
32
+ id: string
33
+ label: string
34
+ order: number
35
+ color: string | null
36
+ icon: string | null
37
+ }
38
+
39
+ export type StageTransitionInfo = {
40
+ stageId: string
41
+ stageLabel: string
42
+ stageOrder: number
43
+ transitionedAt: string
44
+ }
45
+
46
+ export type DealDetailPayload = {
47
+ deal: {
48
+ id: string
49
+ title: string
50
+ description: string | null
51
+ status: string | null
52
+ pipelineStage: string | null
53
+ pipelineId: string | null
54
+ pipelineStageId: string | null
55
+ valueAmount: string | null
56
+ valueCurrency: string | null
57
+ probability: number | null
58
+ expectedCloseAt: string | null
59
+ ownerUserId: string | null
60
+ source: string | null
61
+ closureOutcome: 'won' | 'lost' | null
62
+ lossReasonId: string | null
63
+ lossNotes: string | null
64
+ organizationId: string | null
65
+ tenantId: string | null
66
+ createdAt: string
67
+ updatedAt: string
68
+ }
69
+ people: DealAssociation[]
70
+ companies: DealAssociation[]
71
+ linkedPersonIds: string[]
72
+ linkedCompanyIds: string[]
73
+ counts: {
74
+ people: number
75
+ companies: number
76
+ }
77
+ customFields: Record<string, unknown>
78
+ viewer: {
79
+ userId: string | null
80
+ name: string | null
81
+ email: string | null
82
+ } | null
83
+ pipelineStages: PipelineStageInfo[]
84
+ pipelineName: string | null
85
+ stageTransitions: StageTransitionInfo[]
86
+ owner: { id: string; name: string; email: string } | null
87
+ }
88
+
89
+ export type DealStatsPayload = {
90
+ dealValue: number | null
91
+ dealCurrency: string | null
92
+ closureOutcome: 'won' | 'lost'
93
+ closedAt: string
94
+ pipelineName: string | null
95
+ dealsClosedThisPeriod: number
96
+ salesCycleDays: number | null
97
+ dealRankInQuarter: number | null
98
+ lossReason: string | null
99
+ }
100
+
101
+ export type GuardedMutationRunner = <TResult>(
102
+ operation: () => Promise<TResult>,
103
+ mutationPayload?: Record<string, unknown>,
104
+ ) => Promise<TResult>
@@ -0,0 +1,60 @@
1
+ import * as React from 'react'
2
+ import { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'
3
+ import type { InteractionSummary } from '../../../../../components/detail/types'
4
+ import { useInteractionMutations } from '../../../../../components/detail/hooks/useInteractionMutations'
5
+ import type { GuardedMutationRunner } from './types'
6
+
7
+ type UseDealActivitiesOptions = {
8
+ dealId: string
9
+ runMutationWithContext: GuardedMutationRunner
10
+ }
11
+
12
+ type UseDealActivitiesResult = {
13
+ plannedActivities: InteractionSummary[]
14
+ activityRefreshKey: number
15
+ loadPlannedActivities: () => Promise<void>
16
+ handleActivityCreated: () => Promise<void>
17
+ handleMarkDone: (interactionId: string) => Promise<void>
18
+ handleCancelActivity: (interactionId: string) => Promise<void>
19
+ }
20
+
21
+ export function useDealActivities({
22
+ dealId,
23
+ runMutationWithContext,
24
+ }: UseDealActivitiesOptions): UseDealActivitiesResult {
25
+ const [plannedActivities, setPlannedActivities] = React.useState<InteractionSummary[]>([])
26
+ const [activityRefreshKey, setActivityRefreshKey] = React.useState(0)
27
+
28
+ const loadPlannedActivities = React.useCallback(async () => {
29
+ if (!dealId) return
30
+ try {
31
+ const result = await readApiResultOrThrow<{ items?: InteractionSummary[] }>(
32
+ `/api/customers/interactions?dealId=${encodeURIComponent(dealId)}&status=planned&excludeInteractionType=task&limit=100&sortField=scheduledAt&sortDir=asc`,
33
+ )
34
+ setPlannedActivities(Array.isArray(result.items) ? result.items : [])
35
+ } catch (err) {
36
+ console.warn('[customers.deals.detail] load planned activities failed', err)
37
+ setPlannedActivities([])
38
+ }
39
+ }, [dealId])
40
+
41
+ const handleActivityCreated = React.useCallback(async () => {
42
+ setActivityRefreshKey((value) => value + 1)
43
+ await loadPlannedActivities()
44
+ }, [loadPlannedActivities])
45
+
46
+ const { completeInteraction, cancelInteraction } = useInteractionMutations({
47
+ runMutationWithContext,
48
+ onAfterChange: handleActivityCreated,
49
+ logContext: 'customers.deals.detail',
50
+ })
51
+
52
+ return {
53
+ plannedActivities,
54
+ activityRefreshKey,
55
+ loadPlannedActivities,
56
+ handleActivityCreated,
57
+ handleMarkDone: completeInteraction,
58
+ handleCancelActivity: cancelInteraction,
59
+ }
60
+ }
@@ -0,0 +1,362 @@
1
+ import * as React from 'react'
2
+ import { flash } from '@open-mercato/ui/backend/FlashMessages'
3
+ import { updateCrud } from '@open-mercato/ui/backend/utils/crud'
4
+ import { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'
5
+ import { useT } from '@open-mercato/shared/lib/i18n/context'
6
+ import type {
7
+ CompanyAssociationApiRecord,
8
+ DealAssociation,
9
+ DealDetailPayload,
10
+ GuardedMutationRunner,
11
+ PersonAssociationApiRecord,
12
+ } from './types'
13
+
14
+ export function normalizePersonAssociationRecord(
15
+ record: PersonAssociationApiRecord,
16
+ fallbackId: string,
17
+ ): DealAssociation {
18
+ const displayName =
19
+ typeof record.displayName === 'string' && record.displayName.trim().length
20
+ ? record.displayName.trim()
21
+ : typeof record.display_name === 'string' && record.display_name.trim().length
22
+ ? record.display_name.trim()
23
+ : null
24
+ const email =
25
+ typeof record.primaryEmail === 'string' && record.primaryEmail.trim().length
26
+ ? record.primaryEmail.trim()
27
+ : typeof record.primary_email === 'string' && record.primary_email.trim().length
28
+ ? record.primary_email.trim()
29
+ : null
30
+ const phone =
31
+ typeof record.primaryPhone === 'string' && record.primaryPhone.trim().length
32
+ ? record.primaryPhone.trim()
33
+ : typeof record.primary_phone === 'string' && record.primary_phone.trim().length
34
+ ? record.primary_phone.trim()
35
+ : null
36
+ const profile = record.personProfile ?? record.person_profile ?? null
37
+ const jobTitle =
38
+ profile && typeof profile.jobTitle === 'string' && profile.jobTitle.trim().length
39
+ ? profile.jobTitle.trim()
40
+ : null
41
+ return {
42
+ id: typeof record.id === 'string' ? record.id : fallbackId,
43
+ label: displayName ?? email ?? phone ?? fallbackId,
44
+ subtitle: jobTitle ?? email ?? phone ?? null,
45
+ kind: 'person',
46
+ }
47
+ }
48
+
49
+ export function normalizeCompanyAssociationRecord(
50
+ record: CompanyAssociationApiRecord,
51
+ fallbackId: string,
52
+ ): DealAssociation {
53
+ const displayName =
54
+ typeof record.displayName === 'string' && record.displayName.trim().length
55
+ ? record.displayName.trim()
56
+ : typeof record.display_name === 'string' && record.display_name.trim().length
57
+ ? record.display_name.trim()
58
+ : null
59
+ const profile = record.companyProfile ?? record.company_profile ?? null
60
+ const domain =
61
+ typeof record.domain === 'string' && record.domain.trim().length
62
+ ? record.domain.trim()
63
+ : profile && typeof profile.domain === 'string' && profile.domain.trim().length
64
+ ? profile.domain.trim()
65
+ : null
66
+ const website =
67
+ typeof record.websiteUrl === 'string' && record.websiteUrl.trim().length
68
+ ? record.websiteUrl.trim()
69
+ : typeof record.website_url === 'string' && record.website_url.trim().length
70
+ ? record.website_url.trim()
71
+ : profile && typeof profile.websiteUrl === 'string' && profile.websiteUrl.trim().length
72
+ ? profile.websiteUrl.trim()
73
+ : null
74
+ return {
75
+ id: typeof record.id === 'string' ? record.id : fallbackId,
76
+ label: displayName ?? domain ?? website ?? fallbackId,
77
+ subtitle: domain ?? website ?? null,
78
+ kind: 'company',
79
+ }
80
+ }
81
+
82
+ function sameIdList(left: string[], right: string[]): boolean {
83
+ if (left.length !== right.length) return false
84
+ return left.every((value, index) => value === right[index])
85
+ }
86
+
87
+ type LinkedPageResult = {
88
+ items: DealAssociation[]
89
+ totalPages: number
90
+ total: number
91
+ }
92
+
93
+ type UseDealAssociationsOptions = {
94
+ currentDealId: string | null
95
+ data: DealDetailPayload | null
96
+ setData: React.Dispatch<React.SetStateAction<DealDetailPayload | null>>
97
+ runMutationWithContext: GuardedMutationRunner
98
+ }
99
+
100
+ type UseDealAssociationsResult = {
101
+ peopleEditorIds: string[]
102
+ companiesEditorIds: string[]
103
+ peopleSaving: boolean
104
+ companiesSaving: boolean
105
+ handlePeopleAssociationsChange: (nextIds: string[]) => Promise<void>
106
+ handleCompaniesAssociationsChange: (nextIds: string[]) => Promise<void>
107
+ loadLinkedPeoplePage: (page: number, query: string) => Promise<LinkedPageResult>
108
+ loadLinkedCompaniesPage: (page: number, query: string) => Promise<LinkedPageResult>
109
+ }
110
+
111
+ export function useDealAssociations({
112
+ currentDealId,
113
+ data,
114
+ setData,
115
+ runMutationWithContext,
116
+ }: UseDealAssociationsOptions): UseDealAssociationsResult {
117
+ const t = useT()
118
+ const [peopleEditorIds, setPeopleEditorIds] = React.useState<string[]>([])
119
+ const [companiesEditorIds, setCompaniesEditorIds] = React.useState<string[]>([])
120
+ const [peopleSaving, setPeopleSaving] = React.useState(false)
121
+ const [companiesSaving, setCompaniesSaving] = React.useState(false)
122
+
123
+ React.useEffect(() => {
124
+ setPeopleEditorIds(data?.linkedPersonIds ?? [])
125
+ setCompaniesEditorIds(data?.linkedCompanyIds ?? [])
126
+ }, [data?.linkedCompanyIds, data?.linkedPersonIds])
127
+
128
+ const loadPeopleAssociations = React.useCallback(async (ids: string[]): Promise<DealAssociation[]> => {
129
+ const uniqueIds = Array.from(new Set(ids.map((value) => value.trim()).filter(Boolean)))
130
+ if (!uniqueIds.length) return []
131
+ try {
132
+ const params = new URLSearchParams({
133
+ ids: uniqueIds.join(','),
134
+ pageSize: String(Math.max(uniqueIds.length, 1)),
135
+ })
136
+ const payload = await readApiResultOrThrow<{ items?: PersonAssociationApiRecord[] }>(
137
+ `/api/customers/people?${params.toString()}`,
138
+ )
139
+ const items = Array.isArray(payload.items) ? payload.items : []
140
+ const byId = new Map<string, PersonAssociationApiRecord>()
141
+ items.forEach((record) => {
142
+ if (record && typeof record.id === 'string') byId.set(record.id, record)
143
+ })
144
+ return uniqueIds.map((personId) => {
145
+ const record = byId.get(personId)
146
+ return record
147
+ ? normalizePersonAssociationRecord(record, personId)
148
+ : {
149
+ id: personId,
150
+ label: personId,
151
+ subtitle: null,
152
+ kind: 'person' as const,
153
+ }
154
+ })
155
+ } catch {
156
+ return uniqueIds.map((personId) => ({
157
+ id: personId,
158
+ label: personId,
159
+ subtitle: null,
160
+ kind: 'person' as const,
161
+ }))
162
+ }
163
+ }, [])
164
+
165
+ const loadCompanyAssociations = React.useCallback(async (ids: string[]): Promise<DealAssociation[]> => {
166
+ const uniqueIds = Array.from(new Set(ids.map((value) => value.trim()).filter(Boolean)))
167
+ if (!uniqueIds.length) return []
168
+ try {
169
+ const params = new URLSearchParams({
170
+ ids: uniqueIds.join(','),
171
+ pageSize: String(Math.max(uniqueIds.length, 1)),
172
+ })
173
+ const payload = await readApiResultOrThrow<{ items?: CompanyAssociationApiRecord[] }>(
174
+ `/api/customers/companies?${params.toString()}`,
175
+ )
176
+ const items = Array.isArray(payload.items) ? payload.items : []
177
+ const byId = new Map<string, CompanyAssociationApiRecord>()
178
+ items.forEach((record) => {
179
+ if (record && typeof record.id === 'string') byId.set(record.id, record)
180
+ })
181
+ return uniqueIds.map((companyId) => {
182
+ const record = byId.get(companyId)
183
+ return record
184
+ ? normalizeCompanyAssociationRecord(record, companyId)
185
+ : {
186
+ id: companyId,
187
+ label: companyId,
188
+ subtitle: null,
189
+ kind: 'company' as const,
190
+ }
191
+ })
192
+ } catch {
193
+ return uniqueIds.map((companyId) => ({
194
+ id: companyId,
195
+ label: companyId,
196
+ subtitle: null,
197
+ kind: 'company' as const,
198
+ }))
199
+ }
200
+ }, [])
201
+
202
+ const loadLinkedPeoplePage = React.useCallback(
203
+ async (page: number, query: string): Promise<LinkedPageResult> => {
204
+ if (!currentDealId) {
205
+ return { items: [], totalPages: 1, total: 0 }
206
+ }
207
+ const params = new URLSearchParams({
208
+ page: String(page),
209
+ pageSize: '20',
210
+ sort: 'name-asc',
211
+ })
212
+ if (query.trim().length > 0) {
213
+ params.set('search', query.trim())
214
+ }
215
+ const payload = await readApiResultOrThrow<{
216
+ items?: DealAssociation[]
217
+ total?: number
218
+ totalPages?: number
219
+ }>(`/api/customers/deals/${encodeURIComponent(currentDealId)}/people?${params.toString()}`)
220
+ return {
221
+ items: Array.isArray(payload.items) ? payload.items : [],
222
+ totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,
223
+ total: typeof payload.total === 'number' ? payload.total : 0,
224
+ }
225
+ },
226
+ [currentDealId],
227
+ )
228
+
229
+ const loadLinkedCompaniesPage = React.useCallback(
230
+ async (page: number, query: string): Promise<LinkedPageResult> => {
231
+ if (!currentDealId) {
232
+ return { items: [], totalPages: 1, total: 0 }
233
+ }
234
+ const params = new URLSearchParams({
235
+ page: String(page),
236
+ pageSize: '20',
237
+ sort: 'name-asc',
238
+ })
239
+ if (query.trim().length > 0) {
240
+ params.set('search', query.trim())
241
+ }
242
+ const payload = await readApiResultOrThrow<{
243
+ items?: DealAssociation[]
244
+ total?: number
245
+ totalPages?: number
246
+ }>(`/api/customers/deals/${encodeURIComponent(currentDealId)}/companies?${params.toString()}`)
247
+ return {
248
+ items: Array.isArray(payload.items) ? payload.items : [],
249
+ totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,
250
+ total: typeof payload.total === 'number' ? payload.total : 0,
251
+ }
252
+ },
253
+ [currentDealId],
254
+ )
255
+
256
+ const handlePeopleAssociationsChange = React.useCallback(
257
+ async (nextIds: string[]) => {
258
+ if (!currentDealId) return
259
+ if (sameIdList(nextIds, peopleEditorIds)) return
260
+ const previousIds = peopleEditorIds
261
+ const previousPeople = data?.people ?? []
262
+ setPeopleEditorIds(nextIds)
263
+ setPeopleSaving(true)
264
+ try {
265
+ await runMutationWithContext(
266
+ () => updateCrud('customers/deals', { id: currentDealId, personIds: nextIds }),
267
+ { id: currentDealId, personIds: nextIds, operation: 'updateDealPeople' },
268
+ )
269
+ const nextPeople = await loadPeopleAssociations(nextIds.slice(0, 3))
270
+ setData((prev) =>
271
+ prev
272
+ ? {
273
+ ...prev,
274
+ people: nextPeople,
275
+ linkedPersonIds: nextIds,
276
+ counts: { ...prev.counts, people: nextIds.length },
277
+ }
278
+ : prev,
279
+ )
280
+ } catch {
281
+ setPeopleEditorIds(previousIds)
282
+ setData((prev) =>
283
+ prev
284
+ ? {
285
+ ...prev,
286
+ people: previousPeople,
287
+ linkedPersonIds: previousIds,
288
+ counts: { ...prev.counts, people: previousIds.length },
289
+ }
290
+ : prev,
291
+ )
292
+ flash(t('customers.deals.detail.peopleUpdateError', 'Failed to update linked people.'), 'error')
293
+ } finally {
294
+ setPeopleSaving(false)
295
+ }
296
+ },
297
+ [currentDealId, data?.people, loadPeopleAssociations, peopleEditorIds, runMutationWithContext, setData, t],
298
+ )
299
+
300
+ const handleCompaniesAssociationsChange = React.useCallback(
301
+ async (nextIds: string[]) => {
302
+ if (!currentDealId) return
303
+ if (sameIdList(nextIds, companiesEditorIds)) return
304
+ const previousIds = companiesEditorIds
305
+ const previousCompanies = data?.companies ?? []
306
+ setCompaniesEditorIds(nextIds)
307
+ setCompaniesSaving(true)
308
+ try {
309
+ await runMutationWithContext(
310
+ () => updateCrud('customers/deals', { id: currentDealId, companyIds: nextIds }),
311
+ { id: currentDealId, companyIds: nextIds, operation: 'updateDealCompanies' },
312
+ )
313
+ const nextCompanies = await loadCompanyAssociations(nextIds.slice(0, 3))
314
+ setData((prev) =>
315
+ prev
316
+ ? {
317
+ ...prev,
318
+ companies: nextCompanies,
319
+ linkedCompanyIds: nextIds,
320
+ counts: { ...prev.counts, companies: nextIds.length },
321
+ }
322
+ : prev,
323
+ )
324
+ } catch {
325
+ setCompaniesEditorIds(previousIds)
326
+ setData((prev) =>
327
+ prev
328
+ ? {
329
+ ...prev,
330
+ companies: previousCompanies,
331
+ linkedCompanyIds: previousIds,
332
+ counts: { ...prev.counts, companies: previousIds.length },
333
+ }
334
+ : prev,
335
+ )
336
+ flash(t('customers.deals.detail.companiesUpdateError', 'Failed to update linked companies.'), 'error')
337
+ } finally {
338
+ setCompaniesSaving(false)
339
+ }
340
+ },
341
+ [
342
+ companiesEditorIds,
343
+ currentDealId,
344
+ data?.companies,
345
+ loadCompanyAssociations,
346
+ runMutationWithContext,
347
+ setData,
348
+ t,
349
+ ],
350
+ )
351
+
352
+ return {
353
+ peopleEditorIds,
354
+ companiesEditorIds,
355
+ peopleSaving,
356
+ companiesSaving,
357
+ handlePeopleAssociationsChange,
358
+ handleCompaniesAssociationsChange,
359
+ loadLinkedPeoplePage,
360
+ loadLinkedCompaniesPage,
361
+ }
362
+ }
@@ -0,0 +1,113 @@
1
+ import * as React from 'react'
2
+ import { updateCrud } from '@open-mercato/ui/backend/utils/crud'
3
+ import { readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'
4
+ import type { DealStatsPayload, GuardedMutationRunner } from './types'
5
+
6
+ type UseDealClosureOptions = {
7
+ currentDealId: string | null
8
+ runMutationWithContext: GuardedMutationRunner
9
+ confirmDiscardIfDirty: () => Promise<boolean>
10
+ onClosed: () => Promise<void>
11
+ }
12
+
13
+ type UseDealClosureResult = {
14
+ lostDialogOpen: boolean
15
+ wonPopupOpen: boolean
16
+ lostPopupOpen: boolean
17
+ wonStats: DealStatsPayload | null
18
+ lostStats: DealStatsPayload | null
19
+ openLostDialog: () => void
20
+ closeLostDialog: () => void
21
+ closeWonPopup: () => void
22
+ closeLostPopup: () => void
23
+ handleWon: () => Promise<void>
24
+ handleLostConfirm: (input: { lossReasonId: string; lossNotes?: string }) => Promise<void>
25
+ }
26
+
27
+ export function useDealClosure({
28
+ currentDealId,
29
+ runMutationWithContext,
30
+ confirmDiscardIfDirty,
31
+ onClosed,
32
+ }: UseDealClosureOptions): UseDealClosureResult {
33
+ const [lostDialogOpen, setLostDialogOpen] = React.useState(false)
34
+ const [wonPopupOpen, setWonPopupOpen] = React.useState(false)
35
+ const [lostPopupOpen, setLostPopupOpen] = React.useState(false)
36
+ const [wonStats, setWonStats] = React.useState<DealStatsPayload | null>(null)
37
+ const [lostStats, setLostStats] = React.useState<DealStatsPayload | null>(null)
38
+
39
+ const fetchDealStats = React.useCallback(async (): Promise<DealStatsPayload | null> => {
40
+ if (!currentDealId) return null
41
+ try {
42
+ return await readApiResultOrThrow<DealStatsPayload>(
43
+ `/api/customers/deals/${encodeURIComponent(currentDealId)}/stats`,
44
+ )
45
+ } catch (statsError) {
46
+ console.error('customers.deals.detail.stats failed', statsError)
47
+ return null
48
+ }
49
+ }, [currentDealId])
50
+
51
+ const handleWon = React.useCallback(async () => {
52
+ if (!currentDealId) return
53
+ if (!(await confirmDiscardIfDirty())) return
54
+ await runMutationWithContext(
55
+ () => updateCrud('customers/deals', { id: currentDealId, closureOutcome: 'won', status: 'win' }),
56
+ { id: currentDealId, closureOutcome: 'won', status: 'win', operation: 'closeWon' },
57
+ )
58
+ const stats = await fetchDealStats()
59
+ setWonStats(stats)
60
+ setWonPopupOpen(true)
61
+ await onClosed()
62
+ }, [confirmDiscardIfDirty, currentDealId, fetchDealStats, onClosed, runMutationWithContext])
63
+
64
+ const handleLostConfirm = React.useCallback(
65
+ async (input: { lossReasonId: string; lossNotes?: string }) => {
66
+ if (!currentDealId) return
67
+ if (!(await confirmDiscardIfDirty())) return
68
+ await runMutationWithContext(
69
+ () =>
70
+ updateCrud('customers/deals', {
71
+ id: currentDealId,
72
+ closureOutcome: 'lost',
73
+ status: 'loose',
74
+ lossReasonId: input.lossReasonId,
75
+ lossNotes: input.lossNotes ?? null,
76
+ }),
77
+ {
78
+ id: currentDealId,
79
+ closureOutcome: 'lost',
80
+ status: 'loose',
81
+ lossReasonId: input.lossReasonId,
82
+ lossNotes: input.lossNotes ?? null,
83
+ operation: 'closeLost',
84
+ },
85
+ )
86
+ setLostDialogOpen(false)
87
+ const stats = await fetchDealStats()
88
+ setLostStats(stats)
89
+ setLostPopupOpen(true)
90
+ await onClosed()
91
+ },
92
+ [confirmDiscardIfDirty, currentDealId, fetchDealStats, onClosed, runMutationWithContext],
93
+ )
94
+
95
+ const openLostDialog = React.useCallback(() => setLostDialogOpen(true), [])
96
+ const closeLostDialog = React.useCallback(() => setLostDialogOpen(false), [])
97
+ const closeWonPopup = React.useCallback(() => setWonPopupOpen(false), [])
98
+ const closeLostPopup = React.useCallback(() => setLostPopupOpen(false), [])
99
+
100
+ return {
101
+ lostDialogOpen,
102
+ wonPopupOpen,
103
+ lostPopupOpen,
104
+ wonStats,
105
+ lostStats,
106
+ openLostDialog,
107
+ closeLostDialog,
108
+ closeWonPopup,
109
+ closeLostPopup,
110
+ handleWon,
111
+ handleLostConfirm,
112
+ }
113
+ }