@open-mercato/core 0.4.11-develop.1355.50152f3ee9 → 0.4.11-develop.1365.0acff7b08e

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 (35) hide show
  1. package/dist/modules/customers/api/companies/route.js +141 -3
  2. package/dist/modules/customers/api/companies/route.js.map +2 -2
  3. package/dist/modules/customers/api/deals/route.js +52 -3
  4. package/dist/modules/customers/api/deals/route.js.map +2 -2
  5. package/dist/modules/customers/api/people/route.js +145 -3
  6. package/dist/modules/customers/api/people/route.js.map +2 -2
  7. package/dist/modules/customers/api/utils.js +195 -0
  8. package/dist/modules/customers/api/utils.js.map +2 -2
  9. package/dist/modules/customers/backend/customers/companies/page.js +171 -6
  10. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  11. package/dist/modules/customers/backend/customers/deals/page.js +100 -7
  12. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  13. package/dist/modules/customers/backend/customers/people/page.js +180 -7
  14. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  15. package/dist/modules/customers/commands/interactions.js +7 -0
  16. package/dist/modules/customers/commands/interactions.js.map +2 -2
  17. package/dist/modules/customers/components/detail/DealForm.js +1 -0
  18. package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
  19. package/dist/modules/query_index/lib/engine.js +81 -1
  20. package/dist/modules/query_index/lib/engine.js.map +2 -2
  21. package/package.json +3 -3
  22. package/src/modules/customers/api/companies/route.ts +151 -3
  23. package/src/modules/customers/api/deals/route.ts +54 -3
  24. package/src/modules/customers/api/people/route.ts +160 -3
  25. package/src/modules/customers/api/utils.ts +286 -0
  26. package/src/modules/customers/backend/customers/companies/page.tsx +184 -9
  27. package/src/modules/customers/backend/customers/deals/page.tsx +127 -35
  28. package/src/modules/customers/backend/customers/people/page.tsx +191 -10
  29. package/src/modules/customers/commands/interactions.ts +7 -0
  30. package/src/modules/customers/components/detail/DealForm.tsx +1 -0
  31. package/src/modules/customers/i18n/de.json +12 -0
  32. package/src/modules/customers/i18n/en.json +15 -3
  33. package/src/modules/customers/i18n/es.json +12 -0
  34. package/src/modules/customers/i18n/pl.json +12 -0
  35. package/src/modules/query_index/lib/engine.ts +95 -1
@@ -14,15 +14,20 @@ import { E } from "../../../../../generated/entities.ids.generated.js";
14
14
  import { useOrganizationScopeVersion } from "@open-mercato/shared/lib/frontend/useOrganizationScope";
15
15
  import { useT } from "@open-mercato/shared/lib/i18n/context";
16
16
  import { useConfirmDialog } from "@open-mercato/ui/backend/confirm-dialog";
17
+ import { serializeAdvancedFilter } from "@open-mercato/shared/lib/query/advanced-filter";
17
18
  import {
18
19
  DictionaryValue,
19
20
  renderDictionaryColor,
20
21
  renderDictionaryIcon
21
22
  } from "../../../lib/dictionaries.js";
22
23
  import {
23
- useCustomFieldDefs,
24
- filterCustomFieldDefs
24
+ useCustomFieldDefs
25
25
  } from "@open-mercato/ui/backend/utils/customFieldDefs";
26
+ import {
27
+ mapCustomFieldKindToFilterType,
28
+ normalizeCustomFieldFilterOptions,
29
+ supportsCustomFieldColumn
30
+ } from "@open-mercato/ui/backend/utils/customFieldColumns";
26
31
  import { useQueryClient } from "@tanstack/react-query";
27
32
  import { ensureCustomerDictionary } from "../../../components/detail/hooks/useCustomerDictionary.js";
28
33
  const NO_MATCH_TAG_SENTINEL = "__no_match__";
