@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,1096 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
3
+ import * as React from "react";
4
+ import { Check, Plus, Search, SlidersHorizontal, Tag, X } from "lucide-react";
5
+ import { VisuallyHidden } from "@radix-ui/react-visually-hidden";
6
+ import { useT } from "@open-mercato/shared/lib/i18n/context";
7
+ import { cn, slugifyTagLabel } from "@open-mercato/shared/lib/utils";
8
+ import { apiCall, apiCallOrThrow, readApiResultOrThrow } from "@open-mercato/ui/backend/utils/apiCall";
9
+ import { flash } from "@open-mercato/ui/backend/FlashMessages";
10
+ import { useGuardedMutation } from "@open-mercato/ui/backend/injection/useGuardedMutation";
11
+ import { Button } from "@open-mercato/ui/primitives/button";
12
+ import { IconButton } from "@open-mercato/ui/primitives/icon-button";
13
+ import {
14
+ Dialog,
15
+ DialogContent,
16
+ DialogTitle
17
+ } from "@open-mercato/ui/primitives/dialog";
18
+ import { ManageTagsDialog } from "./ManageTagsDialog.js";
19
+ const CATEGORY_DEFS = [
20
+ {
21
+ kind: "tags",
22
+ source: "tags",
23
+ supportedEntityTypes: ["person", "company"],
24
+ labelKey: "customers.personTags.category.tags",
25
+ labelFallback: "Tags",
26
+ descriptionKey: "customers.personTags.description.tags",
27
+ descriptionFallback: "Shared CRM tags that can be assigned to many records.",
28
+ selectionMode: "multi",
29
+ hasColorDots: true,
30
+ supportsCreate: true
31
+ },
32
+ {
33
+ kind: "labels",
34
+ source: "labels",
35
+ supportedEntityTypes: ["person", "company"],
36
+ labelKey: "customers.personTags.category.labels",
37
+ labelFallback: "Labels",
38
+ descriptionKey: "customers.personTags.description.labels",
39
+ descriptionFallback: "Quick labels you can create inline for this record.",
40
+ selectionMode: "multi",
41
+ hasColorDots: false,
42
+ supportsCreate: true
43
+ },
44
+ {
45
+ kind: "statuses",
46
+ source: "dictionary",
47
+ supportedEntityTypes: ["person", "company"],
48
+ labelKey: "customers.personTags.category.statuses",
49
+ labelFallback: "Status",
50
+ descriptionKey: "customers.personTags.description.statuses",
51
+ descriptionFallback: "Primary CRM status shown in the header badges.",
52
+ routeKind: "statuses",
53
+ settingKind: "status",
54
+ entityField: "status",
55
+ selectionMode: "single",
56
+ hasColorDots: true
57
+ },
58
+ {
59
+ kind: "lifecycle-stages",
60
+ source: "dictionary",
61
+ supportedEntityTypes: ["person", "company"],
62
+ labelKey: "customers.personTags.category.lifecycle-stages",
63
+ labelFallback: "Lifecycle",
64
+ descriptionKey: "customers.personTags.description.lifecycleStages",
65
+ descriptionFallback: "Lifecycle stage used across CRM detail views.",
66
+ routeKind: "lifecycle-stages",
67
+ settingKind: "lifecycle_stage",
68
+ entityField: "lifecycleStage",
69
+ selectionMode: "single",
70
+ hasColorDots: true
71
+ },
72
+ {
73
+ kind: "sources",
74
+ source: "dictionary",
75
+ supportedEntityTypes: ["person", "company"],
76
+ labelKey: "customers.personTags.category.sources",
77
+ labelFallback: "Source",
78
+ descriptionKey: "customers.personTags.description.sources",
79
+ descriptionFallback: "How this record entered the pipeline.",
80
+ routeKind: "sources",
81
+ settingKind: "source",
82
+ entityField: "source",
83
+ selectionMode: "single",
84
+ hasColorDots: true
85
+ },
86
+ {
87
+ kind: "temperature",
88
+ source: "dictionary",
89
+ supportedEntityTypes: ["person", "company"],
90
+ labelKey: "customers.personTags.category.temperature",
91
+ labelFallback: "Temperature",
92
+ descriptionKey: "customers.personTags.description.temperature",
93
+ descriptionFallback: "Sales temperature or engagement level.",
94
+ routeKind: "temperature",
95
+ settingKind: "temperature",
96
+ entityField: "temperature",
97
+ selectionMode: "single",
98
+ hasColorDots: true
99
+ },
100
+ {
101
+ kind: "renewal-quarters",
102
+ source: "dictionary",
103
+ supportedEntityTypes: ["person", "company"],
104
+ labelKey: "customers.personTags.category.renewal-quarters",
105
+ labelFallback: "Renewal quarter",
106
+ descriptionKey: "customers.personTags.description.renewalQuarters",
107
+ descriptionFallback: "Quarter used for renewal planning and badges.",
108
+ routeKind: "renewal-quarters",
109
+ settingKind: "renewal_quarter",
110
+ entityField: "renewalQuarter",
111
+ selectionMode: "single",
112
+ hasColorDots: true
113
+ },
114
+ {
115
+ kind: "job-titles",
116
+ source: "dictionary",
117
+ supportedEntityTypes: ["person"],
118
+ labelKey: "customers.personTags.category.job-titles",
119
+ labelFallback: "Job title",
120
+ descriptionKey: "customers.personTags.description.jobTitles",
121
+ descriptionFallback: "The role or title used for this person.",
122
+ routeKind: "job-titles",
123
+ settingKind: "job_title",
124
+ entityField: "jobTitle",
125
+ selectionMode: "single",
126
+ hasColorDots: true
127
+ },
128
+ {
129
+ kind: "industries",
130
+ source: "dictionary",
131
+ supportedEntityTypes: ["company"],
132
+ labelKey: "customers.personTags.category.industries",
133
+ labelFallback: "Industry",
134
+ descriptionKey: "customers.personTags.description.industries",
135
+ descriptionFallback: "The industry used to classify this company.",
136
+ routeKind: "industries",
137
+ settingKind: "industry",
138
+ entityField: "industry",
139
+ selectionMode: "single",
140
+ hasColorDots: true
141
+ }
142
+ ];
143
+ const REMOTE_CATEGORY_PAGE_SIZE = 50;
144
+ const CUSTOM_CATEGORY_FIELD_PREFIX = "crmTagCategory:";
145
+ function cloneSelectionMap(values) {
146
+ return Object.fromEntries(
147
+ Object.entries(values).map(([key, selection]) => [key, new Set(selection)])
148
+ );
149
+ }
150
+ function sortOptions(entries) {
151
+ return [...entries].sort(
152
+ (left, right) => left.label.localeCompare(right.label, void 0, { sensitivity: "base" })
153
+ );
154
+ }
155
+ function mergeOptions(...groups) {
156
+ const merged = /* @__PURE__ */ new Map();
157
+ groups.flat().forEach((entry) => {
158
+ merged.set(entry.value, entry);
159
+ });
160
+ return sortOptions(Array.from(merged.values()));
161
+ }
162
+ function humanizeCategoryKind(kind) {
163
+ return kind.split(/[-_]+/).filter((part) => part.trim().length > 0).map((part) => `${part.slice(0, 1).toUpperCase()}${part.slice(1)}`).join(" ");
164
+ }
165
+ function resolveCustomCategoryFieldKey(kind) {
166
+ return `${CUSTOM_CATEGORY_FIELD_PREFIX}${kind}`;
167
+ }
168
+ function createCustomCategoryDef(kind, selectionMode) {
169
+ const label = humanizeCategoryKind(kind);
170
+ return {
171
+ kind,
172
+ source: "dictionary",
173
+ supportedEntityTypes: ["person", "company"],
174
+ labelKey: "",
175
+ labelFallback: label,
176
+ descriptionKey: "customers.personTags.description.customCategory",
177
+ descriptionFallback: `Custom CRM category: ${label}.`,
178
+ routeKind: kind,
179
+ settingKind: kind,
180
+ customFieldKey: resolveCustomCategoryFieldKey(kind),
181
+ selectionMode,
182
+ hasColorDots: true
183
+ };
184
+ }
185
+ function normalizeSelectionValues(value, selectionMode) {
186
+ if (typeof value === "string") {
187
+ const trimmed = value.trim();
188
+ return trimmed.length > 0 ? [trimmed] : [];
189
+ }
190
+ if (Array.isArray(value)) {
191
+ const normalized = value.filter((entry) => typeof entry === "string").map((entry) => entry.trim()).filter((entry) => entry.length > 0);
192
+ if (selectionMode === "single") {
193
+ return normalized.length > 0 ? [normalized[0]] : [];
194
+ }
195
+ return normalized;
196
+ }
197
+ return [];
198
+ }
199
+ function readCategorySelectionValues(category, entityData, selectionMode) {
200
+ if (category.entityField) {
201
+ return normalizeSelectionValues(entityData[category.entityField], selectionMode);
202
+ }
203
+ if (category.customFieldKey) {
204
+ return normalizeSelectionValues(entityData.customFields?.[category.customFieldKey], selectionMode);
205
+ }
206
+ return [];
207
+ }
208
+ function areSelectionsEqual(left, right) {
209
+ if (left.size !== right.size) return false;
210
+ for (const value of left) {
211
+ if (!right.has(value)) return false;
212
+ }
213
+ return true;
214
+ }
215
+ function TagChip({
216
+ label,
217
+ color,
218
+ active,
219
+ showColorDot,
220
+ onClick
221
+ }) {
222
+ const activeColorStyle = active && color ? { color, borderColor: color, backgroundColor: `${color}1A` } : void 0;
223
+ return /* @__PURE__ */ jsxs(
224
+ Button,
225
+ {
226
+ type: "button",
227
+ variant: "ghost",
228
+ size: "sm",
229
+ onClick,
230
+ className: cn(
231
+ "inline-flex h-auto items-center gap-1 rounded-full border px-2.5 py-1.5 transition-colors",
232
+ active ? activeColorStyle ? "font-semibold hover:opacity-90" : "border-transparent bg-muted font-semibold text-foreground hover:bg-muted" : "border-border bg-transparent font-normal text-muted-foreground hover:bg-muted/60 hover:text-foreground"
233
+ ),
234
+ style: activeColorStyle,
235
+ children: [
236
+ showColorDot && color ? /* @__PURE__ */ jsx(
237
+ "span",
238
+ {
239
+ className: "inline-block size-2 shrink-0 rounded-full",
240
+ style: { backgroundColor: color }
241
+ }
242
+ ) : null,
243
+ /* @__PURE__ */ jsx("span", { className: "text-xs", children: label }),
244
+ active ? /* @__PURE__ */ jsx(X, { className: "size-2.5 shrink-0" }) : null
245
+ ]
246
+ }
247
+ );
248
+ }
249
+ function buildApplicableCategories(entityType, kindSettings) {
250
+ const baseCategories = CATEGORY_DEFS.filter((category) => category.supportedEntityTypes.includes(entityType));
251
+ const customCategories = kindSettings.filter((setting) => setting.visibleInTags).filter((setting) => !CATEGORY_DEFS.some((category) => category.kind === setting.kind || category.settingKind === setting.kind)).map((setting) => createCustomCategoryDef(setting.kind, setting.selectionMode));
252
+ return [...baseCategories, ...customCategories];
253
+ }
254
+ function EntityTagsDialog({
255
+ open,
256
+ onClose,
257
+ entityId,
258
+ entityType,
259
+ entityOrganizationId,
260
+ entityData,
261
+ onSaved
262
+ }) {
263
+ const t = useT();
264
+ const [loading, setLoading] = React.useState(true);
265
+ const [saving, setSaving] = React.useState(false);
266
+ const [searchValue, setSearchValue] = React.useState("");
267
+ const [categories, setCategories] = React.useState([]);
268
+ const [selectedEntrySeeds, setSelectedEntrySeeds] = React.useState({});
269
+ const [selectedValues, setSelectedValues] = React.useState({});
270
+ const [originalValues, setOriginalValues] = React.useState({});
271
+ const [activeCategoryKind, setActiveCategoryKind] = React.useState(null);
272
+ const [newEntryInputByKind, setNewEntryInputByKind] = React.useState({});
273
+ const [creatingKind, setCreatingKind] = React.useState(null);
274
+ const [manageTagsOpen, setManageTagsOpen] = React.useState(false);
275
+ const [activeCategoryPage, setActiveCategoryPage] = React.useState(1);
276
+ const [activeCategoryTotalPages, setActiveCategoryTotalPages] = React.useState(1);
277
+ const [activeCategoryLoading, setActiveCategoryLoading] = React.useState(false);
278
+ const creationInFlightRef = React.useRef(null);
279
+ const mutationContextId = React.useMemo(
280
+ () => `customer-tags:${entityType}:${entityId}`,
281
+ [entityId, entityType]
282
+ );
283
+ const { runMutation, retryLastMutation } = useGuardedMutation({
284
+ contextId: mutationContextId,
285
+ blockedMessage: t("ui.forms.flash.saveBlocked", "Save blocked by validation")
286
+ });
287
+ const mutationContext = React.useMemo(
288
+ () => ({
289
+ formId: mutationContextId,
290
+ resourceKind: entityType === "person" ? "customers.person" : "customers.company",
291
+ resourceId: entityId,
292
+ entityType,
293
+ retryLastMutation
294
+ }),
295
+ [entityId, entityType, mutationContextId, retryLastMutation]
296
+ );
297
+ const runGuardedMutation = React.useCallback(
298
+ async (operation, mutationPayload) => runMutation({
299
+ operation,
300
+ mutationPayload,
301
+ context: mutationContext
302
+ }),
303
+ [mutationContext, runMutation]
304
+ );
305
+ const updateCategoryEntries = React.useCallback(
306
+ (kind, updater) => {
307
+ setCategories(
308
+ (previous) => previous.map(
309
+ (category) => category.kind === kind ? { ...category, entries: sortOptions(updater(category.entries)) } : category
310
+ )
311
+ );
312
+ },
313
+ []
314
+ );
315
+ const loadData = React.useCallback(async () => {
316
+ setLoading(true);
317
+ try {
318
+ let kindSettings = [];
319
+ const scopedQuery = new URLSearchParams();
320
+ if (entityOrganizationId) {
321
+ scopedQuery.set("organizationId", entityOrganizationId);
322
+ }
323
+ try {
324
+ const settingsCall = await apiCall(
325
+ `/api/customers/dictionaries/kind-settings${scopedQuery.size ? `?${scopedQuery.toString()}` : ""}`,
326
+ { cache: "no-store", headers: { "x-om-unauthorized-redirect": "0" } }
327
+ );
328
+ if (settingsCall.ok && settingsCall.result?.items) {
329
+ kindSettings = settingsCall.result.items;
330
+ }
331
+ } catch {
332
+ }
333
+ const settingsMap = new Map(kindSettings.map((setting) => [setting.kind, setting]));
334
+ const selectedTagEntries = Array.isArray(entityData.tags) ? entityData.tags.map((tag) => ({
335
+ id: tag.id,
336
+ value: tag.id,
337
+ label: tag.label,
338
+ color: tag.color ?? null
339
+ })) : [];
340
+ let assignedLabelIds = [];
341
+ let selectedLabelEntries = [];
342
+ try {
343
+ const labelsQuery = new URLSearchParams();
344
+ labelsQuery.set("entityId", entityId);
345
+ labelsQuery.set("pageSize", "1");
346
+ if (entityOrganizationId) {
347
+ labelsQuery.set("organizationId", entityOrganizationId);
348
+ }
349
+ const labelsCall = await apiCall(`/api/customers/labels?${labelsQuery.toString()}`, {
350
+ cache: "no-store",
351
+ headers: { "x-om-unauthorized-redirect": "0" }
352
+ });
353
+ const labelsData = labelsCall.ok ? labelsCall.result : null;
354
+ assignedLabelIds = labelsData?.assignedIds ?? [];
355
+ if (assignedLabelIds.length > 0) {
356
+ const detailQuery = new URLSearchParams({
357
+ ids: assignedLabelIds.join(","),
358
+ pageSize: String(Math.min(assignedLabelIds.length, 100))
359
+ });
360
+ if (entityOrganizationId) {
361
+ detailQuery.set("organizationId", entityOrganizationId);
362
+ }
363
+ const selectedLabelsCall = await apiCall(
364
+ `/api/customers/labels?${detailQuery.toString()}`,
365
+ {
366
+ cache: "no-store",
367
+ headers: { "x-om-unauthorized-redirect": "0" }
368
+ }
369
+ );
370
+ const selectedLabels = selectedLabelsCall.ok ? selectedLabelsCall.result?.items ?? [] : [];
371
+ selectedLabelEntries = selectedLabels.map((label) => ({
372
+ id: label.id,
373
+ value: label.id,
374
+ label: label.label,
375
+ color: null
376
+ }));
377
+ }
378
+ } catch {
379
+ assignedLabelIds = [];
380
+ selectedLabelEntries = [];
381
+ }
382
+ const categoryDefs = buildApplicableCategories(entityType, kindSettings);
383
+ const loadedCategories = [];
384
+ for (const categoryDef of categoryDefs) {
385
+ if (categoryDef.source === "tags") {
386
+ loadedCategories.push({
387
+ ...categoryDef,
388
+ label: t(categoryDef.labelKey, categoryDef.labelFallback),
389
+ description: t(categoryDef.descriptionKey, categoryDef.descriptionFallback),
390
+ entries: sortOptions(selectedTagEntries),
391
+ selectionMode: categoryDef.selectionMode ?? "multi"
392
+ });
393
+ continue;
394
+ }
395
+ if (categoryDef.source === "labels") {
396
+ loadedCategories.push({
397
+ ...categoryDef,
398
+ label: t(categoryDef.labelKey, categoryDef.labelFallback),
399
+ description: t(categoryDef.descriptionKey, categoryDef.descriptionFallback),
400
+ entries: sortOptions(selectedLabelEntries),
401
+ selectionMode: categoryDef.selectionMode ?? "multi"
402
+ });
403
+ continue;
404
+ }
405
+ try {
406
+ const dictionaryUrl = new URL(`/api/customers/dictionaries/${categoryDef.routeKind}`, "http://localhost");
407
+ if (entityOrganizationId) {
408
+ dictionaryUrl.searchParams.set("organizationId", entityOrganizationId);
409
+ }
410
+ const dictionaryCall = await apiCall(
411
+ `${dictionaryUrl.pathname}${dictionaryUrl.search}`,
412
+ { cache: "no-store", headers: { "x-om-unauthorized-redirect": "0" } }
413
+ );
414
+ const dictionaryItems = dictionaryCall.ok ? dictionaryCall.result?.items ?? [] : [];
415
+ const entries = dictionaryItems.map((entry) => ({
416
+ id: entry.id,
417
+ value: entry.value,
418
+ label: entry.label,
419
+ color: entry.color ?? null
420
+ }));
421
+ const setting = categoryDef.settingKind ? settingsMap.get(categoryDef.settingKind) : void 0;
422
+ const selectionMode = setting?.selectionMode ?? categoryDef.selectionMode ?? "single";
423
+ const currentValues = readCategorySelectionValues(categoryDef, entityData, selectionMode);
424
+ currentValues.forEach((currentValue) => {
425
+ if (entries.some((entry) => entry.value === currentValue)) return;
426
+ entries.push({
427
+ id: `current:${categoryDef.kind}:${currentValue}`,
428
+ value: currentValue,
429
+ label: currentValue,
430
+ color: null
431
+ });
432
+ });
433
+ loadedCategories.push({
434
+ ...categoryDef,
435
+ label: categoryDef.labelKey ? t(categoryDef.labelKey, categoryDef.labelFallback) : categoryDef.labelFallback,
436
+ description: categoryDef.descriptionKey ? t(
437
+ categoryDef.descriptionKey,
438
+ categoryDef.descriptionFallback,
439
+ categoryDef.customFieldKey ? { name: categoryDef.labelFallback } : void 0
440
+ ) : categoryDef.descriptionFallback,
441
+ entries: sortOptions(entries),
442
+ selectionMode
443
+ });
444
+ } catch {
445
+ const setting = categoryDef.settingKind ? settingsMap.get(categoryDef.settingKind) : void 0;
446
+ const selectionMode = setting?.selectionMode ?? categoryDef.selectionMode ?? "single";
447
+ const fallbackEntries = readCategorySelectionValues(categoryDef, entityData, selectionMode).map((value) => ({
448
+ id: `current:${categoryDef.kind}:${value}`,
449
+ value,
450
+ label: value,
451
+ color: null
452
+ }));
453
+ loadedCategories.push({
454
+ ...categoryDef,
455
+ label: categoryDef.labelKey ? t(categoryDef.labelKey, categoryDef.labelFallback) : categoryDef.labelFallback,
456
+ description: categoryDef.descriptionKey ? t(
457
+ categoryDef.descriptionKey,
458
+ categoryDef.descriptionFallback,
459
+ categoryDef.customFieldKey ? { name: categoryDef.labelFallback } : void 0
460
+ ) : categoryDef.descriptionFallback,
461
+ entries: fallbackEntries,
462
+ selectionMode
463
+ });
464
+ }
465
+ }
466
+ loadedCategories.sort((left, right) => {
467
+ const leftSortOrder = left.settingKind && settingsMap.has(left.settingKind) ? settingsMap.get(left.settingKind)?.sortOrder ?? 1e3 : 1e3 + CATEGORY_DEFS.findIndex((category) => category.kind === left.kind);
468
+ const rightSortOrder = right.settingKind && settingsMap.has(right.settingKind) ? settingsMap.get(right.settingKind)?.sortOrder ?? 1e3 : 1e3 + CATEGORY_DEFS.findIndex((category) => category.kind === right.kind);
469
+ return leftSortOrder - rightSortOrder;
470
+ });
471
+ const initialValues = {};
472
+ for (const category of loadedCategories) {
473
+ if (category.source === "dictionary" && (category.entityField || category.customFieldKey)) {
474
+ initialValues[category.kind] = new Set(
475
+ readCategorySelectionValues(category, entityData, category.selectionMode)
476
+ );
477
+ continue;
478
+ }
479
+ if (category.source === "tags") {
480
+ const availableTagIds = new Set(category.entries.map((entry) => entry.value));
481
+ const assignedTagIds = Array.isArray(entityData.tags) ? entityData.tags.map((tag) => tag.id).filter((tagId) => availableTagIds.has(tagId)) : [];
482
+ initialValues[category.kind] = new Set(assignedTagIds);
483
+ continue;
484
+ }
485
+ if (category.source === "labels") {
486
+ const availableLabelIds = new Set(category.entries.map((entry) => entry.value));
487
+ initialValues[category.kind] = new Set(
488
+ assignedLabelIds.filter((labelId) => availableLabelIds.has(labelId))
489
+ );
490
+ continue;
491
+ }
492
+ initialValues[category.kind] = /* @__PURE__ */ new Set();
493
+ }
494
+ setCategories(loadedCategories);
495
+ setSelectedEntrySeeds({
496
+ tags: selectedTagEntries,
497
+ labels: selectedLabelEntries
498
+ });
499
+ setSelectedValues(initialValues);
500
+ setOriginalValues(cloneSelectionMap(initialValues));
501
+ setActiveCategoryKind((previous) => {
502
+ if (previous && loadedCategories.some((category) => category.kind === previous)) {
503
+ return previous;
504
+ }
505
+ return loadedCategories[0]?.kind ?? null;
506
+ });
507
+ } finally {
508
+ setLoading(false);
509
+ }
510
+ }, [entityData, entityId, entityOrganizationId, entityType, t]);
511
+ React.useEffect(() => {
512
+ if (!open) return;
513
+ setSearchValue("");
514
+ setNewEntryInputByKind({});
515
+ loadData().catch((err) => console.warn("[EntityTagsDialog] loadData failed", err));
516
+ }, [loadData, open]);
517
+ React.useEffect(() => {
518
+ if (open) return;
519
+ setManageTagsOpen(false);
520
+ }, [open]);
521
+ const activeCategory = React.useMemo(() => {
522
+ if (!categories.length) return null;
523
+ return categories.find((category) => category.kind === activeCategoryKind) ?? categories[0];
524
+ }, [activeCategoryKind, categories]);
525
+ const activeCategoryKindValue = activeCategory?.kind ?? null;
526
+ const activeCategorySource = activeCategory?.source ?? null;
527
+ React.useEffect(() => {
528
+ if (!open) return;
529
+ setActiveCategoryPage(1);
530
+ }, [activeCategoryKind, open, searchValue]);
531
+ React.useEffect(() => {
532
+ if (!open || !activeCategoryKindValue || activeCategorySource !== "tags" && activeCategorySource !== "labels") {
533
+ setActiveCategoryLoading(false);
534
+ setActiveCategoryTotalPages(1);
535
+ return;
536
+ }
537
+ let cancelled = false;
538
+ const params = new URLSearchParams({
539
+ page: String(activeCategoryPage),
540
+ pageSize: String(REMOTE_CATEGORY_PAGE_SIZE)
541
+ });
542
+ if (searchValue.trim().length > 0) {
543
+ params.set("search", searchValue.trim());
544
+ }
545
+ if (activeCategorySource === "labels") {
546
+ params.set("entityId", entityId);
547
+ if (entityOrganizationId) {
548
+ params.set("organizationId", entityOrganizationId);
549
+ }
550
+ }
551
+ const endpoint = activeCategorySource === "tags" ? `/api/customers/tags?${params.toString()}` : `/api/customers/labels?${params.toString()}`;
552
+ const seedEntries = searchValue.trim().length > 0 ? [] : selectedEntrySeeds[activeCategoryKindValue] ?? [];
553
+ const mapEntries = (items) => items.map((entry) => ({
554
+ id: entry.id,
555
+ value: entry.id,
556
+ label: entry.label,
557
+ color: activeCategorySource === "tags" && "color" in entry ? entry.color ?? null : null
558
+ }));
559
+ setActiveCategoryLoading(true);
560
+ void apiCall(endpoint, {
561
+ cache: "no-store",
562
+ headers: { "x-om-unauthorized-redirect": "0" }
563
+ }).then((response) => {
564
+ if (!response.ok || cancelled) return;
565
+ const fetchedEntries = mapEntries(Array.isArray(response.result?.items) ? response.result.items : []);
566
+ setActiveCategoryTotalPages(
567
+ typeof response.result?.totalPages === "number" ? response.result.totalPages : 1
568
+ );
569
+ updateCategoryEntries(
570
+ activeCategoryKindValue,
571
+ (currentEntries) => activeCategoryPage <= 1 ? mergeOptions(seedEntries, fetchedEntries) : mergeOptions(currentEntries, seedEntries, fetchedEntries)
572
+ );
573
+ }).catch(() => {
574
+ if (cancelled) return;
575
+ setActiveCategoryTotalPages(1);
576
+ updateCategoryEntries(activeCategoryKindValue, () => seedEntries);
577
+ }).finally(() => {
578
+ if (!cancelled) {
579
+ setActiveCategoryLoading(false);
580
+ }
581
+ });
582
+ return () => {
583
+ cancelled = true;
584
+ };
585
+ }, [
586
+ activeCategoryKindValue,
587
+ activeCategoryPage,
588
+ activeCategorySource,
589
+ entityId,
590
+ entityOrganizationId,
591
+ open,
592
+ searchValue,
593
+ selectedEntrySeeds,
594
+ updateCategoryEntries
595
+ ]);
596
+ const activeCount = React.useMemo(
597
+ () => Object.values(selectedValues).reduce((count, values) => count + values.size, 0),
598
+ [selectedValues]
599
+ );
600
+ const hasChanges = React.useMemo(() => {
601
+ const kinds = /* @__PURE__ */ new Set([
602
+ ...Object.keys(selectedValues),
603
+ ...Object.keys(originalValues)
604
+ ]);
605
+ for (const kind of kinds) {
606
+ const current = selectedValues[kind] ?? /* @__PURE__ */ new Set();
607
+ const original = originalValues[kind] ?? /* @__PURE__ */ new Set();
608
+ if (current.size !== original.size) return true;
609
+ for (const value of current) {
610
+ if (!original.has(value)) return true;
611
+ }
612
+ }
613
+ return false;
614
+ }, [originalValues, selectedValues]);
615
+ const filteredEntries = React.useMemo(() => {
616
+ if (!activeCategory) return [];
617
+ const query = searchValue.trim().toLowerCase();
618
+ if (!query) return activeCategory.entries;
619
+ return activeCategory.entries.filter(
620
+ (entry) => entry.label.toLowerCase().includes(query) || entry.value.toLowerCase().includes(query)
621
+ );
622
+ }, [activeCategory, searchValue]);
623
+ const toggleValue = React.useCallback(
624
+ (kind, value, selectionMode) => {
625
+ setSelectedValues((previous) => {
626
+ const next = new Set(previous[kind] ?? []);
627
+ if (next.has(value)) {
628
+ next.delete(value);
629
+ } else {
630
+ if (selectionMode === "single") {
631
+ next.clear();
632
+ }
633
+ next.add(value);
634
+ }
635
+ return { ...previous, [kind]: next };
636
+ });
637
+ },
638
+ []
639
+ );
640
+ const handleCreateEntry = React.useCallback(async () => {
641
+ if (!activeCategory || !activeCategory.supportsCreate) return;
642
+ const draftValue = newEntryInputByKind[activeCategory.kind] ?? "";
643
+ const trimmed = draftValue.trim();
644
+ if (!trimmed || creatingKind || creationInFlightRef.current) {
645
+ if (!trimmed) {
646
+ setNewEntryInputByKind((previous) => ({ ...previous, [activeCategory.kind]: null }));
647
+ }
648
+ return;
649
+ }
650
+ const existingEntry = activeCategory.entries.find(
651
+ (entry) => entry.label.trim().toLowerCase() === trimmed.toLowerCase()
652
+ );
653
+ if (existingEntry) {
654
+ setSelectedValues((previous) => {
655
+ const current = new Set(previous[activeCategory.kind] ?? []);
656
+ current.add(existingEntry.value);
657
+ return { ...previous, [activeCategory.kind]: current };
658
+ });
659
+ setNewEntryInputByKind((previous) => ({ ...previous, [activeCategory.kind]: null }));
660
+ return;
661
+ }
662
+ creationInFlightRef.current = activeCategory.kind;
663
+ setCreatingKind(activeCategory.kind);
664
+ try {
665
+ if (activeCategory.source === "tags") {
666
+ const result = await runGuardedMutation(
667
+ () => readApiResultOrThrow(
668
+ "/api/customers/tags",
669
+ {
670
+ method: "POST",
671
+ headers: { "content-type": "application/json" },
672
+ body: JSON.stringify({
673
+ label: trimmed,
674
+ slug: slugifyTagLabel(trimmed)
675
+ })
676
+ }
677
+ ),
678
+ { categoryKind: activeCategory.kind, operation: "createTag", label: trimmed }
679
+ );
680
+ const id = typeof result?.id === "string" ? result.id : "";
681
+ if (!id) {
682
+ throw new Error(t("customers.people.detail.tags.createError", "Failed to create tag."));
683
+ }
684
+ const option = {
685
+ id,
686
+ value: id,
687
+ label: trimmed,
688
+ color: typeof result?.color === "string" ? result.color : null
689
+ };
690
+ updateCategoryEntries(activeCategory.kind, (entries) => [...entries, option]);
691
+ setSelectedEntrySeeds((previous) => ({
692
+ ...previous,
693
+ [activeCategory.kind]: mergeOptions(previous[activeCategory.kind] ?? [], [option])
694
+ }));
695
+ setSelectedValues((previous) => ({
696
+ ...previous,
697
+ [activeCategory.kind]: /* @__PURE__ */ new Set([...previous[activeCategory.kind] ?? [], option.value])
698
+ }));
699
+ } else if (activeCategory.source === "labels") {
700
+ const payload = entityOrganizationId ? { label: trimmed, organizationId: entityOrganizationId } : { label: trimmed };
701
+ const result = await runGuardedMutation(
702
+ () => readApiResultOrThrow(
703
+ "/api/customers/labels",
704
+ {
705
+ method: "POST",
706
+ headers: { "content-type": "application/json" },
707
+ body: JSON.stringify(payload)
708
+ }
709
+ ),
710
+ { categoryKind: activeCategory.kind, operation: "createLabel", label: trimmed }
711
+ );
712
+ const option = {
713
+ id: result.id,
714
+ value: result.id,
715
+ label: result.label,
716
+ color: null
717
+ };
718
+ updateCategoryEntries(activeCategory.kind, (entries) => [...entries, option]);
719
+ setSelectedEntrySeeds((previous) => ({
720
+ ...previous,
721
+ [activeCategory.kind]: mergeOptions(previous[activeCategory.kind] ?? [], [option])
722
+ }));
723
+ setSelectedValues((previous) => ({
724
+ ...previous,
725
+ [activeCategory.kind]: /* @__PURE__ */ new Set([...previous[activeCategory.kind] ?? [], option.value])
726
+ }));
727
+ }
728
+ setNewEntryInputByKind((previous) => ({ ...previous, [activeCategory.kind]: null }));
729
+ } catch (error) {
730
+ const message = error instanceof Error ? error.message : t("customers.personTags.createLabelError", "Failed to create label");
731
+ flash(message, "error");
732
+ } finally {
733
+ creationInFlightRef.current = null;
734
+ setCreatingKind(null);
735
+ }
736
+ }, [activeCategory, creatingKind, entityOrganizationId, newEntryInputByKind, runGuardedMutation, t, updateCategoryEntries]);
737
+ const handleSave = React.useCallback(async () => {
738
+ if (saving) return;
739
+ setSaving(true);
740
+ try {
741
+ const entityUpdate = {};
742
+ const customFieldUpdate = {};
743
+ categories.forEach((category) => {
744
+ if (category.source !== "dictionary") return;
745
+ const currentSelection = selectedValues[category.kind] ?? /* @__PURE__ */ new Set();
746
+ const originalSelection = originalValues[category.kind] ?? /* @__PURE__ */ new Set();
747
+ if (areSelectionsEqual(currentSelection, originalSelection)) return;
748
+ if (category.entityField) {
749
+ const currentValue = currentSelection.size > 0 ? Array.from(currentSelection)[0] ?? null : null;
750
+ entityUpdate[category.entityField] = currentValue;
751
+ return;
752
+ }
753
+ if (category.customFieldKey) {
754
+ customFieldUpdate[category.customFieldKey] = category.selectionMode === "single" ? Array.from(currentSelection)[0] ?? null : Array.from(currentSelection);
755
+ }
756
+ });
757
+ if (Object.keys(entityUpdate).length > 0 || Object.keys(customFieldUpdate).length > 0) {
758
+ await runGuardedMutation(
759
+ () => apiCallOrThrow(`/api/customers/${entityType === "person" ? "people" : "companies"}`, {
760
+ method: "PUT",
761
+ headers: { "content-type": "application/json" },
762
+ body: JSON.stringify({
763
+ id: entityId,
764
+ ...entityUpdate,
765
+ ...Object.keys(customFieldUpdate).length > 0 ? {
766
+ customFields: {
767
+ ...entityData.customFields ?? {},
768
+ ...customFieldUpdate
769
+ }
770
+ } : {}
771
+ })
772
+ }),
773
+ { operation: "updateEntityTags", entityUpdate, customFieldUpdate }
774
+ );
775
+ }
776
+ const currentTags = selectedValues.tags ?? /* @__PURE__ */ new Set();
777
+ const originalTags = originalValues.tags ?? /* @__PURE__ */ new Set();
778
+ const addedTags = Array.from(currentTags).filter((tagId) => !originalTags.has(tagId));
779
+ const removedTags = Array.from(originalTags).filter((tagId) => !currentTags.has(tagId));
780
+ for (const tagId of addedTags) {
781
+ await runGuardedMutation(
782
+ () => apiCallOrThrow("/api/customers/tags/assign", {
783
+ method: "POST",
784
+ headers: { "content-type": "application/json" },
785
+ body: JSON.stringify({ tagId, entityId })
786
+ }),
787
+ { operation: "assignTag", tagId }
788
+ );
789
+ }
790
+ for (const tagId of removedTags) {
791
+ await runGuardedMutation(
792
+ () => apiCallOrThrow("/api/customers/tags/unassign", {
793
+ method: "POST",
794
+ headers: { "content-type": "application/json" },
795
+ body: JSON.stringify({ tagId, entityId })
796
+ }),
797
+ { operation: "unassignTag", tagId }
798
+ );
799
+ }
800
+ const currentLabels = selectedValues.labels ?? /* @__PURE__ */ new Set();
801
+ const originalLabels = originalValues.labels ?? /* @__PURE__ */ new Set();
802
+ const addedLabels = Array.from(currentLabels).filter((labelId) => !originalLabels.has(labelId));
803
+ const removedLabels = Array.from(originalLabels).filter((labelId) => !currentLabels.has(labelId));
804
+ for (const labelId of addedLabels) {
805
+ const payload = entityOrganizationId ? { labelId, entityId, organizationId: entityOrganizationId } : { labelId, entityId };
806
+ await runGuardedMutation(
807
+ () => apiCallOrThrow("/api/customers/labels/assign", {
808
+ method: "POST",
809
+ headers: { "content-type": "application/json" },
810
+ body: JSON.stringify(payload)
811
+ }),
812
+ { operation: "assignLabel", labelId }
813
+ );
814
+ }
815
+ for (const labelId of removedLabels) {
816
+ const payload = entityOrganizationId ? { labelId, entityId, organizationId: entityOrganizationId } : { labelId, entityId };
817
+ await runGuardedMutation(
818
+ () => apiCallOrThrow("/api/customers/labels/unassign", {
819
+ method: "POST",
820
+ headers: { "content-type": "application/json" },
821
+ body: JSON.stringify(payload)
822
+ }),
823
+ { operation: "unassignLabel", labelId }
824
+ );
825
+ }
826
+ flash(t("customers.personTags.saveSuccess", "Tags updated."), "success");
827
+ onSaved?.();
828
+ onClose();
829
+ } catch (error) {
830
+ const message = error instanceof Error ? error.message : t("customers.personTags.saveError", "Failed to save tags");
831
+ flash(message, "error");
832
+ } finally {
833
+ setSaving(false);
834
+ }
835
+ }, [
836
+ categories,
837
+ entityId,
838
+ entityOrganizationId,
839
+ entityType,
840
+ onClose,
841
+ onSaved,
842
+ originalValues,
843
+ saving,
844
+ selectedValues,
845
+ runGuardedMutation,
846
+ t
847
+ ]);
848
+ React.useEffect(() => {
849
+ if (!open) return;
850
+ const handler = (event) => {
851
+ if ((event.metaKey || event.ctrlKey) && event.key === "Enter") {
852
+ event.preventDefault();
853
+ void handleSave();
854
+ }
855
+ };
856
+ window.addEventListener("keydown", handler);
857
+ return () => window.removeEventListener("keydown", handler);
858
+ }, [handleSave, open]);
859
+ React.useEffect(() => {
860
+ if (!open) return;
861
+ setSearchValue("");
862
+ }, [activeCategoryKind, open]);
863
+ const activeSelection = activeCategory ? selectedValues[activeCategory.kind] ?? /* @__PURE__ */ new Set() : /* @__PURE__ */ new Set();
864
+ return /* @__PURE__ */ jsxs(Dialog, { open, onOpenChange: (next) => {
865
+ if (!next) onClose();
866
+ }, children: [
867
+ /* @__PURE__ */ jsxs(
868
+ DialogContent,
869
+ {
870
+ className: "flex max-h-[85vh] flex-col overflow-hidden border-border bg-background p-0 shadow-[0px_16px_40px_0px_rgba(0,0,0,0.14)] sm:max-w-[760px] sm:rounded-xl [&>[data-dialog-close]]:hidden",
871
+ "aria-describedby": void 0,
872
+ children: [
873
+ /* @__PURE__ */ jsx(VisuallyHidden, { children: /* @__PURE__ */ jsx(DialogTitle, { children: t("customers.personTags.title", "Edit tags") }) }),
874
+ /* @__PURE__ */ jsxs("div", { className: "flex shrink-0 items-center justify-between border-b border-border bg-card px-5 py-4", children: [
875
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
876
+ /* @__PURE__ */ jsx(Tag, { className: "size-4 text-foreground" }),
877
+ /* @__PURE__ */ jsx("span", { className: "text-sm font-bold text-foreground", children: t("customers.personTags.title", "Edit tags") })
878
+ ] }),
879
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
880
+ /* @__PURE__ */ jsxs(
881
+ Button,
882
+ {
883
+ type: "button",
884
+ variant: "outline",
885
+ size: "sm",
886
+ className: "h-auto gap-2 rounded-lg px-3.5 py-2 text-sm font-medium",
887
+ onClick: () => setManageTagsOpen(true),
888
+ children: [
889
+ /* @__PURE__ */ jsx(SlidersHorizontal, { className: "size-3.5" }),
890
+ t("customers.personTags.settingsButton", "Tag settings")
891
+ ]
892
+ }
893
+ ),
894
+ /* @__PURE__ */ jsx(
895
+ IconButton,
896
+ {
897
+ type: "button",
898
+ variant: "outline",
899
+ size: "xs",
900
+ className: "size-7 rounded-sm border-border bg-background",
901
+ onClick: onClose,
902
+ children: /* @__PURE__ */ jsx(X, { className: "size-3.5" })
903
+ }
904
+ )
905
+ ] })
906
+ ] }),
907
+ /* @__PURE__ */ jsx("div", { className: "min-h-0 flex-1 overflow-hidden bg-background", children: loading ? /* @__PURE__ */ jsx("div", { className: "flex h-full items-center justify-center px-6 py-8 text-center text-sm text-muted-foreground", children: t("customers.personTags.loading", "Loading...") }) : /* @__PURE__ */ jsxs("div", { className: "flex h-full flex-col gap-4 px-5 py-4 md:flex-row", children: [
908
+ /* @__PURE__ */ jsx("div", { className: "flex gap-2 overflow-x-auto pb-1 md:w-[220px] md:shrink-0 md:flex-col md:overflow-x-visible md:pb-0", children: categories.map((category) => {
909
+ const count = selectedValues[category.kind]?.size ?? 0;
910
+ const isActive = activeCategory?.kind === category.kind;
911
+ return /* @__PURE__ */ jsxs(
912
+ Button,
913
+ {
914
+ type: "button",
915
+ variant: isActive ? "secondary" : "ghost",
916
+ size: "sm",
917
+ className: cn(
918
+ "h-auto min-w-[140px] justify-between rounded-lg px-3 py-2 text-left md:w-full",
919
+ isActive ? "border border-border bg-muted text-foreground" : "border border-transparent text-muted-foreground"
920
+ ),
921
+ onClick: () => setActiveCategoryKind(category.kind),
922
+ children: [
923
+ /* @__PURE__ */ jsx("span", { className: "truncate text-xs font-medium", children: category.label }),
924
+ /* @__PURE__ */ jsx("span", { className: "ml-3 shrink-0 rounded-full bg-background px-2 py-0.5 text-xs font-semibold text-muted-foreground", children: count })
925
+ ]
926
+ },
927
+ category.kind
928
+ );
929
+ }) }),
930
+ /* @__PURE__ */ jsx("div", { className: "min-h-0 flex-1 overflow-y-auto rounded-xl border border-border bg-card", children: activeCategory ? /* @__PURE__ */ jsxs("div", { className: "flex min-h-full flex-col", children: [
931
+ /* @__PURE__ */ jsx("div", { className: "border-b border-border px-4 py-4", children: /* @__PURE__ */ jsxs("div", { className: "flex flex-wrap items-start justify-between gap-3", children: [
932
+ /* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
933
+ /* @__PURE__ */ jsx("h3", { className: "text-sm font-semibold text-foreground", children: activeCategory.label }),
934
+ /* @__PURE__ */ jsx("p", { className: "max-w-[520px] text-xs leading-5 text-muted-foreground", children: activeCategory.description })
935
+ ] }),
936
+ /* @__PURE__ */ jsx("div", { className: "shrink-0 rounded-full border border-border bg-background px-3 py-1 text-xs font-medium text-muted-foreground", children: t("customers.personTags.activeCount", "{{count}} selected", {
937
+ count: activeSelection.size
938
+ }) })
939
+ ] }) }),
940
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 space-y-4 overflow-y-auto px-4 py-4", children: [
941
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 rounded-lg border border-input bg-background px-3 py-2", children: [
942
+ /* @__PURE__ */ jsx(Search, { className: "size-3.5 shrink-0 text-muted-foreground" }),
943
+ /* @__PURE__ */ jsx(
944
+ "input",
945
+ {
946
+ type: "text",
947
+ value: searchValue,
948
+ onChange: (event) => setSearchValue(event.target.value),
949
+ placeholder: t(
950
+ "customers.personTags.searchPlaceholder",
951
+ "Search {{category}}...",
952
+ { category: activeCategory.label.toLowerCase() }
953
+ ),
954
+ className: "flex-1 bg-transparent text-sm text-foreground outline-none placeholder:text-muted-foreground"
955
+ }
956
+ )
957
+ ] }),
958
+ filteredEntries.length > 0 ? /* @__PURE__ */ jsxs("div", { className: "space-y-3", children: [
959
+ /* @__PURE__ */ jsx("div", { className: "flex flex-wrap gap-x-1.5 gap-y-2", children: filteredEntries.map((entry) => /* @__PURE__ */ jsx(
960
+ TagChip,
961
+ {
962
+ label: entry.label,
963
+ color: entry.color,
964
+ active: activeSelection.has(entry.value),
965
+ showColorDot: activeCategory.hasColorDots,
966
+ onClick: () => toggleValue(
967
+ activeCategory.kind,
968
+ entry.value,
969
+ activeCategory.selectionMode
970
+ )
971
+ },
972
+ `${activeCategory.kind}:${entry.value}`
973
+ )) }),
974
+ activeCategoryLoading ? /* @__PURE__ */ jsx("div", { className: "text-xs text-muted-foreground", children: t("customers.personTags.loading", "Loading...") }) : null,
975
+ (activeCategory.source === "tags" || activeCategory.source === "labels") && activeCategoryPage < activeCategoryTotalPages ? /* @__PURE__ */ jsx(
976
+ Button,
977
+ {
978
+ type: "button",
979
+ variant: "outline",
980
+ size: "sm",
981
+ className: "rounded-lg px-3 text-xs",
982
+ onClick: () => setActiveCategoryPage((current) => current + 1),
983
+ children: t("customers.activities.loadMore", "Load more")
984
+ }
985
+ ) : null
986
+ ] }) : activeCategoryLoading ? /* @__PURE__ */ jsx("div", { className: "rounded-lg border border-dashed border-border bg-background px-4 py-6 text-center text-sm text-muted-foreground", children: t("customers.personTags.loading", "Loading...") }) : /* @__PURE__ */ jsx("div", { className: "rounded-lg border border-dashed border-border bg-background px-4 py-6 text-center text-sm text-muted-foreground", children: searchValue.trim().length > 0 ? t("customers.personTags.emptySearchResults", "No options match the current search.") : activeCategory.source === "dictionary" ? t("customers.personTags.emptyDictionaryCategory", "No options are configured for this category yet.") : t("customers.personTags.emptyCategory", "No items have been added for this category yet.") }),
987
+ activeCategory.supportsCreate ? /* @__PURE__ */ jsx("div", { children: newEntryInputByKind[activeCategory.kind] !== null ? /* @__PURE__ */ jsx("div", { className: "inline-flex items-center rounded-full border border-dashed border-status-success-border bg-status-success-bg/70 px-2.5 py-1", children: /* @__PURE__ */ jsx(
988
+ "input",
989
+ {
990
+ type: "text",
991
+ autoFocus: true,
992
+ value: newEntryInputByKind[activeCategory.kind] ?? "",
993
+ disabled: creatingKind === activeCategory.kind,
994
+ onChange: (event) => setNewEntryInputByKind((previous) => ({
995
+ ...previous,
996
+ [activeCategory.kind]: event.target.value
997
+ })),
998
+ onKeyDown: (event) => {
999
+ if (event.key === "Enter") {
1000
+ event.preventDefault();
1001
+ void handleCreateEntry();
1002
+ }
1003
+ if (event.key === "Escape") {
1004
+ setNewEntryInputByKind((previous) => ({
1005
+ ...previous,
1006
+ [activeCategory.kind]: null
1007
+ }));
1008
+ }
1009
+ },
1010
+ onBlur: () => {
1011
+ const value = newEntryInputByKind[activeCategory.kind] ?? "";
1012
+ if (creationInFlightRef.current === activeCategory.kind) {
1013
+ return;
1014
+ }
1015
+ if (value.trim()) {
1016
+ void handleCreateEntry();
1017
+ } else {
1018
+ setNewEntryInputByKind((previous) => ({
1019
+ ...previous,
1020
+ [activeCategory.kind]: null
1021
+ }));
1022
+ }
1023
+ },
1024
+ placeholder: activeCategory.kind === "tags" ? t("customers.people.detail.tags.placeholder", "Type to add tags") : t("customers.personTags.newLabelPlaceholder", "Label name..."),
1025
+ className: "w-[150px] bg-transparent text-xs font-semibold text-status-success-text outline-none placeholder:text-status-success-text/60 disabled:cursor-wait disabled:opacity-70"
1026
+ }
1027
+ ) }) : /* @__PURE__ */ jsxs(
1028
+ Button,
1029
+ {
1030
+ type: "button",
1031
+ variant: "ghost",
1032
+ size: "sm",
1033
+ disabled: creatingKind === activeCategory.kind,
1034
+ onClick: () => setNewEntryInputByKind((previous) => ({
1035
+ ...previous,
1036
+ [activeCategory.kind]: ""
1037
+ })),
1038
+ className: "inline-flex h-auto items-center gap-1 rounded-full border border-dashed border-status-success-border bg-transparent px-2.5 py-1.5 font-semibold text-status-success-text hover:bg-status-success-bg disabled:opacity-60",
1039
+ children: [
1040
+ /* @__PURE__ */ jsx(Plus, { className: "size-2.5" }),
1041
+ /* @__PURE__ */ jsx("span", { className: "text-xs", children: activeCategory.kind === "tags" ? t("customers.personTags.newTag", "New tag") : t("customers.personTags.newLabel", "New label") })
1042
+ ]
1043
+ }
1044
+ ) }) : null
1045
+ ] })
1046
+ ] }) : /* @__PURE__ */ jsx("div", { className: "flex h-full items-center justify-center px-6 py-8 text-center text-sm text-muted-foreground", children: t("customers.personTags.emptyCategory", "No items have been added for this category yet.") }) })
1047
+ ] }) }),
1048
+ /* @__PURE__ */ jsxs("div", { className: "flex shrink-0 items-center justify-between border-t border-border bg-muted/20 px-5 py-3.5", children: [
1049
+ /* @__PURE__ */ jsx("span", { className: "text-xs text-muted-foreground", children: t("customers.personTags.activeCount", "{{count}} selected", { count: activeCount }) }),
1050
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-6", children: [
1051
+ /* @__PURE__ */ jsx(
1052
+ Button,
1053
+ {
1054
+ type: "button",
1055
+ variant: "outline",
1056
+ onClick: onClose,
1057
+ className: "rounded-md border-border bg-background px-4 py-2 text-sm font-semibold text-foreground",
1058
+ children: t("customers.personTags.cancel", "Cancel")
1059
+ }
1060
+ ),
1061
+ /* @__PURE__ */ jsxs(
1062
+ Button,
1063
+ {
1064
+ type: "button",
1065
+ onClick: () => {
1066
+ void handleSave();
1067
+ },
1068
+ disabled: saving || !hasChanges,
1069
+ className: "rounded-md bg-foreground px-4 py-2 text-sm font-semibold text-background hover:bg-foreground/90",
1070
+ children: [
1071
+ /* @__PURE__ */ jsx(Check, { className: "mr-2 size-3.5" }),
1072
+ saving ? t("customers.personTags.saving", "Saving...") : t("customers.personTags.save", "Save")
1073
+ ]
1074
+ }
1075
+ )
1076
+ ] })
1077
+ ] })
1078
+ ]
1079
+ }
1080
+ ),
1081
+ /* @__PURE__ */ jsx(
1082
+ ManageTagsDialog,
1083
+ {
1084
+ open: manageTagsOpen,
1085
+ onClose: () => {
1086
+ setManageTagsOpen(false);
1087
+ void loadData();
1088
+ }
1089
+ }
1090
+ )
1091
+ ] });
1092
+ }
1093
+ export {
1094
+ EntityTagsDialog
1095
+ };
1096
+ //# sourceMappingURL=EntityTagsDialog.js.map