@open-mercato/core 0.5.1-develop.2912.8d7b1fef24 → 0.5.1-develop.2924.d13908516e

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 (59) hide show
  1. package/dist/modules/customers/api/companies/[id]/people/route.js +12 -7
  2. package/dist/modules/customers/api/companies/[id]/people/route.js.map +2 -2
  3. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js +2 -1
  4. package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js.map +2 -2
  5. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +7 -2
  6. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +2 -2
  7. package/dist/modules/customers/commands/companies.js +93 -19
  8. package/dist/modules/customers/commands/companies.js.map +2 -2
  9. package/dist/modules/customers/commands/people.js +9 -1
  10. package/dist/modules/customers/commands/people.js.map +2 -2
  11. package/dist/modules/customers/commands/personCompanyLinks.js +2 -2
  12. package/dist/modules/customers/commands/personCompanyLinks.js.map +2 -2
  13. package/dist/modules/customers/components/detail/CompanyCard.js +32 -3
  14. package/dist/modules/customers/components/detail/CompanyCard.js.map +2 -2
  15. package/dist/modules/customers/components/detail/CompanyDetailTabs.js +37 -19
  16. package/dist/modules/customers/components/detail/CompanyDetailTabs.js.map +2 -2
  17. package/dist/modules/customers/components/detail/CompanyPeopleSection.js +7 -4
  18. package/dist/modules/customers/components/detail/CompanyPeopleSection.js.map +2 -2
  19. package/dist/modules/customers/components/detail/PersonCompaniesSection.js +63 -2
  20. package/dist/modules/customers/components/detail/PersonCompaniesSection.js.map +2 -2
  21. package/dist/modules/customers/components/detail/PersonDetailTabs.js +37 -19
  22. package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +2 -2
  23. package/dist/modules/customers/components/detail/TasksSection.js +1 -11
  24. package/dist/modules/customers/components/detail/TasksSection.js.map +2 -2
  25. package/dist/modules/customers/components/formConfig.js +50 -39
  26. package/dist/modules/customers/components/formConfig.js.map +2 -2
  27. package/dist/modules/customers/events.js +3 -3
  28. package/dist/modules/customers/events.js.map +2 -2
  29. package/dist/modules/customers/lib/displayName.js +13 -1
  30. package/dist/modules/customers/lib/displayName.js.map +2 -2
  31. package/dist/modules/customers/lib/personCompanies.js +12 -7
  32. package/dist/modules/customers/lib/personCompanies.js.map +2 -2
  33. package/dist/modules/customers/lib/personCompanyLinkTable.js +5 -0
  34. package/dist/modules/customers/lib/personCompanyLinkTable.js.map +2 -2
  35. package/dist/modules/workflows/lib/activity-executor.js +21 -17
  36. package/dist/modules/workflows/lib/activity-executor.js.map +2 -2
  37. package/package.json +3 -3
  38. package/src/modules/customers/api/companies/[id]/people/route.ts +12 -7
  39. package/src/modules/customers/backend/customers/companies-v2/[id]/page.tsx +2 -1
  40. package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +12 -2
  41. package/src/modules/customers/commands/companies.ts +107 -19
  42. package/src/modules/customers/commands/people.ts +16 -1
  43. package/src/modules/customers/commands/personCompanyLinks.ts +3 -2
  44. package/src/modules/customers/components/detail/CompanyCard.tsx +28 -4
  45. package/src/modules/customers/components/detail/CompanyDetailTabs.tsx +18 -2
  46. package/src/modules/customers/components/detail/CompanyPeopleSection.tsx +8 -4
  47. package/src/modules/customers/components/detail/PersonCompaniesSection.tsx +66 -0
  48. package/src/modules/customers/components/detail/PersonDetailTabs.tsx +18 -2
  49. package/src/modules/customers/components/detail/TasksSection.tsx +1 -8
  50. package/src/modules/customers/components/formConfig.tsx +59 -40
  51. package/src/modules/customers/events.ts +3 -3
  52. package/src/modules/customers/i18n/de.json +10 -0
  53. package/src/modules/customers/i18n/en.json +10 -0
  54. package/src/modules/customers/i18n/es.json +10 -0
  55. package/src/modules/customers/i18n/pl.json +10 -0
  56. package/src/modules/customers/lib/displayName.ts +19 -0
  57. package/src/modules/customers/lib/personCompanies.ts +12 -7
  58. package/src/modules/customers/lib/personCompanyLinkTable.ts +14 -0
  59. package/src/modules/workflows/lib/activity-executor.ts +21 -18
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/customers/components/detail/CompanyPeopleSection.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { Users, Link2, Plus, Filter } from 'lucide-react'\nimport { EmptyState } from '@open-mercato/ui/backend/EmptyState'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport {\n readJsonFromLocalStorage,\n writeJsonToLocalStorage,\n} from '@open-mercato/shared/lib/browser/safeLocalStorage'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { createTranslatorWithFallback } from '@open-mercato/shared/lib/i18n/translate'\nimport type { SectionAction, TabEmptyStateConfig, Translator } from './types'\nimport { CreatePersonDialog } from './CreatePersonDialog'\nimport { PersonCard } from './PersonCard'\nimport { DecisionMakersFooter } from './DecisionMakersFooter'\nimport { RolesSection } from './RolesSection'\nimport { LinkEntityDialog, type LinkEntityOption } from '../linking/LinkEntityDialog'\nimport { createPersonLinkAdapter } from '../linking/adapters/personAdapter'\n\ntype GuardedMutationRunner = <T>(\n operation: () => Promise<T>,\n mutationPayload?: Record<string, unknown>,\n) => Promise<T>\n\nexport type CompanyPersonSummary = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n lifecycleStage?: string | null\n jobTitle?: string | null\n department?: string | null\n createdAt?: string | null\n organizationId?: string | null\n temperature?: string | null\n source?: string | null\n linkedAt?: string | null\n}\n\nexport type CompanyPeopleSectionProps = {\n companyId: string\n companyName?: string\n initialPeople: CompanyPersonSummary[]\n addActionLabel: string\n emptyLabel: string\n emptyState: TabEmptyStateConfig\n onPeopleChange?: (next: CompanyPersonSummary[]) => void\n onActionChange?: (action: SectionAction | null) => void\n translator?: Translator\n onLoadingChange?: (isLoading: boolean) => void\n onDataRefresh?: () => Promise<void> | void\n runGuardedMutation?: GuardedMutationRunner\n}\n\nconst COMPANY_PEOPLE_PAGE_SIZE = 20\n\nfunction normalizeCompanyPerson(record: Record<string, unknown>): CompanyPersonSummary | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? record.display_name.trim()\n : null\n if (!displayName) return null\n return {\n id,\n displayName,\n primaryEmail:\n typeof record.primaryEmail === 'string'\n ? record.primaryEmail\n : typeof record.primary_email === 'string'\n ? record.primary_email\n : null,\n primaryPhone:\n typeof record.primaryPhone === 'string'\n ? record.primaryPhone\n : typeof record.primary_phone === 'string'\n ? record.primary_phone\n : null,\n status:\n typeof record.status === 'string'\n ? record.status\n : null,\n lifecycleStage:\n typeof record.lifecycleStage === 'string'\n ? record.lifecycleStage\n : typeof record.lifecycle_stage === 'string'\n ? record.lifecycle_stage\n : null,\n jobTitle:\n typeof record.jobTitle === 'string'\n ? record.jobTitle\n : typeof record.job_title === 'string'\n ? record.job_title\n : null,\n department:\n typeof record.department === 'string'\n ? record.department\n : null,\n createdAt:\n typeof record.createdAt === 'string'\n ? record.createdAt\n : typeof record.created_at === 'string'\n ? record.created_at\n : null,\n organizationId:\n typeof record.organizationId === 'string'\n ? record.organizationId\n : typeof record.organization_id === 'string'\n ? record.organization_id\n : null,\n temperature:\n typeof record.temperature === 'string'\n ? record.temperature\n : null,\n source:\n typeof record.source === 'string'\n ? record.source\n : null,\n linkedAt:\n typeof record.linkedAt === 'string'\n ? record.linkedAt\n : typeof record.linked_at === 'string'\n ? record.linked_at\n : null,\n }\n}\n\nfunction mergeCompanyPeople(items: CompanyPersonSummary[]): CompanyPersonSummary[] {\n const merged = new Map<string, CompanyPersonSummary>()\n items.forEach((item) => merged.set(item.id, item))\n return Array.from(merged.values())\n}\n\nfunction matchesCompanyPersonSearch(person: CompanyPersonSummary, query: string): boolean {\n const normalizedQuery = query.trim().toLowerCase()\n if (!normalizedQuery.length) return true\n const haystack = [\n person.displayName,\n person.jobTitle ?? '',\n person.primaryEmail ?? '',\n person.primaryPhone ?? '',\n person.department ?? '',\n ]\n .join(' ')\n .toLowerCase()\n return haystack.includes(normalizedQuery)\n}\n\nfunction sortCompanyPeople(\n items: CompanyPersonSummary[],\n sortMode: 'name-asc' | 'name-desc' | 'recent',\n): CompanyPersonSummary[] {\n return [...items].sort((left, right) => {\n if (sortMode === 'recent') {\n const leftTimestamp = Date.parse(left.linkedAt ?? left.createdAt ?? '') || 0\n const rightTimestamp = Date.parse(right.linkedAt ?? right.createdAt ?? '') || 0\n return rightTimestamp - leftTimestamp\n }\n const leftLabel = left.displayName.trim().toLowerCase()\n const rightLabel = right.displayName.trim().toLowerCase()\n if (sortMode === 'name-desc') return rightLabel.localeCompare(leftLabel)\n return leftLabel.localeCompare(rightLabel)\n })\n}\n\nexport function CompanyPeopleSection({\n companyId,\n companyName,\n initialPeople,\n addActionLabel,\n emptyLabel,\n emptyState,\n onPeopleChange,\n onActionChange,\n translator,\n onLoadingChange,\n onDataRefresh,\n runGuardedMutation,\n}: CompanyPeopleSectionProps) {\n const tHook = useT()\n const fallbackTranslator = React.useMemo<Translator>(\n () => createTranslatorWithFallback(tHook),\n [tHook],\n )\n const translate: Translator = translator ?? fallbackTranslator\n const [people, setPeople] = React.useState<CompanyPersonSummary[]>(initialPeople)\n const [removingId, setRemovingId] = React.useState<string | null>(null)\n const [linkDialogOpen, setLinkDialogOpen] = React.useState(false)\n const [createDialogOpen, setCreateDialogOpen] = React.useState(false)\n const [searchQuery, setSearchQuery] = React.useState('')\n const [sortMode, setSortMode] = React.useState<'name-asc' | 'name-desc' | 'recent'>('name-asc')\n const [filtersOpen, setFiltersOpen] = React.useState(true)\n const [visiblePeople, setVisiblePeople] = React.useState<CompanyPersonSummary[]>([])\n const [listPage, setListPage] = React.useState(1)\n const [listTotalPages, setListTotalPages] = React.useState(1)\n const [listTotalCount, setListTotalCount] = React.useState(initialPeople.length)\n const [listLoading, setListLoading] = React.useState(true)\n const [starredIds, setStarredIds] = React.useState<Set<string>>(\n () => new Set(readJsonFromLocalStorage<string[]>(`om:starred-people:${companyId}`, [])),\n )\n const pendingPeopleChangeRef = React.useRef(false)\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 toggleStar = React.useCallback(\n (personId: string) => {\n setStarredIds((prev) => {\n const next = new Set(prev)\n if (next.has(personId)) next.delete(personId)\n else next.add(personId)\n writeJsonToLocalStorage(`om:starred-people:${companyId}`, [...next])\n return next\n })\n },\n [companyId],\n )\n\n const displayedPeople = React.useMemo(\n () => (visiblePeople.length > 0 ? visiblePeople : people),\n [people, visiblePeople],\n )\n const totalLinkedPeople = listTotalCount > 0 ? listTotalCount : displayedPeople.length\n const decisionMakerNames = React.useMemo(\n () =>\n displayedPeople\n .filter((person) => starredIds.has(person.id))\n .map((person) => person.displayName),\n [displayedPeople, starredIds],\n )\n\n React.useEffect(() => {\n const action: SectionAction = {\n label: addActionLabel,\n onClick: () => {\n setCreateDialogOpen(true)\n },\n }\n onActionChange?.(action)\n return () => {\n onActionChange?.(null)\n }\n }, [addActionLabel, onActionChange])\n\n React.useEffect(() => {\n pendingPeopleChangeRef.current = false\n setPeople(initialPeople)\n }, [initialPeople])\n\n React.useEffect(() => {\n if (!pendingPeopleChangeRef.current) return\n pendingPeopleChangeRef.current = false\n onPeopleChange?.(people)\n }, [onPeopleChange, people])\n\n const loadVisiblePeople = React.useCallback(async () => {\n setListLoading(true)\n try {\n const params = new URLSearchParams({\n page: String(listPage),\n pageSize: String(COMPANY_PEOPLE_PAGE_SIZE),\n sort: sortMode,\n })\n if (searchQuery.trim().length > 0) {\n params.set('search', searchQuery.trim())\n }\n const payload = await readApiResultOrThrow<{\n items?: CompanyPersonSummary[]\n page?: number\n total?: number\n totalPages?: number\n }>(\n `/api/customers/companies/${encodeURIComponent(companyId)}/people?${params.toString()}`,\n undefined,\n {\n errorMessage: translate(\n 'customers.companies.detail.people.loadError',\n 'Failed to load people.',\n ),\n },\n )\n const nextTotalCount = typeof payload.total === 'number' ? payload.total : 0\n setVisiblePeople(Array.isArray(payload.items) ? payload.items : [])\n setListPage(typeof payload.page === 'number' ? payload.page : listPage)\n setListTotalCount((current) =>\n searchQuery.trim().length > 0 ? Math.max(current, nextTotalCount) : nextTotalCount,\n )\n setListTotalPages(typeof payload.totalPages === 'number' ? payload.totalPages : 1)\n } catch {\n setVisiblePeople([])\n if (searchQuery.trim().length === 0) {\n setListTotalCount(0)\n }\n setListTotalPages(1)\n } finally {\n setListLoading(false)\n }\n }, [companyId, listPage, searchQuery, sortMode, translate])\n\n React.useEffect(() => {\n void loadVisiblePeople()\n }, [loadVisiblePeople])\n\n React.useEffect(() => {\n setListPage(1)\n }, [searchQuery, sortMode])\n\n const applyPeopleChange = React.useCallback(\n (updater: (current: CompanyPersonSummary[]) => CompanyPersonSummary[]) => {\n setPeople((current) => {\n const next = updater(current)\n if (next !== current) {\n pendingPeopleChangeRef.current = true\n }\n return next\n })\n },\n [],\n )\n\n const handleLinkConfirm = React.useCallback(\n async ({\n addedIds,\n optionsById,\n }: {\n addedIds: string[]\n removedIds: string[]\n optionsById: Record<string, LinkEntityOption>\n }) => {\n if (!addedIds.length) return\n onLoadingChange?.(true)\n try {\n for (const personId 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({ companyId }),\n },\n {\n errorMessage: translate(\n 'customers.companies.detail.people.linkError',\n 'Failed to link person to company.',\n ),\n },\n ),\n {\n personId,\n companyId,\n },\n )\n }\n const optimisticPeople: CompanyPersonSummary[] = addedIds\n .map((personId): CompanyPersonSummary | null => {\n const option = optionsById[personId]\n if (!option) return null\n return {\n id: option.id,\n displayName: option.label,\n primaryEmail: null,\n primaryPhone: null,\n jobTitle: option.subtitle ?? null,\n }\n })\n .filter((entry): entry is CompanyPersonSummary => entry !== null)\n if (optimisticPeople.length > 0) {\n applyPeopleChange((current) => mergeCompanyPeople([...current, ...optimisticPeople]))\n setListTotalCount((current) => current + optimisticPeople.length)\n }\n await loadVisiblePeople()\n flash(\n addedIds.length === 1\n ? translate(\n 'customers.companies.detail.people.linkSuccess',\n 'Person linked to company.',\n )\n : translate(\n 'customers.companies.detail.people.linkSuccessMultiple',\n '{{count}} people linked to company.',\n { count: String(addedIds.length) },\n ),\n 'success',\n )\n } catch (err) {\n try {\n await onDataRefresh?.()\n } catch {\n // preserve original linking error for the user\n }\n const message =\n err instanceof Error\n ? err.message\n : translate(\n 'customers.companies.detail.people.linkError',\n 'Failed to link person to company.',\n )\n flash(message, 'error')\n throw err\n } finally {\n onLoadingChange?.(false)\n }\n },\n [\n applyPeopleChange,\n companyId,\n loadVisiblePeople,\n onDataRefresh,\n onLoadingChange,\n runWriteMutation,\n translate,\n ],\n )\n\n const personLinkAdapter = React.useMemo(\n () =>\n createPersonLinkAdapter({\n dialogTitle: translate('customers.linking.person.dialogTitle', 'Link person'),\n dialogSubtitle: companyName\n ? translate(\n 'customers.linking.person.dialogSubtitleFor',\n 'Link an existing contact to {{name}}',\n { name: companyName },\n )\n : translate(\n 'customers.linking.person.dialogSubtitle',\n 'Link an existing contact to this company',\n ),\n sectionLabel: translate('customers.linking.person.sectionLabel', 'MATCHING CONTACTS'),\n searchPlaceholder: translate(\n 'customers.linking.person.searchPlaceholder',\n 'Search all people\u2026',\n ),\n searchEmptyHint: translate(\n 'customers.linking.person.searchEmpty',\n 'No matching people found.',\n ),\n selectedEmptyHint: translate(\n 'customers.linking.person.selectedEmpty',\n 'No people selected.',\n ),\n confirmButtonLabel: translate('customers.linking.person.confirmButton', 'Link person'),\n showLinkSettings: true,\n roleOptions: [\n { id: 'decision_maker', label: 'Decision maker' },\n { id: 'budget_holder', label: 'Budget holder' },\n { id: 'stakeholder', label: 'Stakeholder' },\n { id: 'contact', label: 'Contact' },\n ],\n excludeLinkedCompanyId: companyId,\n addNew: {\n title: translate('customers.linking.person.addNew', 'Add new contact'),\n subtitle: translate(\n 'customers.linking.person.addNewSubtitle',\n 'Company will be filled in automatically',\n ),\n render: ({ onCancel }) => (\n <CreatePersonDialog\n open\n onClose={onCancel}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n // CreatePersonDialog already created and linked the person to this company\n // via the companyEntityId payload field. Refresh the on-page list and close\n // both the nested and outer dialogs so the user can see the new entry.\n void loadVisiblePeople()\n void onDataRefresh?.()\n setLinkDialogOpen(false)\n onCancel()\n }}\n />\n ),\n },\n }),\n [companyId, companyName, loadVisiblePeople, onDataRefresh, runWriteMutation, translate],\n )\n\n const handleRemove = React.useCallback(\n async (personId: string) => {\n if (!personId || removingId) return\n setRemovingId(personId)\n onLoadingChange?.(true)\n try {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,\n { method: 'DELETE' },\n {\n errorMessage: translate(\n 'customers.companies.detail.people.removeError',\n 'Failed to unlink person from company.',\n ),\n },\n ),\n {\n personId,\n companyId,\n },\n )\n applyPeopleChange((current) => current.filter((entry) => entry.id !== personId))\n setVisiblePeople((current) => current.filter((entry) => entry.id !== personId))\n setListTotalCount((current) => Math.max(0, current - 1))\n await loadVisiblePeople()\n flash(\n translate(\n 'customers.companies.detail.people.removeSuccess',\n 'Person unlinked from company.',\n ),\n 'success',\n )\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : translate(\n 'customers.companies.detail.people.removeError',\n 'Failed to unlink person from company.',\n )\n flash(message, 'error')\n } finally {\n setRemovingId(null)\n onLoadingChange?.(false)\n }\n },\n [\n applyPeopleChange,\n companyId,\n loadVisiblePeople,\n onLoadingChange,\n removingId,\n runWriteMutation,\n translate,\n ],\n )\n\n const linkAction = (\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setLinkDialogOpen(true)}\n >\n <Link2 className=\"mr-1.5 h-4 w-4\" />\n {translate(\n 'customers.companies.detail.people.linkAction',\n 'Link existing person',\n )}\n </Button>\n )\n const addPersonAction = (\n <Button type=\"button\" size=\"sm\" onClick={() => setCreateDialogOpen(true)}>\n <Plus className=\"mr-1.5 h-4 w-4\" />\n {addActionLabel}\n </Button>\n )\n\n if (!listLoading && totalLinkedPeople === 0) {\n return (\n <>\n <EmptyState\n icon={<Users className=\"h-10 w-10 text-muted-foreground\" />}\n title={emptyState.title}\n actionLabel={emptyState.actionLabel}\n onAction={() => setCreateDialogOpen(true)}\n >\n <p className=\"text-sm text-muted-foreground\">{emptyLabel}</p>\n <div className=\"mt-4\">{linkAction}</div>\n </EmptyState>\n <LinkEntityDialog\n open={linkDialogOpen}\n onOpenChange={setLinkDialogOpen}\n adapter={personLinkAdapter}\n initialSelectedIds={[]}\n onConfirm={handleLinkConfirm}\n runGuardedMutation={runWriteMutation}\n />\n <CreatePersonDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n setCreateDialogOpen(false)\n void loadVisiblePeople()\n void onDataRefresh?.()\n }}\n />\n </>\n )\n }\n\n return (\n <>\n <div className=\"space-y-4\">\n <RolesSection\n entityType=\"company\"\n entityId={companyId}\n entityName={companyName ?? null}\n />\n\n <section className=\"rounded-lg border bg-card px-4 py-4 sm:px-5\">\n <div className=\"flex flex-col gap-4\">\n <div className=\"flex flex-col gap-3 xl:flex-row xl:items-start xl:justify-between\">\n <div className=\"space-y-1\">\n <div className=\"flex items-center gap-2\">\n <h3 className=\"text-base font-semibold\">\n {translate(\n 'customers.companies.detail.people.sectionTitle',\n 'People',\n )}\n </h3>\n <Badge\n variant=\"secondary\"\n className=\"rounded-full px-2 py-0 text-xs font-semibold\"\n >\n {totalLinkedPeople}\n </Badge>\n </div>\n <p className=\"text-xs text-muted-foreground\">\n {translate(\n 'customers.companies.detail.people.sectionSubtitle',\n 'Employees and decision makers on the client side',\n )}\n </p>\n </div>\n <div className=\"flex flex-wrap items-center gap-2 xl:justify-end\">\n {linkAction}\n {addPersonAction}\n </div>\n </div>\n\n {totalLinkedPeople > 0 ? (\n <div className=\"flex flex-col gap-3 lg:flex-row lg:items-center\">\n {filtersOpen ? (\n <div className=\"min-w-0 flex-1\">\n <input\n type=\"text\"\n value={searchQuery}\n onChange={(event) => setSearchQuery(event.target.value)}\n placeholder={translate(\n 'customers.companies.detail.people.searchPlaceholder',\n 'Search by name, role, email...',\n )}\n className=\"h-10 w-full rounded-md border bg-background px-3 text-sm placeholder:text-muted-foreground focus:outline-none focus:ring-1 focus:ring-ring\"\n />\n </div>\n ) : null}\n <div className=\"flex flex-wrap items-center gap-2 lg:justify-end\">\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setFiltersOpen((current) => !current)}\n className=\"h-10\"\n >\n <Filter className=\"mr-1.5 h-4 w-4\" />\n {translate(\n 'customers.companies.detail.people.filter',\n 'Filters',\n )}\n </Button>\n {filtersOpen ? (\n <select\n value={sortMode}\n onChange={(event) =>\n setSortMode(event.target.value as 'name-asc' | 'name-desc' | 'recent')\n }\n className=\"h-10 min-w-[11rem] 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 {translate(\n 'customers.companies.detail.people.sortNameAsc',\n 'Sort: Name A-Z',\n )}\n </option>\n <option value=\"name-desc\">\n {translate(\n 'customers.companies.detail.people.sortNameDesc',\n 'Sort: Name Z-A',\n )}\n </option>\n <option value=\"recent\">\n {translate(\n 'customers.companies.detail.people.sortRecent',\n 'Sort: Recently linked',\n )}\n </option>\n </select>\n ) : null}\n </div>\n </div>\n ) : null}\n\n {listLoading ? (\n <p className=\"py-6 text-center text-sm text-muted-foreground\">\n {translate('customers.companies.detail.people.loading', 'Loading people\u2026')}\n </p>\n ) : visiblePeople.length > 0 ? (\n <>\n <div\n className=\"grid items-start gap-4\"\n style={{\n gridTemplateColumns: 'repeat(auto-fit, minmax(min(100%, 19.5rem), 1fr))',\n }}\n >\n {visiblePeople.map((person) => (\n <PersonCard\n key={person.id}\n person={person}\n isStarred={starredIds.has(person.id)}\n onToggleStar={toggleStar}\n onUnlink={handleRemove}\n />\n ))}\n </div>\n {listTotalPages > 1 ? (\n <div className=\"flex items-center justify-between border-t border-border/60 pt-3 text-sm text-muted-foreground\">\n <span>\n {translate(\n 'customers.companies.detail.people.pageSummary',\n 'Page {{page}} of {{total}}',\n {\n page: listPage,\n total: listTotalPages,\n },\n )}\n </span>\n <div className=\"flex items-center gap-2\">\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setListPage((current) => Math.max(1, current - 1))}\n disabled={listPage <= 1}\n >\n {translate('customers.companies.detail.people.previous', 'Previous')}\n </Button>\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() =>\n setListPage((current) => Math.min(listTotalPages, current + 1))\n }\n disabled={listPage >= listTotalPages}\n >\n {translate('customers.companies.detail.people.next', 'Next')}\n </Button>\n </div>\n </div>\n ) : null}\n </>\n ) : totalLinkedPeople > 0 ? (\n <p className=\"py-6 text-center text-sm text-muted-foreground\">\n {translate(\n 'customers.companies.detail.people.noSearchResults',\n 'No people match your search.',\n )}\n </p>\n ) : null}\n </div>\n </section>\n\n <DecisionMakersFooter\n names={decisionMakerNames}\n onSendInvitation={() => {\n const starredEmails = displayedPeople\n .filter((person) => starredIds.has(person.id) && person.primaryEmail)\n .map((person) => person.primaryEmail!)\n if (starredEmails.length > 0) {\n window.open(`mailto:${starredEmails.join(',')}`, '_blank')\n }\n }}\n />\n </div>\n\n <LinkEntityDialog\n open={linkDialogOpen}\n onOpenChange={setLinkDialogOpen}\n adapter={personLinkAdapter}\n initialSelectedIds={[]}\n onConfirm={handleLinkConfirm}\n runGuardedMutation={runWriteMutation}\n />\n\n <CreatePersonDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n setCreateDialogOpen(false)\n void loadVisiblePeople()\n void onDataRefresh?.()\n }}\n />\n </>\n )\n}\n\nexport default CompanyPeopleSection\n\nexport { mergeCompanyPeople, matchesCompanyPersonSearch, sortCompanyPeople, normalizeCompanyPerson }\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { Users, Link2, Plus, Filter } from 'lucide-react'\nimport { EmptyState } from '@open-mercato/ui/backend/EmptyState'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport {\n readJsonFromLocalStorage,\n writeJsonToLocalStorage,\n} from '@open-mercato/shared/lib/browser/safeLocalStorage'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { createTranslatorWithFallback } from '@open-mercato/shared/lib/i18n/translate'\nimport { useAppEvent } from '@open-mercato/ui/backend/injection/useAppEvent'\nimport type { SectionAction, TabEmptyStateConfig, Translator } from './types'\nimport { CreatePersonDialog } from './CreatePersonDialog'\nimport { PersonCard } from './PersonCard'\nimport { DecisionMakersFooter } from './DecisionMakersFooter'\nimport { RolesSection } from './RolesSection'\nimport { LinkEntityDialog, type LinkEntityOption } from '../linking/LinkEntityDialog'\nimport { createPersonLinkAdapter } from '../linking/adapters/personAdapter'\n\ntype GuardedMutationRunner = <T>(\n operation: () => Promise<T>,\n mutationPayload?: Record<string, unknown>,\n) => Promise<T>\n\nexport type CompanyPersonSummary = {\n id: string\n displayName: string\n primaryEmail?: string | null\n primaryPhone?: string | null\n status?: string | null\n lifecycleStage?: string | null\n jobTitle?: string | null\n department?: string | null\n createdAt?: string | null\n organizationId?: string | null\n temperature?: string | null\n source?: string | null\n linkedAt?: string | null\n}\n\nexport type CompanyPeopleSectionProps = {\n companyId: string\n companyName?: string\n initialPeople: CompanyPersonSummary[]\n addActionLabel: string\n emptyLabel: string\n emptyState: TabEmptyStateConfig\n onPeopleChange?: (next: CompanyPersonSummary[]) => void\n onActionChange?: (action: SectionAction | null) => void\n translator?: Translator\n onLoadingChange?: (isLoading: boolean) => void\n onDataRefresh?: () => Promise<void> | void\n runGuardedMutation?: GuardedMutationRunner\n}\n\nconst COMPANY_PEOPLE_PAGE_SIZE = 20\n\nfunction normalizeCompanyPerson(record: Record<string, unknown>): CompanyPersonSummary | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? record.display_name.trim()\n : null\n if (!displayName) return null\n return {\n id,\n displayName,\n primaryEmail:\n typeof record.primaryEmail === 'string'\n ? record.primaryEmail\n : typeof record.primary_email === 'string'\n ? record.primary_email\n : null,\n primaryPhone:\n typeof record.primaryPhone === 'string'\n ? record.primaryPhone\n : typeof record.primary_phone === 'string'\n ? record.primary_phone\n : null,\n status:\n typeof record.status === 'string'\n ? record.status\n : null,\n lifecycleStage:\n typeof record.lifecycleStage === 'string'\n ? record.lifecycleStage\n : typeof record.lifecycle_stage === 'string'\n ? record.lifecycle_stage\n : null,\n jobTitle:\n typeof record.jobTitle === 'string'\n ? record.jobTitle\n : typeof record.job_title === 'string'\n ? record.job_title\n : null,\n department:\n typeof record.department === 'string'\n ? record.department\n : null,\n createdAt:\n typeof record.createdAt === 'string'\n ? record.createdAt\n : typeof record.created_at === 'string'\n ? record.created_at\n : null,\n organizationId:\n typeof record.organizationId === 'string'\n ? record.organizationId\n : typeof record.organization_id === 'string'\n ? record.organization_id\n : null,\n temperature:\n typeof record.temperature === 'string'\n ? record.temperature\n : null,\n source:\n typeof record.source === 'string'\n ? record.source\n : null,\n linkedAt:\n typeof record.linkedAt === 'string'\n ? record.linkedAt\n : typeof record.linked_at === 'string'\n ? record.linked_at\n : null,\n }\n}\n\nfunction mergeCompanyPeople(items: CompanyPersonSummary[]): CompanyPersonSummary[] {\n const merged = new Map<string, CompanyPersonSummary>()\n items.forEach((item) => merged.set(item.id, item))\n return Array.from(merged.values())\n}\n\nfunction matchesCompanyPersonSearch(person: CompanyPersonSummary, query: string): boolean {\n const normalizedQuery = query.trim().toLowerCase()\n if (!normalizedQuery.length) return true\n const haystack = [\n person.displayName,\n person.jobTitle ?? '',\n person.primaryEmail ?? '',\n person.primaryPhone ?? '',\n person.department ?? '',\n ]\n .join(' ')\n .toLowerCase()\n return haystack.includes(normalizedQuery)\n}\n\nfunction sortCompanyPeople(\n items: CompanyPersonSummary[],\n sortMode: 'name-asc' | 'name-desc' | 'recent',\n): CompanyPersonSummary[] {\n return [...items].sort((left, right) => {\n if (sortMode === 'recent') {\n const leftTimestamp = Date.parse(left.linkedAt ?? left.createdAt ?? '') || 0\n const rightTimestamp = Date.parse(right.linkedAt ?? right.createdAt ?? '') || 0\n return rightTimestamp - leftTimestamp\n }\n const leftLabel = left.displayName.trim().toLowerCase()\n const rightLabel = right.displayName.trim().toLowerCase()\n if (sortMode === 'name-desc') return rightLabel.localeCompare(leftLabel)\n return leftLabel.localeCompare(rightLabel)\n })\n}\n\nexport function CompanyPeopleSection({\n companyId,\n companyName,\n initialPeople,\n addActionLabel,\n emptyLabel,\n emptyState,\n onPeopleChange,\n onActionChange,\n translator,\n onLoadingChange,\n onDataRefresh,\n runGuardedMutation,\n}: CompanyPeopleSectionProps) {\n const tHook = useT()\n const fallbackTranslator = React.useMemo<Translator>(\n () => createTranslatorWithFallback(tHook),\n [tHook],\n )\n const translate: Translator = translator ?? fallbackTranslator\n const [people, setPeople] = React.useState<CompanyPersonSummary[]>(initialPeople)\n const [removingId, setRemovingId] = React.useState<string | null>(null)\n const [linkDialogOpen, setLinkDialogOpen] = React.useState(false)\n const [createDialogOpen, setCreateDialogOpen] = React.useState(false)\n const [searchQuery, setSearchQuery] = React.useState('')\n const [sortMode, setSortMode] = React.useState<'name-asc' | 'name-desc' | 'recent'>('name-asc')\n const [filtersOpen, setFiltersOpen] = React.useState(true)\n const [visiblePeople, setVisiblePeople] = React.useState<CompanyPersonSummary[]>([])\n const [listPage, setListPage] = React.useState(1)\n const [listTotalPages, setListTotalPages] = React.useState(1)\n const [listTotalCount, setListTotalCount] = React.useState(initialPeople.length)\n const [listLoading, setListLoading] = React.useState(true)\n const [starredIds, setStarredIds] = React.useState<Set<string>>(\n () => new Set(readJsonFromLocalStorage<string[]>(`om:starred-people:${companyId}`, [])),\n )\n const pendingPeopleChangeRef = React.useRef(false)\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 toggleStar = React.useCallback(\n (personId: string) => {\n setStarredIds((prev) => {\n const next = new Set(prev)\n if (next.has(personId)) next.delete(personId)\n else next.add(personId)\n writeJsonToLocalStorage(`om:starred-people:${companyId}`, [...next])\n return next\n })\n },\n [companyId],\n )\n\n const displayedPeople = React.useMemo(\n () => (visiblePeople.length > 0 ? visiblePeople : people),\n [people, visiblePeople],\n )\n const totalLinkedPeople = listTotalCount > 0 ? listTotalCount : displayedPeople.length\n const decisionMakerNames = React.useMemo(\n () =>\n displayedPeople\n .filter((person) => starredIds.has(person.id))\n .map((person) => person.displayName),\n [displayedPeople, starredIds],\n )\n\n React.useEffect(() => {\n const action: SectionAction = {\n label: addActionLabel,\n onClick: () => {\n setCreateDialogOpen(true)\n },\n }\n onActionChange?.(action)\n return () => {\n onActionChange?.(null)\n }\n }, [addActionLabel, onActionChange])\n\n React.useEffect(() => {\n pendingPeopleChangeRef.current = false\n setPeople(initialPeople)\n }, [initialPeople])\n\n React.useEffect(() => {\n if (!pendingPeopleChangeRef.current) return\n pendingPeopleChangeRef.current = false\n onPeopleChange?.(people)\n }, [onPeopleChange, people])\n\n const loadVisiblePeople = React.useCallback(async () => {\n setListLoading(true)\n try {\n const params = new URLSearchParams({\n page: String(listPage),\n pageSize: String(COMPANY_PEOPLE_PAGE_SIZE),\n sort: sortMode,\n })\n if (searchQuery.trim().length > 0) {\n params.set('search', searchQuery.trim())\n }\n const payload = await readApiResultOrThrow<{\n items?: CompanyPersonSummary[]\n page?: number\n total?: number\n totalPages?: number\n }>(\n `/api/customers/companies/${encodeURIComponent(companyId)}/people?${params.toString()}`,\n undefined,\n {\n errorMessage: translate(\n 'customers.companies.detail.people.loadError',\n 'Failed to load people.',\n ),\n },\n )\n const nextTotalCount = typeof payload.total === 'number' ? payload.total : 0\n setVisiblePeople(Array.isArray(payload.items) ? payload.items : [])\n setListPage(typeof payload.page === 'number' ? payload.page : listPage)\n setListTotalCount((current) =>\n searchQuery.trim().length > 0 ? Math.max(current, nextTotalCount) : nextTotalCount,\n )\n setListTotalPages(typeof payload.totalPages === 'number' ? payload.totalPages : 1)\n } catch {\n setVisiblePeople([])\n if (searchQuery.trim().length === 0) {\n setListTotalCount(0)\n }\n setListTotalPages(1)\n } finally {\n setListLoading(false)\n }\n }, [companyId, listPage, searchQuery, sortMode, translate])\n\n React.useEffect(() => {\n void loadVisiblePeople()\n }, [loadVisiblePeople])\n\n useAppEvent('customers.person_company_link.deleted', (event) => {\n const payload = event.payload as { companyEntityId?: string | null } | null | undefined\n if (payload && payload.companyEntityId === companyId) {\n void loadVisiblePeople()\n }\n }, [companyId, loadVisiblePeople])\n\n React.useEffect(() => {\n setListPage(1)\n }, [searchQuery, sortMode])\n\n const applyPeopleChange = React.useCallback(\n (updater: (current: CompanyPersonSummary[]) => CompanyPersonSummary[]) => {\n setPeople((current) => {\n const next = updater(current)\n if (next !== current) {\n pendingPeopleChangeRef.current = true\n }\n return next\n })\n },\n [],\n )\n\n const handleLinkConfirm = React.useCallback(\n async ({\n addedIds,\n optionsById,\n }: {\n addedIds: string[]\n removedIds: string[]\n optionsById: Record<string, LinkEntityOption>\n }) => {\n if (!addedIds.length) return\n onLoadingChange?.(true)\n try {\n for (const personId 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({ companyId }),\n },\n {\n errorMessage: translate(\n 'customers.companies.detail.people.linkError',\n 'Failed to link person to company.',\n ),\n },\n ),\n {\n personId,\n companyId,\n },\n )\n }\n const optimisticPeople: CompanyPersonSummary[] = addedIds\n .map((personId): CompanyPersonSummary | null => {\n const option = optionsById[personId]\n if (!option) return null\n return {\n id: option.id,\n displayName: option.label,\n primaryEmail: null,\n primaryPhone: null,\n jobTitle: option.subtitle ?? null,\n }\n })\n .filter((entry): entry is CompanyPersonSummary => entry !== null)\n if (optimisticPeople.length > 0) {\n applyPeopleChange((current) => mergeCompanyPeople([...current, ...optimisticPeople]))\n setListTotalCount((current) => current + optimisticPeople.length)\n }\n await loadVisiblePeople()\n flash(\n addedIds.length === 1\n ? translate(\n 'customers.companies.detail.people.linkSuccess',\n 'Person linked to company.',\n )\n : translate(\n 'customers.companies.detail.people.linkSuccessMultiple',\n '{{count}} people linked to company.',\n { count: String(addedIds.length) },\n ),\n 'success',\n )\n } catch (err) {\n try {\n await onDataRefresh?.()\n } catch {\n // preserve original linking error for the user\n }\n const message =\n err instanceof Error\n ? err.message\n : translate(\n 'customers.companies.detail.people.linkError',\n 'Failed to link person to company.',\n )\n flash(message, 'error')\n throw err\n } finally {\n onLoadingChange?.(false)\n }\n },\n [\n applyPeopleChange,\n companyId,\n loadVisiblePeople,\n onDataRefresh,\n onLoadingChange,\n runWriteMutation,\n translate,\n ],\n )\n\n const personLinkAdapter = React.useMemo(\n () =>\n createPersonLinkAdapter({\n dialogTitle: translate('customers.linking.person.dialogTitle', 'Link person'),\n dialogSubtitle: companyName\n ? translate(\n 'customers.linking.person.dialogSubtitleFor',\n 'Link an existing contact to {{name}}',\n { name: companyName },\n )\n : translate(\n 'customers.linking.person.dialogSubtitle',\n 'Link an existing contact to this company',\n ),\n sectionLabel: translate('customers.linking.person.sectionLabel', 'MATCHING CONTACTS'),\n searchPlaceholder: translate(\n 'customers.linking.person.searchPlaceholder',\n 'Search all people\u2026',\n ),\n searchEmptyHint: translate(\n 'customers.linking.person.searchEmpty',\n 'No matching people found.',\n ),\n selectedEmptyHint: translate(\n 'customers.linking.person.selectedEmpty',\n 'No people selected.',\n ),\n confirmButtonLabel: translate('customers.linking.person.confirmButton', 'Link person'),\n showLinkSettings: true,\n roleOptions: [\n { id: 'decision_maker', label: 'Decision maker' },\n { id: 'budget_holder', label: 'Budget holder' },\n { id: 'stakeholder', label: 'Stakeholder' },\n { id: 'contact', label: 'Contact' },\n ],\n excludeLinkedCompanyId: companyId,\n addNew: {\n title: translate('customers.linking.person.addNew', 'Add new contact'),\n subtitle: translate(\n 'customers.linking.person.addNewSubtitle',\n 'Company will be filled in automatically',\n ),\n render: ({ onCancel }) => (\n <CreatePersonDialog\n open\n onClose={onCancel}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n // CreatePersonDialog already created and linked the person to this company\n // via the companyEntityId payload field. Refresh the on-page list and close\n // both the nested and outer dialogs so the user can see the new entry.\n void loadVisiblePeople()\n void onDataRefresh?.()\n setLinkDialogOpen(false)\n onCancel()\n }}\n />\n ),\n },\n }),\n [companyId, companyName, loadVisiblePeople, onDataRefresh, runWriteMutation, translate],\n )\n\n const handleRemove = React.useCallback(\n async (personId: string) => {\n if (!personId || removingId) return\n setRemovingId(personId)\n onLoadingChange?.(true)\n try {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,\n { method: 'DELETE' },\n {\n errorMessage: translate(\n 'customers.companies.detail.people.removeError',\n 'Failed to unlink person from company.',\n ),\n },\n ),\n {\n personId,\n companyId,\n },\n )\n await loadVisiblePeople()\n flash(\n translate(\n 'customers.companies.detail.people.removeSuccess',\n 'Person unlinked from company.',\n ),\n 'success',\n )\n } catch (err) {\n const message =\n err instanceof Error\n ? err.message\n : translate(\n 'customers.companies.detail.people.removeError',\n 'Failed to unlink person from company.',\n )\n flash(message, 'error')\n } finally {\n setRemovingId(null)\n onLoadingChange?.(false)\n }\n },\n [\n companyId,\n loadVisiblePeople,\n onLoadingChange,\n removingId,\n runWriteMutation,\n translate,\n ],\n )\n\n const linkAction = (\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setLinkDialogOpen(true)}\n >\n <Link2 className=\"mr-1.5 h-4 w-4\" />\n {translate(\n 'customers.companies.detail.people.linkAction',\n 'Link existing person',\n )}\n </Button>\n )\n const addPersonAction = (\n <Button type=\"button\" size=\"sm\" onClick={() => setCreateDialogOpen(true)}>\n <Plus className=\"mr-1.5 h-4 w-4\" />\n {addActionLabel}\n </Button>\n )\n\n if (!listLoading && totalLinkedPeople === 0) {\n return (\n <>\n <EmptyState\n icon={<Users className=\"h-10 w-10 text-muted-foreground\" />}\n title={emptyState.title}\n actionLabel={emptyState.actionLabel}\n onAction={() => setCreateDialogOpen(true)}\n >\n <p className=\"text-sm text-muted-foreground\">{emptyLabel}</p>\n <div className=\"mt-4\">{linkAction}</div>\n </EmptyState>\n <LinkEntityDialog\n open={linkDialogOpen}\n onOpenChange={setLinkDialogOpen}\n adapter={personLinkAdapter}\n initialSelectedIds={[]}\n onConfirm={handleLinkConfirm}\n runGuardedMutation={runWriteMutation}\n />\n <CreatePersonDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n setCreateDialogOpen(false)\n void loadVisiblePeople()\n void onDataRefresh?.()\n }}\n />\n </>\n )\n }\n\n return (\n <>\n <div className=\"space-y-4\">\n <RolesSection\n entityType=\"company\"\n entityId={companyId}\n entityName={companyName ?? null}\n />\n\n <section className=\"rounded-lg border bg-card px-4 py-4 sm:px-5\">\n <div className=\"flex flex-col gap-4\">\n <div className=\"flex flex-col gap-3 xl:flex-row xl:items-start xl:justify-between\">\n <div className=\"space-y-1\">\n <div className=\"flex items-center gap-2\">\n <h3 className=\"text-base font-semibold\">\n {translate(\n 'customers.companies.detail.people.sectionTitle',\n 'People',\n )}\n </h3>\n <Badge\n variant=\"secondary\"\n className=\"rounded-full px-2 py-0 text-xs font-semibold\"\n >\n {totalLinkedPeople}\n </Badge>\n </div>\n <p className=\"text-xs text-muted-foreground\">\n {translate(\n 'customers.companies.detail.people.sectionSubtitle',\n 'Employees and decision makers on the client side',\n )}\n </p>\n </div>\n <div className=\"flex flex-wrap items-center gap-2 xl:justify-end\">\n {linkAction}\n {addPersonAction}\n </div>\n </div>\n\n {totalLinkedPeople > 0 ? (\n <div className=\"flex flex-col gap-3 lg:flex-row lg:items-center\">\n {filtersOpen ? (\n <div className=\"min-w-0 flex-1\">\n <input\n type=\"text\"\n value={searchQuery}\n onChange={(event) => setSearchQuery(event.target.value)}\n placeholder={translate(\n 'customers.companies.detail.people.searchPlaceholder',\n 'Search by name, role, email...',\n )}\n className=\"h-10 w-full rounded-md border bg-background px-3 text-sm placeholder:text-muted-foreground focus:outline-none focus:ring-1 focus:ring-ring\"\n />\n </div>\n ) : null}\n <div className=\"flex flex-wrap items-center gap-2 lg:justify-end\">\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setFiltersOpen((current) => !current)}\n className=\"h-10\"\n >\n <Filter className=\"mr-1.5 h-4 w-4\" />\n {translate(\n 'customers.companies.detail.people.filter',\n 'Filters',\n )}\n </Button>\n {filtersOpen ? (\n <select\n value={sortMode}\n onChange={(event) =>\n setSortMode(event.target.value as 'name-asc' | 'name-desc' | 'recent')\n }\n className=\"h-10 min-w-[11rem] 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 {translate(\n 'customers.companies.detail.people.sortNameAsc',\n 'Sort: Name A-Z',\n )}\n </option>\n <option value=\"name-desc\">\n {translate(\n 'customers.companies.detail.people.sortNameDesc',\n 'Sort: Name Z-A',\n )}\n </option>\n <option value=\"recent\">\n {translate(\n 'customers.companies.detail.people.sortRecent',\n 'Sort: Recently linked',\n )}\n </option>\n </select>\n ) : null}\n </div>\n </div>\n ) : null}\n\n {listLoading ? (\n <p className=\"py-6 text-center text-sm text-muted-foreground\">\n {translate('customers.companies.detail.people.loading', 'Loading people\u2026')}\n </p>\n ) : visiblePeople.length > 0 ? (\n <>\n <div\n className=\"grid items-start gap-4\"\n style={{\n gridTemplateColumns: 'repeat(auto-fit, minmax(min(100%, 19.5rem), 1fr))',\n }}\n >\n {visiblePeople.map((person) => (\n <PersonCard\n key={person.id}\n person={person}\n isStarred={starredIds.has(person.id)}\n onToggleStar={toggleStar}\n onUnlink={handleRemove}\n />\n ))}\n </div>\n {listTotalPages > 1 ? (\n <div className=\"flex items-center justify-between border-t border-border/60 pt-3 text-sm text-muted-foreground\">\n <span>\n {translate(\n 'customers.companies.detail.people.pageSummary',\n 'Page {{page}} of {{total}}',\n {\n page: listPage,\n total: listTotalPages,\n },\n )}\n </span>\n <div className=\"flex items-center gap-2\">\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() => setListPage((current) => Math.max(1, current - 1))}\n disabled={listPage <= 1}\n >\n {translate('customers.companies.detail.people.previous', 'Previous')}\n </Button>\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n onClick={() =>\n setListPage((current) => Math.min(listTotalPages, current + 1))\n }\n disabled={listPage >= listTotalPages}\n >\n {translate('customers.companies.detail.people.next', 'Next')}\n </Button>\n </div>\n </div>\n ) : null}\n </>\n ) : totalLinkedPeople > 0 ? (\n <p className=\"py-6 text-center text-sm text-muted-foreground\">\n {translate(\n 'customers.companies.detail.people.noSearchResults',\n 'No people match your search.',\n )}\n </p>\n ) : null}\n </div>\n </section>\n\n <DecisionMakersFooter\n names={decisionMakerNames}\n onSendInvitation={() => {\n const starredEmails = displayedPeople\n .filter((person) => starredIds.has(person.id) && person.primaryEmail)\n .map((person) => person.primaryEmail!)\n if (starredEmails.length > 0) {\n window.open(`mailto:${starredEmails.join(',')}`, '_blank')\n }\n }}\n />\n </div>\n\n <LinkEntityDialog\n open={linkDialogOpen}\n onOpenChange={setLinkDialogOpen}\n adapter={personLinkAdapter}\n initialSelectedIds={[]}\n onConfirm={handleLinkConfirm}\n runGuardedMutation={runWriteMutation}\n />\n\n <CreatePersonDialog\n open={createDialogOpen}\n onClose={() => setCreateDialogOpen(false)}\n companyId={companyId}\n companyName={companyName ?? companyId}\n runGuardedMutation={runWriteMutation}\n onPersonCreated={() => {\n setCreateDialogOpen(false)\n void loadVisiblePeople()\n void onDataRefresh?.()\n }}\n />\n </>\n )\n}\n\nexport default CompanyPeopleSection\n\nexport { mergeCompanyPeople, matchesCompanyPersonSearch, sortCompanyPeople, normalizeCompanyPerson }\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -7,6 +7,8 @@ import { flash } from "@open-mercato/ui/backend/FlashMessages";
7
7
  import { apiCallOrThrow, readApiResultOrThrow } from "@open-mercato/ui/backend/utils/apiCall";
8
8
  import { Button } from "@open-mercato/ui/primitives/button";
9
9
  import { Input } from "@open-mercato/ui/primitives/input";
10
+ import { useConfirmDialog } from "@open-mercato/ui/backend/confirm-dialog";
11
+ import { useAppEvent } from "@open-mercato/ui/backend/injection/useAppEvent";
10
12
  import { CompanyCard } from "./CompanyCard.js";
11
13
  import { useCustomerDictionary } from "./hooks/useCustomerDictionary.js";
12
14
  import { LinkEntityDialog } from "../linking/LinkEntityDialog.js";
@@ -70,8 +72,10 @@ function PersonCompaniesSection({
70
72
  runGuardedMutation
71
73
  }) {
72
74
  const t = useT();
75
+ const { confirm, ConfirmDialogElement } = useConfirmDialog();
73
76
  const [items, setItems] = React.useState([]);
74
77
  const [loading, setLoading] = React.useState(true);
78
+ const [unlinkingId, setUnlinkingId] = React.useState(null);
75
79
  const [search, setSearch] = React.useState("");
76
80
  const [sort, setSort] = React.useState("name-asc");
77
81
  const [page, setPage] = React.useState(1);
@@ -242,6 +246,59 @@ function PersonCompaniesSection({
242
246
  },
243
247
  [linkedCompanies, linkedPrimaryId, loadData, onChanged, personId, runWriteMutation, t]
244
248
  );
249
+ const handleUnlink = React.useCallback(
250
+ async (companyId, displayName) => {
251
+ if (!companyId || unlinkingId) return;
252
+ const confirmed = await confirm({
253
+ title: t("customers.people.detail.companies.unlinkConfirmTitle", "Unlink company"),
254
+ description: t(
255
+ "customers.people.detail.companies.unlinkConfirm",
256
+ "Unlink {{company}} from {{person}}?",
257
+ { company: displayName, person: _personName }
258
+ ),
259
+ confirmText: t("customers.people.detail.companies.unlinkAction", "Unlink"),
260
+ cancelText: t("customers.linking.actions.cancel", "Cancel")
261
+ });
262
+ if (!confirmed) return;
263
+ setUnlinkingId(companyId);
264
+ try {
265
+ await runWriteMutation(
266
+ () => apiCallOrThrow(
267
+ `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,
268
+ { method: "DELETE" },
269
+ {
270
+ errorMessage: t(
271
+ "customers.people.detail.companies.unlinkError",
272
+ "Failed to unlink company."
273
+ )
274
+ }
275
+ ),
276
+ { companyId, personId, operation: "unlinkPersonCompanyLink" }
277
+ );
278
+ await loadData({ showLoading: false });
279
+ await onChanged?.();
280
+ flash(
281
+ t("customers.people.detail.companies.unlinkSuccess", "Company unlinked."),
282
+ "success"
283
+ );
284
+ } catch (error) {
285
+ const message = error instanceof Error ? error.message : t(
286
+ "customers.people.detail.companies.unlinkError",
287
+ "Failed to unlink company."
288
+ );
289
+ flash(message, "error");
290
+ } finally {
291
+ setUnlinkingId(null);
292
+ }
293
+ },
294
+ [_personName, confirm, loadData, onChanged, personId, runWriteMutation, t, unlinkingId]
295
+ );
296
+ useAppEvent("customers.person_company_link.deleted", (event) => {
297
+ const payload = event.payload;
298
+ if (payload && payload.personEntityId === personId) {
299
+ void loadData({ showLoading: false });
300
+ }
301
+ }, [personId, loadData]);
245
302
  return /* @__PURE__ */ jsxs(Fragment, { children: [
246
303
  /* @__PURE__ */ jsxs("div", { className: "space-y-4", children: [
247
304
  /* @__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: [
@@ -314,7 +371,10 @@ function PersonCompaniesSection({
314
371
  lifecycleMap: lifecycleDict?.map,
315
372
  temperatureMap: temperatureDict?.map,
316
373
  renewalQuarterMap: renewalQuarterDict?.map,
317
- roleMap: roleDict?.map
374
+ roleMap: roleDict?.map,
375
+ onUnlink: () => handleUnlink(item.companyId, item.displayName),
376
+ unlinkLabel: t("customers.people.detail.companies.unlinkAction", "Unlink"),
377
+ unlinkDisabled: unlinkingId === item.companyId
318
378
  },
319
379
  item.companyId
320
380
  )) }),
@@ -333,7 +393,8 @@ function PersonCompaniesSection({
333
393
  onConfirm: handleLinkConfirm,
334
394
  runGuardedMutation: runWriteMutation
335
395
  }
336
- )
396
+ ),
397
+ ConfirmDialogElement
337
398
  ] });
338
399
  }
339
400
  var PersonCompaniesSection_default = PersonCompaniesSection;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
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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",
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 { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport { useAppEvent } from '@open-mercato/ui/backend/injection/useAppEvent'\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 { confirm, ConfirmDialogElement } = useConfirmDialog()\n const [items, setItems] = React.useState<EnrichedCompanyData[]>([])\n const [loading, setLoading] = React.useState(true)\n const [unlinkingId, setUnlinkingId] = React.useState<string | null>(null)\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 const handleUnlink = React.useCallback(\n async (companyId: string, displayName: string) => {\n if (!companyId || unlinkingId) return\n const confirmed = await confirm({\n title: t('customers.people.detail.companies.unlinkConfirmTitle', 'Unlink company'),\n description: t(\n 'customers.people.detail.companies.unlinkConfirm',\n 'Unlink {{company}} from {{person}}?',\n { company: displayName, person: _personName },\n ),\n confirmText: t('customers.people.detail.companies.unlinkAction', 'Unlink'),\n cancelText: t('customers.linking.actions.cancel', 'Cancel'),\n })\n if (!confirmed) return\n setUnlinkingId(companyId)\n try {\n await runWriteMutation(\n () =>\n apiCallOrThrow(\n `/api/customers/people/${encodeURIComponent(personId)}/companies/${encodeURIComponent(companyId)}`,\n { method: 'DELETE' },\n {\n errorMessage: t(\n 'customers.people.detail.companies.unlinkError',\n 'Failed to unlink company.',\n ),\n },\n ),\n { companyId, personId, operation: 'unlinkPersonCompanyLink' },\n )\n await loadData({ showLoading: false })\n await onChanged?.()\n flash(\n t('customers.people.detail.companies.unlinkSuccess', 'Company unlinked.'),\n 'success',\n )\n } catch (error) {\n const message = error instanceof Error\n ? error.message\n : t(\n 'customers.people.detail.companies.unlinkError',\n 'Failed to unlink company.',\n )\n flash(message, 'error')\n } finally {\n setUnlinkingId(null)\n }\n },\n [_personName, confirm, loadData, onChanged, personId, runWriteMutation, t, unlinkingId],\n )\n\n useAppEvent('customers.person_company_link.deleted', (event) => {\n const payload = event.payload as { personEntityId?: string | null } | null | undefined\n if (payload && payload.personEntityId === personId) {\n void loadData({ showLoading: false })\n }\n }, [personId, loadData])\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 onUnlink={() => handleUnlink(item.companyId, item.displayName)}\n unlinkLabel={t('customers.people.detail.companies.unlinkAction', 'Unlink')}\n unlinkDisabled={unlinkingId === item.companyId}\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 {ConfirmDialogElement}\n </>\n )\n}\n\nexport default PersonCompaniesSection\n\nexport { sameIdSet }\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -10,7 +10,8 @@ import {
10
10
  Building2,
11
11
  Check,
12
12
  History,
13
- Paperclip
13
+ Paperclip,
14
+ Plus
14
15
  } from "lucide-react";
15
16
  const SUPPORTED_TAB_IDS = /* @__PURE__ */ new Set(["activities", "deals", "companies", "tasks", "changelog", "files"]);
16
17
  function resolveLegacyTab(tab) {
@@ -33,6 +34,7 @@ function PersonDetailTabs({
33
34
  companiesCount = 0,
34
35
  tasksCount = 0,
35
36
  filesCount = 0,
37
+ sectionAction = null,
36
38
  children
37
39
  }) {
38
40
  const t = useT();
@@ -88,30 +90,46 @@ function PersonDetailTabs({
88
90
  [builtInTabs, injectedTabs]
89
91
  );
90
92
  return /* @__PURE__ */ jsxs("div", { children: [
91
- /* @__PURE__ */ jsx("div", { className: "border-b", role: "tablist", "aria-label": t("customers.people.detail.tabs.label", "Person detail sections"), children: /* @__PURE__ */ jsx("nav", { className: "-mb-px flex gap-1 overflow-x-auto px-1", children: allTabs.map((tab) => {
92
- const isActive = activeTab === tab.id;
93
- return /* @__PURE__ */ jsxs(
93
+ /* @__PURE__ */ jsxs("div", { className: "flex items-end justify-between gap-2 border-b", role: "tablist", "aria-label": t("customers.people.detail.tabs.label", "Person detail sections"), children: [
94
+ /* @__PURE__ */ jsx("nav", { className: "-mb-px flex flex-1 gap-1 overflow-x-auto px-1", children: allTabs.map((tab) => {
95
+ const isActive = activeTab === tab.id;
96
+ return /* @__PURE__ */ jsxs(
97
+ Button,
98
+ {
99
+ type: "button",
100
+ variant: "ghost",
101
+ size: "sm",
102
+ role: "tab",
103
+ "aria-selected": isActive,
104
+ onClick: () => onTabChange(tab.id),
105
+ className: cn(
106
+ "h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent",
107
+ isActive ? "border-foreground text-foreground font-semibold" : "border-transparent text-muted-foreground hover:text-foreground"
108
+ ),
109
+ children: [
110
+ tab.icon && /* @__PURE__ */ jsx("span", { className: "mr-1.5", children: tab.icon }),
111
+ tab.label,
112
+ tab.badge
113
+ ]
114
+ },
115
+ tab.id
116
+ );
117
+ }) }),
118
+ sectionAction ? /* @__PURE__ */ jsxs(
94
119
  Button,
95
120
  {
96
121
  type: "button",
97
- variant: "ghost",
98
122
  size: "sm",
99
- role: "tab",
100
- "aria-selected": isActive,
101
- onClick: () => onTabChange(tab.id),
102
- className: cn(
103
- "h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent",
104
- isActive ? "border-foreground text-foreground font-semibold" : "border-transparent text-muted-foreground hover:text-foreground"
105
- ),
123
+ onClick: sectionAction.onClick,
124
+ disabled: sectionAction.disabled,
125
+ className: "mb-1.5 mr-1 shrink-0",
106
126
  children: [
107
- tab.icon && /* @__PURE__ */ jsx("span", { className: "mr-1.5", children: tab.icon }),
108
- tab.label,
109
- tab.badge
127
+ /* @__PURE__ */ jsx(Plus, { className: "mr-1.5 h-4 w-4" }),
128
+ sectionAction.label
110
129
  ]
111
- },
112
- tab.id
113
- );
114
- }) }) }),
130
+ }
131
+ ) : null
132
+ ] }),
115
133
  /* @__PURE__ */ jsx("div", { className: "pt-6", children })
116
134
  ] });
117
135
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/customers/components/detail/PersonDetailTabs.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n SquareCheckBig,\n Briefcase,\n Building2,\n Check,\n History,\n Paperclip,\n} from 'lucide-react'\n\nexport type PersonTabId =\n | 'activities'\n | 'deals'\n | 'companies'\n | 'tasks'\n | 'changelog'\n | 'files'\n | string\n\ntype TabDef = {\n id: PersonTabId\n label: string\n icon?: React.ReactNode\n badge?: React.ReactNode\n}\n\ntype PersonDetailTabsProps = {\n activeTab: PersonTabId\n onTabChange: (tab: PersonTabId) => void\n injectedTabs?: Array<{ id: string; label: string }>\n activitiesCount?: number\n dealsCount?: number\n companiesCount?: number\n tasksCount?: number\n filesCount?: number\n children: React.ReactNode\n}\n\nconst SUPPORTED_TAB_IDS = new Set<PersonTabId>(['activities', 'deals', 'companies', 'tasks', 'changelog', 'files'])\n\nexport function resolveLegacyTab(tab: string | null | undefined): PersonTabId {\n if (!tab) return 'activities'\n return SUPPORTED_TAB_IDS.has(tab as PersonTabId) ? (tab as PersonTabId) : 'activities'\n}\n\nfunction CountBadge({ count }: { count: number }) {\n if (count <= 0) return null\n return (\n <span className=\"ml-1 rounded-full bg-muted px-1.5 py-0.5 text-xs font-medium leading-none text-muted-foreground\">\n {count > 999 ? '999+' : count}\n </span>\n )\n}\n\nfunction NewBadge() {\n return (\n <span className=\"ml-1.5 rounded bg-foreground px-1.5 py-0.5 text-overline font-semibold leading-none text-background\">\n NEW\n </span>\n )\n}\n\nexport function PersonDetailTabs({\n activeTab,\n onTabChange,\n injectedTabs = [],\n activitiesCount = 0,\n dealsCount = 0,\n companiesCount = 0,\n tasksCount = 0,\n filesCount = 0,\n children,\n}: PersonDetailTabsProps) {\n const t = useT()\n\n const builtInTabs: TabDef[] = React.useMemo(\n () => [\n {\n id: 'activities',\n label: t('customers.people.detail.tabs.activities', 'Activities'),\n icon: <SquareCheckBig className=\"size-4\" />,\n badge: <CountBadge count={activitiesCount} />,\n },\n {\n id: 'deals',\n label: t('customers.people.detail.tabs.deals', 'Deals'),\n icon: <Briefcase className=\"size-4\" />,\n badge: <CountBadge count={dealsCount} />,\n },\n {\n id: 'companies',\n label: t('customers.people.detail.tabs.companies', 'Companies'),\n icon: <Building2 className=\"size-4\" />,\n badge: <CountBadge count={companiesCount} />,\n },\n {\n id: 'tasks',\n label: t('customers.people.detail.tabs.tasks', 'Tasks'),\n icon: <Check className=\"size-4\" />,\n badge: <CountBadge count={tasksCount} />,\n },\n {\n id: 'changelog',\n label: t('customers.people.detail.tabs.changelog', 'Change log'),\n icon: <History className=\"size-4\" />,\n badge: <NewBadge />,\n },\n {\n id: 'files',\n label: t('customers.people.detail.tabs.files', 'Files'),\n icon: <Paperclip className=\"size-4\" />,\n badge: <CountBadge count={filesCount} />,\n },\n ],\n [t, activitiesCount, dealsCount, companiesCount, tasksCount, filesCount],\n )\n\n const allTabs: TabDef[] = React.useMemo(\n () => [\n ...builtInTabs,\n ...injectedTabs.map((tab) => ({\n id: tab.id as PersonTabId,\n label: tab.label,\n })),\n ],\n [builtInTabs, injectedTabs],\n )\n\n return (\n <div>\n {/* Tab navigation \u2014 full width above both zones */}\n <div className=\"border-b\" role=\"tablist\" aria-label={t('customers.people.detail.tabs.label', 'Person detail sections')}>\n <nav className=\"-mb-px flex gap-1 overflow-x-auto px-1\">\n {allTabs.map((tab) => {\n const isActive = activeTab === tab.id\n return (\n <Button\n key={tab.id}\n type=\"button\"\n variant=\"ghost\"\n size=\"sm\"\n role=\"tab\"\n aria-selected={isActive}\n onClick={() => onTabChange(tab.id)}\n className={cn(\n 'h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent',\n isActive\n ? 'border-foreground text-foreground font-semibold'\n : 'border-transparent text-muted-foreground hover:text-foreground',\n )}\n >\n {tab.icon && <span className=\"mr-1.5\">{tab.icon}</span>}\n {tab.label}\n {tab.badge}\n </Button>\n )\n })}\n </nav>\n </div>\n\n {/* Two-column content below tabs */}\n <div className=\"pt-6\">\n {children}\n </div>\n </div>\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n SquareCheckBig,\n Briefcase,\n Building2,\n Check,\n History,\n Paperclip,\n Plus,\n} from 'lucide-react'\nimport type { SectionAction } from '@open-mercato/ui/backend/detail'\n\nexport type PersonTabId =\n | 'activities'\n | 'deals'\n | 'companies'\n | 'tasks'\n | 'changelog'\n | 'files'\n | string\n\ntype TabDef = {\n id: PersonTabId\n label: string\n icon?: React.ReactNode\n badge?: React.ReactNode\n}\n\ntype PersonDetailTabsProps = {\n activeTab: PersonTabId\n onTabChange: (tab: PersonTabId) => void\n injectedTabs?: Array<{ id: string; label: string }>\n activitiesCount?: number\n dealsCount?: number\n companiesCount?: number\n tasksCount?: number\n filesCount?: number\n sectionAction?: SectionAction | null\n children: React.ReactNode\n}\n\nconst SUPPORTED_TAB_IDS = new Set<PersonTabId>(['activities', 'deals', 'companies', 'tasks', 'changelog', 'files'])\n\nexport function resolveLegacyTab(tab: string | null | undefined): PersonTabId {\n if (!tab) return 'activities'\n return SUPPORTED_TAB_IDS.has(tab as PersonTabId) ? (tab as PersonTabId) : 'activities'\n}\n\nfunction CountBadge({ count }: { count: number }) {\n if (count <= 0) return null\n return (\n <span className=\"ml-1 rounded-full bg-muted px-1.5 py-0.5 text-xs font-medium leading-none text-muted-foreground\">\n {count > 999 ? '999+' : count}\n </span>\n )\n}\n\nfunction NewBadge() {\n return (\n <span className=\"ml-1.5 rounded bg-foreground px-1.5 py-0.5 text-overline font-semibold leading-none text-background\">\n NEW\n </span>\n )\n}\n\nexport function PersonDetailTabs({\n activeTab,\n onTabChange,\n injectedTabs = [],\n activitiesCount = 0,\n dealsCount = 0,\n companiesCount = 0,\n tasksCount = 0,\n filesCount = 0,\n sectionAction = null,\n children,\n}: PersonDetailTabsProps) {\n const t = useT()\n\n const builtInTabs: TabDef[] = React.useMemo(\n () => [\n {\n id: 'activities',\n label: t('customers.people.detail.tabs.activities', 'Activities'),\n icon: <SquareCheckBig className=\"size-4\" />,\n badge: <CountBadge count={activitiesCount} />,\n },\n {\n id: 'deals',\n label: t('customers.people.detail.tabs.deals', 'Deals'),\n icon: <Briefcase className=\"size-4\" />,\n badge: <CountBadge count={dealsCount} />,\n },\n {\n id: 'companies',\n label: t('customers.people.detail.tabs.companies', 'Companies'),\n icon: <Building2 className=\"size-4\" />,\n badge: <CountBadge count={companiesCount} />,\n },\n {\n id: 'tasks',\n label: t('customers.people.detail.tabs.tasks', 'Tasks'),\n icon: <Check className=\"size-4\" />,\n badge: <CountBadge count={tasksCount} />,\n },\n {\n id: 'changelog',\n label: t('customers.people.detail.tabs.changelog', 'Change log'),\n icon: <History className=\"size-4\" />,\n badge: <NewBadge />,\n },\n {\n id: 'files',\n label: t('customers.people.detail.tabs.files', 'Files'),\n icon: <Paperclip className=\"size-4\" />,\n badge: <CountBadge count={filesCount} />,\n },\n ],\n [t, activitiesCount, dealsCount, companiesCount, tasksCount, filesCount],\n )\n\n const allTabs: TabDef[] = React.useMemo(\n () => [\n ...builtInTabs,\n ...injectedTabs.map((tab) => ({\n id: tab.id as PersonTabId,\n label: tab.label,\n })),\n ],\n [builtInTabs, injectedTabs],\n )\n\n return (\n <div>\n {/* Tab navigation \u2014 full width above both zones */}\n <div className=\"flex items-end justify-between gap-2 border-b\" role=\"tablist\" aria-label={t('customers.people.detail.tabs.label', 'Person detail sections')}>\n <nav className=\"-mb-px flex flex-1 gap-1 overflow-x-auto px-1\">\n {allTabs.map((tab) => {\n const isActive = activeTab === tab.id\n return (\n <Button\n key={tab.id}\n type=\"button\"\n variant=\"ghost\"\n size=\"sm\"\n role=\"tab\"\n aria-selected={isActive}\n onClick={() => onTabChange(tab.id)}\n className={cn(\n 'h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent',\n isActive\n ? 'border-foreground text-foreground font-semibold'\n : 'border-transparent text-muted-foreground hover:text-foreground',\n )}\n >\n {tab.icon && <span className=\"mr-1.5\">{tab.icon}</span>}\n {tab.label}\n {tab.badge}\n </Button>\n )\n })}\n </nav>\n {sectionAction ? (\n <Button\n type=\"button\"\n size=\"sm\"\n onClick={sectionAction.onClick}\n disabled={sectionAction.disabled}\n className=\"mb-1.5 mr-1 shrink-0\"\n >\n <Plus className=\"mr-1.5 h-4 w-4\" />\n {sectionAction.label}\n </Button>\n ) : null}\n </div>\n\n {/* Two-column content below tabs */}\n <div className=\"pt-6\">\n {children}\n </div>\n </div>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -361,17 +361,7 @@ function TasksSection({
361
361
  className: "border-0 bg-transparent p-0 py-8 justify-center"
362
362
  }
363
363
  ) : null,
364
- !isInitialLoading && !hasTasks ? /* @__PURE__ */ jsx(
365
- TabEmptyState,
366
- {
367
- title: emptyState.title,
368
- action: {
369
- label: emptyState.actionLabel,
370
- onClick: openCreateDialog,
371
- disabled: isMutating || !entityId
372
- }
373
- }
374
- ) : null,
364
+ !isInitialLoading && !hasTasks ? /* @__PURE__ */ jsx(TabEmptyState, { title: emptyState.title }) : null,
375
365
  !isInitialLoading && hasTasks ? /* @__PURE__ */ jsxs("div", { className: "space-y-4", children: [
376
366
  error ? /* @__PURE__ */ jsx("div", { className: "rounded-md border border-destructive/40 bg-destructive/5 px-3 py-2 text-sm text-destructive", children: error }) : null,
377
367
  sortedTasks.map((task) => {