@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.
- package/dist/modules/customers/api/companies/route.js +141 -3
- package/dist/modules/customers/api/companies/route.js.map +2 -2
- package/dist/modules/customers/api/deals/route.js +52 -3
- package/dist/modules/customers/api/deals/route.js.map +2 -2
- package/dist/modules/customers/api/people/route.js +145 -3
- package/dist/modules/customers/api/people/route.js.map +2 -2
- package/dist/modules/customers/api/utils.js +195 -0
- package/dist/modules/customers/api/utils.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies/page.js +171 -6
- package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/page.js +100 -7
- package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people/page.js +180 -7
- package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
- package/dist/modules/customers/commands/interactions.js +7 -0
- package/dist/modules/customers/commands/interactions.js.map +2 -2
- package/dist/modules/customers/components/detail/DealForm.js +1 -0
- package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
- package/dist/modules/query_index/lib/engine.js +81 -1
- package/dist/modules/query_index/lib/engine.js.map +2 -2
- package/package.json +3 -3
- package/src/modules/customers/api/companies/route.ts +151 -3
- package/src/modules/customers/api/deals/route.ts +54 -3
- package/src/modules/customers/api/people/route.ts +160 -3
- package/src/modules/customers/api/utils.ts +286 -0
- package/src/modules/customers/backend/customers/companies/page.tsx +184 -9
- package/src/modules/customers/backend/customers/deals/page.tsx +127 -35
- package/src/modules/customers/backend/customers/people/page.tsx +191 -10
- package/src/modules/customers/commands/interactions.ts +7 -0
- package/src/modules/customers/components/detail/DealForm.tsx +1 -0
- package/src/modules/customers/i18n/de.json +12 -0
- package/src/modules/customers/i18n/en.json +15 -3
- package/src/modules/customers/i18n/es.json +12 -0
- package/src/modules/customers/i18n/pl.json +12 -0
- 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 =
|
|
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
|
-
|
|
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,
|