@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.
- package/.turbo/turbo-build.log +1 -1
- package/dist/generated/entities/action_log/index.js +8 -0
- package/dist/generated/entities/action_log/index.js.map +2 -2
- package/dist/generated/entities/customer_company_billing/index.js +23 -0
- package/dist/generated/entities/customer_company_billing/index.js.map +7 -0
- package/dist/generated/entities/customer_deal/index.js +8 -0
- package/dist/generated/entities/customer_deal/index.js.map +2 -2
- package/dist/generated/entities/customer_deal_stage_transition/index.js +31 -0
- package/dist/generated/entities/customer_deal_stage_transition/index.js.map +7 -0
- package/dist/generated/entities/customer_dictionary_kind_setting/index.js +21 -0
- package/dist/generated/entities/customer_dictionary_kind_setting/index.js.map +7 -0
- package/dist/generated/entities/customer_entity/index.js +8 -0
- package/dist/generated/entities/customer_entity/index.js.map +2 -2
- package/dist/generated/entities/customer_entity_role/index.js +23 -0
- package/dist/generated/entities/customer_entity_role/index.js.map +7 -0
- package/dist/generated/entities/customer_interaction/index.js +23 -1
- package/dist/generated/entities/customer_interaction/index.js.map +2 -2
- package/dist/generated/entities/customer_label/index.js +19 -0
- package/dist/generated/entities/customer_label/index.js.map +7 -0
- package/dist/generated/entities/customer_label_assignment/index.js +17 -0
- package/dist/generated/entities/customer_label_assignment/index.js.map +7 -0
- package/dist/generated/entities/customer_person_company_link/index.js +21 -0
- package/dist/generated/entities/customer_person_company_link/index.js.map +7 -0
- package/dist/generated/entities/customer_person_company_role/index.js +17 -0
- package/dist/generated/entities/customer_person_company_role/index.js.map +7 -0
- package/dist/generated/entities/dictionary_entry/index.js +4 -0
- package/dist/generated/entities/dictionary_entry/index.js.map +2 -2
- package/dist/generated/entities.ids.generated.js +9 -1
- package/dist/generated/entities.ids.generated.js.map +2 -2
- package/dist/generated/entity-fields-registry.js +116 -1
- package/dist/generated/entity-fields-registry.js.map +2 -2
- package/dist/modules/attachments/api/route.js +46 -8
- package/dist/modules/attachments/api/route.js.map +2 -2
- package/dist/modules/audit_logs/api/audit-logs/actions/export/route.js +208 -0
- package/dist/modules/audit_logs/api/audit-logs/actions/export/route.js.map +7 -0
- package/dist/modules/audit_logs/api/audit-logs/actions/route.js +52 -6
- package/dist/modules/audit_logs/api/audit-logs/actions/route.js.map +2 -2
- package/dist/modules/audit_logs/cli.js +62 -0
- package/dist/modules/audit_logs/cli.js.map +7 -0
- package/dist/modules/audit_logs/data/entities.js +21 -1
- package/dist/modules/audit_logs/data/entities.js.map +2 -2
- package/dist/modules/audit_logs/data/validators.js +9 -1
- package/dist/modules/audit_logs/data/validators.js.map +2 -2
- package/dist/modules/audit_logs/lib/changeRows.js +34 -0
- package/dist/modules/audit_logs/lib/changeRows.js.map +7 -0
- package/dist/modules/audit_logs/lib/display-helpers.js +2 -20
- package/dist/modules/audit_logs/lib/display-helpers.js.map +3 -3
- package/dist/modules/audit_logs/lib/projections.js +58 -0
- package/dist/modules/audit_logs/lib/projections.js.map +7 -0
- package/dist/modules/audit_logs/migrations/Migration20260412160533.js +21 -0
- package/dist/modules/audit_logs/migrations/Migration20260412160533.js.map +7 -0
- package/dist/modules/audit_logs/services/actionLogService.js +313 -79
- package/dist/modules/audit_logs/services/actionLogService.js.map +2 -2
- package/dist/modules/customers/acl.js +3 -1
- package/dist/modules/customers/acl.js.map +2 -2
- package/dist/modules/customers/api/activities/route.js +4 -0
- package/dist/modules/customers/api/activities/route.js.map +2 -2
- package/dist/modules/customers/api/assignable-staff/route.js +208 -0
- package/dist/modules/customers/api/assignable-staff/route.js.map +7 -0
- package/dist/modules/customers/api/companies/[id]/people/route.js +205 -0
- package/dist/modules/customers/api/companies/[id]/people/route.js.map +7 -0
- package/dist/modules/customers/api/companies/[id]/roles/route.js +22 -0
- package/dist/modules/customers/api/companies/[id]/roles/route.js.map +7 -0
- package/dist/modules/customers/api/companies/[id]/route.js +374 -32
- package/dist/modules/customers/api/companies/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/companies/route.js +82 -7
- package/dist/modules/customers/api/companies/route.js.map +2 -2
- package/dist/modules/customers/api/deals/[id]/companies/route.js +172 -0
- package/dist/modules/customers/api/deals/[id]/companies/route.js.map +7 -0
- package/dist/modules/customers/api/deals/[id]/people/route.js +156 -0
- package/dist/modules/customers/api/deals/[id]/people/route.js.map +7 -0
- package/dist/modules/customers/api/deals/[id]/route.js +459 -53
- package/dist/modules/customers/api/deals/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/deals/[id]/stats/route.js +195 -0
- package/dist/modules/customers/api/deals/[id]/stats/route.js.map +7 -0
- package/dist/modules/customers/api/deals/route.js +20 -10
- package/dist/modules/customers/api/deals/route.js.map +3 -3
- package/dist/modules/customers/api/dictionaries/[kind]/[id]/route.js +105 -4
- package/dist/modules/customers/api/dictionaries/[kind]/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/[kind]/route.js +118 -42
- package/dist/modules/customers/api/dictionaries/[kind]/route.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/context.js +30 -6
- package/dist/modules/customers/api/dictionaries/context.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/kind-settings/route.js +207 -0
- package/dist/modules/customers/api/dictionaries/kind-settings/route.js.map +7 -0
- package/dist/modules/customers/api/entity-roles-factory.js +471 -0
- package/dist/modules/customers/api/entity-roles-factory.js.map +7 -0
- package/dist/modules/customers/api/interactions/conflicts/route.js +158 -0
- package/dist/modules/customers/api/interactions/conflicts/route.js.map +7 -0
- package/dist/modules/customers/api/interactions/counts/route.js +92 -0
- package/dist/modules/customers/api/interactions/counts/route.js.map +7 -0
- package/dist/modules/customers/api/interactions/route.js +83 -4
- package/dist/modules/customers/api/interactions/route.js.map +2 -2
- package/dist/modules/customers/api/labels/assign/route.js +189 -0
- package/dist/modules/customers/api/labels/assign/route.js.map +7 -0
- package/dist/modules/customers/api/labels/auth.js +17 -0
- package/dist/modules/customers/api/labels/auth.js.map +7 -0
- package/dist/modules/customers/api/labels/route.js +281 -0
- package/dist/modules/customers/api/labels/route.js.map +7 -0
- package/dist/modules/customers/api/labels/table-errors.js +38 -0
- package/dist/modules/customers/api/labels/table-errors.js.map +7 -0
- package/dist/modules/customers/api/labels/unassign/route.js +184 -0
- package/dist/modules/customers/api/labels/unassign/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/companies/[linkId]/route.js +292 -0
- package/dist/modules/customers/api/people/[id]/companies/[linkId]/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/companies/context.js +66 -0
- package/dist/modules/customers/api/people/[id]/companies/context.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/companies/enriched/route.js +334 -0
- package/dist/modules/customers/api/people/[id]/companies/enriched/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/companies/route.js +205 -0
- package/dist/modules/customers/api/people/[id]/companies/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/roles/route.js +22 -0
- package/dist/modules/customers/api/people/[id]/roles/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/route.js +134 -21
- package/dist/modules/customers/api/people/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/people/route.js +122 -23
- package/dist/modules/customers/api/people/route.js.map +2 -2
- package/dist/modules/customers/api/todos/route.js +4 -0
- package/dist/modules/customers/api/todos/route.js.map +2 -2
- package/dist/modules/customers/api/utils.js +22 -0
- package/dist/modules/customers/api/utils.js.map +2 -2
- package/dist/modules/customers/backend/config/customers/page.js +2 -6
- package/dist/modules/customers/backend/config/customers/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies/page.js +37 -26
- package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js +265 -262
- package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js.map +3 -3
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/formatters.js +23 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/formatters.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/types.js +1 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/types.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js +43 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.js +264 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.js +88 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js +41 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js +66 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.js +39 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.js +49 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.js +43 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.js +28 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/page.js +556 -503
- package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +3 -3
- package/dist/modules/customers/backend/customers/deals/page.js +66 -21
- package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people/page.js +36 -28
- package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +318 -203
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +3 -3
- package/dist/modules/customers/cli.js +105 -13
- package/dist/modules/customers/cli.js.map +2 -2
- package/dist/modules/customers/commands/activities.js +6 -0
- package/dist/modules/customers/commands/activities.js.map +2 -2
- package/dist/modules/customers/commands/deals.js +315 -107
- package/dist/modules/customers/commands/deals.js.map +2 -2
- package/dist/modules/customers/commands/dictionaries.js +166 -32
- package/dist/modules/customers/commands/dictionaries.js.map +2 -2
- package/dist/modules/customers/commands/dictionaryKindSettings.js +208 -0
- package/dist/modules/customers/commands/dictionaryKindSettings.js.map +7 -0
- package/dist/modules/customers/commands/entity-roles.js +415 -0
- package/dist/modules/customers/commands/entity-roles.js.map +7 -0
- package/dist/modules/customers/commands/index.js +4 -0
- package/dist/modules/customers/commands/index.js.map +2 -2
- package/dist/modules/customers/commands/interactions.js +108 -21
- package/dist/modules/customers/commands/interactions.js.map +2 -2
- package/dist/modules/customers/commands/labels.js +539 -0
- package/dist/modules/customers/commands/labels.js.map +7 -0
- package/dist/modules/customers/commands/people.js +560 -463
- package/dist/modules/customers/commands/people.js.map +3 -3
- package/dist/modules/customers/commands/personCompanyLinks.js +568 -0
- package/dist/modules/customers/commands/personCompanyLinks.js.map +7 -0
- package/dist/modules/customers/commands/shared.js +12 -4
- package/dist/modules/customers/commands/shared.js.map +2 -2
- package/dist/modules/customers/commands/todos.js +10 -1
- package/dist/modules/customers/commands/todos.js.map +2 -2
- package/dist/modules/customers/components/AddressEditor.js +1 -1
- package/dist/modules/customers/components/AddressEditor.js.map +2 -2
- package/dist/modules/customers/components/CustomersConfigurationSections.js +31 -0
- package/dist/modules/customers/components/CustomersConfigurationSections.js.map +7 -0
- package/dist/modules/customers/components/DictionarySettings.js +37 -2
- package/dist/modules/customers/components/DictionarySettings.js.map +2 -2
- package/dist/modules/customers/components/detail/ActiveDealCard.js +121 -0
- package/dist/modules/customers/components/detail/ActiveDealCard.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivitiesSection.js +222 -331
- package/dist/modules/customers/components/detail/ActivitiesSection.js.map +3 -3
- package/dist/modules/customers/components/detail/ActivityAiActions.js +36 -0
- package/dist/modules/customers/components/detail/ActivityAiActions.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityCard.js +126 -0
- package/dist/modules/customers/components/detail/ActivityCard.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityHistorySection.js +340 -0
- package/dist/modules/customers/components/detail/ActivityHistorySection.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityLogTab.js +56 -0
- package/dist/modules/customers/components/detail/ActivityLogTab.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityTimeline.js +108 -0
- package/dist/modules/customers/components/detail/ActivityTimeline.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityTimelineFilters.js +139 -0
- package/dist/modules/customers/components/detail/ActivityTimelineFilters.js.map +7 -0
- package/dist/modules/customers/components/detail/ActivityTypeSelector.js +42 -0
- package/dist/modules/customers/components/detail/ActivityTypeSelector.js.map +7 -0
- package/dist/modules/customers/components/detail/AiActionChips.js +38 -0
- package/dist/modules/customers/components/detail/AiActionChips.js.map +7 -0
- package/dist/modules/customers/components/detail/AssignRoleDialog.js +534 -0
- package/dist/modules/customers/components/detail/AssignRoleDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/ChangelogEntryRow.js +79 -0
- package/dist/modules/customers/components/detail/ChangelogEntryRow.js.map +7 -0
- package/dist/modules/customers/components/detail/ChangelogFilters.js +176 -0
- package/dist/modules/customers/components/detail/ChangelogFilters.js.map +7 -0
- package/dist/modules/customers/components/detail/ChangelogKpiCards.js +88 -0
- package/dist/modules/customers/components/detail/ChangelogKpiCards.js.map +7 -0
- package/dist/modules/customers/components/detail/ChangelogTab.js +470 -0
- package/dist/modules/customers/components/detail/ChangelogTab.js.map +7 -0
- package/dist/modules/customers/components/detail/ComingSoonPlaceholder.js +16 -0
- package/dist/modules/customers/components/detail/ComingSoonPlaceholder.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyCard.js +283 -0
- package/dist/modules/customers/components/detail/CompanyCard.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyDashboardTab.js +133 -0
- package/dist/modules/customers/components/detail/CompanyDashboardTab.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyDetailHeader.js +191 -0
- package/dist/modules/customers/components/detail/CompanyDetailHeader.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyDetailTabs.js +123 -0
- package/dist/modules/customers/components/detail/CompanyDetailTabs.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyKpiBar.js +174 -0
- package/dist/modules/customers/components/detail/CompanyKpiBar.js.map +7 -0
- package/dist/modules/customers/components/detail/CompanyPeopleSection.js +514 -230
- package/dist/modules/customers/components/detail/CompanyPeopleSection.js.map +2 -2
- package/dist/modules/customers/components/detail/CompanyTagsDialog.js +22 -0
- package/dist/modules/customers/components/detail/CompanyTagsDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js +159 -0
- package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/CreatePersonDialog.js +135 -0
- package/dist/modules/customers/components/detail/CreatePersonDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/DealClosureActionBar.js +59 -0
- package/dist/modules/customers/components/detail/DealClosureActionBar.js.map +7 -0
- package/dist/modules/customers/components/detail/DealDetailHeader.js +237 -0
- package/dist/modules/customers/components/detail/DealDetailHeader.js.map +7 -0
- package/dist/modules/customers/components/detail/DealDetailTabs.js +109 -0
- package/dist/modules/customers/components/detail/DealDetailTabs.js.map +7 -0
- package/dist/modules/customers/components/detail/DealForm.js +219 -92
- package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
- package/dist/modules/customers/components/detail/DealLinkedEntitiesTab.js +295 -0
- package/dist/modules/customers/components/detail/DealLinkedEntitiesTab.js.map +7 -0
- package/dist/modules/customers/components/detail/DealLostSummaryDialog.js +107 -0
- package/dist/modules/customers/components/detail/DealLostSummaryDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/DealWonPopup.js +113 -0
- package/dist/modules/customers/components/detail/DealWonPopup.js.map +7 -0
- package/dist/modules/customers/components/detail/DealsSection.js +206 -193
- package/dist/modules/customers/components/detail/DealsSection.js.map +2 -2
- package/dist/modules/customers/components/detail/DecisionMakersFooter.js +39 -0
- package/dist/modules/customers/components/detail/DecisionMakersFooter.js.map +7 -0
- package/dist/modules/customers/components/detail/EntityTagsDialog.js +1096 -0
- package/dist/modules/customers/components/detail/EntityTagsDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/InlineActivityComposer.js +197 -0
- package/dist/modules/customers/components/detail/InlineActivityComposer.js.map +7 -0
- package/dist/modules/customers/components/detail/ManageTagsDialog.js +1091 -0
- package/dist/modules/customers/components/detail/ManageTagsDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/MiniWeekCalendar.js +272 -0
- package/dist/modules/customers/components/detail/MiniWeekCalendar.js.map +7 -0
- package/dist/modules/customers/components/detail/MobilePersonDetail.js +106 -0
- package/dist/modules/customers/components/detail/MobilePersonDetail.js.map +7 -0
- package/dist/modules/customers/components/detail/NextStepCard.js +72 -0
- package/dist/modules/customers/components/detail/NextStepCard.js.map +7 -0
- package/dist/modules/customers/components/detail/PersonCard.js +192 -0
- package/dist/modules/customers/components/detail/PersonCard.js.map +7 -0
- package/dist/modules/customers/components/detail/PersonCompaniesSection.js +345 -0
- package/dist/modules/customers/components/detail/PersonCompaniesSection.js.map +7 -0
- package/dist/modules/customers/components/detail/PersonDetailHeader.js +220 -0
- package/dist/modules/customers/components/detail/PersonDetailHeader.js.map +7 -0
- package/dist/modules/customers/components/detail/PersonDetailTabs.js +122 -0
- package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +7 -0
- package/dist/modules/customers/components/detail/PersonTagsDialog.js +24 -0
- package/dist/modules/customers/components/detail/PersonTagsDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/PipelineStepper.js +191 -0
- package/dist/modules/customers/components/detail/PipelineStepper.js.map +7 -0
- package/dist/modules/customers/components/detail/PlannedActivitiesSection.js +222 -0
- package/dist/modules/customers/components/detail/PlannedActivitiesSection.js.map +7 -0
- package/dist/modules/customers/components/detail/RelationshipHealthCard.js +49 -0
- package/dist/modules/customers/components/detail/RelationshipHealthCard.js.map +7 -0
- package/dist/modules/customers/components/detail/RoleAssignmentRow.js +189 -0
- package/dist/modules/customers/components/detail/RoleAssignmentRow.js.map +7 -0
- package/dist/modules/customers/components/detail/RolesSection.js +234 -0
- package/dist/modules/customers/components/detail/RolesSection.js.map +7 -0
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +410 -0
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/aiActionCatalog.js +41 -0
- package/dist/modules/customers/components/detail/aiActionCatalog.js.map +7 -0
- package/dist/modules/customers/components/detail/assignableStaff.js +48 -0
- package/dist/modules/customers/components/detail/assignableStaff.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/ActiveDealWidget.js +48 -0
- package/dist/modules/customers/components/detail/dashboard/ActiveDealWidget.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/OpenTasksWidget.js +86 -0
- package/dist/modules/customers/components/detail/dashboard/OpenTasksWidget.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/RecentActivityWidget.js +53 -0
- package/dist/modules/customers/components/detail/dashboard/RecentActivityWidget.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/RelationshipHealthWidget.js +30 -0
- package/dist/modules/customers/components/detail/dashboard/RelationshipHealthWidget.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.js +43 -0
- package/dist/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.js.map +7 -0
- package/dist/modules/customers/components/detail/dashboard/helpers.js +71 -0
- package/dist/modules/customers/components/detail/dashboard/helpers.js.map +7 -0
- package/dist/modules/customers/components/detail/healthScoreUtils.js +69 -0
- package/dist/modules/customers/components/detail/healthScoreUtils.js.map +7 -0
- package/dist/modules/customers/components/detail/hooks/useCurrencyDictionary.js +5 -5
- package/dist/modules/customers/components/detail/hooks/useCurrencyDictionary.js.map +2 -2
- package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js +9 -8
- package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js.map +3 -3
- package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js +65 -0
- package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js.map +7 -0
- package/dist/modules/customers/components/detail/notesAdapter.js +70 -30
- package/dist/modules/customers/components/detail/notesAdapter.js.map +2 -2
- package/dist/modules/customers/components/detail/pipelineStageUtils.js +26 -0
- package/dist/modules/customers/components/detail/pipelineStageUtils.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/DateTimeFields.js +144 -0
- package/dist/modules/customers/components/detail/schedule/DateTimeFields.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/FooterFields.js +60 -0
- package/dist/modules/customers/components/detail/schedule/FooterFields.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/LinkedEntitiesField.js +216 -0
- package/dist/modules/customers/components/detail/schedule/LinkedEntitiesField.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/LocationField.js +34 -0
- package/dist/modules/customers/components/detail/schedule/LocationField.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/ParticipantsField.js +226 -0
- package/dist/modules/customers/components/detail/schedule/ParticipantsField.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/fieldConfig.js +69 -0
- package/dist/modules/customers/components/detail/schedule/fieldConfig.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/index.js +21 -0
- package/dist/modules/customers/components/detail/schedule/index.js.map +7 -0
- package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js +172 -0
- package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js.map +7 -0
- package/dist/modules/customers/components/detail/utils.js +23 -0
- package/dist/modules/customers/components/detail/utils.js.map +2 -2
- package/dist/modules/customers/components/formConfig.js +144 -22
- package/dist/modules/customers/components/formConfig.js.map +2 -2
- package/dist/modules/customers/components/linking/LinkEntityDialog.js +661 -0
- package/dist/modules/customers/components/linking/LinkEntityDialog.js.map +7 -0
- package/dist/modules/customers/components/linking/adapters/companyAdapter.js +252 -0
- package/dist/modules/customers/components/linking/adapters/companyAdapter.js.map +7 -0
- package/dist/modules/customers/components/linking/adapters/dealAdapter.js +384 -0
- package/dist/modules/customers/components/linking/adapters/dealAdapter.js.map +7 -0
- package/dist/modules/customers/components/linking/adapters/personAdapter.js +324 -0
- package/dist/modules/customers/components/linking/adapters/personAdapter.js.map +7 -0
- package/dist/modules/customers/components/list/CollectionPreviewCell.js +53 -0
- package/dist/modules/customers/components/list/CollectionPreviewCell.js.map +7 -0
- package/dist/modules/customers/data/entities.js +407 -1
- package/dist/modules/customers/data/entities.js.map +2 -2
- package/dist/modules/customers/data/validators.js +139 -21
- package/dist/modules/customers/data/validators.js.map +2 -2
- package/dist/modules/customers/events.js +19 -1
- package/dist/modules/customers/events.js.map +2 -2
- package/dist/modules/customers/lib/customerRoleTypes.js +19 -0
- package/dist/modules/customers/lib/customerRoleTypes.js.map +7 -0
- package/dist/modules/customers/lib/dealClosureNotification.js +39 -0
- package/dist/modules/customers/lib/dealClosureNotification.js.map +7 -0
- package/dist/modules/customers/lib/dealStageTransitionTable.js +29 -0
- package/dist/modules/customers/lib/dealStageTransitionTable.js.map +7 -0
- package/dist/modules/customers/lib/dictionaries.js +25 -0
- package/dist/modules/customers/lib/dictionaries.js.map +2 -2
- package/dist/modules/customers/lib/interactionReadModel.js +10 -0
- package/dist/modules/customers/lib/interactionReadModel.js.map +2 -2
- package/dist/modules/customers/lib/personCompanies.js +235 -0
- package/dist/modules/customers/lib/personCompanies.js.map +7 -0
- package/dist/modules/customers/lib/personCompanyLinkTable.js +42 -0
- package/dist/modules/customers/lib/personCompanyLinkTable.js.map +7 -0
- package/dist/modules/customers/lib/roleTypeUsage.js +104 -0
- package/dist/modules/customers/lib/roleTypeUsage.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260406214502.js +18 -0
- package/dist/modules/customers/migrations/Migration20260406214502.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260408135736.js +17 -0
- package/dist/modules/customers/migrations/Migration20260408135736.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260408225345.js +21 -0
- package/dist/modules/customers/migrations/Migration20260408225345.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260411075533.js +27 -0
- package/dist/modules/customers/migrations/Migration20260411075533.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260411103551.js +13 -0
- package/dist/modules/customers/migrations/Migration20260411103551.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260411130944.js +26 -0
- package/dist/modules/customers/migrations/Migration20260411130944.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260415095203.js +13 -0
- package/dist/modules/customers/migrations/Migration20260415095203.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260415135056.js +20 -0
- package/dist/modules/customers/migrations/Migration20260415135056.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260417140000.js +15 -0
- package/dist/modules/customers/migrations/Migration20260417140000.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260417160000.js +17 -0
- package/dist/modules/customers/migrations/Migration20260417160000.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260417235407.js +13 -0
- package/dist/modules/customers/migrations/Migration20260417235407.js.map +7 -0
- package/dist/modules/customers/setup.js +16 -1
- package/dist/modules/customers/setup.js.map +2 -2
- package/dist/modules/customers/subscribers/deal-closure-notification.js +16 -0
- package/dist/modules/customers/subscribers/deal-closure-notification.js.map +7 -0
- package/dist/modules/customers/subscribers/deal-lost-notification.js +16 -0
- package/dist/modules/customers/subscribers/deal-lost-notification.js.map +7 -0
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js +2 -0
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js.map +2 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.js +154 -0
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.js.map +7 -0
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js +6 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js.map +2 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.js +154 -0
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.js.map +7 -0
- package/dist/modules/dictionaries/api/context.js +8 -1
- package/dist/modules/dictionaries/api/context.js.map +2 -2
- package/dist/modules/dictionaries/api/openapi.js +18 -1
- package/dist/modules/dictionaries/api/openapi.js.map +2 -2
- package/dist/modules/dictionaries/commands/entry-operations.js +388 -0
- package/dist/modules/dictionaries/commands/entry-operations.js.map +7 -0
- package/dist/modules/dictionaries/commands/factory.js +24 -3
- package/dist/modules/dictionaries/commands/factory.js.map +2 -2
- package/dist/modules/dictionaries/commands/index.js +1 -0
- package/dist/modules/dictionaries/commands/index.js.map +2 -2
- package/dist/modules/dictionaries/components/DictionaryTable.js +6 -3
- package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
- package/dist/modules/dictionaries/data/entities.js +11 -1
- package/dist/modules/dictionaries/data/entities.js.map +2 -2
- package/dist/modules/dictionaries/data/validators.js +28 -2
- package/dist/modules/dictionaries/data/validators.js.map +2 -2
- package/dist/modules/dictionaries/events.js +18 -0
- package/dist/modules/dictionaries/events.js.map +7 -0
- package/dist/modules/dictionaries/lib/clientEntries.js +43 -0
- package/dist/modules/dictionaries/lib/clientEntries.js.map +7 -0
- package/dist/modules/dictionaries/migrations/Migration20260410171544.js +45 -0
- package/dist/modules/dictionaries/migrations/Migration20260410171544.js.map +7 -0
- package/dist/modules/inbox_ops/api/proposals/[id]/route.js +4 -1
- package/dist/modules/inbox_ops/api/proposals/[id]/route.js.map +2 -2
- package/dist/modules/query_index/lib/engine.js +1 -1
- package/dist/modules/query_index/lib/engine.js.map +2 -2
- package/dist/modules/sales/components/documents/AddressesSection.js +82 -42
- package/dist/modules/sales/components/documents/AddressesSection.js.map +2 -2
- package/dist/modules/sales/lib/dictionaries.js +16 -0
- package/dist/modules/sales/lib/dictionaries.js.map +2 -2
- package/dist/modules/sales/widgets/injection-table.js +5 -1
- package/dist/modules/sales/widgets/injection-table.js.map +2 -2
- package/generated/entities/action_log/index.ts +4 -0
- package/generated/entities/customer_company_billing/index.ts +10 -0
- package/generated/entities/customer_deal/index.ts +4 -0
- package/generated/entities/customer_deal_stage_transition/index.ts +14 -0
- package/generated/entities/customer_dictionary_kind_setting/index.ts +9 -0
- package/generated/entities/customer_entity/index.ts +4 -0
- package/generated/entities/customer_entity_role/index.ts +10 -0
- package/generated/entities/customer_interaction/index.ts +11 -0
- package/generated/entities/customer_label/index.ts +8 -0
- package/generated/entities/customer_label_assignment/index.ts +7 -0
- package/generated/entities/customer_person_company_link/index.ts +9 -0
- package/generated/entities/customer_person_company_role/index.ts +7 -0
- package/generated/entities/dictionary_entry/index.ts +2 -0
- package/generated/entities.ids.generated.ts +9 -1
- package/generated/entity-fields-registry.ts +116 -1
- package/package.json +3 -3
- package/src/modules/attachments/api/route.ts +48 -6
- package/src/modules/attachments/i18n/de.json +4 -0
- package/src/modules/attachments/i18n/en.json +4 -0
- package/src/modules/attachments/i18n/es.json +4 -0
- package/src/modules/attachments/i18n/pl.json +4 -0
- package/src/modules/audit_logs/api/audit-logs/actions/export/route.ts +260 -0
- package/src/modules/audit_logs/api/audit-logs/actions/route.ts +81 -6
- package/src/modules/audit_logs/cli.ts +79 -0
- package/src/modules/audit_logs/data/entities.ts +17 -0
- package/src/modules/audit_logs/data/validators.ts +9 -1
- package/src/modules/audit_logs/lib/changeRows.ts +47 -0
- package/src/modules/audit_logs/lib/display-helpers.tsx +4 -30
- package/src/modules/audit_logs/lib/projections.ts +110 -0
- package/src/modules/audit_logs/migrations/.snapshot-open-mercato.json +325 -2
- package/src/modules/audit_logs/migrations/Migration20260412160533.ts +21 -0
- package/src/modules/audit_logs/services/actionLogService.ts +455 -85
- package/src/modules/catalog/i18n/de.json +1 -0
- package/src/modules/catalog/i18n/en.json +1 -0
- package/src/modules/catalog/i18n/es.json +1 -0
- package/src/modules/catalog/i18n/pl.json +1 -0
- package/src/modules/customer_accounts/i18n/de.json +2 -0
- package/src/modules/customer_accounts/i18n/en.json +2 -0
- package/src/modules/customer_accounts/i18n/es.json +2 -0
- package/src/modules/customer_accounts/i18n/pl.json +2 -0
- package/src/modules/customers/acl.ts +2 -0
- package/src/modules/customers/api/activities/route.ts +4 -0
- package/src/modules/customers/api/assignable-staff/route.ts +250 -0
- package/src/modules/customers/api/companies/[id]/people/route.ts +244 -0
- package/src/modules/customers/api/companies/[id]/roles/route.ts +15 -0
- package/src/modules/customers/api/companies/[id]/route.ts +458 -40
- package/src/modules/customers/api/companies/route.ts +93 -15
- package/src/modules/customers/api/deals/[id]/companies/route.ts +203 -0
- package/src/modules/customers/api/deals/[id]/people/route.ts +182 -0
- package/src/modules/customers/api/deals/[id]/route.ts +554 -57
- package/src/modules/customers/api/deals/[id]/stats/route.ts +221 -0
- package/src/modules/customers/api/deals/route.ts +35 -46
- package/src/modules/customers/api/dictionaries/[kind]/[id]/route.ts +105 -3
- package/src/modules/customers/api/dictionaries/[kind]/route.ts +143 -44
- package/src/modules/customers/api/dictionaries/context.ts +45 -16
- package/src/modules/customers/api/dictionaries/kind-settings/route.ts +232 -0
- package/src/modules/customers/api/entity-roles-factory.ts +520 -0
- package/src/modules/customers/api/interactions/conflicts/route.ts +196 -0
- package/src/modules/customers/api/interactions/counts/route.ts +112 -0
- package/src/modules/customers/api/interactions/route.ts +95 -2
- package/src/modules/customers/api/labels/assign/route.ts +202 -0
- package/src/modules/customers/api/labels/auth.ts +19 -0
- package/src/modules/customers/api/labels/route.ts +310 -0
- package/src/modules/customers/api/labels/table-errors.ts +36 -0
- package/src/modules/customers/api/labels/unassign/route.ts +197 -0
- package/src/modules/customers/api/people/[id]/companies/[linkId]/route.ts +331 -0
- package/src/modules/customers/api/people/[id]/companies/context.ts +70 -0
- package/src/modules/customers/api/people/[id]/companies/enriched/route.ts +384 -0
- package/src/modules/customers/api/people/[id]/companies/route.ts +215 -0
- package/src/modules/customers/api/people/[id]/roles/route.ts +15 -0
- package/src/modules/customers/api/people/[id]/route.ts +153 -26
- package/src/modules/customers/api/people/route.ts +134 -31
- package/src/modules/customers/api/todos/route.ts +4 -0
- package/src/modules/customers/api/utils.ts +36 -0
- package/src/modules/customers/backend/config/customers/page.tsx +2 -6
- package/src/modules/customers/backend/customers/companies/page.tsx +36 -26
- package/src/modules/customers/backend/customers/companies-v2/[id]/page.tsx +277 -262
- package/src/modules/customers/backend/customers/deals/[id]/hooks/formatters.ts +19 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/types.ts +104 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.ts +60 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealAssociations.ts +362 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealClosure.ts +113 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealData.ts +52 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.ts +86 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealInjectedTabs.tsx +60 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealMutationContext.ts +76 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealPipeline.ts +56 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useScheduleDialog.ts +38 -0
- package/src/modules/customers/backend/customers/deals/[id]/page.tsx +587 -624
- package/src/modules/customers/backend/customers/deals/page.tsx +71 -28
- package/src/modules/customers/backend/customers/people/page.tsx +35 -29
- package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +343 -209
- package/src/modules/customers/cli.ts +107 -12
- package/src/modules/customers/commands/activities.ts +13 -0
- package/src/modules/customers/commands/deals.ts +386 -114
- package/src/modules/customers/commands/dictionaries.ts +175 -32
- package/src/modules/customers/commands/dictionaryKindSettings.ts +268 -0
- package/src/modules/customers/commands/entity-roles.ts +494 -0
- package/src/modules/customers/commands/index.ts +4 -0
- package/src/modules/customers/commands/interactions.ts +125 -21
- package/src/modules/customers/commands/labels.ts +626 -0
- package/src/modules/customers/commands/people.ts +373 -259
- package/src/modules/customers/commands/personCompanyLinks.ts +654 -0
- package/src/modules/customers/commands/shared.ts +16 -15
- package/src/modules/customers/commands/todos.ts +17 -1
- package/src/modules/customers/components/AddressEditor.tsx +1 -1
- package/src/modules/customers/components/CustomersConfigurationSections.tsx +36 -0
- package/src/modules/customers/components/DictionarySettings.tsx +43 -2
- package/src/modules/customers/components/detail/ActiveDealCard.tsx +175 -0
- package/src/modules/customers/components/detail/ActivitiesSection.tsx +267 -361
- package/src/modules/customers/components/detail/ActivityAiActions.tsx +49 -0
- package/src/modules/customers/components/detail/ActivityCard.tsx +154 -0
- package/src/modules/customers/components/detail/ActivityHistorySection.tsx +412 -0
- package/src/modules/customers/components/detail/ActivityLogTab.tsx +67 -0
- package/src/modules/customers/components/detail/ActivityTimeline.tsx +158 -0
- package/src/modules/customers/components/detail/ActivityTimelineFilters.tsx +163 -0
- package/src/modules/customers/components/detail/ActivityTypeSelector.tsx +53 -0
- package/src/modules/customers/components/detail/AiActionChips.tsx +48 -0
- package/src/modules/customers/components/detail/AssignRoleDialog.tsx +672 -0
- package/src/modules/customers/components/detail/ChangelogEntryRow.tsx +132 -0
- package/src/modules/customers/components/detail/ChangelogFilters.tsx +193 -0
- package/src/modules/customers/components/detail/ChangelogKpiCards.tsx +107 -0
- package/src/modules/customers/components/detail/ChangelogTab.tsx +629 -0
- package/src/modules/customers/components/detail/ComingSoonPlaceholder.tsx +21 -0
- package/src/modules/customers/components/detail/CompanyCard.tsx +419 -0
- package/src/modules/customers/components/detail/CompanyDashboardTab.tsx +161 -0
- package/src/modules/customers/components/detail/CompanyDetailHeader.tsx +243 -0
- package/src/modules/customers/components/detail/CompanyDetailTabs.tsx +172 -0
- package/src/modules/customers/components/detail/CompanyKpiBar.tsx +206 -0
- package/src/modules/customers/components/detail/CompanyPeopleSection.tsx +582 -288
- package/src/modules/customers/components/detail/CompanyTagsDialog.tsx +23 -0
- package/src/modules/customers/components/detail/ConfirmDealLostDialog.tsx +210 -0
- package/src/modules/customers/components/detail/CreatePersonDialog.tsx +178 -0
- package/src/modules/customers/components/detail/DealClosureActionBar.tsx +63 -0
- package/src/modules/customers/components/detail/DealDetailHeader.tsx +335 -0
- package/src/modules/customers/components/detail/DealDetailTabs.tsx +154 -0
- package/src/modules/customers/components/detail/DealForm.tsx +253 -101
- package/src/modules/customers/components/detail/DealLinkedEntitiesTab.tsx +349 -0
- package/src/modules/customers/components/detail/DealLostSummaryDialog.tsx +156 -0
- package/src/modules/customers/components/detail/DealWonPopup.tsx +164 -0
- package/src/modules/customers/components/detail/DealsSection.tsx +276 -221
- package/src/modules/customers/components/detail/DecisionMakersFooter.tsx +56 -0
- package/src/modules/customers/components/detail/EntityTagsDialog.tsx +1372 -0
- package/src/modules/customers/components/detail/InlineActivityComposer.tsx +239 -0
- package/src/modules/customers/components/detail/ManageTagsDialog.tsx +1331 -0
- package/src/modules/customers/components/detail/MiniWeekCalendar.tsx +338 -0
- package/src/modules/customers/components/detail/MobilePersonDetail.tsx +124 -0
- package/src/modules/customers/components/detail/NextStepCard.tsx +104 -0
- package/src/modules/customers/components/detail/PersonCard.tsx +238 -0
- package/src/modules/customers/components/detail/PersonCompaniesSection.tsx +426 -0
- package/src/modules/customers/components/detail/PersonDetailHeader.tsx +294 -0
- package/src/modules/customers/components/detail/PersonDetailTabs.tsx +172 -0
- package/src/modules/customers/components/detail/PersonTagsDialog.tsx +26 -0
- package/src/modules/customers/components/detail/PipelineStepper.tsx +245 -0
- package/src/modules/customers/components/detail/PlannedActivitiesSection.tsx +255 -0
- package/src/modules/customers/components/detail/RelationshipHealthCard.tsx +63 -0
- package/src/modules/customers/components/detail/RoleAssignmentRow.tsx +248 -0
- package/src/modules/customers/components/detail/RolesSection.tsx +311 -0
- package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +481 -0
- package/src/modules/customers/components/detail/aiActionCatalog.ts +77 -0
- package/src/modules/customers/components/detail/assignableStaff.ts +124 -0
- package/src/modules/customers/components/detail/dashboard/ActiveDealWidget.tsx +63 -0
- package/src/modules/customers/components/detail/dashboard/OpenTasksWidget.tsx +114 -0
- package/src/modules/customers/components/detail/dashboard/RecentActivityWidget.tsx +69 -0
- package/src/modules/customers/components/detail/dashboard/RelationshipHealthWidget.tsx +40 -0
- package/src/modules/customers/components/detail/dashboard/UpcomingMeetingsWidget.tsx +64 -0
- package/src/modules/customers/components/detail/dashboard/helpers.ts +78 -0
- package/src/modules/customers/components/detail/healthScoreUtils.ts +91 -0
- package/src/modules/customers/components/detail/hooks/useCurrencyDictionary.ts +8 -8
- package/src/modules/customers/components/detail/hooks/useCustomerDictionary.ts +10 -6
- package/src/modules/customers/components/detail/hooks/useInteractionMutations.ts +91 -0
- package/src/modules/customers/components/detail/notesAdapter.ts +91 -30
- package/src/modules/customers/components/detail/pipelineStageUtils.ts +29 -0
- package/src/modules/customers/components/detail/schedule/DateTimeFields.tsx +187 -0
- package/src/modules/customers/components/detail/schedule/FooterFields.tsx +79 -0
- package/src/modules/customers/components/detail/schedule/LinkedEntitiesField.tsx +277 -0
- package/src/modules/customers/components/detail/schedule/LocationField.tsx +42 -0
- package/src/modules/customers/components/detail/schedule/ParticipantsField.tsx +255 -0
- package/src/modules/customers/components/detail/schedule/fieldConfig.ts +70 -0
- package/src/modules/customers/components/detail/schedule/index.ts +9 -0
- package/src/modules/customers/components/detail/schedule/useScheduleFormState.ts +221 -0
- package/src/modules/customers/components/detail/types.ts +16 -0
- package/src/modules/customers/components/detail/utils.ts +25 -0
- package/src/modules/customers/components/formConfig.tsx +223 -28
- package/src/modules/customers/components/linking/LinkEntityDialog.tsx +920 -0
- package/src/modules/customers/components/linking/adapters/companyAdapter.tsx +398 -0
- package/src/modules/customers/components/linking/adapters/dealAdapter.tsx +578 -0
- package/src/modules/customers/components/linking/adapters/personAdapter.tsx +512 -0
- package/src/modules/customers/components/list/CollectionPreviewCell.tsx +66 -0
- package/src/modules/customers/data/entities.ts +353 -1
- package/src/modules/customers/data/validators.ts +170 -19
- package/src/modules/customers/events.ts +22 -0
- package/src/modules/customers/i18n/de.json +841 -2
- package/src/modules/customers/i18n/en.json +841 -2
- package/src/modules/customers/i18n/es.json +840 -1
- package/src/modules/customers/i18n/pl.json +841 -2
- package/src/modules/customers/lib/customerRoleTypes.ts +24 -0
- package/src/modules/customers/lib/dealClosureNotification.ts +64 -0
- package/src/modules/customers/lib/dealStageTransitionTable.ts +32 -0
- package/src/modules/customers/lib/dictionaries.ts +26 -10
- package/src/modules/customers/lib/interactionReadModel.ts +10 -0
- package/src/modules/customers/lib/personCompanies.ts +317 -0
- package/src/modules/customers/lib/personCompanyLinkTable.ts +58 -0
- package/src/modules/customers/lib/roleTypeUsage.ts +146 -0
- package/src/modules/customers/migrations/.snapshot-open-mercato.json +2747 -798
- package/src/modules/customers/migrations/Migration20260406214502.ts +19 -0
- package/src/modules/customers/migrations/Migration20260408135736.ts +15 -0
- package/src/modules/customers/migrations/Migration20260408225345.ts +23 -0
- package/src/modules/customers/migrations/Migration20260411075533.ts +30 -0
- package/src/modules/customers/migrations/Migration20260411103551.ts +13 -0
- package/src/modules/customers/migrations/Migration20260411130944.ts +30 -0
- package/src/modules/customers/migrations/Migration20260415095203.ts +13 -0
- package/src/modules/customers/migrations/Migration20260415135056.ts +22 -0
- package/src/modules/customers/migrations/Migration20260417140000.ts +15 -0
- package/src/modules/customers/migrations/Migration20260417160000.ts +17 -0
- package/src/modules/customers/migrations/Migration20260417235407.ts +13 -0
- package/src/modules/customers/setup.ts +15 -0
- package/src/modules/customers/subscribers/deal-closure-notification.ts +22 -0
- package/src/modules/customers/subscribers/deal-lost-notification.ts +22 -0
- package/src/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.ts +2 -0
- package/src/modules/dictionaries/api/[dictionaryId]/entries/reorder/route.ts +162 -0
- package/src/modules/dictionaries/api/[dictionaryId]/entries/route.ts +6 -2
- package/src/modules/dictionaries/api/[dictionaryId]/entries/set-default/route.ts +162 -0
- package/src/modules/dictionaries/api/context.ts +9 -0
- package/src/modules/dictionaries/api/openapi.ts +17 -0
- package/src/modules/dictionaries/commands/entry-operations.ts +457 -0
- package/src/modules/dictionaries/commands/factory.ts +31 -3
- package/src/modules/dictionaries/commands/index.ts +1 -0
- package/src/modules/dictionaries/components/DictionaryTable.tsx +15 -6
- package/src/modules/dictionaries/data/entities.ts +9 -0
- package/src/modules/dictionaries/data/validators.ts +34 -0
- package/src/modules/dictionaries/events.ts +20 -0
- package/src/modules/dictionaries/i18n/de.json +2 -0
- package/src/modules/dictionaries/i18n/en.json +2 -0
- package/src/modules/dictionaries/i18n/es.json +2 -0
- package/src/modules/dictionaries/i18n/pl.json +2 -0
- package/src/modules/dictionaries/lib/clientEntries.ts +66 -0
- package/src/modules/dictionaries/migrations/.snapshot-open-mercato.json +185 -3
- package/src/modules/dictionaries/migrations/Migration20260410171544.ts +49 -0
- package/src/modules/inbox_ops/api/proposals/[id]/route.ts +4 -1
- package/src/modules/query_index/lib/engine.ts +9 -1
- package/src/modules/sales/components/documents/AddressesSection.tsx +92 -42
- package/src/modules/sales/i18n/de.json +28 -0
- package/src/modules/sales/i18n/en.json +28 -0
- package/src/modules/sales/i18n/es.json +28 -0
- package/src/modules/sales/i18n/pl.json +28 -0
- package/src/modules/sales/lib/dictionaries.ts +18 -0
- package/src/modules/sales/widgets/injection-table.ts +4 -0
|
@@ -0,0 +1,345 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { Fragment, jsx, jsxs } from "react/jsx-runtime";
|
|
3
|
+
import * as React from "react";
|
|
4
|
+
import { ArrowLeft, ArrowRight, Link2 } from "lucide-react";
|
|
5
|
+
import { useT } from "@open-mercato/shared/lib/i18n/context";
|
|
6
|
+
import { flash } from "@open-mercato/ui/backend/FlashMessages";
|
|
7
|
+
import { apiCallOrThrow, readApiResultOrThrow } from "@open-mercato/ui/backend/utils/apiCall";
|
|
8
|
+
import { Button } from "@open-mercato/ui/primitives/button";
|
|
9
|
+
import { Input } from "@open-mercato/ui/primitives/input";
|
|
10
|
+
import { CompanyCard } from "./CompanyCard.js";
|
|
11
|
+
import { useCustomerDictionary } from "./hooks/useCustomerDictionary.js";
|
|
12
|
+
import { LinkEntityDialog } from "../linking/LinkEntityDialog.js";
|
|
13
|
+
import { createCompanyLinkAdapter } from "../linking/adapters/companyAdapter.js";
|
|
14
|
+
const LINKED_PAGE_SIZE = 20;
|
|
15
|
+
function sameIdSet(left, right) {
|
|
16
|
+
if (left.length !== right.length) return false;
|
|
17
|
+
const rightSet = new Set(right);
|
|
18
|
+
return left.every((value) => rightSet.has(value));
|
|
19
|
+
}
|
|
20
|
+
function Pagination({
|
|
21
|
+
page,
|
|
22
|
+
totalPages,
|
|
23
|
+
onPageChange
|
|
24
|
+
}) {
|
|
25
|
+
if (totalPages <= 1) return null;
|
|
26
|
+
return /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between border-t border-border/60 pt-3 text-sm text-muted-foreground", children: [
|
|
27
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
28
|
+
"Page ",
|
|
29
|
+
page,
|
|
30
|
+
" of ",
|
|
31
|
+
totalPages
|
|
32
|
+
] }),
|
|
33
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
34
|
+
/* @__PURE__ */ jsxs(
|
|
35
|
+
Button,
|
|
36
|
+
{
|
|
37
|
+
type: "button",
|
|
38
|
+
variant: "outline",
|
|
39
|
+
size: "sm",
|
|
40
|
+
onClick: () => onPageChange(Math.max(1, page - 1)),
|
|
41
|
+
disabled: page <= 1,
|
|
42
|
+
children: [
|
|
43
|
+
/* @__PURE__ */ jsx(ArrowLeft, { className: "mr-1.5 size-3.5" }),
|
|
44
|
+
"Previous"
|
|
45
|
+
]
|
|
46
|
+
}
|
|
47
|
+
),
|
|
48
|
+
/* @__PURE__ */ jsxs(
|
|
49
|
+
Button,
|
|
50
|
+
{
|
|
51
|
+
type: "button",
|
|
52
|
+
variant: "outline",
|
|
53
|
+
size: "sm",
|
|
54
|
+
onClick: () => onPageChange(Math.min(totalPages, page + 1)),
|
|
55
|
+
disabled: page >= totalPages,
|
|
56
|
+
children: [
|
|
57
|
+
"Next",
|
|
58
|
+
/* @__PURE__ */ jsx(ArrowRight, { className: "ml-1.5 size-3.5" })
|
|
59
|
+
]
|
|
60
|
+
}
|
|
61
|
+
)
|
|
62
|
+
] })
|
|
63
|
+
] });
|
|
64
|
+
}
|
|
65
|
+
function PersonCompaniesSection({
|
|
66
|
+
personId,
|
|
67
|
+
personName: _personName,
|
|
68
|
+
initialLinkedCompanies = [],
|
|
69
|
+
onChanged,
|
|
70
|
+
runGuardedMutation
|
|
71
|
+
}) {
|
|
72
|
+
const t = useT();
|
|
73
|
+
const [items, setItems] = React.useState([]);
|
|
74
|
+
const [loading, setLoading] = React.useState(true);
|
|
75
|
+
const [search, setSearch] = React.useState("");
|
|
76
|
+
const [sort, setSort] = React.useState("name-asc");
|
|
77
|
+
const [page, setPage] = React.useState(1);
|
|
78
|
+
const [totalPages, setTotalPages] = React.useState(1);
|
|
79
|
+
const [linkedCompanies, setLinkedCompanies] = React.useState(
|
|
80
|
+
initialLinkedCompanies
|
|
81
|
+
);
|
|
82
|
+
const [dialogOpen, setDialogOpen] = React.useState(false);
|
|
83
|
+
const { data: statusDict } = useCustomerDictionary("statuses");
|
|
84
|
+
const { data: lifecycleDict } = useCustomerDictionary("lifecycle-stages");
|
|
85
|
+
const { data: temperatureDict } = useCustomerDictionary("temperature");
|
|
86
|
+
const { data: renewalQuarterDict } = useCustomerDictionary("renewal-quarters");
|
|
87
|
+
const { data: roleDict } = useCustomerDictionary("person-company-roles");
|
|
88
|
+
React.useEffect(() => {
|
|
89
|
+
setLinkedCompanies(initialLinkedCompanies);
|
|
90
|
+
}, [initialLinkedCompanies]);
|
|
91
|
+
const runWriteMutation = React.useCallback(
|
|
92
|
+
async (operation, mutationPayload) => {
|
|
93
|
+
if (!runGuardedMutation) {
|
|
94
|
+
return operation();
|
|
95
|
+
}
|
|
96
|
+
return runGuardedMutation(operation, mutationPayload);
|
|
97
|
+
},
|
|
98
|
+
[runGuardedMutation]
|
|
99
|
+
);
|
|
100
|
+
const loadData = React.useCallback(
|
|
101
|
+
async (options) => {
|
|
102
|
+
const showLoading = options?.showLoading ?? true;
|
|
103
|
+
if (showLoading) setLoading(true);
|
|
104
|
+
try {
|
|
105
|
+
const params = new URLSearchParams({
|
|
106
|
+
page: String(page),
|
|
107
|
+
pageSize: String(LINKED_PAGE_SIZE),
|
|
108
|
+
sort
|
|
109
|
+
});
|
|
110
|
+
if (search.trim().length > 0) {
|
|
111
|
+
params.set("search", search.trim());
|
|
112
|
+
}
|
|
113
|
+
const payload = await readApiResultOrThrow(
|
|
114
|
+
`/api/customers/people/${encodeURIComponent(personId)}/companies/enriched?${params.toString()}`,
|
|
115
|
+
{ cache: "no-store" }
|
|
116
|
+
);
|
|
117
|
+
const nextItems = Array.isArray(payload?.items) ? payload.items : [];
|
|
118
|
+
setItems(nextItems);
|
|
119
|
+
setTotalPages(typeof payload?.totalPages === "number" ? payload.totalPages : 1);
|
|
120
|
+
} catch (error) {
|
|
121
|
+
const message = error instanceof Error ? error.message : t("customers.people.detail.companies.loadError", "Failed to load companies.");
|
|
122
|
+
flash(message, "error");
|
|
123
|
+
setItems([]);
|
|
124
|
+
setTotalPages(1);
|
|
125
|
+
} finally {
|
|
126
|
+
if (showLoading) setLoading(false);
|
|
127
|
+
}
|
|
128
|
+
},
|
|
129
|
+
[page, personId, search, sort, t]
|
|
130
|
+
);
|
|
131
|
+
React.useEffect(() => {
|
|
132
|
+
void loadData();
|
|
133
|
+
}, [loadData]);
|
|
134
|
+
React.useEffect(() => {
|
|
135
|
+
setPage(1);
|
|
136
|
+
}, [search, sort]);
|
|
137
|
+
const linkedIds = React.useMemo(
|
|
138
|
+
() => linkedCompanies.map((entry) => entry.id),
|
|
139
|
+
[linkedCompanies]
|
|
140
|
+
);
|
|
141
|
+
const linkedPrimaryId = React.useMemo(
|
|
142
|
+
() => linkedCompanies.find((entry) => entry.isPrimary)?.id ?? linkedCompanies[0]?.id ?? null,
|
|
143
|
+
[linkedCompanies]
|
|
144
|
+
);
|
|
145
|
+
const companyLinkAdapter = React.useMemo(
|
|
146
|
+
() => createCompanyLinkAdapter({
|
|
147
|
+
dialogTitle: t("customers.linking.company.dialogTitle", "Link company"),
|
|
148
|
+
dialogSubtitle: _personName ? t("customers.linking.company.dialogSubtitleFor", "Link an existing company to {{name}}", {
|
|
149
|
+
name: _personName
|
|
150
|
+
}) : t("customers.linking.company.dialogSubtitle", "Link an existing company to this person"),
|
|
151
|
+
sectionLabel: t("customers.linking.company.sectionLabel", "MATCHING COMPANIES"),
|
|
152
|
+
searchPlaceholder: t(
|
|
153
|
+
"customers.linking.company.searchPlaceholder",
|
|
154
|
+
"Search all companies\u2026"
|
|
155
|
+
),
|
|
156
|
+
searchEmptyHint: t(
|
|
157
|
+
"customers.linking.company.searchEmpty",
|
|
158
|
+
"No matching companies found."
|
|
159
|
+
),
|
|
160
|
+
selectedEmptyHint: t(
|
|
161
|
+
"customers.linking.company.selectedEmpty",
|
|
162
|
+
"No companies selected."
|
|
163
|
+
),
|
|
164
|
+
confirmButtonLabel: t("customers.linking.company.confirmButton", "Link company"),
|
|
165
|
+
excludeLinkedPersonId: personId
|
|
166
|
+
}),
|
|
167
|
+
[_personName, personId, t]
|
|
168
|
+
);
|
|
169
|
+
const handleLinkConfirm = React.useCallback(
|
|
170
|
+
async ({
|
|
171
|
+
addedIds,
|
|
172
|
+
removedIds,
|
|
173
|
+
nextSelectedIds,
|
|
174
|
+
primaryId,
|
|
175
|
+
optionsById
|
|
176
|
+
}) => {
|
|
177
|
+
const currentPrimaryId = linkedPrimaryId;
|
|
178
|
+
const nextPrimaryId = nextSelectedIds.length ? primaryId && nextSelectedIds.includes(primaryId) ? primaryId : nextSelectedIds[0] : null;
|
|
179
|
+
try {
|
|
180
|
+
for (const companyId of removedIds) {
|
|
181
|
+
await runWriteMutation(
|
|
182
|
+
() => apiCallOrThrow(
|
|
183
|
+
`/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,
|
|
184
|
+
{ method: "DELETE" }
|
|
185
|
+
),
|
|
186
|
+
{ companyId, personId, operation: "removePersonCompanyLink" }
|
|
187
|
+
);
|
|
188
|
+
}
|
|
189
|
+
for (const companyId of addedIds) {
|
|
190
|
+
await runWriteMutation(
|
|
191
|
+
() => apiCallOrThrow(
|
|
192
|
+
`/api/customers/people/${encodeURIComponent(personId)}/companies`,
|
|
193
|
+
{
|
|
194
|
+
method: "POST",
|
|
195
|
+
headers: { "content-type": "application/json" },
|
|
196
|
+
body: JSON.stringify({
|
|
197
|
+
companyId,
|
|
198
|
+
isPrimary: nextPrimaryId === companyId
|
|
199
|
+
})
|
|
200
|
+
}
|
|
201
|
+
),
|
|
202
|
+
{ companyId, personId, operation: "addPersonCompanyLink" }
|
|
203
|
+
);
|
|
204
|
+
}
|
|
205
|
+
if (nextPrimaryId && nextPrimaryId !== currentPrimaryId && !addedIds.includes(nextPrimaryId)) {
|
|
206
|
+
await runWriteMutation(
|
|
207
|
+
() => apiCallOrThrow(
|
|
208
|
+
`/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(nextPrimaryId)}`,
|
|
209
|
+
{
|
|
210
|
+
method: "PATCH",
|
|
211
|
+
headers: { "content-type": "application/json" },
|
|
212
|
+
body: JSON.stringify({ isPrimary: true })
|
|
213
|
+
}
|
|
214
|
+
),
|
|
215
|
+
{ companyId: nextPrimaryId, personId, operation: "setPrimaryPersonCompanyLink" }
|
|
216
|
+
);
|
|
217
|
+
}
|
|
218
|
+
const nextLinkedCompanies = nextSelectedIds.map((id) => {
|
|
219
|
+
const option = optionsById[id];
|
|
220
|
+
const fromCurrent = linkedCompanies.find((entry) => entry.id === id);
|
|
221
|
+
return {
|
|
222
|
+
id,
|
|
223
|
+
displayName: option?.label ?? fromCurrent?.displayName ?? id,
|
|
224
|
+
isPrimary: id === nextPrimaryId
|
|
225
|
+
};
|
|
226
|
+
});
|
|
227
|
+
setLinkedCompanies(nextLinkedCompanies);
|
|
228
|
+
await loadData({ showLoading: false });
|
|
229
|
+
await onChanged?.();
|
|
230
|
+
flash(
|
|
231
|
+
t("customers.people.detail.companies.manageSuccess", "Linked companies updated."),
|
|
232
|
+
"success"
|
|
233
|
+
);
|
|
234
|
+
} catch (error) {
|
|
235
|
+
const message = error instanceof Error ? error.message : t(
|
|
236
|
+
"customers.people.detail.companies.manageError",
|
|
237
|
+
"Failed to update linked companies."
|
|
238
|
+
);
|
|
239
|
+
flash(message, "error");
|
|
240
|
+
throw error;
|
|
241
|
+
}
|
|
242
|
+
},
|
|
243
|
+
[linkedCompanies, linkedPrimaryId, loadData, onChanged, personId, runWriteMutation, t]
|
|
244
|
+
);
|
|
245
|
+
return /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
246
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-4", children: [
|
|
247
|
+
/* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-3 rounded-2xl border border-border/70 bg-card px-4 py-4 sm:flex-row sm:items-center sm:justify-between", children: [
|
|
248
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
249
|
+
/* @__PURE__ */ jsx("div", { className: "text-sm font-semibold text-foreground", children: t("customers.people.detail.companies.manageTitle", "Manage linked companies") }),
|
|
250
|
+
/* @__PURE__ */ jsx("div", { className: "text-sm text-muted-foreground", children: t("customers.people.detail.companies.summary", "{{count}} linked companies", {
|
|
251
|
+
count: linkedCompanies.length
|
|
252
|
+
}) })
|
|
253
|
+
] }),
|
|
254
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
255
|
+
/* @__PURE__ */ jsx(
|
|
256
|
+
Input,
|
|
257
|
+
{
|
|
258
|
+
value: search,
|
|
259
|
+
onChange: (event) => setSearch(event.target.value),
|
|
260
|
+
placeholder: t(
|
|
261
|
+
"customers.people.detail.companies.searchPlaceholder",
|
|
262
|
+
"Search linked companies\u2026"
|
|
263
|
+
),
|
|
264
|
+
className: "sm:w-[260px]"
|
|
265
|
+
}
|
|
266
|
+
),
|
|
267
|
+
/* @__PURE__ */ jsxs(
|
|
268
|
+
Button,
|
|
269
|
+
{
|
|
270
|
+
type: "button",
|
|
271
|
+
variant: "outline",
|
|
272
|
+
size: "sm",
|
|
273
|
+
onClick: () => setDialogOpen(true),
|
|
274
|
+
children: [
|
|
275
|
+
/* @__PURE__ */ jsx(Link2, { className: "mr-2 size-4" }),
|
|
276
|
+
t("customers.people.detail.companies.manageAction", "Manage links")
|
|
277
|
+
]
|
|
278
|
+
}
|
|
279
|
+
)
|
|
280
|
+
] })
|
|
281
|
+
] }),
|
|
282
|
+
/* @__PURE__ */ jsx("div", { className: "flex items-center justify-end", children: /* @__PURE__ */ jsxs(
|
|
283
|
+
"select",
|
|
284
|
+
{
|
|
285
|
+
value: sort,
|
|
286
|
+
onChange: (event) => setSort(event.target.value),
|
|
287
|
+
className: "h-10 rounded-md border bg-background px-3 text-sm focus:outline-none focus:ring-1 focus:ring-ring",
|
|
288
|
+
children: [
|
|
289
|
+
/* @__PURE__ */ jsx("option", { value: "name-asc", children: t("customers.people.detail.companies.sortNameAsc", "Sort: Name A-Z") }),
|
|
290
|
+
/* @__PURE__ */ jsx("option", { value: "name-desc", children: t("customers.people.detail.companies.sortNameDesc", "Sort: Name Z-A") }),
|
|
291
|
+
/* @__PURE__ */ jsx("option", { value: "recent", children: t("customers.people.detail.companies.sortRecent", "Sort: Recently active") })
|
|
292
|
+
]
|
|
293
|
+
}
|
|
294
|
+
) }),
|
|
295
|
+
loading ? /* @__PURE__ */ jsx("div", { className: "space-y-4", children: [1, 2].map((idx) => /* @__PURE__ */ jsx(
|
|
296
|
+
"div",
|
|
297
|
+
{
|
|
298
|
+
className: "h-[320px] animate-pulse rounded-2xl border border-border/60 bg-muted/30"
|
|
299
|
+
},
|
|
300
|
+
idx
|
|
301
|
+
)) }) : items.length === 0 ? /* @__PURE__ */ jsx("div", { className: "rounded-2xl border border-dashed border-border/60 px-6 py-12 text-center text-sm text-muted-foreground", children: search.trim().length ? t(
|
|
302
|
+
"customers.people.detail.companies.noSearchResults",
|
|
303
|
+
"No linked companies match your search."
|
|
304
|
+
) : t(
|
|
305
|
+
"customers.people.detail.empty.companies",
|
|
306
|
+
"No company linked to this person."
|
|
307
|
+
) }) : /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
308
|
+
/* @__PURE__ */ jsx("div", { className: "space-y-4", children: items.map((item) => /* @__PURE__ */ jsx(
|
|
309
|
+
CompanyCard,
|
|
310
|
+
{
|
|
311
|
+
data: item,
|
|
312
|
+
personName: _personName,
|
|
313
|
+
statusMap: statusDict?.map,
|
|
314
|
+
lifecycleMap: lifecycleDict?.map,
|
|
315
|
+
temperatureMap: temperatureDict?.map,
|
|
316
|
+
renewalQuarterMap: renewalQuarterDict?.map,
|
|
317
|
+
roleMap: roleDict?.map
|
|
318
|
+
},
|
|
319
|
+
item.companyId
|
|
320
|
+
)) }),
|
|
321
|
+
/* @__PURE__ */ jsx(Pagination, { page, totalPages, onPageChange: setPage })
|
|
322
|
+
] })
|
|
323
|
+
] }),
|
|
324
|
+
/* @__PURE__ */ jsx(
|
|
325
|
+
LinkEntityDialog,
|
|
326
|
+
{
|
|
327
|
+
open: dialogOpen,
|
|
328
|
+
onOpenChange: setDialogOpen,
|
|
329
|
+
adapter: companyLinkAdapter,
|
|
330
|
+
initialSelectedIds: linkedIds,
|
|
331
|
+
initialPrimaryId: linkedPrimaryId,
|
|
332
|
+
primarySupported: true,
|
|
333
|
+
onConfirm: handleLinkConfirm,
|
|
334
|
+
runGuardedMutation: runWriteMutation
|
|
335
|
+
}
|
|
336
|
+
)
|
|
337
|
+
] });
|
|
338
|
+
}
|
|
339
|
+
var PersonCompaniesSection_default = PersonCompaniesSection;
|
|
340
|
+
export {
|
|
341
|
+
PersonCompaniesSection,
|
|
342
|
+
PersonCompaniesSection_default as default,
|
|
343
|
+
sameIdSet
|
|
344
|
+
};
|
|
345
|
+
//# sourceMappingURL=PersonCompaniesSection.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../../src/modules/customers/components/detail/PersonCompaniesSection.tsx"],
|
|
4
|
+
"sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport { ArrowLeft, ArrowRight, Link2 } from 'lucide-react'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { CompanyCard, type EnrichedCompanyData } from './CompanyCard'\nimport { useCustomerDictionary } from './hooks/useCustomerDictionary'\nimport { LinkEntityDialog, type LinkEntityOption } from '../linking/LinkEntityDialog'\nimport { createCompanyLinkAdapter } from '../linking/adapters/companyAdapter'\n\ntype GuardedMutationRunner = <T,>(\n operation: () => Promise<T>,\n mutationPayload?: Record<string, unknown>,\n) => Promise<T>\n\ntype LinkedCompanySummary = {\n id: string\n displayName: string\n isPrimary: boolean\n}\n\ntype PersonCompaniesSectionProps = {\n personId: string\n personName: string\n initialLinkedCompanies?: LinkedCompanySummary[]\n onChanged?: () => Promise<void> | void\n runGuardedMutation?: GuardedMutationRunner\n}\n\nconst LINKED_PAGE_SIZE = 20\n\nfunction sameIdSet(left: string[], right: string[]): boolean {\n if (left.length !== right.length) return false\n const rightSet = new Set(right)\n return left.every((value) => rightSet.has(value))\n}\n\nfunction Pagination({\n page,\n totalPages,\n onPageChange,\n}: {\n page: number\n totalPages: number\n onPageChange: (page: number) => void\n}) {\n if (totalPages <= 1) return null\n return (\n <div className=\"flex items-center justify-between border-t border-border/60 pt-3 text-sm text-muted-foreground\">\n <span>\n Page {page} of {totalPages}\n </span>\n <div className=\"flex items-center gap-2\">\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => onPageChange(Math.max(1, page - 1))}\n disabled={page <= 1}\n >\n <ArrowLeft className=\"mr-1.5 size-3.5\" />\n Previous\n </Button>\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => onPageChange(Math.min(totalPages, page + 1))}\n disabled={page >= totalPages}\n >\n Next\n <ArrowRight className=\"ml-1.5 size-3.5\" />\n </Button>\n </div>\n </div>\n )\n}\n\nexport function PersonCompaniesSection({\n personId,\n personName: _personName,\n initialLinkedCompanies = [],\n onChanged,\n runGuardedMutation,\n}: PersonCompaniesSectionProps) {\n const t = useT()\n const [items, setItems] = React.useState<EnrichedCompanyData[]>([])\n const [loading, setLoading] = React.useState(true)\n const [search, setSearch] = React.useState('')\n const [sort, setSort] = React.useState<'name-asc' | 'name-desc' | 'recent'>('name-asc')\n const [page, setPage] = React.useState(1)\n const [totalPages, setTotalPages] = React.useState(1)\n const [linkedCompanies, setLinkedCompanies] = React.useState<LinkedCompanySummary[]>(\n initialLinkedCompanies,\n )\n const [dialogOpen, setDialogOpen] = React.useState(false)\n\n const { data: statusDict } = useCustomerDictionary('statuses')\n const { data: lifecycleDict } = useCustomerDictionary('lifecycle-stages')\n const { data: temperatureDict } = useCustomerDictionary('temperature')\n const { data: renewalQuarterDict } = useCustomerDictionary('renewal-quarters')\n const { data: roleDict } = useCustomerDictionary('person-company-roles')\n\n React.useEffect(() => {\n setLinkedCompanies(initialLinkedCompanies)\n }, [initialLinkedCompanies])\n\n const runWriteMutation = React.useCallback(\n async <T,>(\n operation: () => Promise<T>,\n mutationPayload?: Record<string, unknown>,\n ): Promise<T> => {\n if (!runGuardedMutation) {\n return operation()\n }\n return runGuardedMutation(operation, mutationPayload)\n },\n [runGuardedMutation],\n )\n\n const loadData = React.useCallback(\n async (options?: { showLoading?: boolean }) => {\n const showLoading = options?.showLoading ?? true\n if (showLoading) setLoading(true)\n try {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: String(LINKED_PAGE_SIZE),\n sort,\n })\n if (search.trim().length > 0) {\n params.set('search', search.trim())\n }\n const payload = await readApiResultOrThrow<{\n items?: EnrichedCompanyData[]\n totalPages?: number\n }>(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/enriched?${params.toString()}`,\n { cache: 'no-store' },\n )\n const nextItems = Array.isArray(payload?.items) ? payload.items : []\n setItems(nextItems)\n setTotalPages(typeof payload?.totalPages === 'number' ? payload.totalPages : 1)\n } catch (error) {\n const message =\n error instanceof Error\n ? error.message\n : t('customers.people.detail.companies.loadError', 'Failed to load companies.')\n flash(message, 'error')\n setItems([])\n setTotalPages(1)\n } finally {\n if (showLoading) setLoading(false)\n }\n },\n [page, personId, search, sort, t],\n )\n\n React.useEffect(() => {\n void loadData()\n }, [loadData])\n\n React.useEffect(() => {\n setPage(1)\n }, [search, sort])\n\n const linkedIds = React.useMemo(\n () => linkedCompanies.map((entry) => entry.id),\n [linkedCompanies],\n )\n const linkedPrimaryId = React.useMemo(\n () =>\n linkedCompanies.find((entry) => entry.isPrimary)?.id ??\n linkedCompanies[0]?.id ??\n null,\n [linkedCompanies],\n )\n\n const companyLinkAdapter = React.useMemo(\n () =>\n createCompanyLinkAdapter({\n dialogTitle: t('customers.linking.company.dialogTitle', 'Link company'),\n dialogSubtitle: _personName\n ? t('customers.linking.company.dialogSubtitleFor', 'Link an existing company to {{name}}', {\n name: _personName,\n })\n : t('customers.linking.company.dialogSubtitle', 'Link an existing company to this person'),\n sectionLabel: t('customers.linking.company.sectionLabel', 'MATCHING COMPANIES'),\n searchPlaceholder: t(\n 'customers.linking.company.searchPlaceholder',\n 'Search all companies\u2026',\n ),\n searchEmptyHint: t(\n 'customers.linking.company.searchEmpty',\n 'No matching companies found.',\n ),\n selectedEmptyHint: t(\n 'customers.linking.company.selectedEmpty',\n 'No companies selected.',\n ),\n confirmButtonLabel: t('customers.linking.company.confirmButton', 'Link company'),\n excludeLinkedPersonId: personId,\n }),\n [_personName, personId, t],\n )\n\n const handleLinkConfirm = React.useCallback(\n async ({\n addedIds,\n removedIds,\n nextSelectedIds,\n primaryId,\n optionsById,\n }: {\n addedIds: string[]\n removedIds: string[]\n nextSelectedIds: string[]\n primaryId?: string | null\n optionsById: Record<string, LinkEntityOption>\n }) => {\n const currentPrimaryId = linkedPrimaryId\n const nextPrimaryId = nextSelectedIds.length\n ? primaryId && nextSelectedIds.includes(primaryId)\n ? primaryId\n : nextSelectedIds[0]\n : null\n\n try {\n for (const companyId of removedIds) {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,\n { method: 'DELETE' },\n ),\n { companyId, personId, operation: 'removePersonCompanyLink' },\n )\n }\n\n for (const companyId of addedIds) {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies`,\n {\n method: 'POST',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({\n companyId,\n isPrimary: nextPrimaryId === companyId,\n }),\n },\n ),\n { companyId, personId, operation: 'addPersonCompanyLink' },\n )\n }\n\n if (\n nextPrimaryId &&\n nextPrimaryId !== currentPrimaryId &&\n !addedIds.includes(nextPrimaryId)\n ) {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(nextPrimaryId)}`,\n {\n method: 'PATCH',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({ isPrimary: true }),\n },\n ),\n { companyId: nextPrimaryId, personId, operation: 'setPrimaryPersonCompanyLink' },\n )\n }\n\n const nextLinkedCompanies: LinkedCompanySummary[] = nextSelectedIds.map((id) => {\n const option = optionsById[id]\n const fromCurrent = linkedCompanies.find((entry) => entry.id === id)\n return {\n id,\n displayName: option?.label ?? fromCurrent?.displayName ?? id,\n isPrimary: id === nextPrimaryId,\n }\n })\n setLinkedCompanies(nextLinkedCompanies)\n await loadData({ showLoading: false })\n await onChanged?.()\n flash(\n t('customers.people.detail.companies.manageSuccess', 'Linked companies updated.'),\n 'success',\n )\n } catch (error) {\n const message =\n error instanceof Error\n ? error.message\n : t(\n 'customers.people.detail.companies.manageError',\n 'Failed to update linked companies.',\n )\n flash(message, 'error')\n throw error\n }\n },\n [linkedCompanies, linkedPrimaryId, loadData, onChanged, personId, runWriteMutation, t],\n )\n\n return (\n <>\n <div className=\"space-y-4\">\n <div className=\"flex flex-col gap-3 rounded-2xl border border-border/70 bg-card px-4 py-4 sm:flex-row sm:items-center sm:justify-between\">\n <div className=\"space-y-1\">\n <div className=\"text-sm font-semibold text-foreground\">\n {t('customers.people.detail.companies.manageTitle', 'Manage linked companies')}\n </div>\n <div className=\"text-sm text-muted-foreground\">\n {t('customers.people.detail.companies.summary', '{{count}} linked companies', {\n count: linkedCompanies.length,\n })}\n </div>\n </div>\n <div className=\"flex items-center gap-2\">\n <Input\n value={search}\n onChange={(event) => setSearch(event.target.value)}\n placeholder={t(\n 'customers.people.detail.companies.searchPlaceholder',\n 'Search linked companies\u2026',\n )}\n className=\"sm:w-[260px]\"\n />\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setDialogOpen(true)}\n >\n <Link2 className=\"mr-2 size-4\" />\n {t('customers.people.detail.companies.manageAction', 'Manage links')}\n </Button>\n </div>\n </div>\n\n <div className=\"flex items-center justify-end\">\n <select\n value={sort}\n onChange={(event) =>\n setSort(event.target.value as 'name-asc' | 'name-desc' | 'recent')\n }\n className=\"h-10 rounded-md border bg-background px-3 text-sm focus:outline-none focus:ring-1 focus:ring-ring\"\n >\n <option value=\"name-asc\">\n {t('customers.people.detail.companies.sortNameAsc', 'Sort: Name A-Z')}\n </option>\n <option value=\"name-desc\">\n {t('customers.people.detail.companies.sortNameDesc', 'Sort: Name Z-A')}\n </option>\n <option value=\"recent\">\n {t('customers.people.detail.companies.sortRecent', 'Sort: Recently active')}\n </option>\n </select>\n </div>\n\n {loading ? (\n <div className=\"space-y-4\">\n {[1, 2].map((idx) => (\n <div\n key={idx}\n className=\"h-[320px] animate-pulse rounded-2xl border border-border/60 bg-muted/30\"\n />\n ))}\n </div>\n ) : items.length === 0 ? (\n <div className=\"rounded-2xl border border-dashed border-border/60 px-6 py-12 text-center text-sm text-muted-foreground\">\n {search.trim().length\n ? t(\n 'customers.people.detail.companies.noSearchResults',\n 'No linked companies match your search.',\n )\n : t(\n 'customers.people.detail.empty.companies',\n 'No company linked to this person.',\n )}\n </div>\n ) : (\n <>\n <div className=\"space-y-4\">\n {items.map((item) => (\n <CompanyCard\n key={item.companyId}\n data={item}\n personName={_personName}\n statusMap={statusDict?.map}\n lifecycleMap={lifecycleDict?.map}\n temperatureMap={temperatureDict?.map}\n renewalQuarterMap={renewalQuarterDict?.map}\n roleMap={roleDict?.map}\n />\n ))}\n </div>\n <Pagination page={page} totalPages={totalPages} onPageChange={setPage} />\n </>\n )}\n </div>\n\n <LinkEntityDialog\n open={dialogOpen}\n onOpenChange={setDialogOpen}\n adapter={companyLinkAdapter}\n initialSelectedIds={linkedIds}\n initialPrimaryId={linkedPrimaryId}\n primarySupported\n onConfirm={handleLinkConfirm}\n runGuardedMutation={runWriteMutation}\n />\n </>\n )\n}\n\nexport default PersonCompaniesSection\n\nexport { sameIdSet }\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
3
|
+
import * as React from "react";
|
|
4
|
+
import Link from "next/link";
|
|
5
|
+
import { Phone, Mail, Building2, Trash2, Pencil } from "lucide-react";
|
|
6
|
+
import { cn } from "@open-mercato/shared/lib/utils";
|
|
7
|
+
import { useT } from "@open-mercato/shared/lib/i18n/context";
|
|
8
|
+
import { Button } from "@open-mercato/ui/primitives/button";
|
|
9
|
+
import { IconButton } from "@open-mercato/ui/primitives/icon-button";
|
|
10
|
+
import { Badge } from "@open-mercato/ui/primitives/badge";
|
|
11
|
+
import { useQueryClient } from "@tanstack/react-query";
|
|
12
|
+
import { PersonTagsDialog } from "./PersonTagsDialog.js";
|
|
13
|
+
import { useCustomerDictionary, invalidateCustomerDictionary } from "./hooks/useCustomerDictionary.js";
|
|
14
|
+
import { renderDictionaryIcon } from "../../../dictionaries/components/dictionaryAppearance.js";
|
|
15
|
+
import { getInitials, formatFallbackLabel } from "./utils.js";
|
|
16
|
+
function DictionaryBadge({ value, map, categoryIcon, className }) {
|
|
17
|
+
const entry = map?.[value];
|
|
18
|
+
const color = entry?.color ?? null;
|
|
19
|
+
const icon = entry?.icon ?? null;
|
|
20
|
+
const label = entry?.label ?? formatFallbackLabel(value);
|
|
21
|
+
const colorStyle = color ? { color, borderColor: color, backgroundColor: `${color}1A` } : void 0;
|
|
22
|
+
return /* @__PURE__ */ jsxs(
|
|
23
|
+
Badge,
|
|
24
|
+
{
|
|
25
|
+
variant: "outline",
|
|
26
|
+
className: cn(
|
|
27
|
+
"rounded-sm gap-1.5 text-xs font-medium",
|
|
28
|
+
className
|
|
29
|
+
),
|
|
30
|
+
style: colorStyle,
|
|
31
|
+
children: [
|
|
32
|
+
icon ? renderDictionaryIcon(icon, "size-2.5") : categoryIcon ?? null,
|
|
33
|
+
label
|
|
34
|
+
]
|
|
35
|
+
}
|
|
36
|
+
);
|
|
37
|
+
}
|
|
38
|
+
function TagBadge({ tag }) {
|
|
39
|
+
const colorStyle = tag.color ? { color: tag.color, borderColor: tag.color, backgroundColor: `${tag.color}1A` } : void 0;
|
|
40
|
+
return /* @__PURE__ */ jsx(Badge, { variant: "outline", className: "rounded-sm gap-1.5 text-xs font-medium", style: colorStyle, children: tag.label });
|
|
41
|
+
}
|
|
42
|
+
function PersonDetailHeader({
|
|
43
|
+
data,
|
|
44
|
+
onTagsChange,
|
|
45
|
+
tagsSectionControllerRef,
|
|
46
|
+
onSave,
|
|
47
|
+
onDelete,
|
|
48
|
+
isDirty,
|
|
49
|
+
isSaving,
|
|
50
|
+
onFocusField,
|
|
51
|
+
onOpenCompaniesTab,
|
|
52
|
+
onDataReload
|
|
53
|
+
}) {
|
|
54
|
+
const t = useT();
|
|
55
|
+
const queryClient = useQueryClient();
|
|
56
|
+
const [manageTagsOpen, setManageTagsOpen] = React.useState(false);
|
|
57
|
+
const person = data.person;
|
|
58
|
+
const profile = data.profile;
|
|
59
|
+
const displayName = person.displayName || t("customers.people.detail.untitled", "Untitled");
|
|
60
|
+
const jobTitle = profile?.jobTitle ?? null;
|
|
61
|
+
const linkedCompanies = React.useMemo(() => {
|
|
62
|
+
const items = Array.isArray(data.companies) && data.companies.length > 0 ? data.companies : data.company ? [{ ...data.company, isPrimary: Boolean(data.isPrimary) }] : [];
|
|
63
|
+
return items;
|
|
64
|
+
}, [data.companies, data.company, data.isPrimary]);
|
|
65
|
+
const visibleCompanies = React.useMemo(() => linkedCompanies.slice(0, 3), [linkedCompanies]);
|
|
66
|
+
const hiddenCompaniesCount = Math.max(0, linkedCompanies.length - visibleCompanies.length);
|
|
67
|
+
const visibleTags = React.useMemo(() => data.tags.slice(0, 6), [data.tags]);
|
|
68
|
+
const hiddenTagsCount = Math.max(0, data.tags.length - visibleTags.length);
|
|
69
|
+
const primaryCompany = linkedCompanies.find((entry) => entry.isPrimary) ?? linkedCompanies[0] ?? null;
|
|
70
|
+
const companyName = primaryCompany?.displayName ?? null;
|
|
71
|
+
const companyId = primaryCompany?.id ?? profile?.companyEntityId ?? null;
|
|
72
|
+
const personOrgId = person.organizationId ?? null;
|
|
73
|
+
const { data: statusDict } = useCustomerDictionary("statuses", 0, personOrgId);
|
|
74
|
+
const { data: lifecycleDict } = useCustomerDictionary("lifecycle-stages", 0, personOrgId);
|
|
75
|
+
const { data: sourceDict } = useCustomerDictionary("sources", 0, personOrgId);
|
|
76
|
+
const { data: temperatureDict } = useCustomerDictionary("temperature", 0, personOrgId);
|
|
77
|
+
const { data: renewalQuarterDict } = useCustomerDictionary("renewal-quarters", 0, personOrgId);
|
|
78
|
+
return /* @__PURE__ */ jsxs("div", { className: "rounded-lg border bg-card px-6 py-5", children: [
|
|
79
|
+
/* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-4 sm:flex-row sm:items-start sm:gap-5", children: [
|
|
80
|
+
/* @__PURE__ */ jsx("div", { className: "flex size-18 shrink-0 items-center justify-center rounded-full bg-muted text-xl font-bold text-muted-foreground", children: getInitials(displayName) }),
|
|
81
|
+
/* @__PURE__ */ jsxs("div", { className: "min-w-0 flex-1", children: [
|
|
82
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
83
|
+
/* @__PURE__ */ jsx("h1", { className: "truncate text-2xl font-bold text-foreground", children: displayName }),
|
|
84
|
+
data.isPrimary && /* @__PURE__ */ jsx("span", { className: "shrink-0 rounded-sm bg-status-warning-bg px-1.5 py-0.5 text-overline font-bold text-status-warning-text", children: t("customers.people.detail.header.primary", "PRIMARY") })
|
|
85
|
+
] }),
|
|
86
|
+
(jobTitle || companyName) && /* @__PURE__ */ jsxs("p", { className: "mt-0.5 text-sm text-muted-foreground", children: [
|
|
87
|
+
jobTitle,
|
|
88
|
+
jobTitle && companyName && ` ${t("customers.people.detail.header.at", "at")} `,
|
|
89
|
+
companyName && companyId && /* @__PURE__ */ jsx(Link, { href: `/backend/customers/companies-v2/${companyId}`, className: "text-primary hover:underline", children: companyName }),
|
|
90
|
+
companyName && !companyId && companyName
|
|
91
|
+
] }),
|
|
92
|
+
/* @__PURE__ */ jsxs("div", { className: "mt-1.5 flex flex-wrap items-center gap-x-5 gap-y-1 text-sm text-muted-foreground", children: [
|
|
93
|
+
person.primaryPhone && /* @__PURE__ */ jsxs("span", { className: "inline-flex items-center gap-1.5", children: [
|
|
94
|
+
/* @__PURE__ */ jsx(Phone, { className: "size-3.5" }),
|
|
95
|
+
/* @__PURE__ */ jsx("a", { href: `tel:${person.primaryPhone}`, className: "hover:text-foreground", children: person.primaryPhone })
|
|
96
|
+
] }),
|
|
97
|
+
person.primaryEmail && /* @__PURE__ */ jsxs("span", { className: "inline-flex items-center gap-1.5", children: [
|
|
98
|
+
/* @__PURE__ */ jsx(Mail, { className: "size-3.5" }),
|
|
99
|
+
/* @__PURE__ */ jsx("a", { href: `mailto:${person.primaryEmail}`, className: "hover:text-foreground", children: person.primaryEmail })
|
|
100
|
+
] })
|
|
101
|
+
] }),
|
|
102
|
+
linkedCompanies.length > 0 && /* @__PURE__ */ jsxs("div", { className: "mt-1.5 flex flex-wrap items-center gap-2 text-sm", children: [
|
|
103
|
+
/* @__PURE__ */ jsxs("span", { className: "text-muted-foreground", children: [
|
|
104
|
+
/* @__PURE__ */ jsx(Building2, { className: "mr-1 inline size-3.5" }),
|
|
105
|
+
t("customers.people.detail.header.companies", "Companies"),
|
|
106
|
+
" (",
|
|
107
|
+
linkedCompanies.length,
|
|
108
|
+
"):"
|
|
109
|
+
] }),
|
|
110
|
+
visibleCompanies.map((company) => /* @__PURE__ */ jsxs(
|
|
111
|
+
Link,
|
|
112
|
+
{
|
|
113
|
+
href: `/backend/customers/companies-v2/${company.id}`,
|
|
114
|
+
className: cn(
|
|
115
|
+
"inline-flex items-center gap-1.5 rounded-sm border px-2 py-0.5 text-xs font-semibold transition-colors hover:bg-status-info-bg",
|
|
116
|
+
company.isPrimary ? "border-status-info-border bg-status-info-bg text-status-info-text" : "border-border bg-background text-foreground"
|
|
117
|
+
),
|
|
118
|
+
children: [
|
|
119
|
+
/* @__PURE__ */ jsx(Building2, { className: "size-3" }),
|
|
120
|
+
company.displayName,
|
|
121
|
+
company.isPrimary ? /* @__PURE__ */ jsx("span", { className: "rounded-sm bg-status-info-icon px-1 py-px text-overline font-bold text-white", children: t("customers.people.detail.header.primary", "PRIMARY") }) : null
|
|
122
|
+
]
|
|
123
|
+
},
|
|
124
|
+
company.id
|
|
125
|
+
)),
|
|
126
|
+
hiddenCompaniesCount > 0 ? /* @__PURE__ */ jsxs(Badge, { variant: "outline", className: "rounded-sm text-xs font-semibold", children: [
|
|
127
|
+
"+",
|
|
128
|
+
hiddenCompaniesCount,
|
|
129
|
+
" ",
|
|
130
|
+
t("customers.people.detail.header.more", "more")
|
|
131
|
+
] }) : null
|
|
132
|
+
] }),
|
|
133
|
+
/* @__PURE__ */ jsxs("div", { className: "mt-2.5 flex flex-wrap items-center gap-2", children: [
|
|
134
|
+
person.status && /* @__PURE__ */ jsx(DictionaryBadge, { value: person.status, map: statusDict?.map }),
|
|
135
|
+
person.lifecycleStage && /* @__PURE__ */ jsx(DictionaryBadge, { value: person.lifecycleStage, map: lifecycleDict?.map }),
|
|
136
|
+
person.source && /* @__PURE__ */ jsx(DictionaryBadge, { value: person.source, map: sourceDict?.map }),
|
|
137
|
+
person.temperature && /* @__PURE__ */ jsx(DictionaryBadge, { value: person.temperature, map: temperatureDict?.map }),
|
|
138
|
+
person.renewalQuarter && /* @__PURE__ */ jsx(DictionaryBadge, { value: person.renewalQuarter, map: renewalQuarterDict?.map }),
|
|
139
|
+
visibleTags.map((tag) => /* @__PURE__ */ jsx(TagBadge, { tag }, tag.id ?? tag.label)),
|
|
140
|
+
hiddenTagsCount > 0 ? /* @__PURE__ */ jsxs(Badge, { variant: "outline", className: "rounded-sm gap-1.5 text-xs font-medium", children: [
|
|
141
|
+
"+",
|
|
142
|
+
hiddenTagsCount,
|
|
143
|
+
" ",
|
|
144
|
+
t("customers.people.detail.header.more", "more")
|
|
145
|
+
] }) : null,
|
|
146
|
+
/* @__PURE__ */ jsxs(
|
|
147
|
+
Button,
|
|
148
|
+
{
|
|
149
|
+
type: "button",
|
|
150
|
+
variant: "outline",
|
|
151
|
+
size: "sm",
|
|
152
|
+
className: "h-auto rounded-sm px-2 py-1 text-xs font-medium text-muted-foreground hover:text-foreground",
|
|
153
|
+
onClick: () => setManageTagsOpen(true),
|
|
154
|
+
children: [
|
|
155
|
+
/* @__PURE__ */ jsx(Pencil, { className: "mr-1 size-3" }),
|
|
156
|
+
t("customers.people.detail.actions.manageTags", "Edit tags")
|
|
157
|
+
]
|
|
158
|
+
}
|
|
159
|
+
)
|
|
160
|
+
] })
|
|
161
|
+
] }),
|
|
162
|
+
/* @__PURE__ */ jsxs("div", { className: "flex w-full shrink-0 items-center justify-start gap-2 sm:w-auto sm:justify-end", children: [
|
|
163
|
+
/* @__PURE__ */ jsx(
|
|
164
|
+
IconButton,
|
|
165
|
+
{
|
|
166
|
+
variant: "outline",
|
|
167
|
+
size: "sm",
|
|
168
|
+
type: "button",
|
|
169
|
+
"aria-label": t("customers.people.detail.actions.delete", "Delete"),
|
|
170
|
+
onClick: () => {
|
|
171
|
+
void onDelete();
|
|
172
|
+
},
|
|
173
|
+
children: /* @__PURE__ */ jsx(Trash2, { className: "size-4" })
|
|
174
|
+
}
|
|
175
|
+
),
|
|
176
|
+
/* @__PURE__ */ jsx(
|
|
177
|
+
Button,
|
|
178
|
+
{
|
|
179
|
+
type: "button",
|
|
180
|
+
size: "sm",
|
|
181
|
+
onClick: onSave,
|
|
182
|
+
disabled: !isDirty || isSaving,
|
|
183
|
+
children: t("customers.people.detail.actions.save", "Save")
|
|
184
|
+
}
|
|
185
|
+
)
|
|
186
|
+
] })
|
|
187
|
+
] }),
|
|
188
|
+
/* @__PURE__ */ jsx(
|
|
189
|
+
PersonTagsDialog,
|
|
190
|
+
{
|
|
191
|
+
open: manageTagsOpen,
|
|
192
|
+
onClose: () => setManageTagsOpen(false),
|
|
193
|
+
personId: person.id,
|
|
194
|
+
personOrganizationId: person.organizationId ?? null,
|
|
195
|
+
personData: {
|
|
196
|
+
status: person.status,
|
|
197
|
+
lifecycleStage: person.lifecycleStage,
|
|
198
|
+
source: person.source,
|
|
199
|
+
temperature: person.temperature,
|
|
200
|
+
renewalQuarter: person.renewalQuarter,
|
|
201
|
+
jobTitle: data.profile?.jobTitle ?? null,
|
|
202
|
+
customFields: data.customFields,
|
|
203
|
+
tags: data.tags
|
|
204
|
+
},
|
|
205
|
+
onSaved: () => {
|
|
206
|
+
void invalidateCustomerDictionary(queryClient, "statuses");
|
|
207
|
+
void invalidateCustomerDictionary(queryClient, "lifecycle-stages");
|
|
208
|
+
void invalidateCustomerDictionary(queryClient, "sources");
|
|
209
|
+
void invalidateCustomerDictionary(queryClient, "temperature");
|
|
210
|
+
void invalidateCustomerDictionary(queryClient, "renewal-quarters");
|
|
211
|
+
onDataReload?.();
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
)
|
|
215
|
+
] });
|
|
216
|
+
}
|
|
217
|
+
export {
|
|
218
|
+
PersonDetailHeader
|
|
219
|
+
};
|
|
220
|
+
//# sourceMappingURL=PersonDetailHeader.js.map
|