@@ -52,6 +57,16 @@ function mapApiItem(item) {
52
57
  const description = typeof item.description === "string" ? item.description : null;
53
58
  const email = typeof item.primary_email === "string" ? item.primary_email : null;
54
59
  const phone = typeof item.primary_phone === "string" ? item.primary_phone : null;
60
+ const firstName = typeof item.first_name === "string" ? item.first_name : null;
61
+ const lastName = typeof item.last_name === "string" ? item.last_name : null;
62
+ const preferredName = typeof item.preferred_name === "string" ? item.preferred_name : null;
63
+ const jobTitle = typeof item.job_title === "string" ? item.job_title : null;
64
+ const department = typeof item.department === "string" ? item.department : null;
65
+ const seniority = typeof item.seniority === "string" ? item.seniority : null;
66
+ const timezone = typeof item.timezone === "string" ? item.timezone : null;
67
+ const linkedInUrl = typeof item.linked_in_url === "string" ? item.linked_in_url : null;
68
+ const twitterUrl = typeof item.twitter_url === "string" ? item.twitter_url : null;
69
+ const companyEntityId = typeof item.company_entity_id === "string" ? item.company_entity_id : null;
55
70
  const status = typeof item.status === "string" ? item.status : null;
56
71
  const lifecycleStage = typeof item.lifecycle_stage === "string" ? item.lifecycle_stage : null;
57
72
  const nextInteractionAt = typeof item.next_interaction_at === "string" ? item.next_interaction_at : null;
@@ -72,6 +87,16 @@ function mapApiItem(item) {
72
87
  description,
73
88
  email,
74
89
  phone,
90
+ firstName,
91
+ lastName,
92
+ preferredName,
93
+ jobTitle,
94
+ department,
95
+ seniority,
96
+ timezone,
97
+ linkedInUrl,
98
+ twitterUrl,
99
+ companyEntityId,
75
100
  status,
76
101
  lifecycleStage,
77
102
  nextInteractionAt,
@@ -87,11 +112,13 @@ function CustomersPeoplePage() {
87
112
  const { confirm, ConfirmDialogElement } = useConfirmDialog();
88
113
  const [rows, setRows] = React.useState([]);
89
114
  const [page, setPage] = React.useState(1);
90
- const [pageSize] = React.useState(20);
115
+ const [pageSize, setPageSize] = React.useState(20);
116
+ const [sorting, setSorting] = React.useState([]);
91
117
  const [total, setTotal] = React.useState(0);
92
118
  const [totalPages, setTotalPages] = React.useState(1);
93
119
  const [search, setSearch] = React.useState("");
94
120
  const [filterValues, setFilterValues] = React.useState({});
121
+ const [advancedFilterState, setAdvancedFilterState] = React.useState({ logic: "and", conditions: [] });
95
122
  const [isLoading, setIsLoading] = React.useState(true);
96
123
  const [reloadToken, setReloadToken] = React.useState(0);
97
124
  const [cacheStatus, setCacheStatus] = React.useState(null);
@@ -101,6 +128,10 @@ function CustomersPeoplePage() {
101
128
  const queryClient = useQueryClient();
102
129
  const t = useT();
103
130
  const router = useRouter();
131
+ const handlePageSizeChange = React.useCallback((newSize) => {
132
+ setPageSize(newSize);
133
+ setPage(1);
134
+ }, []);
104
135
  const fetchDictionaryEntries = React.useCallback(async (kind) => {
105
136
  try {
106
137
  const data = await ensureCustomerDictionary(queryClient, kind, scopeVersion);
@@ -252,6 +283,10 @@ function CustomersPeoplePage() {
252
283
  const params = new URLSearchParams();
253
284
  params.set("page", String(page));
254
285
  params.set("pageSize", String(pageSize));
286
+ if (sorting.length > 0) {
287
+ params.set("sort", sorting[0].id);
288
+ params.set("order", sorting[0].desc ? "desc" : "asc");
289
+ }
255
290
  if (search.trim()) params.set("search", search.trim());
256
291
  const status = filterValues.status;
257
292
  if (typeof status === "string" && status.trim()) params.set("status", status);
@@ -303,8 +338,12 @@ function CustomersPeoplePage() {
303
338
  if (stringValue) params.set(key, stringValue);
304
339
  }
305
340
  });
341
+ const advancedParams = serializeAdvancedFilter(advancedFilterState);
342
+ for (const [key, val] of Object.entries(advancedParams)) {
343
+ params.set(key, val);
344
+ }
306
345
  return params.toString();
307
- }, [filterValues, page, pageSize, search, tagIdToLabel, tagLabelToId]);
346
+ }, [advancedFilterState, filterValues, page, pageSize, search, sorting, tagIdToLabel, tagLabelToId]);
308
347
  const currentParams = React.useMemo(() => Object.fromEntries(new URLSearchParams(queryParams)), [queryParams]);
309
348
  const exportConfig = React.useMemo(() => ({
310
349
  view: {
@@ -380,6 +419,35 @@ function CustomersPeoplePage() {
380
419
  flash(message, "error");
381
420
  }
382
421
  }, [confirm, handleRefresh, t]);
422
+ const handleBulkDelete = React.useCallback(async (selectedRows) => {
423
+ const confirmed = await confirm({
424
+ title: t("customers.people.list.bulkDelete.title", "Delete {count} people?", { count: selectedRows.length }),
425
+ description: t("customers.people.list.bulkDelete.description", "This action cannot be undone."),
426
+ variant: "destructive"
427
+ });
428
+ if (!confirmed) return false;
429
+ let deletedCount = 0;
430
+ for (const row of selectedRows) {
431
+ try {
432
+ await apiCallOrThrow(`/api/customers/people?id=${encodeURIComponent(row.id)}`, {
433
+ method: "DELETE",
434
+ headers: { "content-type": "application/json" }
435
+ });
436
+ deletedCount++;
437
+ } catch {
438
+ }
439
+ }
440
+ if (deletedCount > 0) {
441
+ setRows((prev) => {
442
+ const deletedIds = new Set(selectedRows.map((r) => r.id));
443
+ return prev.filter((r) => !deletedIds.has(r.id));
444
+ });
445
+ setTotal((prev) => Math.max(0, prev - deletedCount));
446
+ flash(t("customers.people.list.bulkDelete.success", "{count} people deleted", { count: deletedCount }), "success");
447
+ setReloadToken((prev) => prev + 1);
448
+ }
449
+ return deletedCount > 0;
450
+ }, [confirm, t]);
383
451
  const handleFiltersApply = React.useCallback((values) => {
384
452
  const next = {};
385
453
  Object.entries(values).forEach(([key, value]) => {
@@ -439,27 +507,38 @@ function CustomersPeoplePage() {
439
507
  {
440
508
  accessorKey: "name",
441
509
  header: t("customers.people.list.columns.name"),
510
+ meta: { alwaysVisible: true, columnChooserGroup: "Basic Info", filterKey: "display_name" },
442
511
  cell: ({ row }) => /* @__PURE__ */ jsx(Link, { href: `/backend/customers/people-v2/${row.original.id}`, className: "font-medium hover:underline", children: row.original.name })
443
512
  },
444
513
  {
445
514
  accessorKey: "email",
446
515
  header: t("customers.people.list.columns.email"),
516
+ meta: { columnChooserGroup: "Contact", filterKey: "primary_email" },
447
517
  cell: ({ row }) => row.original.email || /* @__PURE__ */ jsx("span", { className: "text-muted-foreground text-sm", children: t("customers.people.list.noValue") })
448
518
  },
449
519
  {
450
520
  accessorKey: "status",
451
521
  header: t("customers.people.list.columns.status"),
522
+ meta: { filterType: "select", filterOptions: dictionaryOptions.statuses, columnChooserGroup: "Basic Info" },
452
523
  cell: ({ row }) => renderDictionaryCell("statuses", row.original.status)
453
524
  },
454
525
  {
455
526
  accessorKey: "lifecycleStage",
456
527
  header: t("customers.people.list.columns.lifecycleStage"),
528
+ meta: {
529
+ filterType: "select",
530
+ filterOptions: dictionaryOptions.lifecycleStages,
531
+ columnChooserGroup: "Basic Info",
532
+ filterKey: "lifecycle_stage"
533
+ },
457
534
  cell: ({ row }) => renderDictionaryCell("lifecycle-stages", row.original.lifecycleStage)
458
535
  },
459
536
  {
460
537
  accessorKey: "nextInteractionAt",
461
538
  header: t("customers.people.list.columns.nextInteraction"),
462
539
  meta: {
540
+ columnChooserGroup: "Dates",
541
+ filterKey: "next_interaction_at",
463
542
  tooltipContent: (row) => {
464
543
  if (!row.nextInteractionAt) return void 0;
465
544
  const date = formatDate(row.nextInteractionAt, "");
@@ -479,20 +558,89 @@ function CustomersPeoplePage() {
479
558
  {
480
559
  accessorKey: "source",
481
560
  header: t("customers.people.list.columns.source"),
561
+ meta: { filterType: "select", filterOptions: dictionaryOptions.sources, columnChooserGroup: "Basic Info" },
482
562
  cell: ({ row }) => renderDictionaryCell("sources", row.original.source)
563
+ },
564
+ {
565
+ accessorKey: "firstName",
566
+ header: t("customers.people.form.firstName", "First name"),
567
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.first_name" },
568
+ cell: ({ row }) => row.original.firstName || noValue
569
+ },
570
+ {
571
+ accessorKey: "lastName",
572
+ header: t("customers.people.form.lastName", "Last name"),
573
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.last_name" },
574
+ cell: ({ row }) => row.original.lastName || noValue
575
+ },
576
+ {
577
+ accessorKey: "preferredName",
578
+ header: t("customers.people.form.preferredName", "Preferred name"),
579
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.preferred_name" },
580
+ cell: ({ row }) => row.original.preferredName || noValue
581
+ },
582
+ {
583
+ accessorKey: "jobTitle",
584
+ header: t("customers.people.form.jobTitle", "Job title"),
585
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.job_title" },
586
+ cell: ({ row }) => row.original.jobTitle || noValue
587
+ },
588
+ {
589
+ accessorKey: "department",
590
+ header: t("customers.people.detail.fields.department", "Department"),
591
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.department" },
592
+ cell: ({ row }) => row.original.department || noValue
593
+ },
594
+ {
595
+ accessorKey: "seniority",
596
+ header: t("customers.people.detail.fields.seniority", "Seniority"),
597
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.seniority" },
598
+ cell: ({ row }) => row.original.seniority || noValue
599
+ },
600
+ {
601
+ accessorKey: "timezone",
602
+ header: t("customers.people.detail.fields.timezone", "Timezone"),
603
+ meta: { columnChooserGroup: "Profile", hidden: true, filterKey: "person_profile.timezone" },
604
+ cell: ({ row }) => row.original.timezone || noValue
605
+ },
606
+ {
607
+ accessorKey: "linkedInUrl",
608
+ header: t("customers.people.detail.fields.linkedIn", "LinkedIn"),
609
+ meta: { columnChooserGroup: "Socials", hidden: true, filterKey: "person_profile.linked_in_url" },
610
+ cell: ({ row }) => row.original.linkedInUrl || noValue
611
+ },
612
+ {
613
+ accessorKey: "twitterUrl",
614
+ header: t("customers.people.detail.fields.twitter", "Twitter"),
615
+ meta: { columnChooserGroup: "Socials", hidden: true, filterKey: "person_profile.twitter_url" },
616
+ cell: ({ row }) => row.original.twitterUrl || noValue
617
+ },
618
+ {
619
+ accessorKey: "description",
620
+ header: t("customers.people.form.description", "Description"),
621
+ meta: { columnChooserGroup: "Notes", hidden: true, filterKey: "description" },
622
+ cell: ({ row }) => row.original.description || noValue
483
623
  }
484
624
  ];
485
- const customColumns = filterCustomFieldDefs(customFieldDefs, "list").map((def) => ({
625
+ const customColumns = customFieldDefs.filter((def) => supportsCustomFieldColumn(def)).map((def) => ({
486
626
  accessorKey: `cf_${def.key}`,
487
627
  header: def.label || def.key,
628
+ meta: {
629
+ columnChooserGroup: def.group?.title ?? "Custom Fields",
630
+ filterGroup: def.group?.title ?? "Custom Fields",
631
+ filterType: mapCustomFieldKindToFilterType(def.kind),
632
+ filterOptions: normalizeCustomFieldFilterOptions(def.options),
633
+ hidden: def.listVisible === false
634
+ },
488
635
  cell: ({ getValue }) => renderCustomFieldCell(getValue())
489
636
  }));
490
637
  return [...baseColumns, ...customColumns];
491
- }, [customFieldDefs, dictionaryMaps, t]);
638
+ }, [customFieldDefs, dictionaryMaps, dictionaryOptions, t]);
492
639
  return /* @__PURE__ */ jsxs(Page, { children: [
493
640
  /* @__PURE__ */ jsx(PageBody, { children: /* @__PURE__ */ jsx(
494
641
  DataTable,
495
642
  {
643
+ stickyFirstColumn: true,
496
644
  title: t("customers.people.list.title"),
497
645
  refreshButton: {
498
646
  label: t("customers.people.list.actions.refresh"),
@@ -504,6 +652,7 @@ function CustomersPeoplePage() {
504
652
  },
505
653
  actions: /* @__PURE__ */ jsx(Button, { asChild: true, children: /* @__PURE__ */ jsx(Link, { href: "/backend/customers/people/create", children: t("customers.people.list.actions.new") }) }),
506
654
  columns,
655
+ columnChooser: { auto: true },
507
656
  data: rows,
508
657
  exporter: exportConfig,
509
658
  searchValue: search,
@@ -519,6 +668,17 @@ function CustomersPeoplePage() {
519
668
  entityIds: [E.customers.customer_entity, E.customers.customer_person_profile],
520
669
  perspective: { tableId: "customers.people.list" },
521
670
  onRowClick: (row) => router.push(`/backend/customers/people-v2/${row.id}`),
671
+ sortable: true,
672
+ sorting,
673
+ onSortingChange: setSorting,
674
+ bulkActions: [
675
+ {
676
+ id: "delete",
677
+ label: t("customers.people.list.bulkDelete.action", "Delete selected"),
678
+ destructive: true,
679
+ onExecute: handleBulkDelete
680
+ }
681
+ ],
522
682
  rowActions: (row) => /* @__PURE__ */ jsx(
523
683
  RowActions,
524
684
  {
@@ -544,7 +704,20 @@ function CustomersPeoplePage() {
544
704
  ]
545
705
  }
546
706
  ),
547
- pagination: { page, pageSize, total, totalPages, onPageChange: setPage, cacheStatus },
707
+ advancedFilter: {
708
+ auto: true,
709
+ value: advancedFilterState,
710
+ onChange: setAdvancedFilterState,
711
+ onApply: () => {
712
+ setPage(1);
713
+ },
714
+ onClear: () => {
715
+ setAdvancedFilterState({ logic: "and", conditions: [] });
716
+ setPage(1);
717
+ }
718
+ },
719
+ virtualized: true,
720
+ pagination: { page, pageSize, total, totalPages, onPageChange: setPage, cacheStatus, pageSizeOptions: [10, 25, 50, 100], onPageSizeChange: handlePageSizeChange },
548
721
  isLoading
549
722
  }
550
723
  ) }),
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/modules/customers/backend/customers/people/page.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { useRouter } from 'next/navigation'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable, type DataTableExportFormat, withDataTableNamespaces } from '@open-mercato/ui/backend/DataTable'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { apiCall, apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { buildCrudExportUrl } from '@open-mercato/ui/backend/utils/crud'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { E } from '#generated/entities.ids.generated'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport type { FilterOption } from '@open-mercato/ui/backend/FilterOverlay'\nimport {\n DictionaryValue,\n renderDictionaryColor,\n renderDictionaryIcon,\n type CustomerDictionaryKind,\n type CustomerDictionaryMap,\n} from '../../../lib/dictionaries'\nimport {\n useCustomFieldDefs,\n filterCustomFieldDefs,\n} from '@open-mercato/ui/backend/utils/customFieldDefs'\nimport { useQueryClient } from '@tanstack/react-query'\nimport { ensureCustomerDictionary } from '../../../components/detail/hooks/useCustomerDictionary'\n\ntype PersonRow = {\n id: string\n name: string\n description?: string | null\n email?: string | null\n phone?: string | null\n status?: string | null\n lifecycleStage?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n source?: string | null\n} & Record<string, unknown>\n\ntype PeopleResponse = {\n items?: Array<Record<string, unknown>>\n total?: number\n page?: number\n totalPages?: number\n}\n\ntype DictionaryKindKey = CustomerDictionaryKind\ntype DictionaryMap = CustomerDictionaryMap\n\nconst NO_MATCH_TAG_SENTINEL = '__no_match__'\n\nfunction createEmptyDictionaryMaps(): Record<DictionaryKindKey, DictionaryMap> {\n return {\n statuses: {},\n sources: {},\n 'lifecycle-stages': {},\n 'address-types': {},\n 'activity-types': {},\n 'deal-statuses': {},\n 'pipeline-stages': {},\n 'job-titles': {},\n industries: {},\n }\n}\n\nfunction formatDate(value: string | null | undefined, fallback: string): string {\n if (!value) return fallback\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return fallback\n return date.toLocaleDateString()\n}\n\nfunction mapApiItem(item: Record<string, unknown>): PersonRow | null {\n const id = typeof item.id === 'string' ? item.id : null\n if (!id) return null\n const name = typeof item.display_name === 'string' ? item.display_name : ''\n const description = typeof item.description === 'string' ? item.description : null\n const email = typeof item.primary_email === 'string' ? item.primary_email : null\n const phone = typeof item.primary_phone === 'string' ? item.primary_phone : null\n const status = typeof item.status === 'string' ? item.status : null\n const lifecycleStage = typeof item.lifecycle_stage === 'string' ? item.lifecycle_stage : null\n const nextInteractionAt = typeof item.next_interaction_at === 'string' ? item.next_interaction_at : null\n const nextInteractionName = typeof item.next_interaction_name === 'string' ? item.next_interaction_name : null\n const nextInteractionIcon = typeof item.next_interaction_icon === 'string' ? item.next_interaction_icon : null\n const nextInteractionColor = typeof item.next_interaction_color === 'string' ? item.next_interaction_color : null\n const organizationId = typeof item.organization_id === 'string' ? item.organization_id : null\n const source = typeof item.source === 'string' ? item.source : null\n const customFields: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(item)) {\n if (key.startsWith('cf_')) {\n customFields[key] = value\n }\n }\n return withDataTableNamespaces({\n id,\n name,\n description,\n email,\n phone,\n status,\n lifecycleStage,\n nextInteractionAt,\n nextInteractionName,\n nextInteractionIcon,\n nextInteractionColor,\n organizationId,\n source,\n ...customFields,\n }, item)\n}\n\nexport default function CustomersPeoplePage() {\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n const [rows, setRows] = React.useState<PersonRow[]>([])\n const [page, setPage] = React.useState(1)\n const [pageSize] = React.useState(20)\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [isLoading, setIsLoading] = React.useState(true)\n const [reloadToken, setReloadToken] = React.useState(0)\n const [cacheStatus, setCacheStatus] = React.useState<'hit' | 'miss' | null>(null)\n const [dictionaryMaps, setDictionaryMaps] = React.useState<Record<DictionaryKindKey, DictionaryMap>>(createEmptyDictionaryMaps())\n const [tagIdToLabel, setTagIdToLabel] = React.useState<Record<string, string>>({})\n const scopeVersion = useOrganizationScopeVersion()\n const queryClient = useQueryClient()\n const t = useT()\n const router = useRouter()\n const fetchDictionaryEntries = React.useCallback(async (kind: DictionaryKindKey) => {\n try {\n const data = await ensureCustomerDictionary(queryClient, kind, scopeVersion)\n setDictionaryMaps((prev) => ({\n ...prev,\n [kind]: data.map,\n }))\n return data.entries\n } catch {\n return []\n }\n }, [queryClient, scopeVersion])\n const loadDictionaryOptions = React.useCallback(async (kind: 'statuses' | 'sources' | 'lifecycle-stages') => {\n const entries = await fetchDictionaryEntries(kind)\n return entries.map((entry) => ({ value: entry.value, label: entry.label }))\n }, [fetchDictionaryEntries])\n\n const dictionaryOptions = React.useMemo(() => {\n const toOptions = (map?: DictionaryMap | null): FilterOption[] =>\n Object.values(map ?? {})\n .map((entry) => ({ value: entry.value, label: entry.label }))\n .sort((a, b) => a.label.localeCompare(b.label, undefined, { sensitivity: 'base' }))\n return {\n statuses: toOptions(dictionaryMaps.statuses),\n sources: toOptions(dictionaryMaps.sources),\n lifecycleStages: toOptions(dictionaryMaps['lifecycle-stages']),\n }\n }, [dictionaryMaps])\n\n const loadTagOptions = React.useCallback(async (query?: string): Promise<FilterOption[]> => {\n try {\n const params = new URLSearchParams({ pageSize: '100' })\n const trimmedQuery = typeof query === 'string' ? query.trim() : ''\n if (trimmedQuery) params.set('search', trimmedQuery)\n const payload = await readApiResultOrThrow<{ items?: unknown[] }>(\n `/api/customers/tags?${params.toString()}`,\n undefined,\n { errorMessage: t('customers.people.detail.tags.loadError', 'Failed to load tags.') },\n )\n const items = Array.isArray(payload?.items) ? payload.items : []\n const options: FilterOption[] = []\n for (const item of items) {\n if (!item || typeof item !== 'object') continue\n const raw = item as { id?: unknown; tagId?: unknown; label?: unknown; slug?: unknown }\n const rawId = typeof raw.id === 'string'\n ? raw.id\n : typeof raw.tagId === 'string'\n ? raw.tagId\n : null\n if (!rawId) continue\n const label = typeof raw.label === 'string' && raw.label.trim().length\n ? raw.label.trim()\n : typeof raw.slug === 'string' && raw.slug.trim().length\n ? raw.slug.trim()\n : rawId\n options.push({ value: rawId, label })\n }\n if (options.length) {\n setTagIdToLabel((prev) => {\n let changed = false\n const next = { ...prev }\n for (const option of options) {\n if (next[option.value] !== option.label) {\n next[option.value] = option.label\n changed = true\n }\n }\n return changed ? next : prev\n })\n }\n return options\n } catch (err) {\n console.error('customers.people.list.loadTagOptions', err)\n return []\n }\n }, [setTagIdToLabel, t])\n\n const tagLabelToId = React.useMemo(() => {\n const map: Record<string, string> = {}\n for (const [id, label] of Object.entries(tagIdToLabel)) {\n if (!label) continue\n map[label] = id\n }\n return map\n }, [tagIdToLabel])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadAll() {\n if (cancelled) return\n setDictionaryMaps(createEmptyDictionaryMaps())\n await Promise.all([\n fetchDictionaryEntries('statuses'),\n fetchDictionaryEntries('sources'),\n fetchDictionaryEntries('lifecycle-stages'),\n ])\n }\n loadAll().catch(() => {})\n return () => {\n cancelled = true\n }\n }, [fetchDictionaryEntries, scopeVersion, reloadToken])\n\n const { data: customFieldDefs = [] } = useCustomFieldDefs(\n [E.customers.customer_entity, E.customers.customer_person_profile],\n { keyExtras: [scopeVersion, reloadToken] },\n )\n\n const filters = React.useMemo<FilterDef[]>(() => [\n {\n id: 'status',\n label: t('customers.people.list.filters.status'),\n type: 'select',\n options: dictionaryOptions.statuses,\n loadOptions: () => loadDictionaryOptions('statuses'),\n },\n {\n id: 'source',\n label: t('customers.people.list.filters.source'),\n type: 'select',\n options: dictionaryOptions.sources,\n loadOptions: () => loadDictionaryOptions('sources'),\n },\n {\n id: 'lifecycleStage',\n label: t('customers.people.list.filters.lifecycleStage'),\n type: 'select',\n options: dictionaryOptions.lifecycleStages,\n loadOptions: () => loadDictionaryOptions('lifecycle-stages'),\n },\n {\n id: 'tagIds',\n label: t('customers.people.list.filters.tags'),\n type: 'tags',\n loadOptions: loadTagOptions,\n },\n {\n id: 'createdAt',\n label: t('customers.people.list.filters.createdAt'),\n type: 'dateRange',\n },\n {\n id: 'emailContains',\n label: t('customers.people.list.filters.emailContains'),\n type: 'text',\n placeholder: t('customers.people.list.filters.emailContainsPlaceholder'),\n },\n {\n id: 'hasEmail',\n label: t('customers.people.list.filters.hasEmail'),\n type: 'checkbox',\n },\n {\n id: 'hasPhone',\n label: t('customers.people.list.filters.hasPhone'),\n type: 'checkbox',\n },\n {\n id: 'hasNextInteraction',\n label: t('customers.people.list.filters.hasNextInteraction'),\n type: 'checkbox',\n },\n ], [dictionaryOptions.lifecycleStages, dictionaryOptions.sources, dictionaryOptions.statuses, loadDictionaryOptions, loadTagOptions, t])\n\n const queryParams = React.useMemo(() => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(pageSize))\n if (search.trim()) params.set('search', search.trim())\n const status = filterValues.status\n if (typeof status === 'string' && status.trim()) params.set('status', status)\n const source = filterValues.source\n if (typeof source === 'string' && source.trim()) params.set('source', source)\n const lifecycleStage = filterValues.lifecycleStage\n if (typeof lifecycleStage === 'string' && lifecycleStage.trim()) params.set('lifecycleStage', lifecycleStage)\n const createdAt = filterValues.createdAt\n if (createdAt && typeof createdAt === 'object') {\n if (createdAt.from) params.set('createdFrom', createdAt.from)\n if (createdAt.to) params.set('createdTo', createdAt.to)\n }\n const emailContains = filterValues.emailContains\n if (typeof emailContains === 'string' && emailContains.trim()) {\n params.set('emailContains', emailContains.trim())\n }\n const booleanFilters: Array<['hasEmail' | 'hasPhone' | 'hasNextInteraction', string]> = [\n ['hasEmail', 'hasEmail'],\n ['hasPhone', 'hasPhone'],\n ['hasNextInteraction', 'hasNextInteraction'],\n ]\n for (const [key, queryKey] of booleanFilters) {\n const value = filterValues[key]\n if (value === true) params.set(queryKey, 'true')\n if (value === false) params.set(queryKey, 'false')\n }\n const tagValues = Array.isArray(filterValues.tagIds)\n ? filterValues.tagIds\n .map((value) => (typeof value === 'string' ? value.trim() : String(value || '').trim()))\n .filter((value) => value.length > 0)\n : []\n if (tagValues.length > 0) {\n const normalizedTagIds = tagValues\n .map((value) => (typeof tagIdToLabel[value] === 'string' ? value : tagLabelToId[value]))\n .filter((id): id is string => typeof id === 'string' && id.length > 0)\n if (normalizedTagIds.length === tagValues.length && normalizedTagIds.length > 0) {\n params.set('tagIds', normalizedTagIds.join(','))\n } else {\n params.set('tagIdsEmpty', 'true')\n }\n }\n Object.entries(filterValues).forEach(([key, value]) => {\n if (!key.startsWith('cf_') || value == null) return\n if (Array.isArray(value)) {\n const normalized = value\n .map((item) => {\n if (item == null) return ''\n if (typeof item === 'string') return item.trim()\n return String(item).trim()\n })\n .filter((item) => item.length > 0)\n if (normalized.length) params.set(key, normalized.join(','))\n } else if (typeof value === 'object') {\n return\n } else if (value !== '') {\n const stringValue = typeof value === 'string' ? value.trim() : String(value)\n if (stringValue) params.set(key, stringValue)\n }\n })\n return params.toString()\n }, [filterValues, page, pageSize, search, tagIdToLabel, tagLabelToId])\n\n const currentParams = React.useMemo(() => Object.fromEntries(new URLSearchParams(queryParams)), [queryParams])\n const exportConfig = React.useMemo(() => ({\n view: {\n getUrl: (format: DataTableExportFormat) =>\n buildCrudExportUrl('customers/people', { ...currentParams, exportScope: 'view' }, format),\n },\n full: {\n getUrl: (format: DataTableExportFormat) =>\n buildCrudExportUrl('customers/people', { ...currentParams, exportScope: 'full', all: 'true' }, format),\n },\n }), [currentParams])\n\n React.useEffect(() => {\n let cancelled = false\n async function load() {\n setIsLoading(true)\n setCacheStatus(null)\n try {\n const fallback: PeopleResponse = { items: [], total: 0, totalPages: 1 }\n const call = await apiCall<PeopleResponse>(`/api/customers/people?${queryParams}`, undefined, { fallback })\n if (!call.ok) {\n const errorPayload = call.result as { error?: string } | undefined\n const message = typeof errorPayload?.error === 'string' ? errorPayload.error : t('customers.people.list.error.load')\n flash(message, 'error')\n if (!cancelled) setCacheStatus(null)\n return\n }\n const payload = call.result ?? fallback\n if (cancelled) return\n setCacheStatus(call.cacheStatus ?? null)\n const items = Array.isArray(payload.items) ? payload.items : []\n setRows(items.map((item) => mapApiItem(item as Record<string, unknown>)).filter((row): row is PersonRow => !!row))\n setTotal(typeof payload.total === 'number' ? payload.total : items.length)\n setTotalPages(typeof payload.totalPages === 'number' ? payload.totalPages : 1)\n } catch (err) {\n if (!cancelled) {\n setCacheStatus(null)\n const message = err instanceof Error ? err.message : t('customers.people.list.error.load')\n flash(message, 'error')\n }\n } finally {\n if (!cancelled) setIsLoading(false)\n }\n }\n load()\n return () => { cancelled = true }\n }, [queryParams, reloadToken, scopeVersion, t])\n\n const handleRefresh = React.useCallback(() => {\n setReloadToken((token) => token + 1)\n }, [])\n\n const handleDelete = React.useCallback(async (person: PersonRow) => {\n if (!person?.id) return\n const name = person.name || t('customers.people.list.deleteFallbackName')\n const confirmed = await confirm({\n title: t('customers.people.list.deleteConfirm', undefined, { name }),\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n await apiCallOrThrow(\n `/api/customers/people?id=${encodeURIComponent(person.id)}`,\n {\n method: 'DELETE',\n headers: { 'content-type': 'application/json' },\n },\n { errorMessage: t('customers.people.list.deleteError') },\n )\n setRows((prev) => prev.filter((row) => row.id !== person.id))\n setTotal((prev) => Math.max(prev - 1, 0))\n handleRefresh()\n flash(t('customers.people.list.deleteSuccess'), 'success')\n } catch (err) {\n const message = err instanceof Error ? err.message : t('customers.people.list.deleteError')\n flash(message, 'error')\n }\n }, [confirm, handleRefresh, t])\n\n const handleFiltersApply = React.useCallback((values: FilterValues) => {\n const next: FilterValues = {}\n Object.entries(values).forEach(([key, value]) => {\n if (value !== undefined) {\n next[key] = value\n }\n })\n const rawTags = Array.isArray(values.tagIds) ? (values.tagIds as string[]) : []\n const sanitizedTags = rawTags\n .map((tag) => {\n const normalized = typeof tag === 'string' ? tag.trim() : ''\n if (!normalized) return ''\n return tagIdToLabel[normalized] ?? normalized\n })\n .filter((tag) => tag.length > 0)\n if (sanitizedTags.length) next.tagIds = sanitizedTags\n else delete next.tagIds\n setFilterValues(next)\n setPage(1)\n }, [setFilterValues, setPage, tagIdToLabel])\n\n const handleFiltersClear = React.useCallback(() => {\n setFilterValues({})\n setPage(1)\n }, [setFilterValues, setPage])\n\n const columns = React.useMemo<ColumnDef<PersonRow>[]>(() => {\n const noValue = <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>\n const renderDictionaryCell = (kind: DictionaryKindKey, rawValue: string | null | undefined) => (\n <DictionaryValue\n value={rawValue}\n map={dictionaryMaps[kind]}\n fallback={rawValue ? <span>{rawValue}</span> : noValue}\n className=\"text-sm\"\n iconWrapperClassName=\"inline-flex h-6 w-6 items-center justify-center rounded border border-border bg-card\"\n iconClassName=\"h-4 w-4\"\n colorClassName=\"h-3 w-3 rounded-full\"\n />\n )\n\n const renderCustomFieldCell = (value: unknown) => {\n if (value == null) return noValue\n if (Array.isArray(value)) {\n if (!value.length) return noValue\n const normalized = value\n .map((item) => {\n if (item == null) return ''\n if (typeof item === 'string') return item.trim()\n return String(item).trim()\n })\n .filter((item) => item.length > 0)\n if (!normalized.length) return noValue\n return <span className=\"text-sm\">{normalized.join(', ')}</span>\n }\n if (typeof value === 'boolean') {\n return (\n <span className=\"text-sm\">\n {value\n ? t('customers.people.list.booleanYes', 'Yes')\n : t('customers.people.list.booleanNo', 'No')}\n </span>\n )\n }\n const stringValue = typeof value === 'string' ? value.trim() : String(value)\n if (!stringValue) return noValue\n return <span className=\"text-sm\">{stringValue}</span>\n }\n\n const baseColumns: ColumnDef<PersonRow>[] = [\n {\n accessorKey: 'name',\n header: t('customers.people.list.columns.name'),\n cell: ({ row }) => (\n <Link href={`/backend/customers/people-v2/${row.original.id}`} className=\"font-medium hover:underline\">\n {row.original.name}\n </Link>\n ),\n },\n {\n accessorKey: 'email',\n header: t('customers.people.list.columns.email'),\n cell: ({ row }) => row.original.email || <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>,\n },\n {\n accessorKey: 'status',\n header: t('customers.people.list.columns.status'),\n cell: ({ row }) => renderDictionaryCell('statuses', row.original.status),\n },\n {\n accessorKey: 'lifecycleStage',\n header: t('customers.people.list.columns.lifecycleStage'),\n cell: ({ row }) => renderDictionaryCell('lifecycle-stages', row.original.lifecycleStage),\n },\n {\n accessorKey: 'nextInteractionAt',\n header: t('customers.people.list.columns.nextInteraction'),\n meta: {\n tooltipContent: (row: PersonRow) => {\n if (!row.nextInteractionAt) return undefined\n const date = formatDate(row.nextInteractionAt, '')\n const name = row.nextInteractionName || ''\n return [date, name].filter(Boolean).join(' - ')\n },\n },\n cell: ({ row }) =>\n row.original.nextInteractionAt\n ? (\n <div className=\"flex items-start gap-2 text-sm\">\n {row.original.nextInteractionIcon ? (\n <span className=\"mt-0.5 inline-flex h-6 w-6 items-center justify-center rounded border border-border bg-card\">\n {renderDictionaryIcon(row.original.nextInteractionIcon, 'h-4 w-4')}\n </span>\n ) : null}\n <div className=\"flex flex-col\">\n <span>{formatDate(row.original.nextInteractionAt, t('customers.people.list.noValue'))}</span>\n {row.original.nextInteractionName ? (\n <span className=\"text-xs text-muted-foreground\">{row.original.nextInteractionName}</span>\n ) : null}\n </div>\n {row.original.nextInteractionColor ? (\n <span className=\"mt-1\">\n {renderDictionaryColor(row.original.nextInteractionColor, 'h-3 w-3 rounded-full border border-border')}\n </span>\n ) : null}\n </div>\n )\n : <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>,\n },\n {\n accessorKey: 'source',\n header: t('customers.people.list.columns.source'),\n cell: ({ row }) => renderDictionaryCell('sources', row.original.source),\n },\n ]\n\n const customColumns = filterCustomFieldDefs(customFieldDefs, 'list').map<ColumnDef<PersonRow>>((def) => ({\n accessorKey: `cf_${def.key}`,\n header: def.label || def.key,\n cell: ({ getValue }) => renderCustomFieldCell(getValue()),\n }))\n\n return [...baseColumns, ...customColumns]\n }, [customFieldDefs, dictionaryMaps, t])\n\n return (\n <Page>\n <PageBody>\n <DataTable<PersonRow>\n title={t('customers.people.list.title')}\n refreshButton={{\n label: t('customers.people.list.actions.refresh'),\n onRefresh: () => { setSearch(''); setPage(1); handleRefresh() },\n }}\n actions={(\n <Button asChild>\n <Link href=\"/backend/customers/people/create\">\n {t('customers.people.list.actions.new')}\n </Link>\n </Button>\n )}\n columns={columns}\n data={rows}\n exporter={exportConfig}\n searchValue={search}\n onSearchChange={(value) => { setSearch(value); setPage(1) }}\n searchPlaceholder={t('customers.people.list.searchPlaceholder')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={handleFiltersApply}\n onFiltersClear={handleFiltersClear}\n entityIds={[E.customers.customer_entity, E.customers.customer_person_profile]}\n perspective={{ tableId: 'customers.people.list' }}\n onRowClick={(row) => router.push(`/backend/customers/people-v2/${row.id}`)}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'view',\n label: t('customers.people.list.actions.view'),\n onSelect: () => { router.push(`/backend/customers/people-v2/${row.id}`) },\n },\n {\n id: 'open-new-tab',\n label: t('customers.people.list.actions.openInNewTab'),\n onSelect: () => window.open(`/backend/customers/people-v2/${row.id}`, '_blank', 'noopener'),\n },\n {\n id: 'delete',\n label: t('customers.people.list.actions.delete'),\n destructive: true,\n onSelect: () => handleDelete(row),\n },\n ]}\n />\n )}\n pagination={{ page, pageSize, total, totalPages, onPageChange: setPage, cacheStatus }}\n isLoading={isLoading}\n />\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { useRouter } from 'next/navigation'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable, type DataTableExportFormat, withDataTableNamespaces } from '@open-mercato/ui/backend/DataTable'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { apiCall, apiCallOrThrow, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { buildCrudExportUrl } from '@open-mercato/ui/backend/utils/crud'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { E } from '#generated/entities.ids.generated'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport type { FilterOption } from '@open-mercato/ui/backend/FilterOverlay'\nimport type { AdvancedFilterState } from '@open-mercato/shared/lib/query/advanced-filter'\nimport { serializeAdvancedFilter } from '@open-mercato/shared/lib/query/advanced-filter'\nimport {\n DictionaryValue,\n renderDictionaryColor,\n renderDictionaryIcon,\n type CustomerDictionaryKind,\n type CustomerDictionaryMap,\n} from '../../../lib/dictionaries'\nimport {\n useCustomFieldDefs,\n} from '@open-mercato/ui/backend/utils/customFieldDefs'\nimport {\n mapCustomFieldKindToFilterType,\n normalizeCustomFieldFilterOptions,\n supportsCustomFieldColumn,\n} from '@open-mercato/ui/backend/utils/customFieldColumns'\nimport { useQueryClient } from '@tanstack/react-query'\nimport { ensureCustomerDictionary } from '../../../components/detail/hooks/useCustomerDictionary'\n\ntype PersonRow = {\n id: string\n name: string\n description?: string | null\n email?: string | null\n phone?: string | null\n firstName?: string | null\n lastName?: string | null\n preferredName?: string | null\n jobTitle?: string | null\n department?: string | null\n seniority?: string | null\n timezone?: string | null\n linkedInUrl?: string | null\n twitterUrl?: string | null\n companyEntityId?: string | null\n status?: string | null\n lifecycleStage?: string | null\n nextInteractionAt?: string | null\n nextInteractionName?: string | null\n nextInteractionIcon?: string | null\n nextInteractionColor?: string | null\n organizationId?: string | null\n source?: string | null\n} & Record<string, unknown>\n\ntype PeopleResponse = {\n items?: Array<Record<string, unknown>>\n total?: number\n page?: number\n totalPages?: number\n}\n\ntype DictionaryKindKey = CustomerDictionaryKind\ntype DictionaryMap = CustomerDictionaryMap\n\nconst NO_MATCH_TAG_SENTINEL = '__no_match__'\n\nfunction createEmptyDictionaryMaps(): Record<DictionaryKindKey, DictionaryMap> {\n return {\n statuses: {},\n sources: {},\n 'lifecycle-stages': {},\n 'address-types': {},\n 'activity-types': {},\n 'deal-statuses': {},\n 'pipeline-stages': {},\n 'job-titles': {},\n industries: {},\n }\n}\n\nfunction formatDate(value: string | null | undefined, fallback: string): string {\n if (!value) return fallback\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return fallback\n return date.toLocaleDateString()\n}\n\nfunction mapApiItem(item: Record<string, unknown>): PersonRow | null {\n const id = typeof item.id === 'string' ? item.id : null\n if (!id) return null\n const name = typeof item.display_name === 'string' ? item.display_name : ''\n const description = typeof item.description === 'string' ? item.description : null\n const email = typeof item.primary_email === 'string' ? item.primary_email : null\n const phone = typeof item.primary_phone === 'string' ? item.primary_phone : null\n const firstName = typeof item.first_name === 'string' ? item.first_name : null\n const lastName = typeof item.last_name === 'string' ? item.last_name : null\n const preferredName = typeof item.preferred_name === 'string' ? item.preferred_name : null\n const jobTitle = typeof item.job_title === 'string' ? item.job_title : null\n const department = typeof item.department === 'string' ? item.department : null\n const seniority = typeof item.seniority === 'string' ? item.seniority : null\n const timezone = typeof item.timezone === 'string' ? item.timezone : null\n const linkedInUrl = typeof item.linked_in_url === 'string' ? item.linked_in_url : null\n const twitterUrl = typeof item.twitter_url === 'string' ? item.twitter_url : null\n const companyEntityId = typeof item.company_entity_id === 'string' ? item.company_entity_id : null\n const status = typeof item.status === 'string' ? item.status : null\n const lifecycleStage = typeof item.lifecycle_stage === 'string' ? item.lifecycle_stage : null\n const nextInteractionAt = typeof item.next_interaction_at === 'string' ? item.next_interaction_at : null\n const nextInteractionName = typeof item.next_interaction_name === 'string' ? item.next_interaction_name : null\n const nextInteractionIcon = typeof item.next_interaction_icon === 'string' ? item.next_interaction_icon : null\n const nextInteractionColor = typeof item.next_interaction_color === 'string' ? item.next_interaction_color : null\n const organizationId = typeof item.organization_id === 'string' ? item.organization_id : null\n const source = typeof item.source === 'string' ? item.source : null\n const customFields: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(item)) {\n if (key.startsWith('cf_')) {\n customFields[key] = value\n }\n }\n return withDataTableNamespaces({\n id,\n name,\n description,\n email,\n phone,\n firstName,\n lastName,\n preferredName,\n jobTitle,\n department,\n seniority,\n timezone,\n linkedInUrl,\n twitterUrl,\n companyEntityId,\n status,\n lifecycleStage,\n nextInteractionAt,\n nextInteractionName,\n nextInteractionIcon,\n nextInteractionColor,\n organizationId,\n source,\n ...customFields,\n }, item)\n}\n\nexport default function CustomersPeoplePage() {\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n const [rows, setRows] = React.useState<PersonRow[]>([])\n const [page, setPage] = React.useState(1)\n const [pageSize, setPageSize] = React.useState(20)\n const [sorting, setSorting] = React.useState<import('@tanstack/react-table').SortingState>([])\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [advancedFilterState, setAdvancedFilterState] = React.useState<AdvancedFilterState>({ logic: 'and', conditions: [] })\n const [isLoading, setIsLoading] = React.useState(true)\n const [reloadToken, setReloadToken] = React.useState(0)\n const [cacheStatus, setCacheStatus] = React.useState<'hit' | 'miss' | null>(null)\n const [dictionaryMaps, setDictionaryMaps] = React.useState<Record<DictionaryKindKey, DictionaryMap>>(createEmptyDictionaryMaps())\n const [tagIdToLabel, setTagIdToLabel] = React.useState<Record<string, string>>({})\n const scopeVersion = useOrganizationScopeVersion()\n const queryClient = useQueryClient()\n const t = useT()\n const router = useRouter()\n const handlePageSizeChange = React.useCallback((newSize: number) => {\n setPageSize(newSize)\n setPage(1)\n }, [])\n const fetchDictionaryEntries = React.useCallback(async (kind: DictionaryKindKey) => {\n try {\n const data = await ensureCustomerDictionary(queryClient, kind, scopeVersion)\n setDictionaryMaps((prev) => ({\n ...prev,\n [kind]: data.map,\n }))\n return data.entries\n } catch {\n return []\n }\n }, [queryClient, scopeVersion])\n const loadDictionaryOptions = React.useCallback(async (kind: 'statuses' | 'sources' | 'lifecycle-stages') => {\n const entries = await fetchDictionaryEntries(kind)\n return entries.map((entry) => ({ value: entry.value, label: entry.label }))\n }, [fetchDictionaryEntries])\n\n const dictionaryOptions = React.useMemo(() => {\n const toOptions = (map?: DictionaryMap | null): FilterOption[] =>\n Object.values(map ?? {})\n .map((entry) => ({ value: entry.value, label: entry.label }))\n .sort((a, b) => a.label.localeCompare(b.label, undefined, { sensitivity: 'base' }))\n return {\n statuses: toOptions(dictionaryMaps.statuses),\n sources: toOptions(dictionaryMaps.sources),\n lifecycleStages: toOptions(dictionaryMaps['lifecycle-stages']),\n }\n }, [dictionaryMaps])\n\n const loadTagOptions = React.useCallback(async (query?: string): Promise<FilterOption[]> => {\n try {\n const params = new URLSearchParams({ pageSize: '100' })\n const trimmedQuery = typeof query === 'string' ? query.trim() : ''\n if (trimmedQuery) params.set('search', trimmedQuery)\n const payload = await readApiResultOrThrow<{ items?: unknown[] }>(\n `/api/customers/tags?${params.toString()}`,\n undefined,\n { errorMessage: t('customers.people.detail.tags.loadError', 'Failed to load tags.') },\n )\n const items = Array.isArray(payload?.items) ? payload.items : []\n const options: FilterOption[] = []\n for (const item of items) {\n if (!item || typeof item !== 'object') continue\n const raw = item as { id?: unknown; tagId?: unknown; label?: unknown; slug?: unknown }\n const rawId = typeof raw.id === 'string'\n ? raw.id\n : typeof raw.tagId === 'string'\n ? raw.tagId\n : null\n if (!rawId) continue\n const label = typeof raw.label === 'string' && raw.label.trim().length\n ? raw.label.trim()\n : typeof raw.slug === 'string' && raw.slug.trim().length\n ? raw.slug.trim()\n : rawId\n options.push({ value: rawId, label })\n }\n if (options.length) {\n setTagIdToLabel((prev) => {\n let changed = false\n const next = { ...prev }\n for (const option of options) {\n if (next[option.value] !== option.label) {\n next[option.value] = option.label\n changed = true\n }\n }\n return changed ? next : prev\n })\n }\n return options\n } catch (err) {\n console.error('customers.people.list.loadTagOptions', err)\n return []\n }\n }, [setTagIdToLabel, t])\n\n const tagLabelToId = React.useMemo(() => {\n const map: Record<string, string> = {}\n for (const [id, label] of Object.entries(tagIdToLabel)) {\n if (!label) continue\n map[label] = id\n }\n return map\n }, [tagIdToLabel])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadAll() {\n if (cancelled) return\n setDictionaryMaps(createEmptyDictionaryMaps())\n await Promise.all([\n fetchDictionaryEntries('statuses'),\n fetchDictionaryEntries('sources'),\n fetchDictionaryEntries('lifecycle-stages'),\n ])\n }\n loadAll().catch(() => {})\n return () => {\n cancelled = true\n }\n }, [fetchDictionaryEntries, scopeVersion, reloadToken])\n\n const { data: customFieldDefs = [] } = useCustomFieldDefs(\n [E.customers.customer_entity, E.customers.customer_person_profile],\n { keyExtras: [scopeVersion, reloadToken] },\n )\n\n const filters = React.useMemo<FilterDef[]>(() => [\n {\n id: 'status',\n label: t('customers.people.list.filters.status'),\n type: 'select',\n options: dictionaryOptions.statuses,\n loadOptions: () => loadDictionaryOptions('statuses'),\n },\n {\n id: 'source',\n label: t('customers.people.list.filters.source'),\n type: 'select',\n options: dictionaryOptions.sources,\n loadOptions: () => loadDictionaryOptions('sources'),\n },\n {\n id: 'lifecycleStage',\n label: t('customers.people.list.filters.lifecycleStage'),\n type: 'select',\n options: dictionaryOptions.lifecycleStages,\n loadOptions: () => loadDictionaryOptions('lifecycle-stages'),\n },\n {\n id: 'tagIds',\n label: t('customers.people.list.filters.tags'),\n type: 'tags',\n loadOptions: loadTagOptions,\n },\n {\n id: 'createdAt',\n label: t('customers.people.list.filters.createdAt'),\n type: 'dateRange',\n },\n {\n id: 'emailContains',\n label: t('customers.people.list.filters.emailContains'),\n type: 'text',\n placeholder: t('customers.people.list.filters.emailContainsPlaceholder'),\n },\n {\n id: 'hasEmail',\n label: t('customers.people.list.filters.hasEmail'),\n type: 'checkbox',\n },\n {\n id: 'hasPhone',\n label: t('customers.people.list.filters.hasPhone'),\n type: 'checkbox',\n },\n {\n id: 'hasNextInteraction',\n label: t('customers.people.list.filters.hasNextInteraction'),\n type: 'checkbox',\n },\n ], [dictionaryOptions.lifecycleStages, dictionaryOptions.sources, dictionaryOptions.statuses, loadDictionaryOptions, loadTagOptions, t])\n\n const queryParams = React.useMemo(() => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(pageSize))\n if (sorting.length > 0) {\n params.set('sort', sorting[0].id)\n params.set('order', sorting[0].desc ? 'desc' : 'asc')\n }\n if (search.trim()) params.set('search', search.trim())\n const status = filterValues.status\n if (typeof status === 'string' && status.trim()) params.set('status', status)\n const source = filterValues.source\n if (typeof source === 'string' && source.trim()) params.set('source', source)\n const lifecycleStage = filterValues.lifecycleStage\n if (typeof lifecycleStage === 'string' && lifecycleStage.trim()) params.set('lifecycleStage', lifecycleStage)\n const createdAt = filterValues.createdAt\n if (createdAt && typeof createdAt === 'object') {\n if (createdAt.from) params.set('createdFrom', createdAt.from)\n if (createdAt.to) params.set('createdTo', createdAt.to)\n }\n const emailContains = filterValues.emailContains\n if (typeof emailContains === 'string' && emailContains.trim()) {\n params.set('emailContains', emailContains.trim())\n }\n const booleanFilters: Array<['hasEmail' | 'hasPhone' | 'hasNextInteraction', string]> = [\n ['hasEmail', 'hasEmail'],\n ['hasPhone', 'hasPhone'],\n ['hasNextInteraction', 'hasNextInteraction'],\n ]\n for (const [key, queryKey] of booleanFilters) {\n const value = filterValues[key]\n if (value === true) params.set(queryKey, 'true')\n if (value === false) params.set(queryKey, 'false')\n }\n const tagValues = Array.isArray(filterValues.tagIds)\n ? filterValues.tagIds\n .map((value) => (typeof value === 'string' ? value.trim() : String(value || '').trim()))\n .filter((value) => value.length > 0)\n : []\n if (tagValues.length > 0) {\n const normalizedTagIds = tagValues\n .map((value) => (typeof tagIdToLabel[value] === 'string' ? value : tagLabelToId[value]))\n .filter((id): id is string => typeof id === 'string' && id.length > 0)\n if (normalizedTagIds.length === tagValues.length && normalizedTagIds.length > 0) {\n params.set('tagIds', normalizedTagIds.join(','))\n } else {\n params.set('tagIdsEmpty', 'true')\n }\n }\n Object.entries(filterValues).forEach(([key, value]) => {\n if (!key.startsWith('cf_') || value == null) return\n if (Array.isArray(value)) {\n const normalized = value\n .map((item) => {\n if (item == null) return ''\n if (typeof item === 'string') return item.trim()\n return String(item).trim()\n })\n .filter((item) => item.length > 0)\n if (normalized.length) params.set(key, normalized.join(','))\n } else if (typeof value === 'object') {\n return\n } else if (value !== '') {\n const stringValue = typeof value === 'string' ? value.trim() : String(value)\n if (stringValue) params.set(key, stringValue)\n }\n })\n const advancedParams = serializeAdvancedFilter(advancedFilterState)\n for (const [key, val] of Object.entries(advancedParams)) {\n params.set(key, val)\n }\n return params.toString()\n }, [advancedFilterState, filterValues, page, pageSize, search, sorting, tagIdToLabel, tagLabelToId])\n\n const currentParams = React.useMemo(() => Object.fromEntries(new URLSearchParams(queryParams)), [queryParams])\n const exportConfig = React.useMemo(() => ({\n view: {\n getUrl: (format: DataTableExportFormat) =>\n buildCrudExportUrl('customers/people', { ...currentParams, exportScope: 'view' }, format),\n },\n full: {\n getUrl: (format: DataTableExportFormat) =>\n buildCrudExportUrl('customers/people', { ...currentParams, exportScope: 'full', all: 'true' }, format),\n },\n }), [currentParams])\n\n React.useEffect(() => {\n let cancelled = false\n async function load() {\n setIsLoading(true)\n setCacheStatus(null)\n try {\n const fallback: PeopleResponse = { items: [], total: 0, totalPages: 1 }\n const call = await apiCall<PeopleResponse>(`/api/customers/people?${queryParams}`, undefined, { fallback })\n if (!call.ok) {\n const errorPayload = call.result as { error?: string } | undefined\n const message = typeof errorPayload?.error === 'string' ? errorPayload.error : t('customers.people.list.error.load')\n flash(message, 'error')\n if (!cancelled) setCacheStatus(null)\n return\n }\n const payload = call.result ?? fallback\n if (cancelled) return\n setCacheStatus(call.cacheStatus ?? null)\n const items = Array.isArray(payload.items) ? payload.items : []\n setRows(items.map((item) => mapApiItem(item as Record<string, unknown>)).filter((row): row is PersonRow => !!row))\n setTotal(typeof payload.total === 'number' ? payload.total : items.length)\n setTotalPages(typeof payload.totalPages === 'number' ? payload.totalPages : 1)\n } catch (err) {\n if (!cancelled) {\n setCacheStatus(null)\n const message = err instanceof Error ? err.message : t('customers.people.list.error.load')\n flash(message, 'error')\n }\n } finally {\n if (!cancelled) setIsLoading(false)\n }\n }\n load()\n return () => { cancelled = true }\n }, [queryParams, reloadToken, scopeVersion, t])\n\n const handleRefresh = React.useCallback(() => {\n setReloadToken((token) => token + 1)\n }, [])\n\n const handleDelete = React.useCallback(async (person: PersonRow) => {\n if (!person?.id) return\n const name = person.name || t('customers.people.list.deleteFallbackName')\n const confirmed = await confirm({\n title: t('customers.people.list.deleteConfirm', undefined, { name }),\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n await apiCallOrThrow(\n `/api/customers/people?id=${encodeURIComponent(person.id)}`,\n {\n method: 'DELETE',\n headers: { 'content-type': 'application/json' },\n },\n { errorMessage: t('customers.people.list.deleteError') },\n )\n setRows((prev) => prev.filter((row) => row.id !== person.id))\n setTotal((prev) => Math.max(prev - 1, 0))\n handleRefresh()\n flash(t('customers.people.list.deleteSuccess'), 'success')\n } catch (err) {\n const message = err instanceof Error ? err.message : t('customers.people.list.deleteError')\n flash(message, 'error')\n }\n }, [confirm, handleRefresh, t])\n\n const handleBulkDelete = React.useCallback(async (selectedRows: PersonRow[]) => {\n const confirmed = await confirm({\n title: t('customers.people.list.bulkDelete.title', 'Delete {count} people?', { count: selectedRows.length }),\n description: t('customers.people.list.bulkDelete.description', 'This action cannot be undone.'),\n variant: 'destructive',\n })\n if (!confirmed) return false\n let deletedCount = 0\n for (const row of selectedRows) {\n try {\n await apiCallOrThrow(`/api/customers/people?id=${encodeURIComponent(row.id)}`, {\n method: 'DELETE',\n headers: { 'content-type': 'application/json' },\n })\n deletedCount++\n } catch {}\n }\n if (deletedCount > 0) {\n setRows((prev) => {\n const deletedIds = new Set(selectedRows.map((r) => r.id))\n return prev.filter((r) => !deletedIds.has(r.id))\n })\n setTotal((prev) => Math.max(0, prev - deletedCount))\n flash(t('customers.people.list.bulkDelete.success', '{count} people deleted', { count: deletedCount }), 'success')\n setReloadToken((prev) => prev + 1)\n }\n return deletedCount > 0\n }, [confirm, t])\n\n const handleFiltersApply = React.useCallback((values: FilterValues) => {\n const next: FilterValues = {}\n Object.entries(values).forEach(([key, value]) => {\n if (value !== undefined) {\n next[key] = value\n }\n })\n const rawTags = Array.isArray(values.tagIds) ? (values.tagIds as string[]) : []\n const sanitizedTags = rawTags\n .map((tag) => {\n const normalized = typeof tag === 'string' ? tag.trim() : ''\n if (!normalized) return ''\n return tagIdToLabel[normalized] ?? normalized\n })\n .filter((tag) => tag.length > 0)\n if (sanitizedTags.length) next.tagIds = sanitizedTags\n else delete next.tagIds\n setFilterValues(next)\n setPage(1)\n }, [setFilterValues, setPage, tagIdToLabel])\n\n const handleFiltersClear = React.useCallback(() => {\n setFilterValues({})\n setPage(1)\n }, [setFilterValues, setPage])\n\n const columns = React.useMemo<ColumnDef<PersonRow>[]>(() => {\n const noValue = <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>\n const renderDictionaryCell = (kind: DictionaryKindKey, rawValue: string | null | undefined) => (\n <DictionaryValue\n value={rawValue}\n map={dictionaryMaps[kind]}\n fallback={rawValue ? <span>{rawValue}</span> : noValue}\n className=\"text-sm\"\n iconWrapperClassName=\"inline-flex h-6 w-6 items-center justify-center rounded border border-border bg-card\"\n iconClassName=\"h-4 w-4\"\n colorClassName=\"h-3 w-3 rounded-full\"\n />\n )\n\n const renderCustomFieldCell = (value: unknown) => {\n if (value == null) return noValue\n if (Array.isArray(value)) {\n if (!value.length) return noValue\n const normalized = value\n .map((item) => {\n if (item == null) return ''\n if (typeof item === 'string') return item.trim()\n return String(item).trim()\n })\n .filter((item) => item.length > 0)\n if (!normalized.length) return noValue\n return <span className=\"text-sm\">{normalized.join(', ')}</span>\n }\n if (typeof value === 'boolean') {\n return (\n <span className=\"text-sm\">\n {value\n ? t('customers.people.list.booleanYes', 'Yes')\n : t('customers.people.list.booleanNo', 'No')}\n </span>\n )\n }\n const stringValue = typeof value === 'string' ? value.trim() : String(value)\n if (!stringValue) return noValue\n return <span className=\"text-sm\">{stringValue}</span>\n }\n\n const baseColumns: ColumnDef<PersonRow>[] = [\n {\n accessorKey: 'name',\n header: t('customers.people.list.columns.name'),\n meta: { alwaysVisible: true, columnChooserGroup: 'Basic Info', filterKey: 'display_name' },\n cell: ({ row }) => (\n <Link href={`/backend/customers/people-v2/${row.original.id}`} className=\"font-medium hover:underline\">\n {row.original.name}\n </Link>\n ),\n },\n {\n accessorKey: 'email',\n header: t('customers.people.list.columns.email'),\n meta: { columnChooserGroup: 'Contact', filterKey: 'primary_email' },\n cell: ({ row }) => row.original.email || <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>,\n },\n {\n accessorKey: 'status',\n header: t('customers.people.list.columns.status'),\n meta: { filterType: 'select' as const, filterOptions: dictionaryOptions.statuses, columnChooserGroup: 'Basic Info' },\n cell: ({ row }) => renderDictionaryCell('statuses', row.original.status),\n },\n {\n accessorKey: 'lifecycleStage',\n header: t('customers.people.list.columns.lifecycleStage'),\n meta: {\n filterType: 'select' as const,\n filterOptions: dictionaryOptions.lifecycleStages,\n columnChooserGroup: 'Basic Info',\n filterKey: 'lifecycle_stage',\n },\n cell: ({ row }) => renderDictionaryCell('lifecycle-stages', row.original.lifecycleStage),\n },\n {\n accessorKey: 'nextInteractionAt',\n header: t('customers.people.list.columns.nextInteraction'),\n meta: {\n columnChooserGroup: 'Dates',\n filterKey: 'next_interaction_at',\n tooltipContent: (row: PersonRow) => {\n if (!row.nextInteractionAt) return undefined\n const date = formatDate(row.nextInteractionAt, '')\n const name = row.nextInteractionName || ''\n return [date, name].filter(Boolean).join(' - ')\n },\n },\n cell: ({ row }) =>\n row.original.nextInteractionAt\n ? (\n <div className=\"flex items-start gap-2 text-sm\">\n {row.original.nextInteractionIcon ? (\n <span className=\"mt-0.5 inline-flex h-6 w-6 items-center justify-center rounded border border-border bg-card\">\n {renderDictionaryIcon(row.original.nextInteractionIcon, 'h-4 w-4')}\n </span>\n ) : null}\n <div className=\"flex flex-col\">\n <span>{formatDate(row.original.nextInteractionAt, t('customers.people.list.noValue'))}</span>\n {row.original.nextInteractionName ? (\n <span className=\"text-xs text-muted-foreground\">{row.original.nextInteractionName}</span>\n ) : null}\n </div>\n {row.original.nextInteractionColor ? (\n <span className=\"mt-1\">\n {renderDictionaryColor(row.original.nextInteractionColor, 'h-3 w-3 rounded-full border border-border')}\n </span>\n ) : null}\n </div>\n )\n : <span className=\"text-muted-foreground text-sm\">{t('customers.people.list.noValue')}</span>,\n },\n {\n accessorKey: 'source',\n header: t('customers.people.list.columns.source'),\n meta: { filterType: 'select' as const, filterOptions: dictionaryOptions.sources, columnChooserGroup: 'Basic Info' },\n cell: ({ row }) => renderDictionaryCell('sources', row.original.source),\n },\n {\n accessorKey: 'firstName',\n header: t('customers.people.form.firstName', 'First name'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.first_name' },\n cell: ({ row }) => row.original.firstName || noValue,\n },\n {\n accessorKey: 'lastName',\n header: t('customers.people.form.lastName', 'Last name'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.last_name' },\n cell: ({ row }) => row.original.lastName || noValue,\n },\n {\n accessorKey: 'preferredName',\n header: t('customers.people.form.preferredName', 'Preferred name'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.preferred_name' },\n cell: ({ row }) => row.original.preferredName || noValue,\n },\n {\n accessorKey: 'jobTitle',\n header: t('customers.people.form.jobTitle', 'Job title'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.job_title' },\n cell: ({ row }) => row.original.jobTitle || noValue,\n },\n {\n accessorKey: 'department',\n header: t('customers.people.detail.fields.department', 'Department'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.department' },\n cell: ({ row }) => row.original.department || noValue,\n },\n {\n accessorKey: 'seniority',\n header: t('customers.people.detail.fields.seniority', 'Seniority'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.seniority' },\n cell: ({ row }) => row.original.seniority || noValue,\n },\n {\n accessorKey: 'timezone',\n header: t('customers.people.detail.fields.timezone', 'Timezone'),\n meta: { columnChooserGroup: 'Profile', hidden: true, filterKey: 'person_profile.timezone' },\n cell: ({ row }) => row.original.timezone || noValue,\n },\n {\n accessorKey: 'linkedInUrl',\n header: t('customers.people.detail.fields.linkedIn', 'LinkedIn'),\n meta: { columnChooserGroup: 'Socials', hidden: true, filterKey: 'person_profile.linked_in_url' },\n cell: ({ row }) => row.original.linkedInUrl || noValue,\n },\n {\n accessorKey: 'twitterUrl',\n header: t('customers.people.detail.fields.twitter', 'Twitter'),\n meta: { columnChooserGroup: 'Socials', hidden: true, filterKey: 'person_profile.twitter_url' },\n cell: ({ row }) => row.original.twitterUrl || noValue,\n },\n {\n accessorKey: 'description',\n header: t('customers.people.form.description', 'Description'),\n meta: { columnChooserGroup: 'Notes', hidden: true, filterKey: 'description' },\n cell: ({ row }) => row.original.description || noValue,\n },\n ]\n\n const customColumns = customFieldDefs\n .filter((def) => supportsCustomFieldColumn(def))\n .map<ColumnDef<PersonRow>>((def) => ({\n accessorKey: `cf_${def.key}`,\n header: def.label || def.key,\n meta: {\n columnChooserGroup: def.group?.title ?? 'Custom Fields',\n filterGroup: def.group?.title ?? 'Custom Fields',\n filterType: mapCustomFieldKindToFilterType(def.kind),\n filterOptions: normalizeCustomFieldFilterOptions(def.options),\n hidden: def.listVisible === false,\n },\n cell: ({ getValue }) => renderCustomFieldCell(getValue()),\n }))\n\n return [...baseColumns, ...customColumns]\n }, [customFieldDefs, dictionaryMaps, dictionaryOptions, t])\n\n return (\n <Page>\n <PageBody>\n <DataTable<PersonRow>\n stickyFirstColumn\n title={t('customers.people.list.title')}\n refreshButton={{\n label: t('customers.people.list.actions.refresh'),\n onRefresh: () => { setSearch(''); setPage(1); handleRefresh() },\n }}\n actions={(\n <Button asChild>\n <Link href=\"/backend/customers/people/create\">\n {t('customers.people.list.actions.new')}\n </Link>\n </Button>\n )}\n columns={columns}\n columnChooser={{ auto: true }}\n data={rows}\n exporter={exportConfig}\n searchValue={search}\n onSearchChange={(value) => { setSearch(value); setPage(1) }}\n searchPlaceholder={t('customers.people.list.searchPlaceholder')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={handleFiltersApply}\n onFiltersClear={handleFiltersClear}\n entityIds={[E.customers.customer_entity, E.customers.customer_person_profile]}\n perspective={{ tableId: 'customers.people.list' }}\n onRowClick={(row) => router.push(`/backend/customers/people-v2/${row.id}`)}\n sortable\n sorting={sorting}\n onSortingChange={setSorting}\n bulkActions={[\n {\n id: 'delete',\n label: t('customers.people.list.bulkDelete.action', 'Delete selected'),\n destructive: true,\n onExecute: handleBulkDelete,\n },\n ]}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'view',\n label: t('customers.people.list.actions.view'),\n onSelect: () => { router.push(`/backend/customers/people-v2/${row.id}`) },\n },\n {\n id: 'open-new-tab',\n label: t('customers.people.list.actions.openInNewTab'),\n onSelect: () => window.open(`/backend/customers/people-v2/${row.id}`, '_blank', 'noopener'),\n },\n {\n id: 'delete',\n label: t('customers.people.list.actions.delete'),\n destructive: true,\n onSelect: () => handleDelete(row),\n },\n ]}\n />\n )}\n advancedFilter={{\n auto: true,\n value: advancedFilterState,\n onChange: setAdvancedFilterState,\n onApply: () => { setPage(1) },\n onClear: () => { setAdvancedFilterState({ logic: 'and', conditions: [] }); setPage(1) },\n }}\n virtualized\n pagination={{ page, pageSize, total, totalPages, onPageChange: setPage, cacheStatus, pageSizeOptions: [10, 25, 50, 100], onPageSizeChange: handlePageSizeChange }}\n isLoading={isLoading}\n />\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -21,6 +21,7 @@ import {
21
21
  ensureTenantScope,
22
22
  requireCustomerEntity,
23
23
  extractUndoPayload,
24
+ emitQueryIndexUpsertEvents,
24
25
  requireDealInScope,
25
26
  resolveParentResourceKind
26
27
  } from "./shared.js";
@@ -138,6 +139,12 @@ async function runInTransaction(em, operation) {
138
139
  return operation(em);
139
140
  }
140
141
  async function emitNextInteractionUpdatedEvent(ctx, projection, identifiers) {
142
+ await emitQueryIndexUpsertEvents(ctx, [{
143
+ entityType: "customers:customer_entity",
144
+ recordId: projection.entityId,
145
+ organizationId: identifiers.organizationId,
146
+ tenantId: identifiers.tenantId
147
+ }]);
141
148
  await emitLifecycleEvent(ctx, "customers.next_interaction.updated", {
142
149
  id: projection.entityId,
143
150
  entityId: projection.entityId,