@open-mercato/core 0.6.5-develop.5382.1.f542de69af → 0.6.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +1 -1
- package/dist/bootstrap.js +46 -6
- package/dist/bootstrap.js.map +2 -2
- package/dist/generated/entities/organization/index.js +2 -0
- package/dist/generated/entities/organization/index.js.map +2 -2
- package/dist/generated/entity-fields-registry.js +1 -0
- package/dist/generated/entity-fields-registry.js.map +2 -2
- package/dist/helpers/integration/crmFixtures.js +4 -0
- package/dist/helpers/integration/crmFixtures.js.map +2 -2
- package/dist/modules/attachments/api/route.js +2 -0
- package/dist/modules/attachments/api/route.js.map +2 -2
- package/dist/modules/attachments/lib/access.js +18 -0
- package/dist/modules/attachments/lib/access.js.map +2 -2
- package/dist/modules/audit_logs/data/entities.js +2 -1
- package/dist/modules/audit_logs/data/entities.js.map +2 -2
- package/dist/modules/audit_logs/migrations/Migration20260611104500.js +13 -0
- package/dist/modules/audit_logs/migrations/Migration20260611104500.js.map +7 -0
- package/dist/modules/audit_logs/services/accessLogService.js +10 -0
- package/dist/modules/audit_logs/services/accessLogService.js.map +2 -2
- package/dist/modules/auth/api/admin/nav.js +9 -0
- package/dist/modules/auth/api/admin/nav.js.map +2 -2
- package/dist/modules/auth/api/login.js +4 -13
- package/dist/modules/auth/api/login.js.map +2 -2
- package/dist/modules/auth/data/entities.js +3 -1
- package/dist/modules/auth/data/entities.js.map +2 -2
- package/dist/modules/auth/lib/backendChrome.js +35 -2
- package/dist/modules/auth/lib/backendChrome.js.map +2 -2
- package/dist/modules/auth/lib/consentIntegrity.js +3 -3
- package/dist/modules/auth/lib/consentIntegrity.js.map +2 -2
- package/dist/modules/auth/migrations/Migration20260611103000.js +15 -0
- package/dist/modules/auth/migrations/Migration20260611103000.js.map +7 -0
- package/dist/modules/auth/services/authService.js +5 -3
- package/dist/modules/auth/services/authService.js.map +2 -2
- package/dist/modules/auth/services/rbacService.js +3 -2
- package/dist/modules/auth/services/rbacService.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.js +0 -3
- package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.js.map +2 -2
- package/dist/modules/customers/api/deals/route.js +43 -2
- package/dist/modules/customers/api/deals/route.js.map +2 -2
- package/dist/modules/customers/api/deals/summary/route.js +402 -0
- package/dist/modules/customers/api/deals/summary/route.js.map +7 -0
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js +16 -5
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js +22 -5
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/[id]/page.js +12 -2
- package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/page.js +221 -56
- package/dist/modules/customers/backend/customers/deals/page.js.map +3 -3
- package/dist/modules/customers/backend/customers/deals/pipeline/page.js +1 -1
- package/dist/modules/customers/backend/customers/deals/pipeline/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +18 -0
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +2 -2
- package/dist/modules/customers/cli.js +15 -9
- package/dist/modules/customers/cli.js.map +2 -2
- package/dist/modules/customers/components/DealsKpiStrip.js +282 -0
- package/dist/modules/customers/components/DealsKpiStrip.js.map +7 -0
- package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js +0 -1
- package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js.map +2 -2
- package/dist/modules/customers/components/detail/DealForm.js +100 -17
- package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
- package/dist/modules/customers/components/detail/PersonDetailTabs.js +11 -3
- package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +2 -2
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +1 -2
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +2 -2
- package/dist/modules/customers/components/kpi/PipelineStageBar.js +63 -0
- package/dist/modules/customers/components/kpi/PipelineStageBar.js.map +7 -0
- package/dist/modules/customers/lib/dealsMetrics.js +82 -0
- package/dist/modules/customers/lib/dealsMetrics.js.map +7 -0
- package/dist/modules/directory/api/organization-branding/route.js +214 -0
- package/dist/modules/directory/api/organization-branding/route.js.map +7 -0
- package/dist/modules/directory/api/organizations/route.js +7 -0
- package/dist/modules/directory/api/organizations/route.js.map +3 -3
- package/dist/modules/directory/backend/directory/branding/page.js +214 -0
- package/dist/modules/directory/backend/directory/branding/page.js.map +7 -0
- package/dist/modules/directory/backend/directory/branding/page.meta.js +26 -0
- package/dist/modules/directory/backend/directory/branding/page.meta.js.map +7 -0
- package/dist/modules/directory/commands/organizations.js +8 -1
- package/dist/modules/directory/commands/organizations.js.map +2 -2
- package/dist/modules/directory/data/entities.js +3 -0
- package/dist/modules/directory/data/entities.js.map +2 -2
- package/dist/modules/directory/data/validators.js +9 -0
- package/dist/modules/directory/data/validators.js.map +2 -2
- package/dist/modules/directory/migrations/Migration20260607222259_directory.js +13 -0
- package/dist/modules/directory/migrations/Migration20260607222259_directory.js.map +7 -0
- package/dist/modules/directory/subscribers/invalidateOrgScopeCache.js +2 -1
- package/dist/modules/directory/subscribers/invalidateOrgScopeCache.js.map +2 -2
- package/dist/modules/directory/utils/organizationScope.js +59 -27
- package/dist/modules/directory/utils/organizationScope.js.map +2 -2
- package/dist/modules/entities/api/definitions.batch.js +2 -1
- package/dist/modules/entities/api/definitions.batch.js.map +2 -2
- package/dist/modules/entities/api/entities.js +7 -0
- package/dist/modules/entities/api/entities.js.map +2 -2
- package/dist/modules/entities/api/records.js +26 -15
- package/dist/modules/entities/api/records.js.map +2 -2
- package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js +14 -0
- package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js.map +2 -2
- package/dist/modules/entities/backend/entities/user/[entityId]/records/create/page.js +14 -0
- package/dist/modules/entities/backend/entities/user/[entityId]/records/create/page.js.map +2 -2
- package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +12 -0
- package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
- package/dist/modules/entities/components/useRecordsEntityGuard.js +30 -0
- package/dist/modules/entities/components/useRecordsEntityGuard.js.map +7 -0
- package/dist/modules/query_index/data/entities.js +2 -1
- package/dist/modules/query_index/data/entities.js.map +2 -2
- package/dist/modules/query_index/lib/engine.js +4 -2
- package/dist/modules/query_index/lib/engine.js.map +2 -2
- package/dist/modules/query_index/migrations/Migration20260611103000_query_index.js +16 -0
- package/dist/modules/query_index/migrations/Migration20260611103000_query_index.js.map +7 -0
- package/dist/modules/sales/commands/documents.js +7 -5
- package/dist/modules/sales/commands/documents.js.map +2 -2
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -1
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
- package/dist/modules/sales/components/documents/salesDocumentsColumns.js +10 -0
- package/dist/modules/sales/components/documents/salesDocumentsColumns.js.map +7 -0
- package/dist/modules/staff/api/team-members.js +9 -2
- package/dist/modules/staff/api/team-members.js.map +2 -2
- package/dist/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.js +24 -1
- package/dist/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-members/[id]/page.js +11 -6
- package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
- package/dist/modules/staff/commands/team-members.js +1 -1
- package/dist/modules/staff/commands/team-members.js.map +2 -2
- package/dist/modules/staff/components/TeamMemberForm.js +1 -1
- package/dist/modules/staff/components/TeamMemberForm.js.map +2 -2
- package/dist/modules/staff/lib/scheduleSwitch.js +23 -0
- package/dist/modules/staff/lib/scheduleSwitch.js.map +7 -0
- package/dist/modules/workflows/backend/definitions/create/page.js +1 -2
- package/dist/modules/workflows/backend/definitions/create/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js +1 -2
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js +1 -2
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +2 -2
- package/dist/modules/workflows/components/NodeEditDialog.js +4 -13
- package/dist/modules/workflows/components/NodeEditDialog.js.map +2 -2
- package/dist/modules/workflows/components/NodeEditDialogCrudForm.js +4 -13
- package/dist/modules/workflows/components/NodeEditDialogCrudForm.js.map +2 -2
- package/dist/modules/workflows/components/WorkflowGraphImpl.js +1 -4
- package/dist/modules/workflows/components/WorkflowGraphImpl.js.map +2 -2
- package/dist/modules/workflows/components/fields/FormFieldArrayEditor.js +2 -5
- package/dist/modules/workflows/components/fields/FormFieldArrayEditor.js.map +2 -2
- package/generated/entities/organization/index.ts +1 -0
- package/generated/entity-fields-registry.ts +1 -0
- package/package.json +11 -12
- package/src/bootstrap.ts +65 -7
- package/src/helpers/integration/crmFixtures.ts +21 -1
- package/src/modules/attachments/AGENTS.md +79 -0
- package/src/modules/attachments/api/route.ts +2 -0
- package/src/modules/attachments/lib/access.ts +36 -0
- package/src/modules/audit_logs/data/entities.ts +1 -0
- package/src/modules/audit_logs/migrations/.snapshot-open-mercato.json +10 -0
- package/src/modules/audit_logs/migrations/Migration20260611104500.ts +13 -0
- package/src/modules/audit_logs/services/accessLogService.ts +15 -0
- package/src/modules/auth/api/admin/nav.ts +9 -0
- package/src/modules/auth/api/login.ts +13 -13
- package/src/modules/auth/data/entities.ts +2 -0
- package/src/modules/auth/i18n/de.json +0 -1
- package/src/modules/auth/i18n/en.json +0 -1
- package/src/modules/auth/i18n/es.json +0 -1
- package/src/modules/auth/i18n/pl.json +0 -1
- package/src/modules/auth/lib/backendChrome.tsx +37 -1
- package/src/modules/auth/lib/consentIntegrity.ts +6 -3
- package/src/modules/auth/migrations/.snapshot-open-mercato.json +20 -0
- package/src/modules/auth/migrations/Migration20260611103000.ts +21 -0
- package/src/modules/auth/services/authService.ts +24 -4
- package/src/modules/auth/services/rbacService.ts +11 -2
- package/src/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.tsx +0 -3
- package/src/modules/customers/api/deals/route.ts +51 -2
- package/src/modules/customers/api/deals/summary/route.ts +496 -0
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.ts +28 -6
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealData.ts +33 -6
- package/src/modules/customers/backend/customers/deals/[id]/page.tsx +17 -2
- package/src/modules/customers/backend/customers/deals/page.tsx +254 -66
- package/src/modules/customers/backend/customers/deals/pipeline/page.tsx +1 -2
- package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +18 -0
- package/src/modules/customers/cli.ts +15 -15
- package/src/modules/customers/components/DealsKpiStrip.tsx +389 -0
- package/src/modules/customers/components/detail/ConfirmDealLostDialog.tsx +0 -1
- package/src/modules/customers/components/detail/DealForm.tsx +121 -19
- package/src/modules/customers/components/detail/PersonDetailTabs.tsx +12 -2
- package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +1 -2
- package/src/modules/customers/components/kpi/PipelineStageBar.tsx +77 -0
- package/src/modules/customers/i18n/de.json +43 -0
- package/src/modules/customers/i18n/en.json +43 -0
- package/src/modules/customers/i18n/es.json +43 -0
- package/src/modules/customers/i18n/pl.json +43 -0
- package/src/modules/customers/lib/dealsMetrics.ts +159 -0
- package/src/modules/directory/api/organization-branding/route.ts +238 -0
- package/src/modules/directory/api/organizations/route.ts +7 -0
- package/src/modules/directory/backend/directory/branding/page.meta.ts +24 -0
- package/src/modules/directory/backend/directory/branding/page.tsx +248 -0
- package/src/modules/directory/commands/organizations.ts +9 -1
- package/src/modules/directory/data/entities.ts +3 -0
- package/src/modules/directory/data/validators.ts +12 -0
- package/src/modules/directory/i18n/de.json +21 -0
- package/src/modules/directory/i18n/en.json +21 -0
- package/src/modules/directory/i18n/es.json +21 -0
- package/src/modules/directory/i18n/pl.json +21 -0
- package/src/modules/directory/migrations/.snapshot-open-mercato.json +40 -0
- package/src/modules/directory/migrations/Migration20260607222259_directory.ts +13 -0
- package/src/modules/directory/subscribers/invalidateOrgScopeCache.ts +3 -1
- package/src/modules/directory/utils/organizationScope.ts +85 -30
- package/src/modules/entities/api/definitions.batch.ts +11 -7
- package/src/modules/entities/api/entities.ts +11 -0
- package/src/modules/entities/api/records.ts +46 -25
- package/src/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.tsx +15 -0
- package/src/modules/entities/backend/entities/user/[entityId]/records/create/page.tsx +15 -0
- package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +23 -0
- package/src/modules/entities/components/useRecordsEntityGuard.ts +41 -0
- package/src/modules/entities/i18n/de.json +1 -0
- package/src/modules/entities/i18n/en.json +1 -0
- package/src/modules/entities/i18n/es.json +1 -0
- package/src/modules/entities/i18n/pl.json +1 -0
- package/src/modules/query_index/data/entities.ts +1 -0
- package/src/modules/query_index/lib/engine.ts +11 -5
- package/src/modules/query_index/migrations/.snapshot-open-mercato.json +11 -0
- package/src/modules/query_index/migrations/Migration20260611103000_query_index.ts +29 -0
- package/src/modules/sales/commands/documents.ts +7 -5
- package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -1
- package/src/modules/sales/components/documents/salesDocumentsColumns.ts +6 -0
- package/src/modules/staff/api/team-members.ts +9 -2
- package/src/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.ts +31 -1
- package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +18 -8
- package/src/modules/staff/commands/team-members.ts +5 -2
- package/src/modules/staff/components/TeamMemberForm.tsx +4 -1
- package/src/modules/staff/i18n/de.json +1 -0
- package/src/modules/staff/i18n/en.json +1 -0
- package/src/modules/staff/i18n/es.json +1 -0
- package/src/modules/staff/i18n/pl.json +1 -0
- package/src/modules/staff/lib/scheduleSwitch.ts +46 -0
- package/src/modules/workflows/backend/definitions/create/page.tsx +1 -2
- package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +1 -2
- package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +1 -2
- package/src/modules/workflows/components/NodeEditDialog.tsx +1 -4
- package/src/modules/workflows/components/NodeEditDialogCrudForm.tsx +4 -7
- package/src/modules/workflows/components/WorkflowGraphImpl.tsx +1 -2
- package/src/modules/workflows/components/fields/FormFieldArrayEditor.tsx +2 -3
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../../src/modules/customers/components/detail/DealForm.tsx"],
|
|
4
|
-
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { z } from 'zod'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { IconButton } from '@open-mercato/ui/primitives/icon-button'\nimport {\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n SelectValue,\n} from '@open-mercato/ui/primitives/select'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { createCrudFormError } from '@open-mercato/ui/backend/utils/serverErrors'\nimport { DictionarySelectField } from '../formConfig'\nimport { createDictionarySelectLabels } from './utils'\nimport { E } from '#generated/entities.ids.generated'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { useCurrencyDictionary } from './hooks/useCurrencyDictionary'\nimport { DictionaryEntrySelect } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport { normalizeCustomFieldSubmitValue } from './customFieldUtils'\n\nexport type DealFormBaseValues = {\n title: string\n status?: string | null\n pipelineStage?: string | null\n pipelineId?: string | null\n pipelineStageId?: string | null\n valueAmount?: number | null\n valueCurrency?: string | null\n probability?: number | null\n expectedCloseAt?: string | null\n description?: string | null\n personIds?: string[]\n companyIds?: string[]\n}\n\nexport type DealFormSubmitPayload = {\n base: DealFormBaseValues\n custom: Record<string, unknown>\n}\n\nexport type DealFormProps = {\n mode: 'create' | 'edit'\n initialValues?: Partial<DealFormBaseValues & Record<string, unknown>>\n onSubmit: (payload: DealFormSubmitPayload) => Promise<void>\n onCancel: () => void\n onDelete?: () => Promise<void> | void\n submitLabel?: string\n cancelLabel?: string\n isSubmitting?: boolean\n embedded?: boolean\n trackDirtyWhenEmbedded?: boolean\n title?: string\n backHref?: string\n hideFooterActions?: boolean\n onDirtyChange?: (dirty: boolean) => void\n collapsibleGroups?: boolean | { pageType: string; chevronPosition?: 'left' | 'right' }\n sortableGroups?: boolean | { pageType: string }\n singleColumnGroups?: boolean\n showAssociationsGroup?: boolean\n showVersionHistory?: boolean\n showCancelAction?: boolean\n}\n\ntype EntityOption = {\n id: string\n label: string\n subtitle?: string | null\n}\n\nexport type DealLookupOption = EntityOption\n\ntype EntityMultiSelectProps = {\n value: string[]\n onChange: (next: string[]) => void\n initialOptions?: EntityOption[]\n placeholder: string\n emptyLabel: string\n loadingLabel: string\n noResultsLabel: string\n removeLabel: string\n errorLabel: string\n search: (query: string) => Promise<EntityOption[]>\n fetchByIds: (ids: string[]) => Promise<EntityOption[]>\n disabled?: boolean\n autoFocus?: boolean\n}\n\nconst DEAL_ENTITY_IDS = [E.customers.customer_deal]\nconst CURRENCY_PRIORITY = ['EUR', 'USD', 'GBP', 'PLN'] as const\n\nconst schema = z.object({\n title: z\n .string()\n .trim()\n .min(1, 'customers.people.detail.deals.titleRequired')\n .max(200, 'customers.people.detail.deals.titleTooLong'),\n status: z\n .string()\n .trim()\n .max(50, 'customers.people.detail.deals.statusTooLong')\n .optional(),\n pipelineStage: z\n .string()\n .trim()\n .max(100, 'customers.people.detail.deals.pipelineTooLong')\n .optional(),\n pipelineId: z.preprocess(\n (v) => (typeof v === 'string' && !v.trim() ? undefined : v),\n z.string().uuid('customers.people.detail.deals.pipelineIdInvalid').optional(),\n ),\n pipelineStageId: z.preprocess(\n (v) => (typeof v === 'string' && !v.trim() ? undefined : v),\n z.string().uuid('customers.people.detail.deals.pipelineStageIdInvalid').optional(),\n ),\n valueAmount: z\n .preprocess((value) => {\n if (value === '' || value === null || value === undefined) return undefined\n if (typeof value === 'number') return value\n if (typeof value === 'string') {\n const trimmed = value.trim()\n if (!trimmed) return undefined\n const parsed = Number(trimmed)\n if (Number.isNaN(parsed)) return value\n return parsed\n }\n return value\n }, z\n .number()\n .min(0, 'customers.people.detail.deals.valueInvalid')\n .optional())\n .optional(),\n valueCurrency: z\n .string()\n .transform((value) => value.trim().toUpperCase())\n .refine(\n (value) => !value || /^[A-Z]{3}$/.test(value),\n 'customers.people.detail.deals.currencyInvalid',\n )\n .optional(),\n probability: z\n .preprocess((value) => {\n if (value === '' || value === null || value === undefined) return undefined\n if (typeof value === 'number') return value\n if (typeof value === 'string') {\n const trimmed = value.trim()\n if (!trimmed) return undefined\n const parsed = Number(trimmed)\n if (Number.isNaN(parsed)) return value\n return parsed\n }\n return value\n }, z\n .number()\n .min(0, 'customers.people.detail.deals.probabilityInvalid')\n .max(100, 'customers.people.detail.deals.probabilityInvalid')\n .optional())\n .optional(),\n expectedCloseAt: z\n .string()\n .transform((value) => value.trim())\n .refine(\n (value) => {\n if (!value) return true\n const parsed = new Date(value)\n return !Number.isNaN(parsed.getTime())\n },\n 'customers.people.detail.deals.expectedCloseInvalid',\n )\n .optional(),\n description: z.string().max(4000, 'customers.people.detail.deals.descriptionTooLong').optional(),\n personIds: z.array(z.string().trim().min(1)).optional(),\n companyIds: z.array(z.string().trim().min(1)).optional(),\n}).passthrough()\n\nexport const dealFormSchema = schema\n\nimport { toDateInputValue as toDateInputValueOrNull } from '@open-mercato/shared/lib/date/format'\n\nfunction toDateInputValue(value: string | null | undefined): string {\n return toDateInputValueOrNull(value) ?? ''\n}\n\nfunction normalizeCurrency(value: string | null | undefined): string {\n if (!value) return ''\n return value.trim().slice(0, 3).toUpperCase()\n}\n\nfunction sanitizeIdList(input: unknown): string[] {\n if (!Array.isArray(input)) return []\n const set = new Set<string>()\n input.forEach((candidate) => {\n if (typeof candidate !== 'string') return\n const trimmed = candidate.trim()\n if (!trimmed.length) return\n set.add(trimmed)\n })\n return Array.from(set)\n}\n\nfunction extractPersonOption(record: Record<string, unknown>): EntityOption | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? (record.display_name as string).trim()\n : null\n const email =\n typeof record.primaryEmail === 'string' && record.primaryEmail.trim().length\n ? record.primaryEmail.trim()\n : typeof record.primary_email === 'string' && record.primary_email.trim().length\n ? (record.primary_email as string).trim()\n : null\n const label = displayName ?? email ?? id\n const subtitle = email && email !== label ? email : null\n return { id, label, subtitle }\n}\n\nfunction extractCompanyOption(record: Record<string, unknown>): EntityOption | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? (record.display_name as string).trim()\n : null\n const domain =\n typeof record.domain === 'string' && record.domain.trim().length\n ? record.domain.trim()\n : typeof record.websiteUrl === 'string' && record.websiteUrl.trim().length\n ? record.websiteUrl.trim()\n : typeof record.website_url === 'string' && record.website_url.trim().length\n ? (record.website_url as string).trim()\n : null\n const label = displayName ?? domain ?? id\n const subtitle = domain && domain !== label ? domain : null\n return { id, label, subtitle }\n}\n\nfunction EntityMultiSelect({\n value,\n onChange,\n initialOptions = [],\n placeholder,\n emptyLabel,\n loadingLabel,\n noResultsLabel,\n removeLabel,\n errorLabel,\n search,\n fetchByIds,\n disabled = false,\n autoFocus = false,\n}: EntityMultiSelectProps) {\n const [input, setInput] = React.useState('')\n const [suggestions, setSuggestions] = React.useState<EntityOption[]>([])\n const [cache, setCache] = React.useState<Map<string, EntityOption>>(() => new Map())\n const [loading, setLoading] = React.useState(false)\n const [error, setError] = React.useState<string | null>(null)\n\n const normalizedValue = React.useMemo(() => sanitizeIdList(value), [value])\n\n React.useEffect(() => {\n if (!initialOptions.length) return\n setCache((prev) => {\n const next = new Map(prev)\n initialOptions.forEach((option) => {\n if (option?.id) next.set(option.id, option)\n })\n return next\n })\n }, [initialOptions])\n\n React.useEffect(() => {\n if (!normalizedValue.length) return\n const missing = normalizedValue.filter((id) => !cache.has(id))\n if (!missing.length) return\n let cancelled = false\n ;(async () => {\n try {\n const entries = await fetchByIds(missing)\n if (cancelled) return\n setCache((prev) => {\n const next = new Map(prev)\n entries.forEach((entry) => {\n if (entry?.id) next.set(entry.id, entry)\n })\n return next\n })\n } catch {\n if (!cancelled) setError(errorLabel)\n }\n })().catch(() => {})\n return () => { cancelled = true }\n }, [cache, errorLabel, fetchByIds, normalizedValue])\n\n React.useEffect(() => {\n if (disabled) {\n setLoading(false)\n return\n }\n let cancelled = false\n const handler = window.setTimeout(async () => {\n setLoading(true)\n try {\n const results = await search(input.trim())\n if (cancelled) return\n setSuggestions(results)\n setCache((prev) => {\n const next = new Map(prev)\n results.forEach((entry) => {\n if (entry?.id) next.set(entry.id, entry)\n })\n return next\n })\n setError(null)\n } catch {\n if (!cancelled) {\n setError(errorLabel)\n setSuggestions([])\n }\n } finally {\n if (!cancelled) setLoading(false)\n }\n }, 200)\n return () => {\n cancelled = true\n window.clearTimeout(handler)\n }\n }, [disabled, errorLabel, input, search])\n\n const filteredSuggestions = React.useMemo(\n () => suggestions.filter((option) => !normalizedValue.includes(option.id)),\n [normalizedValue, suggestions],\n )\n\n const selectedOptions = React.useMemo(\n () => normalizedValue.map((id) => cache.get(id) ?? { id, label: id }),\n [cache, normalizedValue],\n )\n\n const addOption = React.useCallback(\n (option: EntityOption) => {\n if (!option?.id) return\n if (normalizedValue.includes(option.id)) return\n const next = [...normalizedValue, option.id]\n onChange(next)\n setCache((prev) => {\n const nextCache = new Map(prev)\n nextCache.set(option.id, option)\n return nextCache\n })\n setInput('')\n setSuggestions([])\n },\n [normalizedValue, onChange],\n )\n\n const removeOption = React.useCallback(\n (id: string) => {\n const next = normalizedValue.filter((candidate) => candidate !== id)\n onChange(next)\n },\n [normalizedValue, onChange],\n )\n\n return (\n <div className=\"space-y-2\">\n <div className=\"flex flex-wrap items-center gap-2 rounded border px-2 py-1\">\n {selectedOptions.map((option) => (\n <span key={option.id} className=\"inline-flex items-center gap-1 rounded bg-muted px-2 py-0.5 text-xs\">\n {option.label}\n <IconButton\n variant=\"ghost\"\n size=\"xs\"\n className=\"opacity-60 hover:opacity-100\"\n onClick={() => removeOption(option.id)}\n aria-label={`${removeLabel} ${option.label}`}\n disabled={disabled}\n >\n \u00D7\n </IconButton>\n </span>\n ))}\n <input\n type=\"text\"\n className=\"flex-1 min-w-[160px] border-0 bg-transparent py-1 text-sm outline-none\"\n value={input}\n placeholder={placeholder}\n onChange={(event) => setInput(event.target.value)}\n onKeyDown={(event) => {\n if (event.key === 'Enter') {\n event.preventDefault()\n const nextOption = filteredSuggestions[0]\n if (nextOption) addOption(nextOption)\n } else if (event.key === 'Backspace' && !input.length && normalizedValue.length) {\n removeOption(normalizedValue[normalizedValue.length - 1])\n }\n }}\n disabled={disabled}\n autoFocus={autoFocus}\n data-crud-focus-target=\"\"\n />\n </div>\n {loading ? <div className=\"text-xs text-muted-foreground\">{loadingLabel}</div> : null}\n {!loading && filteredSuggestions.length ? (\n <div className=\"flex flex-wrap gap-2\">\n {filteredSuggestions.slice(0, 10).map((option) => (\n <Button\n key={option.id}\n variant=\"outline\"\n size=\"sm\"\n className=\"h-auto px-2 py-1 text-xs font-normal\"\n onMouseDown={(event) => event.preventDefault()}\n onClick={() => addOption(option)}\n disabled={disabled}\n aria-label={option.label}\n >\n <span className=\"flex flex-col items-start\">\n <span>{option.label}</span>\n {option.subtitle ? (\n <span className=\"text-overline text-muted-foreground\">{option.subtitle}</span>\n ) : null}\n </span>\n </Button>\n ))}\n </div>\n ) : null}\n {!loading && !filteredSuggestions.length && input.trim().length ? (\n <div className=\"text-xs text-muted-foreground\">{noResultsLabel}</div>\n ) : null}\n {error ? <div className=\"text-xs text-status-error-text\">{error}</div> : null}\n {!normalizedValue.length && !input.trim().length ? (\n <div className=\"text-xs text-muted-foreground\">{emptyLabel}</div>\n ) : null}\n </div>\n )\n}\n\nexport function useDealAssociationLookups(options?: { excludeLinkedDealId?: string | null }) {\n const searchPeoplePage = React.useCallback(async (query: string, page = 1): Promise<{ items: EntityOption[]; totalPages: number }> => {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: '20',\n sortField: 'name',\n sortDir: 'asc',\n })\n if (query.trim().length) params.set('search', query.trim())\n if (options?.excludeLinkedDealId) params.set('excludeLinkedDealId', options.excludeLinkedDealId)\n const call = await apiCall<Record<string, unknown>>(`/api/customers/people?${params.toString()}`)\n if (!call.ok) {\n throw new Error(typeof call.result?.error === 'string' ? String(call.result?.error) : 'Failed to search people')\n }\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n return {\n items: items\n .map((item: unknown) => (item && typeof item === 'object' ? extractPersonOption(item as Record<string, unknown>) : null))\n .filter((entry: EntityOption | null): entry is EntityOption => entry !== null),\n totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,\n }\n }, [options?.excludeLinkedDealId])\n\n const searchPeople = React.useCallback(async (query: string): Promise<EntityOption[]> => {\n const result = await searchPeoplePage(query, 1)\n return result.items\n }, [searchPeoplePage])\n\n const fetchPeopleByIds = React.useCallback(async (ids: string[]): Promise<EntityOption[]> => {\n const unique = sanitizeIdList(ids)\n if (!unique.length) return []\n try {\n const params = new URLSearchParams({\n ids: unique.join(','),\n pageSize: String(Math.max(unique.length, 1)),\n })\n const call = await apiCall<Record<string, unknown>>(`/api/customers/people?${params.toString()}`)\n if (!call.ok) throw new Error()\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n const optionMap = new Map<string, EntityOption>()\n items.forEach((item: unknown) => {\n if (!item || typeof item !== 'object') return\n const option = extractPersonOption(item as Record<string, unknown>)\n if (option?.id) optionMap.set(option.id, option)\n })\n return unique.map((id) => optionMap.get(id) ?? { id, label: id })\n } catch {\n return unique.map((id) => ({ id, label: id }))\n }\n }, [])\n\n const searchCompaniesPage = React.useCallback(async (query: string, page = 1): Promise<{ items: EntityOption[]; totalPages: number }> => {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: '20',\n sortField: 'name',\n sortDir: 'asc',\n })\n if (query.trim().length) params.set('search', query.trim())\n if (options?.excludeLinkedDealId) params.set('excludeLinkedDealId', options.excludeLinkedDealId)\n const call = await apiCall<Record<string, unknown>>(`/api/customers/companies?${params.toString()}`)\n if (!call.ok) {\n throw new Error(typeof call.result?.error === 'string' ? String(call.result?.error) : 'Failed to search companies')\n }\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n return {\n items: items\n .map((item: unknown) => (item && typeof item === 'object' ? extractCompanyOption(item as Record<string, unknown>) : null))\n .filter((entry: EntityOption | null): entry is EntityOption => entry !== null),\n totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,\n }\n }, [options?.excludeLinkedDealId])\n\n const searchCompanies = React.useCallback(async (query: string): Promise<EntityOption[]> => {\n const result = await searchCompaniesPage(query, 1)\n return result.items\n }, [searchCompaniesPage])\n\n const fetchCompaniesByIds = React.useCallback(async (ids: string[]): Promise<EntityOption[]> => {\n const unique = sanitizeIdList(ids)\n if (!unique.length) return []\n try {\n const params = new URLSearchParams({\n ids: unique.join(','),\n pageSize: String(Math.max(unique.length, 1)),\n })\n const call = await apiCall<Record<string, unknown>>(`/api/customers/companies?${params.toString()}`)\n if (!call.ok) throw new Error()\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n const optionMap = new Map<string, EntityOption>()\n items.forEach((item: unknown) => {\n if (!item || typeof item !== 'object') return\n const option = extractCompanyOption(item as Record<string, unknown>)\n if (option?.id) optionMap.set(option.id, option)\n })\n return unique.map((id) => optionMap.get(id) ?? { id, label: id })\n } catch {\n return unique.map((id) => ({ id, label: id }))\n }\n }, [])\n\n return {\n searchPeople,\n searchPeoplePage,\n fetchPeopleByIds,\n searchCompanies,\n searchCompaniesPage,\n fetchCompaniesByIds,\n }\n}\n\nexport function DealPeopleSelector({\n value,\n onChange,\n options = [],\n disabled = false,\n autoFocus = false,\n}: {\n value: string[]\n onChange: (next: string[]) => void\n options?: EntityOption[]\n disabled?: boolean\n autoFocus?: boolean\n}) {\n const t = useT()\n const { searchPeople, fetchPeopleByIds } = useDealAssociationLookups()\n\n return (\n <EntityMultiSelect\n value={value}\n onChange={onChange}\n initialOptions={options}\n placeholder={t('customers.deals.form.people.searchPlaceholder', 'Search people\u2026')}\n emptyLabel={t('customers.deals.form.people.empty', 'No people linked yet.')}\n loadingLabel={t('customers.deals.form.people.loading', 'Searching people\u2026')}\n noResultsLabel={t('customers.deals.form.people.noResults', 'No people match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.people.error', 'Failed to load people.')}\n search={searchPeople}\n fetchByIds={fetchPeopleByIds}\n disabled={disabled}\n autoFocus={autoFocus}\n />\n )\n}\n\nexport function DealCompaniesSelector({\n value,\n onChange,\n options = [],\n disabled = false,\n}: {\n value: string[]\n onChange: (next: string[]) => void\n options?: EntityOption[]\n disabled?: boolean\n}) {\n const t = useT()\n const { searchCompanies, fetchCompaniesByIds } = useDealAssociationLookups()\n\n return (\n <EntityMultiSelect\n value={value}\n onChange={onChange}\n initialOptions={options}\n placeholder={t('customers.deals.form.companies.searchPlaceholder', 'Search companies\u2026')}\n emptyLabel={t('customers.deals.form.companies.empty', 'No companies linked yet.')}\n loadingLabel={t('customers.deals.form.companies.loading', 'Searching companies\u2026')}\n noResultsLabel={t('customers.deals.form.companies.noResults', 'No companies match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.companies.error', 'Failed to load companies.')}\n search={searchCompanies}\n fetchByIds={fetchCompaniesByIds}\n disabled={disabled}\n />\n )\n}\n\nexport function DealForm({\n mode,\n initialValues,\n onSubmit,\n onCancel,\n onDelete,\n submitLabel,\n cancelLabel,\n isSubmitting = false,\n embedded = true,\n trackDirtyWhenEmbedded = false,\n title,\n backHref,\n hideFooterActions = false,\n onDirtyChange,\n collapsibleGroups,\n sortableGroups,\n singleColumnGroups = false,\n showAssociationsGroup = true,\n showVersionHistory = true,\n showCancelAction = true,\n}: DealFormProps) {\n const t = useT()\n const [pending, setPending] = React.useState(false)\n const {\n data: currencyDictionaryData,\n error: currencyDictionaryErrorRaw,\n isLoading: currencyDictionaryLoading,\n refetch: refetchCurrencyDictionary,\n } = useCurrencyDictionary()\n const currencyDictionaryError = currencyDictionaryErrorRaw\n ? currencyDictionaryErrorRaw instanceof Error\n ? currencyDictionaryErrorRaw.message\n : String(currencyDictionaryErrorRaw)\n : null\n\n const translate = React.useCallback(\n (key: string, fallback: string) => {\n const value = t(key)\n return value === key ? fallback : value\n },\n [t],\n )\n\n const dictionaryLabels = React.useMemo(() => ({\n status: createDictionarySelectLabels('deal-statuses', translate),\n }), [translate])\n\n const resolvedCurrencyError = React.useMemo(() => {\n if (currencyDictionaryError) return currencyDictionaryError\n if (!currencyDictionaryLoading && !currencyDictionaryData) {\n return t('customers.deals.form.currency.missing', 'Currency dictionary is not configured yet.')\n }\n return null\n }, [currencyDictionaryData, currencyDictionaryError, currencyDictionaryLoading, t])\n\n const fetchCurrencyOptions = React.useCallback(async () => {\n let payload = currencyDictionaryData ?? null\n if (!payload) {\n try {\n payload = await refetchCurrencyDictionary()\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err ?? '')\n throw new Error(message || t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'))\n }\n }\n if (!payload) {\n throw new Error(t('customers.deals.form.currency.missing', 'Currency dictionary is not configured yet.'))\n }\n const priorityOrder = new Map<string, number>()\n CURRENCY_PRIORITY.forEach((code, index) => priorityOrder.set(code, index))\n const prioritized: { value: string; label: string; color: string | null; icon: string | null }[] = []\n const remainder: { value: string; label: string; color: string | null; icon: string | null }[] = []\n payload.entries.forEach((entry) => {\n const value = entry.value.toUpperCase()\n const label = entry.label && entry.label.length ? `${value} \u2013 ${entry.label}` : value\n const option = { value, label, color: null, icon: null }\n if (priorityOrder.has(value)) prioritized.push(option)\n else remainder.push(option)\n })\n prioritized.sort((a, b) => (priorityOrder.get(a.value)! - priorityOrder.get(b.value)!))\n remainder.sort((a, b) => a.label.localeCompare(b.label, undefined, { sensitivity: 'base' }))\n return [...prioritized, ...remainder]\n }, [currencyDictionaryData, refetchCurrencyDictionary, t])\n\n const currencyDictionaryLabels = React.useMemo(() => ({\n placeholder: t('customers.deals.form.currency.placeholder', 'Select currency\u2026'),\n addLabel: t('customers.deals.form.currency.add', 'Add currency'),\n dialogTitle: t('customers.deals.form.currency.dialogTitle', 'Add currency'),\n valueLabel: t('customers.deals.form.currency.valueLabel', 'Currency code'),\n valuePlaceholder: t('customers.deals.form.currency.valuePlaceholder', 'e.g. USD'),\n labelLabel: t('customers.deals.form.currency.labelLabel', 'Label'),\n labelPlaceholder: t('customers.deals.form.currency.labelPlaceholder', 'Display name shown in UI'),\n emptyError: t('customers.deals.form.currency.error.required', 'Currency code is required.'),\n cancelLabel: t('customers.deals.form.currency.cancel', 'Cancel'),\n saveLabel: t('customers.deals.form.currency.save', 'Save'),\n errorLoad: t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'),\n errorSave: t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'),\n loadingLabel: t('customers.deals.form.currency.loading', 'Loading currencies\u2026'),\n manageTitle: t('customers.deals.form.currency.manage', 'Manage currency dictionary'),\n }), [t])\n\n const { searchPeople, fetchPeopleByIds, searchCompanies, fetchCompaniesByIds } = useDealAssociationLookups()\n\n const disabled = pending || isSubmitting\n const canDelete = mode === 'edit' && typeof onDelete === 'function'\n\n type PipelineOption = { id: string; name: string; isDefault: boolean }\n type PipelineStageOption = { id: string; label: string; order: number }\n\n const [pipelines, setPipelines] = React.useState<PipelineOption[]>([])\n const [pipelineStages, setPipelineStages] = React.useState<PipelineStageOption[]>([])\n\n const loadStagesForPipeline = React.useCallback(async (pipelineId: string) => {\n if (!pipelineId) {\n setPipelineStages([])\n return\n }\n try {\n const call = await apiCall<{ items: PipelineStageOption[] }>(`/api/customers/pipeline-stages?pipelineId=${encodeURIComponent(pipelineId)}`)\n if (call.ok && call.result?.items) {\n const sorted = [...call.result.items].sort((a, b) => a.order - b.order)\n setPipelineStages(sorted)\n }\n } catch {\n setPipelineStages([])\n }\n }, [])\n\n React.useEffect(() => {\n let cancelled = false\n ;(async () => {\n try {\n const call = await apiCall<{ items: PipelineOption[] }>('/api/customers/pipelines')\n if (cancelled) return\n if (call.ok && call.result?.items) {\n setPipelines(call.result.items)\n }\n } catch {\n // ignore\n }\n })().catch(() => {})\n return () => { cancelled = true }\n }, [])\n\n React.useEffect(() => {\n const pid = initialValues?.pipelineId\n if (typeof pid === 'string' && pid.length) {\n loadStagesForPipeline(pid).catch(() => {})\n }\n }, [initialValues?.pipelineId, loadStagesForPipeline])\n\n const baseFields = React.useMemo<CrudField[]>(() => [\n {\n id: 'title',\n label: t('customers.people.detail.deals.fields.title', 'Title'),\n type: 'text',\n required: true,\n },\n {\n id: 'status',\n label: t('customers.people.detail.deals.fields.status', 'Status'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <DictionarySelectField\n kind=\"deal-statuses\"\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => setValue(next ?? '')}\n labels={dictionaryLabels.status}\n selectClassName=\"w-full\"\n />\n ),\n } as CrudField,\n {\n id: 'pipelineId',\n label: t('customers.people.detail.deals.fields.pipeline', 'Pipeline'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue, setFormValue }) => (\n <Select\n value={typeof value === 'string' && value ? value : undefined}\n onValueChange={(next) => {\n setValue(next ?? '')\n setFormValue?.('pipelineStageId', '')\n loadStagesForPipeline(next ?? '').catch(() => {})\n }}\n disabled={disabled}\n >\n <SelectTrigger>\n <SelectValue placeholder={t('customers.deals.form.pipeline.placeholder', 'Select pipeline\u2026')} />\n </SelectTrigger>\n <SelectContent>\n {pipelines.map((p) => (\n <SelectItem key={p.id} value={p.id}>{p.name}</SelectItem>\n ))}\n </SelectContent>\n </Select>\n ),\n } as CrudField,\n {\n id: 'pipelineStageId',\n label: t('customers.people.detail.deals.fields.pipelineStage', 'Pipeline stage'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <Select\n value={typeof value === 'string' && value ? value : undefined}\n onValueChange={(next) => setValue(next ?? '')}\n disabled={disabled || !pipelineStages.length}\n >\n <SelectTrigger>\n <SelectValue placeholder={t('customers.deals.form.pipelineStage.placeholder', 'Select stage\u2026')} />\n </SelectTrigger>\n <SelectContent>\n {pipelineStages.map((s) => (\n <SelectItem key={s.id} value={s.id}>{s.label}</SelectItem>\n ))}\n </SelectContent>\n </Select>\n ),\n } as CrudField,\n {\n id: 'valueAmount',\n label: t('customers.people.detail.deals.fields.valueAmount', 'Amount'),\n type: 'number',\n layout: 'half',\n },\n {\n id: 'valueCurrency',\n label: t('customers.people.detail.deals.fields.valueCurrency', 'Currency'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <div className=\"space-y-1\">\n <DictionaryEntrySelect\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => setValue(next ?? '')}\n fetchOptions={fetchCurrencyOptions}\n labels={currencyDictionaryLabels}\n manageHref=\"/backend/config/dictionaries?key=currency\"\n allowInlineCreate={false}\n allowAppearance={false}\n selectClassName=\"w-full\"\n sortOptions=\"none\"\n disabled={disabled}\n showLabelInput={false}\n />\n {resolvedCurrencyError ? (\n <div className=\"text-xs text-muted-foreground\">{resolvedCurrencyError}</div>\n ) : null}\n </div>\n ),\n } as CrudField,\n {\n id: 'probability',\n label: t('customers.people.detail.deals.fields.probability', 'Probability (%)'),\n type: 'number',\n layout: 'half',\n },\n {\n id: 'expectedCloseAt',\n label: t('customers.people.detail.deals.fields.expectedCloseAt', 'Expected close'),\n type: 'date',\n layout: 'half',\n },\n {\n id: 'description',\n label: t('customers.people.detail.deals.fields.description', 'Description'),\n type: 'textarea',\n },\n {\n id: 'personIds',\n label: t('customers.people.detail.deals.fields.people', 'People'),\n type: 'custom',\n component: ({ value, setValue, autoFocus }) => (\n <EntityMultiSelect\n value={Array.isArray(value) ? value : []}\n onChange={(next) => setValue(next)}\n placeholder={t('customers.deals.form.people.searchPlaceholder', 'Search people\u2026')}\n emptyLabel={t('customers.deals.form.people.empty', 'No people linked yet.')}\n loadingLabel={t('customers.deals.form.people.loading', 'Searching people\u2026')}\n noResultsLabel={t('customers.deals.form.people.noResults', 'No people match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.people.error', 'Failed to load people.')}\n search={searchPeople}\n fetchByIds={fetchPeopleByIds}\n disabled={disabled}\n autoFocus={autoFocus}\n />\n ),\n } as CrudField,\n {\n id: 'companyIds',\n label: t('customers.people.detail.deals.fields.companies', 'Companies'),\n type: 'custom',\n component: ({ value, setValue }) => (\n <EntityMultiSelect\n value={Array.isArray(value) ? value : []}\n onChange={(next) => setValue(next)}\n placeholder={t('customers.deals.form.companies.searchPlaceholder', 'Search companies\u2026')}\n emptyLabel={t('customers.deals.form.companies.empty', 'No companies linked yet.')}\n loadingLabel={t('customers.deals.form.companies.loading', 'Searching companies\u2026')}\n noResultsLabel={t('customers.deals.form.companies.noResults', 'No companies match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.companies.error', 'Failed to load companies.')}\n search={searchCompanies}\n fetchByIds={fetchCompaniesByIds}\n disabled={disabled}\n />\n ),\n } as CrudField,\n ], [currencyDictionaryLabels, fetchCurrencyOptions, resolvedCurrencyError, pipelines, pipelineStages, loadStagesForPipeline, dictionaryLabels.status, disabled, fetchCompaniesByIds, fetchPeopleByIds, searchCompanies, searchPeople, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n const nextGroups: CrudFormGroup[] = [\n {\n id: 'details',\n title: t('customers.people.detail.deals.form.details', 'Deal details'),\n column: 1,\n fields: ['title', 'status', 'pipelineId', 'pipelineStageId', 'valueAmount', 'valueCurrency', 'probability', 'expectedCloseAt', 'description'],\n },\n ...(showAssociationsGroup\n ? [{\n id: 'associations',\n title: t('customers.people.detail.deals.form.associations', 'Associations'),\n column: 1,\n fields: ['personIds', 'companyIds'],\n } satisfies CrudFormGroup]\n : []),\n {\n id: 'custom',\n title: t('customers.people.detail.deals.form.customFields', 'Custom fields'),\n column: 2,\n kind: 'customFields',\n },\n ]\n return nextGroups.map((group) => (singleColumnGroups ? { ...group, column: 1 } : group))\n }, [showAssociationsGroup, singleColumnGroups, t])\n\n const embeddedInitialValues = React.useMemo(() => {\n const normalizeNumber = (value: unknown): number | null => {\n if (typeof value === 'number' && Number.isFinite(value)) return value\n if (typeof value === 'string') {\n const parsed = Number(value)\n return Number.isNaN(parsed) ? null : parsed\n }\n return null\n }\n\n const resolveIdsFromSource = (source: unknown): string[] => {\n if (Array.isArray(source)) {\n return sanitizeIdList(\n source.map((entry) => {\n if (typeof entry === 'string') return entry\n if (entry && typeof entry === 'object' && 'id' in entry) {\n const candidate = (entry as { id?: unknown }).id\n if (typeof candidate === 'string') return candidate\n }\n return null\n }),\n )\n }\n return []\n }\n\n return {\n id: typeof initialValues?.id === 'string' ? initialValues.id : undefined,\n title: initialValues?.title ?? '',\n status: initialValues?.status ?? '',\n pipelineStage: initialValues?.pipelineStage ?? '',\n pipelineId: initialValues?.pipelineId ?? (typeof (initialValues as Record<string, unknown>)?.pipeline_id === 'string' ? (initialValues as Record<string, unknown>).pipeline_id as string : ''),\n pipelineStageId: initialValues?.pipelineStageId ?? (typeof (initialValues as Record<string, unknown>)?.pipeline_stage_id === 'string' ? (initialValues as Record<string, unknown>).pipeline_stage_id as string : ''),\n valueAmount: normalizeNumber(initialValues?.valueAmount ?? null),\n valueCurrency: normalizeCurrency(initialValues?.valueCurrency ?? null),\n probability: normalizeNumber(initialValues?.probability ?? null),\n expectedCloseAt: toDateInputValue(initialValues?.expectedCloseAt ?? null),\n description: initialValues?.description ?? '',\n personIds: sanitizeIdList(initialValues?.personIds ?? resolveIdsFromSource(initialValues?.people)),\n companyIds: sanitizeIdList(initialValues?.companyIds ?? resolveIdsFromSource(initialValues?.companies)),\n ...Object.fromEntries(\n Object.entries(initialValues ?? {})\n .filter(([key]) => key.startsWith('cf_'))\n .map(([key, value]) => [key, value]),\n ),\n }\n }, [initialValues])\n\n const handleSubmit = React.useCallback(\n async (values: Record<string, unknown>) => {\n if (pending || isSubmitting) return\n setPending(true)\n try {\n const parsed = schema.safeParse(values)\n if (!parsed.success) {\n throw buildDealValidationError(parsed.error.issues, t)\n }\n const expectedCloseAt =\n parsed.data.expectedCloseAt && parsed.data.expectedCloseAt.length\n ? new Date(parsed.data.expectedCloseAt).toISOString()\n : undefined\n const personIds = sanitizeIdList(parsed.data.personIds)\n const companyIds = sanitizeIdList(parsed.data.companyIds)\n const base: DealFormBaseValues = {\n title: parsed.data.title,\n status: parsed.data.status || undefined,\n pipelineStage: parsed.data.pipelineStage || undefined,\n pipelineId: parsed.data.pipelineId || undefined,\n pipelineStageId: parsed.data.pipelineStageId || undefined,\n valueAmount:\n typeof parsed.data.valueAmount === 'number' ? parsed.data.valueAmount : undefined,\n valueCurrency: parsed.data.valueCurrency || undefined,\n probability:\n typeof parsed.data.probability === 'number' ? parsed.data.probability : undefined,\n expectedCloseAt,\n description: parsed.data.description && parsed.data.description.length\n ? parsed.data.description\n : undefined,\n personIds,\n companyIds,\n }\n const customEntries = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n await onSubmit({ base, custom: customEntries })\n } finally {\n setPending(false)\n }\n },\n [isSubmitting, onSubmit, pending, t],\n )\n\n return (\n <CrudForm<Record<string, unknown>>\n embedded={embedded}\n trackDirtyWhenEmbedded={trackDirtyWhenEmbedded}\n title={title}\n backHref={backHref}\n hideFooterActions={hideFooterActions}\n onDirtyChange={onDirtyChange}\n collapsibleGroups={collapsibleGroups}\n sortableGroups={sortableGroups}\n versionHistory={showVersionHistory && mode === 'edit' && initialValues?.id\n ? { resourceKind: 'customers.deal', resourceId: String(initialValues.id) }\n : undefined}\n schema={schema}\n fields={baseFields}\n groups={groups}\n entityIds={DEAL_ENTITY_IDS}\n initialValues={embeddedInitialValues}\n onSubmit={handleSubmit}\n onDelete={canDelete ? onDelete : undefined}\n deleteVisible={canDelete}\n submitLabel={\n submitLabel ??\n (mode === 'edit'\n ? t('customers.people.detail.deals.update', 'Update deal (\u2318/Ctrl + Enter)')\n : t('customers.people.detail.deals.save', 'Save deal (\u2318/Ctrl + Enter)'))\n }\n extraActions={showCancelAction ? (\n <Button\n type=\"button\"\n variant=\"outline\"\n onClick={onCancel}\n disabled={pending || isSubmitting}\n >\n {cancelLabel ?? t('customers.people.detail.deals.cancel', 'Cancel')}\n </Button>\n ) : undefined}\n />\n )\n}\n\nexport function buildDealValidationError(issues: z.ZodIssue[], t: (key: string, fallback?: string) => string) {\n const issue = issues[0]\n const message =\n typeof issue?.message === 'string'\n ? issue.message\n : t('customers.people.detail.deals.error', 'Failed to save deal.')\n const firstPath = Array.isArray(issue?.path) ? issue?.path?.[0] : undefined\n const field = typeof firstPath === 'string' ? firstPath : undefined\n throw createCrudFormError(message, field ? { [field]: message } : undefined)\n}\n\nexport default DealForm\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { z } from 'zod'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { IconButton } from '@open-mercato/ui/primitives/icon-button'\nimport {\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n SelectValue,\n} from '@open-mercato/ui/primitives/select'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { createCrudFormError } from '@open-mercato/ui/backend/utils/serverErrors'\nimport { DictionarySelectField } from '../formConfig'\nimport { createDictionarySelectLabels } from './utils'\nimport { E } from '#generated/entities.ids.generated'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { useCurrencyDictionary } from './hooks/useCurrencyDictionary'\nimport { DictionaryEntrySelect } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport { normalizeCustomFieldSubmitValue } from './customFieldUtils'\n\nexport type DealFormBaseValues = {\n title: string\n status?: string | null\n pipelineStage?: string | null\n pipelineId?: string | null\n pipelineStageId?: string | null\n valueAmount?: number | null\n valueCurrency?: string | null\n probability?: number | null\n expectedCloseAt?: string | null\n description?: string | null\n personIds?: string[]\n companyIds?: string[]\n}\n\ntype PipelineOption = { id: string; name: string; isDefault: boolean }\ntype PipelineStageOption = { id: string; label: string; order: number }\n\nexport type DealFormSubmitPayload = {\n base: DealFormBaseValues\n custom: Record<string, unknown>\n}\n\nexport type DealFormProps = {\n mode: 'create' | 'edit'\n initialValues?: Partial<DealFormBaseValues & Record<string, unknown>>\n onSubmit: (payload: DealFormSubmitPayload) => Promise<void>\n onCancel: () => void\n onDelete?: () => Promise<void> | void\n submitLabel?: string\n cancelLabel?: string\n isSubmitting?: boolean\n embedded?: boolean\n trackDirtyWhenEmbedded?: boolean\n title?: string\n backHref?: string\n hideFooterActions?: boolean\n onDirtyChange?: (dirty: boolean) => void\n collapsibleGroups?: boolean | { pageType: string; chevronPosition?: 'left' | 'right' }\n sortableGroups?: boolean | { pageType: string }\n singleColumnGroups?: boolean\n showAssociationsGroup?: boolean\n showVersionHistory?: boolean\n showCancelAction?: boolean\n initialPipelineOptions?: PipelineOption[]\n initialPipelineStageOptions?: PipelineStageOption[]\n}\n\ntype EntityOption = {\n id: string\n label: string\n subtitle?: string | null\n}\n\nexport type DealLookupOption = EntityOption\n\ntype EntityMultiSelectProps = {\n value: string[]\n onChange: (next: string[]) => void\n initialOptions?: EntityOption[]\n placeholder: string\n emptyLabel: string\n loadingLabel: string\n noResultsLabel: string\n removeLabel: string\n errorLabel: string\n search: (query: string) => Promise<EntityOption[]>\n fetchByIds: (ids: string[]) => Promise<EntityOption[]>\n disabled?: boolean\n autoFocus?: boolean\n}\n\nconst DEAL_ENTITY_IDS = [E.customers.customer_deal]\nconst CURRENCY_PRIORITY = ['EUR', 'USD', 'GBP', 'PLN'] as const\nconst PIPELINE_OPTIONS_TTL_MS = 60_000\nconst PIPELINE_STAGE_OPTIONS_TTL_MS = 30_000\n\ntype MetadataCacheEntry<T> = {\n expiresAt: number\n promise: Promise<T>\n}\n\nlet pipelineOptionsCache: MetadataCacheEntry<PipelineOption[]> | null = null\nconst pipelineStageOptionsCache = new Map<string, MetadataCacheEntry<PipelineStageOption[]>>()\n\nfunction isFreshCacheEntry<T>(entry: MetadataCacheEntry<T> | null | undefined): entry is MetadataCacheEntry<T> {\n return Boolean(entry && entry.expiresAt > Date.now())\n}\n\nfunction normalizePipelineOptions(options: PipelineOption[] | undefined): PipelineOption[] {\n const byId = new Map<string, PipelineOption>()\n for (const option of options ?? []) {\n if (!option.id) continue\n byId.set(option.id, {\n id: option.id,\n name: option.name,\n isDefault: option.isDefault === true,\n })\n }\n return Array.from(byId.values())\n}\n\nfunction mergePipelineOptions(seed: PipelineOption[], loaded: PipelineOption[]): PipelineOption[] {\n const byId = new Map<string, PipelineOption>()\n for (const option of seed) byId.set(option.id, option)\n for (const option of loaded) byId.set(option.id, option)\n return Array.from(byId.values())\n}\n\nfunction normalizePipelineStageOptions(options: PipelineStageOption[] | undefined): PipelineStageOption[] {\n return [...(options ?? [])]\n .filter((option) => option.id)\n .sort((left, right) => left.order - right.order)\n}\n\nasync function fetchPipelineOptions(): Promise<PipelineOption[]> {\n if (isFreshCacheEntry(pipelineOptionsCache)) return pipelineOptionsCache.promise\n const entry: MetadataCacheEntry<PipelineOption[]> = {\n expiresAt: Date.now() + PIPELINE_OPTIONS_TTL_MS,\n promise: apiCall<{ items: PipelineOption[] }>('/api/customers/pipelines')\n .then((call) => (call.ok && call.result?.items ? normalizePipelineOptions(call.result.items) : [])),\n }\n pipelineOptionsCache = entry\n try {\n return await entry.promise\n } catch (error) {\n if (pipelineOptionsCache === entry) pipelineOptionsCache = null\n throw error\n }\n}\n\nasync function fetchPipelineStageOptions(pipelineId: string): Promise<PipelineStageOption[]> {\n const cached = pipelineStageOptionsCache.get(pipelineId)\n if (isFreshCacheEntry(cached)) return cached.promise\n const entry: MetadataCacheEntry<PipelineStageOption[]> = {\n expiresAt: Date.now() + PIPELINE_STAGE_OPTIONS_TTL_MS,\n promise: apiCall<{ items: PipelineStageOption[] }>(`/api/customers/pipeline-stages?pipelineId=${encodeURIComponent(pipelineId)}`)\n .then((call) => (call.ok && call.result?.items ? normalizePipelineStageOptions(call.result.items) : [])),\n }\n pipelineStageOptionsCache.set(pipelineId, entry)\n try {\n return await entry.promise\n } catch (error) {\n if (pipelineStageOptionsCache.get(pipelineId) === entry) pipelineStageOptionsCache.delete(pipelineId)\n throw error\n }\n}\n\nexport function resetDealPipelineMetadataCacheForTests() {\n pipelineOptionsCache = null\n pipelineStageOptionsCache.clear()\n}\n\nconst schema = z.object({\n title: z\n .string()\n .trim()\n .min(1, 'customers.people.detail.deals.titleRequired')\n .max(200, 'customers.people.detail.deals.titleTooLong'),\n status: z\n .string()\n .trim()\n .max(50, 'customers.people.detail.deals.statusTooLong')\n .optional(),\n pipelineStage: z\n .string()\n .trim()\n .max(100, 'customers.people.detail.deals.pipelineTooLong')\n .optional(),\n pipelineId: z.preprocess(\n (v) => (typeof v === 'string' && !v.trim() ? undefined : v),\n z.string().uuid('customers.people.detail.deals.pipelineIdInvalid').optional(),\n ),\n pipelineStageId: z.preprocess(\n (v) => (typeof v === 'string' && !v.trim() ? undefined : v),\n z.string().uuid('customers.people.detail.deals.pipelineStageIdInvalid').optional(),\n ),\n valueAmount: z\n .preprocess((value) => {\n if (value === '' || value === null || value === undefined) return undefined\n if (typeof value === 'number') return value\n if (typeof value === 'string') {\n const trimmed = value.trim()\n if (!trimmed) return undefined\n const parsed = Number(trimmed)\n if (Number.isNaN(parsed)) return value\n return parsed\n }\n return value\n }, z\n .number()\n .min(0, 'customers.people.detail.deals.valueInvalid')\n .optional())\n .optional(),\n valueCurrency: z\n .string()\n .transform((value) => value.trim().toUpperCase())\n .refine(\n (value) => !value || /^[A-Z]{3}$/.test(value),\n 'customers.people.detail.deals.currencyInvalid',\n )\n .optional(),\n probability: z\n .preprocess((value) => {\n if (value === '' || value === null || value === undefined) return undefined\n if (typeof value === 'number') return value\n if (typeof value === 'string') {\n const trimmed = value.trim()\n if (!trimmed) return undefined\n const parsed = Number(trimmed)\n if (Number.isNaN(parsed)) return value\n return parsed\n }\n return value\n }, z\n .number()\n .min(0, 'customers.people.detail.deals.probabilityInvalid')\n .max(100, 'customers.people.detail.deals.probabilityInvalid')\n .optional())\n .optional(),\n expectedCloseAt: z\n .string()\n .transform((value) => value.trim())\n .refine(\n (value) => {\n if (!value) return true\n const parsed = new Date(value)\n return !Number.isNaN(parsed.getTime())\n },\n 'customers.people.detail.deals.expectedCloseInvalid',\n )\n .optional(),\n description: z.string().max(4000, 'customers.people.detail.deals.descriptionTooLong').optional(),\n personIds: z.array(z.string().trim().min(1)).optional(),\n companyIds: z.array(z.string().trim().min(1)).optional(),\n}).passthrough()\n\nexport const dealFormSchema = schema\n\nimport { toDateInputValue as toDateInputValueOrNull } from '@open-mercato/shared/lib/date/format'\n\nfunction toDateInputValue(value: string | null | undefined): string {\n return toDateInputValueOrNull(value) ?? ''\n}\n\nfunction normalizeCurrency(value: string | null | undefined): string {\n if (!value) return ''\n return value.trim().slice(0, 3).toUpperCase()\n}\n\nfunction sanitizeIdList(input: unknown): string[] {\n if (!Array.isArray(input)) return []\n const set = new Set<string>()\n input.forEach((candidate) => {\n if (typeof candidate !== 'string') return\n const trimmed = candidate.trim()\n if (!trimmed.length) return\n set.add(trimmed)\n })\n return Array.from(set)\n}\n\nfunction extractPersonOption(record: Record<string, unknown>): EntityOption | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? (record.display_name as string).trim()\n : null\n const email =\n typeof record.primaryEmail === 'string' && record.primaryEmail.trim().length\n ? record.primaryEmail.trim()\n : typeof record.primary_email === 'string' && record.primary_email.trim().length\n ? (record.primary_email as string).trim()\n : null\n const label = displayName ?? email ?? id\n const subtitle = email && email !== label ? email : null\n return { id, label, subtitle }\n}\n\nfunction extractCompanyOption(record: Record<string, unknown>): EntityOption | null {\n const id = typeof record.id === 'string' ? record.id : null\n if (!id) return null\n const displayName =\n typeof record.displayName === 'string' && record.displayName.trim().length\n ? record.displayName.trim()\n : typeof record.display_name === 'string' && record.display_name.trim().length\n ? (record.display_name as string).trim()\n : null\n const domain =\n typeof record.domain === 'string' && record.domain.trim().length\n ? record.domain.trim()\n : typeof record.websiteUrl === 'string' && record.websiteUrl.trim().length\n ? record.websiteUrl.trim()\n : typeof record.website_url === 'string' && record.website_url.trim().length\n ? (record.website_url as string).trim()\n : null\n const label = displayName ?? domain ?? id\n const subtitle = domain && domain !== label ? domain : null\n return { id, label, subtitle }\n}\n\nfunction EntityMultiSelect({\n value,\n onChange,\n initialOptions = [],\n placeholder,\n emptyLabel,\n loadingLabel,\n noResultsLabel,\n removeLabel,\n errorLabel,\n search,\n fetchByIds,\n disabled = false,\n autoFocus = false,\n}: EntityMultiSelectProps) {\n const [input, setInput] = React.useState('')\n const [suggestions, setSuggestions] = React.useState<EntityOption[]>([])\n const [cache, setCache] = React.useState<Map<string, EntityOption>>(() => new Map())\n const [loading, setLoading] = React.useState(false)\n const [error, setError] = React.useState<string | null>(null)\n\n const normalizedValue = React.useMemo(() => sanitizeIdList(value), [value])\n\n React.useEffect(() => {\n if (!initialOptions.length) return\n setCache((prev) => {\n const next = new Map(prev)\n initialOptions.forEach((option) => {\n if (option?.id) next.set(option.id, option)\n })\n return next\n })\n }, [initialOptions])\n\n React.useEffect(() => {\n if (!normalizedValue.length) return\n const missing = normalizedValue.filter((id) => !cache.has(id))\n if (!missing.length) return\n let cancelled = false\n ;(async () => {\n try {\n const entries = await fetchByIds(missing)\n if (cancelled) return\n setCache((prev) => {\n const next = new Map(prev)\n entries.forEach((entry) => {\n if (entry?.id) next.set(entry.id, entry)\n })\n return next\n })\n } catch {\n if (!cancelled) setError(errorLabel)\n }\n })().catch(() => {})\n return () => { cancelled = true }\n }, [cache, errorLabel, fetchByIds, normalizedValue])\n\n React.useEffect(() => {\n if (disabled) {\n setLoading(false)\n return\n }\n let cancelled = false\n const handler = window.setTimeout(async () => {\n setLoading(true)\n try {\n const results = await search(input.trim())\n if (cancelled) return\n setSuggestions(results)\n setCache((prev) => {\n const next = new Map(prev)\n results.forEach((entry) => {\n if (entry?.id) next.set(entry.id, entry)\n })\n return next\n })\n setError(null)\n } catch {\n if (!cancelled) {\n setError(errorLabel)\n setSuggestions([])\n }\n } finally {\n if (!cancelled) setLoading(false)\n }\n }, 200)\n return () => {\n cancelled = true\n window.clearTimeout(handler)\n }\n }, [disabled, errorLabel, input, search])\n\n const filteredSuggestions = React.useMemo(\n () => suggestions.filter((option) => !normalizedValue.includes(option.id)),\n [normalizedValue, suggestions],\n )\n\n const selectedOptions = React.useMemo(\n () => normalizedValue.map((id) => cache.get(id) ?? { id, label: id }),\n [cache, normalizedValue],\n )\n\n const addOption = React.useCallback(\n (option: EntityOption) => {\n if (!option?.id) return\n if (normalizedValue.includes(option.id)) return\n const next = [...normalizedValue, option.id]\n onChange(next)\n setCache((prev) => {\n const nextCache = new Map(prev)\n nextCache.set(option.id, option)\n return nextCache\n })\n setInput('')\n setSuggestions([])\n },\n [normalizedValue, onChange],\n )\n\n const removeOption = React.useCallback(\n (id: string) => {\n const next = normalizedValue.filter((candidate) => candidate !== id)\n onChange(next)\n },\n [normalizedValue, onChange],\n )\n\n return (\n <div className=\"space-y-2\">\n <div className=\"flex flex-wrap items-center gap-2 rounded border px-2 py-1\">\n {selectedOptions.map((option) => (\n <span key={option.id} className=\"inline-flex items-center gap-1 rounded bg-muted px-2 py-0.5 text-xs\">\n {option.label}\n <IconButton\n variant=\"ghost\"\n size=\"xs\"\n className=\"opacity-60 hover:opacity-100\"\n onClick={() => removeOption(option.id)}\n aria-label={`${removeLabel} ${option.label}`}\n disabled={disabled}\n >\n \u00D7\n </IconButton>\n </span>\n ))}\n <input\n type=\"text\"\n className=\"flex-1 min-w-[160px] border-0 bg-transparent py-1 text-sm outline-none\"\n value={input}\n placeholder={placeholder}\n onChange={(event) => setInput(event.target.value)}\n onKeyDown={(event) => {\n if (event.key === 'Enter') {\n event.preventDefault()\n const nextOption = filteredSuggestions[0]\n if (nextOption) addOption(nextOption)\n } else if (event.key === 'Backspace' && !input.length && normalizedValue.length) {\n removeOption(normalizedValue[normalizedValue.length - 1])\n }\n }}\n disabled={disabled}\n autoFocus={autoFocus}\n data-crud-focus-target=\"\"\n />\n </div>\n {loading ? <div className=\"text-xs text-muted-foreground\">{loadingLabel}</div> : null}\n {!loading && filteredSuggestions.length ? (\n <div className=\"flex flex-wrap gap-2\">\n {filteredSuggestions.slice(0, 10).map((option) => (\n <Button\n key={option.id}\n variant=\"outline\"\n size=\"sm\"\n className=\"h-auto px-2 py-1 text-xs font-normal\"\n onMouseDown={(event) => event.preventDefault()}\n onClick={() => addOption(option)}\n disabled={disabled}\n aria-label={option.label}\n >\n <span className=\"flex flex-col items-start\">\n <span>{option.label}</span>\n {option.subtitle ? (\n <span className=\"text-overline text-muted-foreground\">{option.subtitle}</span>\n ) : null}\n </span>\n </Button>\n ))}\n </div>\n ) : null}\n {!loading && !filteredSuggestions.length && input.trim().length ? (\n <div className=\"text-xs text-muted-foreground\">{noResultsLabel}</div>\n ) : null}\n {error ? <div className=\"text-xs text-status-error-text\">{error}</div> : null}\n {!normalizedValue.length && !input.trim().length ? (\n <div className=\"text-xs text-muted-foreground\">{emptyLabel}</div>\n ) : null}\n </div>\n )\n}\n\nexport function useDealAssociationLookups(options?: { excludeLinkedDealId?: string | null }) {\n const searchPeoplePage = React.useCallback(async (query: string, page = 1): Promise<{ items: EntityOption[]; totalPages: number }> => {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: '20',\n sortField: 'name',\n sortDir: 'asc',\n })\n if (query.trim().length) params.set('search', query.trim())\n if (options?.excludeLinkedDealId) params.set('excludeLinkedDealId', options.excludeLinkedDealId)\n const call = await apiCall<Record<string, unknown>>(`/api/customers/people?${params.toString()}`)\n if (!call.ok) {\n throw new Error(typeof call.result?.error === 'string' ? String(call.result?.error) : 'Failed to search people')\n }\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n return {\n items: items\n .map((item: unknown) => (item && typeof item === 'object' ? extractPersonOption(item as Record<string, unknown>) : null))\n .filter((entry: EntityOption | null): entry is EntityOption => entry !== null),\n totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,\n }\n }, [options?.excludeLinkedDealId])\n\n const searchPeople = React.useCallback(async (query: string): Promise<EntityOption[]> => {\n const result = await searchPeoplePage(query, 1)\n return result.items\n }, [searchPeoplePage])\n\n const fetchPeopleByIds = React.useCallback(async (ids: string[]): Promise<EntityOption[]> => {\n const unique = sanitizeIdList(ids)\n if (!unique.length) return []\n try {\n const params = new URLSearchParams({\n ids: unique.join(','),\n pageSize: String(Math.max(unique.length, 1)),\n })\n const call = await apiCall<Record<string, unknown>>(`/api/customers/people?${params.toString()}`)\n if (!call.ok) throw new Error()\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n const optionMap = new Map<string, EntityOption>()\n items.forEach((item: unknown) => {\n if (!item || typeof item !== 'object') return\n const option = extractPersonOption(item as Record<string, unknown>)\n if (option?.id) optionMap.set(option.id, option)\n })\n return unique.map((id) => optionMap.get(id) ?? { id, label: id })\n } catch {\n return unique.map((id) => ({ id, label: id }))\n }\n }, [])\n\n const searchCompaniesPage = React.useCallback(async (query: string, page = 1): Promise<{ items: EntityOption[]; totalPages: number }> => {\n const params = new URLSearchParams({\n page: String(page),\n pageSize: '20',\n sortField: 'name',\n sortDir: 'asc',\n })\n if (query.trim().length) params.set('search', query.trim())\n if (options?.excludeLinkedDealId) params.set('excludeLinkedDealId', options.excludeLinkedDealId)\n const call = await apiCall<Record<string, unknown>>(`/api/customers/companies?${params.toString()}`)\n if (!call.ok) {\n throw new Error(typeof call.result?.error === 'string' ? String(call.result?.error) : 'Failed to search companies')\n }\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n return {\n items: items\n .map((item: unknown) => (item && typeof item === 'object' ? extractCompanyOption(item as Record<string, unknown>) : null))\n .filter((entry: EntityOption | null): entry is EntityOption => entry !== null),\n totalPages: typeof payload.totalPages === 'number' ? payload.totalPages : 1,\n }\n }, [options?.excludeLinkedDealId])\n\n const searchCompanies = React.useCallback(async (query: string): Promise<EntityOption[]> => {\n const result = await searchCompaniesPage(query, 1)\n return result.items\n }, [searchCompaniesPage])\n\n const fetchCompaniesByIds = React.useCallback(async (ids: string[]): Promise<EntityOption[]> => {\n const unique = sanitizeIdList(ids)\n if (!unique.length) return []\n try {\n const params = new URLSearchParams({\n ids: unique.join(','),\n pageSize: String(Math.max(unique.length, 1)),\n })\n const call = await apiCall<Record<string, unknown>>(`/api/customers/companies?${params.toString()}`)\n if (!call.ok) throw new Error()\n const payload = call.result ?? {}\n const items = Array.isArray(payload.items) ? payload.items : []\n const optionMap = new Map<string, EntityOption>()\n items.forEach((item: unknown) => {\n if (!item || typeof item !== 'object') return\n const option = extractCompanyOption(item as Record<string, unknown>)\n if (option?.id) optionMap.set(option.id, option)\n })\n return unique.map((id) => optionMap.get(id) ?? { id, label: id })\n } catch {\n return unique.map((id) => ({ id, label: id }))\n }\n }, [])\n\n return {\n searchPeople,\n searchPeoplePage,\n fetchPeopleByIds,\n searchCompanies,\n searchCompaniesPage,\n fetchCompaniesByIds,\n }\n}\n\nexport function DealPeopleSelector({\n value,\n onChange,\n options = [],\n disabled = false,\n autoFocus = false,\n}: {\n value: string[]\n onChange: (next: string[]) => void\n options?: EntityOption[]\n disabled?: boolean\n autoFocus?: boolean\n}) {\n const t = useT()\n const { searchPeople, fetchPeopleByIds } = useDealAssociationLookups()\n\n return (\n <EntityMultiSelect\n value={value}\n onChange={onChange}\n initialOptions={options}\n placeholder={t('customers.deals.form.people.searchPlaceholder', 'Search people\u2026')}\n emptyLabel={t('customers.deals.form.people.empty', 'No people linked yet.')}\n loadingLabel={t('customers.deals.form.people.loading', 'Searching people\u2026')}\n noResultsLabel={t('customers.deals.form.people.noResults', 'No people match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.people.error', 'Failed to load people.')}\n search={searchPeople}\n fetchByIds={fetchPeopleByIds}\n disabled={disabled}\n autoFocus={autoFocus}\n />\n )\n}\n\nexport function DealCompaniesSelector({\n value,\n onChange,\n options = [],\n disabled = false,\n}: {\n value: string[]\n onChange: (next: string[]) => void\n options?: EntityOption[]\n disabled?: boolean\n}) {\n const t = useT()\n const { searchCompanies, fetchCompaniesByIds } = useDealAssociationLookups()\n\n return (\n <EntityMultiSelect\n value={value}\n onChange={onChange}\n initialOptions={options}\n placeholder={t('customers.deals.form.companies.searchPlaceholder', 'Search companies\u2026')}\n emptyLabel={t('customers.deals.form.companies.empty', 'No companies linked yet.')}\n loadingLabel={t('customers.deals.form.companies.loading', 'Searching companies\u2026')}\n noResultsLabel={t('customers.deals.form.companies.noResults', 'No companies match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.companies.error', 'Failed to load companies.')}\n search={searchCompanies}\n fetchByIds={fetchCompaniesByIds}\n disabled={disabled}\n />\n )\n}\n\nexport function DealForm({\n mode,\n initialValues,\n onSubmit,\n onCancel,\n onDelete,\n submitLabel,\n cancelLabel,\n isSubmitting = false,\n embedded = true,\n trackDirtyWhenEmbedded = false,\n title,\n backHref,\n hideFooterActions = false,\n onDirtyChange,\n collapsibleGroups,\n sortableGroups,\n singleColumnGroups = false,\n showAssociationsGroup = true,\n showVersionHistory = true,\n showCancelAction = true,\n initialPipelineOptions,\n initialPipelineStageOptions,\n}: DealFormProps) {\n const t = useT()\n const [pending, setPending] = React.useState(false)\n const {\n data: currencyDictionaryData,\n error: currencyDictionaryErrorRaw,\n isLoading: currencyDictionaryLoading,\n refetch: refetchCurrencyDictionary,\n } = useCurrencyDictionary()\n const currencyDictionaryError = currencyDictionaryErrorRaw\n ? currencyDictionaryErrorRaw instanceof Error\n ? currencyDictionaryErrorRaw.message\n : String(currencyDictionaryErrorRaw)\n : null\n\n const translate = React.useCallback(\n (key: string, fallback: string) => {\n const value = t(key)\n return value === key ? fallback : value\n },\n [t],\n )\n\n const dictionaryLabels = React.useMemo(() => ({\n status: createDictionarySelectLabels('deal-statuses', translate),\n }), [translate])\n\n const resolvedCurrencyError = React.useMemo(() => {\n if (currencyDictionaryError) return currencyDictionaryError\n if (!currencyDictionaryLoading && !currencyDictionaryData) {\n return t('customers.deals.form.currency.missing', 'Currency dictionary is not configured yet.')\n }\n return null\n }, [currencyDictionaryData, currencyDictionaryError, currencyDictionaryLoading, t])\n\n const fetchCurrencyOptions = React.useCallback(async () => {\n let payload = currencyDictionaryData ?? null\n if (!payload) {\n try {\n payload = await refetchCurrencyDictionary()\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err ?? '')\n throw new Error(message || t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'))\n }\n }\n if (!payload) {\n throw new Error(t('customers.deals.form.currency.missing', 'Currency dictionary is not configured yet.'))\n }\n const priorityOrder = new Map<string, number>()\n CURRENCY_PRIORITY.forEach((code, index) => priorityOrder.set(code, index))\n const prioritized: { value: string; label: string; color: string | null; icon: string | null }[] = []\n const remainder: { value: string; label: string; color: string | null; icon: string | null }[] = []\n payload.entries.forEach((entry) => {\n const value = entry.value.toUpperCase()\n const label = entry.label && entry.label.length ? `${value} \u2013 ${entry.label}` : value\n const option = { value, label, color: null, icon: null }\n if (priorityOrder.has(value)) prioritized.push(option)\n else remainder.push(option)\n })\n prioritized.sort((a, b) => (priorityOrder.get(a.value)! - priorityOrder.get(b.value)!))\n remainder.sort((a, b) => a.label.localeCompare(b.label, undefined, { sensitivity: 'base' }))\n return [...prioritized, ...remainder]\n }, [currencyDictionaryData, refetchCurrencyDictionary, t])\n\n const currencyDictionaryLabels = React.useMemo(() => ({\n placeholder: t('customers.deals.form.currency.placeholder', 'Select currency\u2026'),\n addLabel: t('customers.deals.form.currency.add', 'Add currency'),\n dialogTitle: t('customers.deals.form.currency.dialogTitle', 'Add currency'),\n valueLabel: t('customers.deals.form.currency.valueLabel', 'Currency code'),\n valuePlaceholder: t('customers.deals.form.currency.valuePlaceholder', 'e.g. USD'),\n labelLabel: t('customers.deals.form.currency.labelLabel', 'Label'),\n labelPlaceholder: t('customers.deals.form.currency.labelPlaceholder', 'Display name shown in UI'),\n emptyError: t('customers.deals.form.currency.error.required', 'Currency code is required.'),\n cancelLabel: t('customers.deals.form.currency.cancel', 'Cancel'),\n saveLabel: t('customers.deals.form.currency.save', 'Save'),\n errorLoad: t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'),\n errorSave: t('customers.deals.form.currency.error', 'Failed to load currency dictionary.'),\n loadingLabel: t('customers.deals.form.currency.loading', 'Loading currencies\u2026'),\n manageTitle: t('customers.deals.form.currency.manage', 'Manage currency dictionary'),\n }), [t])\n\n const { searchPeople, fetchPeopleByIds, searchCompanies, fetchCompaniesByIds } = useDealAssociationLookups()\n\n const disabled = pending || isSubmitting\n const canDelete = mode === 'edit' && typeof onDelete === 'function'\n\n const mountedRef = React.useRef(false)\n const seedPipelineOptions = React.useMemo(\n () => normalizePipelineOptions(initialPipelineOptions),\n [initialPipelineOptions],\n )\n const seedPipelineStageOptions = React.useMemo(\n () => Array.isArray(initialPipelineStageOptions) ? normalizePipelineStageOptions(initialPipelineStageOptions) : null,\n [initialPipelineStageOptions],\n )\n\n const [pipelines, setPipelines] = React.useState<PipelineOption[]>(() => seedPipelineOptions)\n const [pipelineStages, setPipelineStages] = React.useState<PipelineStageOption[]>(() => seedPipelineStageOptions ?? [])\n\n React.useEffect(() => {\n mountedRef.current = true\n return () => {\n mountedRef.current = false\n }\n }, [])\n\n const loadStagesForPipeline = React.useCallback(async (pipelineId: string) => {\n if (!pipelineId) {\n if (mountedRef.current) setPipelineStages([])\n return\n }\n try {\n const stages = await fetchPipelineStageOptions(pipelineId)\n if (mountedRef.current) setPipelineStages(stages)\n } catch {\n if (mountedRef.current) setPipelineStages([])\n }\n }, [])\n\n React.useEffect(() => {\n let cancelled = false\n ;(async () => {\n try {\n const loaded = await fetchPipelineOptions()\n if (cancelled || !mountedRef.current) return\n setPipelines(mergePipelineOptions(seedPipelineOptions, loaded))\n } catch {\n if (!cancelled && mountedRef.current && seedPipelineOptions.length > 0) {\n setPipelines(seedPipelineOptions)\n }\n }\n })().catch(() => {})\n return () => { cancelled = true }\n }, [seedPipelineOptions])\n\n React.useEffect(() => {\n const pid = initialValues?.pipelineId\n if (typeof pid === 'string' && pid.length) {\n if (seedPipelineStageOptions) {\n setPipelineStages(seedPipelineStageOptions)\n return\n }\n loadStagesForPipeline(pid).catch(() => {})\n } else {\n setPipelineStages([])\n }\n }, [initialValues?.pipelineId, loadStagesForPipeline, seedPipelineStageOptions])\n\n const baseFields = React.useMemo<CrudField[]>(() => [\n {\n id: 'title',\n label: t('customers.people.detail.deals.fields.title', 'Title'),\n type: 'text',\n required: true,\n },\n {\n id: 'status',\n label: t('customers.people.detail.deals.fields.status', 'Status'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <DictionarySelectField\n kind=\"deal-statuses\"\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => setValue(next ?? '')}\n labels={dictionaryLabels.status}\n selectClassName=\"w-full\"\n />\n ),\n } as CrudField,\n {\n id: 'pipelineId',\n label: t('customers.people.detail.deals.fields.pipeline', 'Pipeline'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue, setFormValue }) => (\n <Select\n value={typeof value === 'string' && value ? value : undefined}\n onValueChange={(next) => {\n setValue(next ?? '')\n setFormValue?.('pipelineStageId', '')\n loadStagesForPipeline(next ?? '').catch(() => {})\n }}\n disabled={disabled}\n >\n <SelectTrigger>\n <SelectValue placeholder={t('customers.deals.form.pipeline.placeholder', 'Select pipeline\u2026')} />\n </SelectTrigger>\n <SelectContent>\n {pipelines.map((p) => (\n <SelectItem key={p.id} value={p.id}>{p.name}</SelectItem>\n ))}\n </SelectContent>\n </Select>\n ),\n } as CrudField,\n {\n id: 'pipelineStageId',\n label: t('customers.people.detail.deals.fields.pipelineStage', 'Pipeline stage'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <Select\n value={typeof value === 'string' && value ? value : undefined}\n onValueChange={(next) => setValue(next ?? '')}\n disabled={disabled || !pipelineStages.length}\n >\n <SelectTrigger>\n <SelectValue placeholder={t('customers.deals.form.pipelineStage.placeholder', 'Select stage\u2026')} />\n </SelectTrigger>\n <SelectContent>\n {pipelineStages.map((s) => (\n <SelectItem key={s.id} value={s.id}>{s.label}</SelectItem>\n ))}\n </SelectContent>\n </Select>\n ),\n } as CrudField,\n {\n id: 'valueAmount',\n label: t('customers.people.detail.deals.fields.valueAmount', 'Amount'),\n type: 'number',\n layout: 'half',\n },\n {\n id: 'valueCurrency',\n label: t('customers.people.detail.deals.fields.valueCurrency', 'Currency'),\n type: 'custom',\n layout: 'half',\n component: ({ value, setValue }) => (\n <div className=\"space-y-1\">\n <DictionaryEntrySelect\n value={typeof value === 'string' ? value : undefined}\n onChange={(next) => setValue(next ?? '')}\n fetchOptions={fetchCurrencyOptions}\n labels={currencyDictionaryLabels}\n manageHref=\"/backend/config/dictionaries?key=currency\"\n allowInlineCreate={false}\n allowAppearance={false}\n selectClassName=\"w-full\"\n sortOptions=\"none\"\n disabled={disabled}\n showLabelInput={false}\n />\n {resolvedCurrencyError ? (\n <div className=\"text-xs text-muted-foreground\">{resolvedCurrencyError}</div>\n ) : null}\n </div>\n ),\n } as CrudField,\n {\n id: 'probability',\n label: t('customers.people.detail.deals.fields.probability', 'Probability (%)'),\n type: 'number',\n layout: 'half',\n },\n {\n id: 'expectedCloseAt',\n label: t('customers.people.detail.deals.fields.expectedCloseAt', 'Expected close'),\n type: 'date',\n layout: 'half',\n },\n {\n id: 'description',\n label: t('customers.people.detail.deals.fields.description', 'Description'),\n type: 'textarea',\n },\n {\n id: 'personIds',\n label: t('customers.people.detail.deals.fields.people', 'People'),\n type: 'custom',\n component: ({ value, setValue, autoFocus }) => (\n <EntityMultiSelect\n value={Array.isArray(value) ? value : []}\n onChange={(next) => setValue(next)}\n placeholder={t('customers.deals.form.people.searchPlaceholder', 'Search people\u2026')}\n emptyLabel={t('customers.deals.form.people.empty', 'No people linked yet.')}\n loadingLabel={t('customers.deals.form.people.loading', 'Searching people\u2026')}\n noResultsLabel={t('customers.deals.form.people.noResults', 'No people match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.people.error', 'Failed to load people.')}\n search={searchPeople}\n fetchByIds={fetchPeopleByIds}\n disabled={disabled}\n autoFocus={autoFocus}\n />\n ),\n } as CrudField,\n {\n id: 'companyIds',\n label: t('customers.people.detail.deals.fields.companies', 'Companies'),\n type: 'custom',\n component: ({ value, setValue }) => (\n <EntityMultiSelect\n value={Array.isArray(value) ? value : []}\n onChange={(next) => setValue(next)}\n placeholder={t('customers.deals.form.companies.searchPlaceholder', 'Search companies\u2026')}\n emptyLabel={t('customers.deals.form.companies.empty', 'No companies linked yet.')}\n loadingLabel={t('customers.deals.form.companies.loading', 'Searching companies\u2026')}\n noResultsLabel={t('customers.deals.form.companies.noResults', 'No companies match your search.')}\n removeLabel={t('customers.deals.form.assignees.remove', 'Remove')}\n errorLabel={t('customers.deals.form.companies.error', 'Failed to load companies.')}\n search={searchCompanies}\n fetchByIds={fetchCompaniesByIds}\n disabled={disabled}\n />\n ),\n } as CrudField,\n ], [currencyDictionaryLabels, fetchCurrencyOptions, resolvedCurrencyError, pipelines, pipelineStages, loadStagesForPipeline, dictionaryLabels.status, disabled, fetchCompaniesByIds, fetchPeopleByIds, searchCompanies, searchPeople, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n const nextGroups: CrudFormGroup[] = [\n {\n id: 'details',\n title: t('customers.people.detail.deals.form.details', 'Deal details'),\n column: 1,\n fields: ['title', 'status', 'pipelineId', 'pipelineStageId', 'valueAmount', 'valueCurrency', 'probability', 'expectedCloseAt', 'description'],\n },\n ...(showAssociationsGroup\n ? [{\n id: 'associations',\n title: t('customers.people.detail.deals.form.associations', 'Associations'),\n column: 1,\n fields: ['personIds', 'companyIds'],\n } satisfies CrudFormGroup]\n : []),\n {\n id: 'custom',\n title: t('customers.people.detail.deals.form.customFields', 'Custom fields'),\n column: 2,\n kind: 'customFields',\n },\n ]\n return nextGroups.map((group) => (singleColumnGroups ? { ...group, column: 1 } : group))\n }, [showAssociationsGroup, singleColumnGroups, t])\n\n const embeddedInitialValues = React.useMemo(() => {\n const normalizeNumber = (value: unknown): number | null => {\n if (typeof value === 'number' && Number.isFinite(value)) return value\n if (typeof value === 'string') {\n const parsed = Number(value)\n return Number.isNaN(parsed) ? null : parsed\n }\n return null\n }\n\n const resolveIdsFromSource = (source: unknown): string[] => {\n if (Array.isArray(source)) {\n return sanitizeIdList(\n source.map((entry) => {\n if (typeof entry === 'string') return entry\n if (entry && typeof entry === 'object' && 'id' in entry) {\n const candidate = (entry as { id?: unknown }).id\n if (typeof candidate === 'string') return candidate\n }\n return null\n }),\n )\n }\n return []\n }\n\n return {\n id: typeof initialValues?.id === 'string' ? initialValues.id : undefined,\n title: initialValues?.title ?? '',\n status: initialValues?.status ?? '',\n pipelineStage: initialValues?.pipelineStage ?? '',\n pipelineId: initialValues?.pipelineId ?? (typeof (initialValues as Record<string, unknown>)?.pipeline_id === 'string' ? (initialValues as Record<string, unknown>).pipeline_id as string : ''),\n pipelineStageId: initialValues?.pipelineStageId ?? (typeof (initialValues as Record<string, unknown>)?.pipeline_stage_id === 'string' ? (initialValues as Record<string, unknown>).pipeline_stage_id as string : ''),\n valueAmount: normalizeNumber(initialValues?.valueAmount ?? null),\n valueCurrency: normalizeCurrency(initialValues?.valueCurrency ?? null),\n probability: normalizeNumber(initialValues?.probability ?? null),\n expectedCloseAt: toDateInputValue(initialValues?.expectedCloseAt ?? null),\n description: initialValues?.description ?? '',\n personIds: sanitizeIdList(initialValues?.personIds ?? resolveIdsFromSource(initialValues?.people)),\n companyIds: sanitizeIdList(initialValues?.companyIds ?? resolveIdsFromSource(initialValues?.companies)),\n ...Object.fromEntries(\n Object.entries(initialValues ?? {})\n .filter(([key]) => key.startsWith('cf_'))\n .map(([key, value]) => [key, value]),\n ),\n }\n }, [initialValues])\n\n const handleSubmit = React.useCallback(\n async (values: Record<string, unknown>) => {\n if (pending || isSubmitting) return\n setPending(true)\n try {\n const parsed = schema.safeParse(values)\n if (!parsed.success) {\n throw buildDealValidationError(parsed.error.issues, t)\n }\n const expectedCloseAt =\n parsed.data.expectedCloseAt && parsed.data.expectedCloseAt.length\n ? new Date(parsed.data.expectedCloseAt).toISOString()\n : undefined\n const personIds = sanitizeIdList(parsed.data.personIds)\n const companyIds = sanitizeIdList(parsed.data.companyIds)\n const base: DealFormBaseValues = {\n title: parsed.data.title,\n status: parsed.data.status || undefined,\n pipelineStage: parsed.data.pipelineStage || undefined,\n pipelineId: parsed.data.pipelineId || undefined,\n pipelineStageId: parsed.data.pipelineStageId || undefined,\n valueAmount:\n typeof parsed.data.valueAmount === 'number' ? parsed.data.valueAmount : undefined,\n valueCurrency: parsed.data.valueCurrency || undefined,\n probability:\n typeof parsed.data.probability === 'number' ? parsed.data.probability : undefined,\n expectedCloseAt,\n description: parsed.data.description && parsed.data.description.length\n ? parsed.data.description\n : undefined,\n personIds,\n companyIds,\n }\n const customEntries = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n await onSubmit({ base, custom: customEntries })\n } finally {\n setPending(false)\n }\n },\n [isSubmitting, onSubmit, pending, t],\n )\n\n return (\n <CrudForm<Record<string, unknown>>\n embedded={embedded}\n trackDirtyWhenEmbedded={trackDirtyWhenEmbedded}\n title={title}\n backHref={backHref}\n hideFooterActions={hideFooterActions}\n onDirtyChange={onDirtyChange}\n collapsibleGroups={collapsibleGroups}\n sortableGroups={sortableGroups}\n versionHistory={showVersionHistory && mode === 'edit' && initialValues?.id\n ? { resourceKind: 'customers.deal', resourceId: String(initialValues.id) }\n : undefined}\n schema={schema}\n fields={baseFields}\n groups={groups}\n entityIds={DEAL_ENTITY_IDS}\n initialValues={embeddedInitialValues}\n onSubmit={handleSubmit}\n onDelete={canDelete ? onDelete : undefined}\n deleteVisible={canDelete}\n submitLabel={\n submitLabel ??\n (mode === 'edit'\n ? t('customers.people.detail.deals.update', 'Update deal (\u2318/Ctrl + Enter)')\n : t('customers.people.detail.deals.save', 'Save deal (\u2318/Ctrl + Enter)'))\n }\n extraActions={showCancelAction ? (\n <Button\n type=\"button\"\n variant=\"outline\"\n onClick={onCancel}\n disabled={pending || isSubmitting}\n >\n {cancelLabel ?? t('customers.people.detail.deals.cancel', 'Cancel')}\n </Button>\n ) : undefined}\n />\n )\n}\n\nexport function buildDealValidationError(issues: z.ZodIssue[], t: (key: string, fallback?: string) => string) {\n const issue = issues[0]\n const message =\n typeof issue?.message === 'string'\n ? issue.message\n : t('customers.people.detail.deals.error', 'Failed to save deal.')\n const firstPath = Array.isArray(issue?.path) ? issue?.path?.[0] : undefined\n const field = typeof firstPath === 'string' ? firstPath : undefined\n throw createCrudFormError(message, field ? { [field]: message } : undefined)\n}\n\nexport default DealForm\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -12,9 +12,10 @@ import {
|
|
|
12
12
|
Check,
|
|
13
13
|
History,
|
|
14
14
|
Paperclip,
|
|
15
|
-
Plus
|
|
15
|
+
Plus,
|
|
16
|
+
MapPin
|
|
16
17
|
} from "lucide-react";
|
|
17
|
-
const SUPPORTED_TAB_IDS = /* @__PURE__ */ new Set(["activities", "emails", "deals", "companies", "tasks", "changelog", "files"]);
|
|
18
|
+
const SUPPORTED_TAB_IDS = /* @__PURE__ */ new Set(["activities", "emails", "deals", "companies", "addresses", "tasks", "changelog", "files"]);
|
|
18
19
|
function resolveLegacyTab(tab) {
|
|
19
20
|
if (!tab) return "activities";
|
|
20
21
|
return SUPPORTED_TAB_IDS.has(tab) ? tab : "activities";
|
|
@@ -33,6 +34,7 @@ function PersonDetailTabs({
|
|
|
33
34
|
activitiesCount = 0,
|
|
34
35
|
dealsCount = 0,
|
|
35
36
|
companiesCount = 0,
|
|
37
|
+
addressesCount = 0,
|
|
36
38
|
tasksCount = 0,
|
|
37
39
|
filesCount = 0,
|
|
38
40
|
sectionAction = null,
|
|
@@ -64,6 +66,12 @@ function PersonDetailTabs({
|
|
|
64
66
|
icon: /* @__PURE__ */ jsx(Building2, { className: "size-4" }),
|
|
65
67
|
badge: /* @__PURE__ */ jsx(CountBadge, { count: companiesCount })
|
|
66
68
|
},
|
|
69
|
+
{
|
|
70
|
+
id: "addresses",
|
|
71
|
+
label: t("customers.people.detail.tabs.addresses", "Addresses"),
|
|
72
|
+
icon: /* @__PURE__ */ jsx(MapPin, { className: "size-4" }),
|
|
73
|
+
badge: /* @__PURE__ */ jsx(CountBadge, { count: addressesCount })
|
|
74
|
+
},
|
|
67
75
|
{
|
|
68
76
|
id: "tasks",
|
|
69
77
|
label: t("customers.people.detail.tabs.tasks", "Tasks"),
|
|
@@ -83,7 +91,7 @@ function PersonDetailTabs({
|
|
|
83
91
|
badge: /* @__PURE__ */ jsx(CountBadge, { count: filesCount })
|
|
84
92
|
}
|
|
85
93
|
],
|
|
86
|
-
[t, activitiesCount, dealsCount, companiesCount, tasksCount, filesCount]
|
|
94
|
+
[t, activitiesCount, dealsCount, companiesCount, addressesCount, tasksCount, filesCount]
|
|
87
95
|
);
|
|
88
96
|
const allTabs = React.useMemo(
|
|
89
97
|
() => [
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../../src/modules/customers/components/detail/PersonDetailTabs.tsx"],
|
|
4
|
-
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n SquareCheckBig,\n Mail,\n Briefcase,\n Building2,\n Check,\n History,\n Paperclip,\n Plus,\n} from 'lucide-react'\nimport type { SectionAction } from '@open-mercato/ui/backend/detail'\n\nexport type PersonTabId =\n | 'activities'\n | 'emails'\n | 'deals'\n | 'companies'\n | 'tasks'\n | 'changelog'\n | 'files'\n | string\n\ntype TabDef = {\n id: PersonTabId\n label: string\n icon?: React.ReactNode\n badge?: React.ReactNode\n}\n\ntype PersonDetailTabsProps = {\n activeTab: PersonTabId\n onTabChange: (tab: PersonTabId) => void\n injectedTabs?: Array<{ id: string; label: string }>\n activitiesCount?: number\n dealsCount?: number\n companiesCount?: number\n tasksCount?: number\n filesCount?: number\n sectionAction?: SectionAction | null\n children: React.ReactNode\n}\n\nconst SUPPORTED_TAB_IDS = new Set<PersonTabId>(['activities', 'emails', 'deals', 'companies', 'tasks', 'changelog', 'files'])\n\nexport function resolveLegacyTab(tab: string | null | undefined): PersonTabId {\n if (!tab) return 'activities'\n return SUPPORTED_TAB_IDS.has(tab as PersonTabId) ? (tab as PersonTabId) : 'activities'\n}\n\nfunction CountBadge({ count }: { count: number }) {\n if (count <= 0) return null\n return (\n <span className=\"ml-1 rounded-full bg-muted px-1.5 py-0.5 text-xs font-medium leading-none text-muted-foreground\">\n {count > 999 ? '999+' : count}\n </span>\n )\n}\n\nfunction NewBadge() {\n return (\n <span className=\"ml-1.5 rounded bg-foreground px-1.5 py-0.5 text-overline font-semibold leading-none text-background\">\n NEW\n </span>\n )\n}\n\nexport function PersonDetailTabs({\n activeTab,\n onTabChange,\n injectedTabs = [],\n activitiesCount = 0,\n dealsCount = 0,\n companiesCount = 0,\n tasksCount = 0,\n filesCount = 0,\n sectionAction = null,\n children,\n}: PersonDetailTabsProps) {\n const t = useT()\n\n const builtInTabs: TabDef[] = React.useMemo(\n () => [\n {\n id: 'activities',\n label: t('customers.people.detail.tabs.activities', 'Activities'),\n icon: <SquareCheckBig className=\"size-4\" />,\n badge: <CountBadge count={activitiesCount} />,\n },\n {\n id: 'emails',\n label: t('customers.people.detail.tabs.emails', 'Emails'),\n icon: <Mail className=\"size-4\" />,\n },\n {\n id: 'deals',\n label: t('customers.people.detail.tabs.deals', 'Deals'),\n icon: <Briefcase className=\"size-4\" />,\n badge: <CountBadge count={dealsCount} />,\n },\n {\n id: 'companies',\n label: t('customers.people.detail.tabs.companies', 'Companies'),\n icon: <Building2 className=\"size-4\" />,\n badge: <CountBadge count={companiesCount} />,\n },\n {\n id: 'tasks',\n label: t('customers.people.detail.tabs.tasks', 'Tasks'),\n icon: <Check className=\"size-4\" />,\n badge: <CountBadge count={tasksCount} />,\n },\n {\n id: 'changelog',\n label: t('customers.people.detail.tabs.changelog', 'Change log'),\n icon: <History className=\"size-4\" />,\n badge: <NewBadge />,\n },\n {\n id: 'files',\n label: t('customers.people.detail.tabs.files', 'Files'),\n icon: <Paperclip className=\"size-4\" />,\n badge: <CountBadge count={filesCount} />,\n },\n ],\n [t, activitiesCount, dealsCount, companiesCount, tasksCount, filesCount],\n )\n\n const allTabs: TabDef[] = React.useMemo(\n () => [\n ...builtInTabs,\n ...injectedTabs.map((tab) => ({\n id: tab.id as PersonTabId,\n label: tab.label,\n })),\n ],\n [builtInTabs, injectedTabs],\n )\n\n return (\n <div>\n {/* Tab navigation \u2014 full width above both zones */}\n <div className=\"flex items-end justify-between gap-2 border-b\" role=\"tablist\" aria-label={t('customers.people.detail.tabs.label', 'Person detail sections')}>\n <nav className=\"-mb-px flex flex-1 gap-1 overflow-x-auto px-1\">\n {allTabs.map((tab) => {\n const isActive = activeTab === tab.id\n return (\n <Button\n key={tab.id}\n type=\"button\"\n variant=\"ghost\"\n size=\"sm\"\n role=\"tab\"\n aria-selected={isActive}\n onClick={() => onTabChange(tab.id)}\n className={cn(\n 'h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent',\n isActive\n ? 'border-foreground text-foreground font-semibold'\n : 'border-transparent text-muted-foreground hover:text-foreground',\n )}\n >\n {tab.icon && <span className=\"mr-1.5\">{tab.icon}</span>}\n {tab.label}\n {tab.badge}\n </Button>\n )\n })}\n </nav>\n {sectionAction ? (\n <Button\n type=\"button\"\n size=\"sm\"\n onClick={sectionAction.onClick}\n disabled={sectionAction.disabled}\n className=\"mb-1.5 mr-1 shrink-0\"\n >\n <Plus className=\"mr-1.5 h-4 w-4\" />\n {sectionAction.label}\n </Button>\n ) : null}\n </div>\n\n {/* Two-column content below tabs */}\n <div className=\"pt-6\">\n {children}\n </div>\n </div>\n )\n}\n"],
|
|
5
|
-
"mappings": ";
|
|
4
|
+
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n SquareCheckBig,\n Mail,\n Briefcase,\n Building2,\n Check,\n History,\n Paperclip,\n Plus,\n MapPin,\n} from 'lucide-react'\nimport type { SectionAction } from '@open-mercato/ui/backend/detail'\n\nexport type PersonTabId =\n | 'activities'\n | 'emails'\n | 'deals'\n | 'companies'\n | 'addresses'\n | 'tasks'\n | 'changelog'\n | 'files'\n | string\n\ntype TabDef = {\n id: PersonTabId\n label: string\n icon?: React.ReactNode\n badge?: React.ReactNode\n}\n\ntype PersonDetailTabsProps = {\n activeTab: PersonTabId\n onTabChange: (tab: PersonTabId) => void\n injectedTabs?: Array<{ id: string; label: string }>\n activitiesCount?: number\n dealsCount?: number\n companiesCount?: number\n addressesCount?: number\n tasksCount?: number\n filesCount?: number\n sectionAction?: SectionAction | null\n children: React.ReactNode\n}\n\nconst SUPPORTED_TAB_IDS = new Set<PersonTabId>(['activities', 'emails', 'deals', 'companies', 'addresses', 'tasks', 'changelog', 'files'])\n\nexport function resolveLegacyTab(tab: string | null | undefined): PersonTabId {\n if (!tab) return 'activities'\n return SUPPORTED_TAB_IDS.has(tab as PersonTabId) ? (tab as PersonTabId) : 'activities'\n}\n\nfunction CountBadge({ count }: { count: number }) {\n if (count <= 0) return null\n return (\n <span className=\"ml-1 rounded-full bg-muted px-1.5 py-0.5 text-xs font-medium leading-none text-muted-foreground\">\n {count > 999 ? '999+' : count}\n </span>\n )\n}\n\nfunction NewBadge() {\n return (\n <span className=\"ml-1.5 rounded bg-foreground px-1.5 py-0.5 text-overline font-semibold leading-none text-background\">\n NEW\n </span>\n )\n}\n\nexport function PersonDetailTabs({\n activeTab,\n onTabChange,\n injectedTabs = [],\n activitiesCount = 0,\n dealsCount = 0,\n companiesCount = 0,\n addressesCount = 0,\n tasksCount = 0,\n filesCount = 0,\n sectionAction = null,\n children,\n}: PersonDetailTabsProps) {\n const t = useT()\n\n const builtInTabs: TabDef[] = React.useMemo(\n () => [\n {\n id: 'activities',\n label: t('customers.people.detail.tabs.activities', 'Activities'),\n icon: <SquareCheckBig className=\"size-4\" />,\n badge: <CountBadge count={activitiesCount} />,\n },\n {\n id: 'emails',\n label: t('customers.people.detail.tabs.emails', 'Emails'),\n icon: <Mail className=\"size-4\" />,\n },\n {\n id: 'deals',\n label: t('customers.people.detail.tabs.deals', 'Deals'),\n icon: <Briefcase className=\"size-4\" />,\n badge: <CountBadge count={dealsCount} />,\n },\n {\n id: 'companies',\n label: t('customers.people.detail.tabs.companies', 'Companies'),\n icon: <Building2 className=\"size-4\" />,\n badge: <CountBadge count={companiesCount} />,\n },\n {\n id: 'addresses',\n label: t('customers.people.detail.tabs.addresses', 'Addresses'),\n icon: <MapPin className=\"size-4\" />,\n badge: <CountBadge count={addressesCount} />,\n },\n {\n id: 'tasks',\n label: t('customers.people.detail.tabs.tasks', 'Tasks'),\n icon: <Check className=\"size-4\" />,\n badge: <CountBadge count={tasksCount} />,\n },\n {\n id: 'changelog',\n label: t('customers.people.detail.tabs.changelog', 'Change log'),\n icon: <History className=\"size-4\" />,\n badge: <NewBadge />,\n },\n {\n id: 'files',\n label: t('customers.people.detail.tabs.files', 'Files'),\n icon: <Paperclip className=\"size-4\" />,\n badge: <CountBadge count={filesCount} />,\n },\n ],\n [t, activitiesCount, dealsCount, companiesCount, addressesCount, tasksCount, filesCount],\n )\n\n const allTabs: TabDef[] = React.useMemo(\n () => [\n ...builtInTabs,\n ...injectedTabs.map((tab) => ({\n id: tab.id as PersonTabId,\n label: tab.label,\n })),\n ],\n [builtInTabs, injectedTabs],\n )\n\n return (\n <div>\n {/* Tab navigation \u2014 full width above both zones */}\n <div className=\"flex items-end justify-between gap-2 border-b\" role=\"tablist\" aria-label={t('customers.people.detail.tabs.label', 'Person detail sections')}>\n <nav className=\"-mb-px flex flex-1 gap-1 overflow-x-auto px-1\">\n {allTabs.map((tab) => {\n const isActive = activeTab === tab.id\n return (\n <Button\n key={tab.id}\n type=\"button\"\n variant=\"ghost\"\n size=\"sm\"\n role=\"tab\"\n aria-selected={isActive}\n onClick={() => onTabChange(tab.id)}\n className={cn(\n 'h-auto shrink-0 rounded-none border-b-2 px-3 py-2.5 hover:bg-transparent',\n isActive\n ? 'border-foreground text-foreground font-semibold'\n : 'border-transparent text-muted-foreground hover:text-foreground',\n )}\n >\n {tab.icon && <span className=\"mr-1.5\">{tab.icon}</span>}\n {tab.label}\n {tab.badge}\n </Button>\n )\n })}\n </nav>\n {sectionAction ? (\n <Button\n type=\"button\"\n size=\"sm\"\n onClick={sectionAction.onClick}\n disabled={sectionAction.disabled}\n className=\"mb-1.5 mr-1 shrink-0\"\n >\n <Plus className=\"mr-1.5 h-4 w-4\" />\n {sectionAction.label}\n </Button>\n ) : null}\n </div>\n\n {/* Two-column content below tabs */}\n <div className=\"pt-6\">\n {children}\n </div>\n </div>\n )\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
"use client";
|
|
2
2
|
import { jsx, jsxs } from "react/jsx-runtime";
|
|
3
3
|
import * as React from "react";
|
|
4
|
-
import { Users, Phone, Check, Mail, Calendar,
|
|
4
|
+
import { Users, Phone, Check, Mail, Calendar, X, StickyNote } from "lucide-react";
|
|
5
5
|
import { cn } from "@open-mercato/shared/lib/utils";
|
|
6
6
|
import { useT } from "@open-mercato/shared/lib/i18n/context";
|
|
7
7
|
import { validatePhoneNumber } from "@open-mercato/shared/lib/phone";
|
|
@@ -431,7 +431,6 @@ function ScheduleActivityDialog({
|
|
|
431
431
|
] }),
|
|
432
432
|
/* @__PURE__ */ jsx("div", { className: "flex-1 overflow-y-auto", children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-4 bg-background p-6", children: [
|
|
433
433
|
state.conflict && /* @__PURE__ */ jsxs(Alert, { variant: "warning", className: "rounded-lg", children: [
|
|
434
|
-
/* @__PURE__ */ jsx(AlertTriangle, { className: "size-5" }),
|
|
435
434
|
/* @__PURE__ */ jsx(AlertTitle, { children: t("customers.schedule.conflict.title", "Calendar conflict") }),
|
|
436
435
|
/* @__PURE__ */ jsx(AlertDescription, { children: state.conflict })
|
|
437
436
|
] }),
|