@open-mercato/core 0.6.5-develop.5116.1.f0af9e5080 → 0.6.5-develop.5155.1.148d10a46d

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (29) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/dist/modules/communication_channels/lib/email-mime.js +18 -1
  3. package/dist/modules/communication_channels/lib/email-mime.js.map +2 -2
  4. package/dist/modules/customers/components/formConfig.js +4 -1
  5. package/dist/modules/customers/components/formConfig.js.map +2 -2
  6. package/dist/modules/customers/data/validators.js +5 -1
  7. package/dist/modules/customers/data/validators.js.map +2 -2
  8. package/dist/modules/sales/commands/configuration.js +2 -2
  9. package/dist/modules/sales/commands/configuration.js.map +2 -2
  10. package/dist/modules/sales/commands/documents.js +17 -13
  11. package/dist/modules/sales/commands/documents.js.map +2 -2
  12. package/dist/modules/sales/commands/payments.js +6 -6
  13. package/dist/modules/sales/commands/payments.js.map +2 -2
  14. package/dist/modules/sales/commands/shipments.js +6 -6
  15. package/dist/modules/sales/commands/shipments.js.map +2 -2
  16. package/dist/modules/sales/lib/dictionaries.js +2 -2
  17. package/dist/modules/sales/lib/dictionaries.js.map +2 -2
  18. package/dist/modules/staff/components/TeamMemberForm.js +5 -2
  19. package/dist/modules/staff/components/TeamMemberForm.js.map +2 -2
  20. package/package.json +14 -14
  21. package/src/modules/communication_channels/lib/email-mime.ts +33 -6
  22. package/src/modules/customers/components/formConfig.tsx +16 -1
  23. package/src/modules/customers/data/validators.ts +8 -1
  24. package/src/modules/sales/commands/configuration.ts +2 -2
  25. package/src/modules/sales/commands/documents.ts +13 -9
  26. package/src/modules/sales/commands/payments.ts +6 -6
  27. package/src/modules/sales/commands/shipments.ts +6 -6
  28. package/src/modules/sales/lib/dictionaries.ts +3 -2
  29. package/src/modules/staff/components/TeamMemberForm.tsx +6 -1
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/staff/components/TeamMemberForm.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { useRouter } from 'next/navigation'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { normalizeCustomFieldValues } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { LookupSelect, type LookupSelectItem } from '@open-mercato/ui/backend/inputs'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n SelectValue,\n} from '@open-mercato/ui/primitives/select'\nimport { AttachmentsSection, TagsSection, type TagOption, type TagsSectionLabels } from '@open-mercato/ui/backend/detail'\nimport { E } from '#generated/entities.ids.generated'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { Plus } from 'lucide-react'\n\nconst DEFAULT_PAGE_SIZE = 100\n\nexport type TeamMemberFormValues = {\n id?: string\n teamId?: string | null\n userId?: string | null\n displayName?: string | null\n description?: string | null\n roleIds?: string[]\n tags?: string[]\n isActive?: boolean\n updatedAt?: string | null\n} & Record<string, unknown>\n\nexport type TeamMemberFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n embedded?: boolean\n initialValues: TeamMemberFormValues\n onSubmit: (values: TeamMemberFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n tagsSection?: TeamMemberTagsSectionConfig\n}\n\ntype TeamRoleRow = {\n id: string\n name: string\n teamId: string | null\n}\n\ntype TeamRolesResponse = {\n items?: Array<Record<string, unknown>>\n}\n\ntype UsersResponse = {\n items?: Array<{ id?: string; email?: string; organizationName?: string | null }>\n}\n\ntype TeamsResponse = {\n items?: Array<{ id?: string; name?: string }>\n}\n\ntype TeamMemberTagsSectionConfig = {\n title: string\n tags: TagOption[]\n onChange: (next: TagOption[]) => void\n loadOptions: (query?: string) => Promise<TagOption[]>\n createTag: (label: string) => Promise<TagOption>\n onSave: (payload: { next: TagOption[]; added: TagOption[]; removed: TagOption[] }) => Promise<void>\n labels: TagsSectionLabels\n}\n\nconst normalizeCustomFieldSubmitValue = (value: unknown): unknown => {\n const normalized = normalizeCustomFieldValues({ value })\n return normalized.value\n}\n\nexport const buildTeamMemberPayload = (\n values: TeamMemberFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> => {\n const roleIds = Array.isArray(values.roleIds)\n ? values.roleIds.filter((item): item is string => typeof item === 'string')\n : []\n const tags = Array.isArray(values.tags)\n ? values.tags.filter((item): item is string => typeof item === 'string' && item.trim().length > 0)\n : null\n const customFields = collectCustomFieldValues(values, { transform: normalizeCustomFieldSubmitValue })\n return {\n ...(options.id ? { id: options.id } : {}),\n teamId: values.teamId ? String(values.teamId) : null,\n userId: values.userId ? String(values.userId) : null,\n displayName: typeof values.displayName === 'string' ? values.displayName : '',\n description: typeof values.description === 'string' && values.description.trim().length ? values.description : null,\n roleIds,\n ...(tags ? { tags } : {}),\n isActive: values.isActive ?? true,\n ...(Object.keys(customFields).length ? { customFields } : {}),\n }\n}\n\nexport function TeamMemberForm(props: TeamMemberFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n embedded = false,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n tagsSection,\n } = props\n const translate = useT()\n const recordId = typeof initialValues?.id === 'string' ? initialValues.id : null\n const router = useRouter()\n const scopeVersion = useOrganizationScopeVersion()\n const [roles, setRoles] = React.useState<TeamRoleRow[]>([])\n const [selectedTeamId, setSelectedTeamId] = React.useState<string | null>(null)\n const [userOptions, setUserOptions] = React.useState<LookupSelectItem[]>([])\n const [teamOptions, setTeamOptions] = React.useState<Array<{ value: string; label: string }>>([])\n const [roleSearch, setRoleSearch] = React.useState('')\n\n const resolvedTeamId = typeof initialValues.teamId === 'string' && initialValues.teamId.trim().length\n ? initialValues.teamId\n : null\n const resolvedUserId = typeof initialValues.userId === 'string' && initialValues.userId.trim().length\n ? initialValues.userId\n : null\n\n React.useEffect(() => {\n setSelectedTeamId(resolvedTeamId)\n }, [resolvedTeamId])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadRoles() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: String(DEFAULT_PAGE_SIZE) })\n const call = await apiCall<TeamRolesResponse>(`/api/staff/team-roles?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const nextRoles = items\n .map(mapTeamRole)\n .filter((role): role is TeamRoleRow => role !== null)\n if (!cancelled) setRoles(nextRoles)\n } catch {\n if (!cancelled) setRoles([])\n }\n }\n loadRoles()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadTeams() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: '100' })\n const call = await apiCall<TeamsResponse>(`/api/staff/teams?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((team) => {\n const id = typeof team.id === 'string' ? team.id : null\n const name = typeof team.name === 'string' ? team.name : null\n if (!id || !name) return null\n return { value: id, label: name }\n })\n .filter((entry): entry is { value: string; label: string } => entry !== null)\n if (!cancelled) setTeamOptions(options)\n } catch {\n if (!cancelled) setTeamOptions([])\n }\n }\n loadTeams()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n if (!resolvedTeamId) return\n if (teamOptions.some((option) => option.value === resolvedTeamId)) return\n const selectedTeamId = resolvedTeamId\n let cancelled = false\n async function loadSelectedTeam() {\n try {\n const call = await apiCall<TeamsResponse>(`/api/staff/teams?ids=${encodeURIComponent(selectedTeamId)}&pageSize=1`)\n const entry = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const entryId = typeof entry?.id === 'string' ? entry.id : null\n const entryName = typeof entry?.name === 'string' ? entry.name : null\n if (!entryId || !entryName) return\n if (!cancelled) {\n setTeamOptions((prev) => {\n if (prev.some((option) => option.value === entryId)) return prev\n return [{ value: entryId, label: entryName }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setTeamOptions((prev) => prev)\n }\n }\n loadSelectedTeam()\n return () => { cancelled = true }\n }, [resolvedTeamId, teamOptions])\n\n React.useEffect(() => {\n if (!resolvedUserId) return\n const userId = resolvedUserId\n if (userOptions.some((option) => option.id === resolvedUserId)) return\n let cancelled = false\n async function loadSelectedUser() {\n try {\n const call = await apiCall<UsersResponse>(`/api/auth/users?id=${encodeURIComponent(userId)}`)\n const entry = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const entryId = typeof entry?.id === 'string' ? entry.id : null\n const entryEmail = typeof entry?.email === 'string' ? entry.email : null\n if (!entryId || !entryEmail) return\n if (!cancelled) {\n setUserOptions((prev) => {\n if (prev.some((option) => option.id === entryId)) return prev\n return [{ id: entryId, title: entryEmail, subtitle: entry?.organizationName ?? null }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setUserOptions((prev) => prev)\n }\n }\n loadSelectedUser()\n return () => { cancelled = true }\n }, [resolvedUserId, userOptions])\n\n const filteredRoles = React.useMemo(\n () => filterRolesByTeam(roles, selectedTeamId),\n [roles, selectedTeamId],\n )\n const roleOptions = React.useMemo(\n () => filteredRoles.map((role) => ({ value: role.id, label: role.name })),\n [filteredRoles],\n )\n const filteredRoleOptions = React.useMemo(() => {\n const query = roleSearch.trim().toLowerCase()\n if (!query) return roleOptions\n return roleOptions.filter((option) => (\n option.label.toLowerCase().includes(query) || option.value.toLowerCase().includes(query)\n ))\n }, [roleOptions, roleSearch])\n const createRoleHref = React.useMemo(() => {\n const params = new URLSearchParams()\n if (selectedTeamId) params.set('teamId', selectedTeamId)\n const query = params.toString()\n return `/backend/staff/team-roles/create${query ? `?${query}` : ''}`\n }, [selectedTeamId])\n const createTeamHref = '/backend/staff/teams/create'\n\n React.useEffect(() => {\n setRoleSearch('')\n }, [selectedTeamId])\n\n const fetchUserOptions = React.useCallback(async (query?: string): Promise<LookupSelectItem[]> => {\n const params = new URLSearchParams({ page: '1', pageSize: '50' })\n if (query && query.trim().length) params.set('search', query.trim())\n const call = await apiCall<UsersResponse>(`/api/auth/users?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((user): LookupSelectItem | null => {\n const id = typeof user?.id === 'string' ? user.id : null\n const email = typeof user?.email === 'string' ? user.email : null\n if (!id || !email) return null\n return {\n id,\n title: email,\n subtitle: user.organizationName ?? null,\n }\n })\n .filter((option): option is LookupSelectItem => option !== null)\n setUserOptions(options)\n return options\n }, [])\n\n const fields = React.useMemo<CrudField[]>(() => {\n const baseFields: CrudField[] = [\n {\n id: 'userId',\n label: translate('staff.teamMembers.form.fields.user', 'User'),\n type: 'custom',\n component: ({ value, setValue }) => (\n <LookupSelect\n value={typeof value === 'string' ? value : null}\n onChange={(next) => setValue(next)}\n options={userOptions}\n fetchOptions={fetchUserOptions}\n placeholder={translate('staff.teamMembers.form.fields.user.placeholder', 'Select a user')}\n searchPlaceholder={translate('staff.teamMembers.form.fields.user.search', 'Search users')}\n emptyLabel={translate('staff.teamMembers.form.fields.user.empty', 'No users found')}\n selectedHintLabel={(id) => translate('staff.teamMembers.form.fields.user.selected', 'Selected user: {{id}}', { id })}\n />\n ),\n },\n {\n id: 'teamId',\n label: '',\n type: 'custom',\n component: ({ value, setValue, setFormValue, values, disabled }) => {\n const currentValue = typeof value === 'string' ? value : ''\n return (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between gap-3\">\n <span className=\"text-sm font-medium\">\n {translate('staff.teamMembers.form.fields.team', 'Team')}\n </span>\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n onClick={() => router.push(createTeamHref)}\n disabled={disabled}\n >\n <Plus className=\"mr-2 h-4 w-4\" aria-hidden />\n {translate('staff.teamMembers.form.actions.createTeam', 'Create new team')}\n </Button>\n </div>\n <Select\n value={currentValue}\n onValueChange={(value) => {\n const nextValue = value || undefined\n const nextTeamId = value || null\n setValue(nextValue)\n setSelectedTeamId(nextTeamId)\n if (!setFormValue) return\n const roleIds = Array.isArray(values?.roleIds)\n ? values?.roleIds.filter((item): item is string => typeof item === 'string')\n : []\n const allowedRoleIds = buildAllowedRoleIdSet(roles, nextTeamId)\n const nextRoleIds = roleIds.filter((roleId) => allowedRoleIds.has(roleId))\n if (nextRoleIds.length !== roleIds.length) {\n setFormValue('roleIds', nextRoleIds)\n }\n }}\n disabled={disabled}\n >\n <SelectTrigger data-crud-focus-target=\"\">\n <SelectValue placeholder={translate('ui.forms.select.emptyOption', '\u2014')} />\n </SelectTrigger>\n <SelectContent>\n {teamOptions.map((option) => (\n <SelectItem key={option.value} value={option.value}>\n {option.label}\n </SelectItem>\n ))}\n </SelectContent>\n </Select>\n </div>\n )\n },\n },\n {\n id: 'displayName',\n label: translate('staff.teamMembers.form.fields.displayName', 'Display name'),\n type: 'text',\n required: true,\n },\n {\n id: 'description',\n label: translate('staff.teamMembers.form.fields.description', 'Description'),\n type: 'richtext',\n editor: 'uiw',\n },\n {\n id: 'roleIds',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => {\n const selectedValues = Array.isArray(value)\n ? value.filter((item): item is string => typeof item === 'string')\n : []\n return (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between gap-3\">\n <span className=\"text-sm font-medium\">\n {translate('staff.teamMembers.form.fields.roles', 'Roles')}\n </span>\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n onClick={() => router.push(createRoleHref)}\n disabled={disabled}\n >\n <Plus className=\"mr-2 h-4 w-4\" aria-hidden />\n {translate('staff.teamMembers.form.actions.defineRole', 'Define new role')}\n </Button>\n </div>\n <input\n className=\"w-full h-8 rounded border px-2 text-sm\"\n placeholder={translate('ui.forms.listbox.searchPlaceholder', 'Search...')}\n value={roleSearch}\n onChange={(event) => setRoleSearch(event.target.value)}\n data-crud-focus-target=\"\"\n disabled={disabled}\n />\n <div className=\"rounded border max-h-48 overflow-auto divide-y\">\n {filteredRoleOptions.map((option) => {\n const isSelected = selectedValues.includes(option.value)\n return (\n <button\n key={option.value}\n type=\"button\"\n onClick={() => {\n const next = new Set(selectedValues)\n if (isSelected) {\n next.delete(option.value)\n } else {\n next.add(option.value)\n }\n setValue(Array.from(next))\n }}\n className={`w-full text-left px-3 py-2 text-sm hover:bg-muted ${isSelected ? 'bg-muted' : ''}`}\n disabled={disabled}\n >\n <span className=\"inline-flex items-center gap-2\">\n <input type=\"checkbox\" className=\"size-4\" readOnly checked={isSelected} />\n <span>{option.label}</span>\n </span>\n </button>\n )\n })}\n {!filteredRoleOptions.length ? (\n <div className=\"px-3 py-2 text-sm text-muted-foreground\">\n {translate('ui.forms.listbox.noMatches', 'No matches')}\n </div>\n ) : null}\n </div>\n </div>\n )\n },\n },\n {\n id: 'isActive',\n label: translate('staff.teamMembers.form.fields.active', 'Active'),\n type: 'checkbox',\n },\n ]\n\n if (!tagsSection) {\n baseFields.splice(5, 0, {\n id: 'tags',\n label: translate('staff.teamMembers.form.fields.tags', 'Tags'),\n type: 'tags',\n placeholder: translate('staff.teamMembers.form.fields.tags.placeholder', 'Add tags'),\n })\n }\n\n return baseFields\n }, [\n createRoleHref,\n fetchUserOptions,\n filteredRoleOptions,\n roleSearch,\n roles,\n router,\n tagsSection,\n translate,\n teamOptions,\n userOptions,\n ])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n const baseGroups: CrudFormGroup[] = [\n {\n id: 'details',\n column: 1,\n fields: ['userId', 'teamId', 'displayName', 'description', 'roleIds', 'isActive'],\n },\n {\n id: 'custom',\n title: translate('entities.customFields.title', 'Custom Attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n\n if (tagsSection) {\n baseGroups.push({\n id: 'tags',\n column: 2,\n bare: true,\n component: () => (\n <TagsSection\n title={tagsSection.title}\n tags={tagsSection.tags}\n onChange={tagsSection.onChange}\n loadOptions={tagsSection.loadOptions}\n createTag={tagsSection.createTag}\n onSave={tagsSection.onSave}\n labels={tagsSection.labels}\n />\n ),\n })\n } else {\n baseGroups.splice(1, 0, {\n id: 'tags',\n title: translate('staff.teamMembers.form.fields.tags', 'Tags'),\n column: 2,\n fields: ['tags'],\n })\n }\n\n baseGroups.push({\n id: 'attachments',\n title: translate('attachments.library.title', 'Attachments'),\n column: 1,\n component: () => (\n <AttachmentsSection\n entityId={E.staff.staff_team_member}\n recordId={recordId}\n showHeader={false}\n />\n ),\n })\n\n return baseGroups\n }, [recordId, tagsSection, translate])\n\n return (\n <CrudForm<TeamMemberFormValues>\n embedded={embedded}\n title={title}\n backHref={backHref}\n cancelHref={cancelHref}\n versionHistory={initialValues.id\n ? { resourceKind: 'staff.teamMember', resourceId: String(initialValues.id) }\n : undefined}\n submitLabel={submitLabel}\n fields={fields}\n groups={groups}\n entityId={E.staff.staff_team_member}\n initialValues={initialValues}\n optimisticLockUpdatedAt={initialValues.updatedAt}\n onSubmit={onSubmit}\n onDelete={onDelete}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n />\n )\n}\n\nfunction mapTeamRole(item: Record<string, unknown>): TeamRoleRow | null {\n const id = typeof item.id === 'string' ? item.id : ''\n if (!id) return null\n const name = typeof item.name === 'string' && item.name.trim().length ? item.name.trim() : id\n const teamId = typeof item.teamId === 'string'\n ? item.teamId\n : typeof item.team_id === 'string'\n ? item.team_id\n : null\n return { id, name, teamId }\n}\n\nfunction filterRolesByTeam(roles: TeamRoleRow[], teamId: string | null): TeamRoleRow[] {\n if (!teamId) return roles.filter((role) => role.teamId == null)\n return roles.filter((role) => role.teamId == null || role.teamId === teamId)\n}\n\nfunction buildAllowedRoleIdSet(roles: TeamRoleRow[], teamId: string | null): Set<string> {\n return new Set(filterRolesByTeam(roles, teamId).map((role) => role.id))\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { useRouter } from 'next/navigation'\nimport { CrudForm, type CrudField, type CrudFormGroup } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { normalizeCustomFieldValues } from '@open-mercato/shared/lib/custom-fields/normalize'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { LookupSelect, type LookupSelectItem } from '@open-mercato/ui/backend/inputs'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport {\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n SelectValue,\n} from '@open-mercato/ui/primitives/select'\nimport { AttachmentsSection, TagsSection, type TagOption, type TagsSectionLabels } from '@open-mercato/ui/backend/detail'\nimport { E } from '#generated/entities.ids.generated'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { Plus } from 'lucide-react'\n\nconst DEFAULT_PAGE_SIZE = 100\n\nexport type TeamMemberFormValues = {\n id?: string\n teamId?: string | null\n userId?: string | null\n displayName?: string | null\n description?: string | null\n roleIds?: string[]\n tags?: string[]\n isActive?: boolean\n updatedAt?: string | null\n} & Record<string, unknown>\n\nexport type TeamMemberFormProps = {\n title: string\n submitLabel?: string\n backHref: string\n cancelHref: string\n embedded?: boolean\n initialValues: TeamMemberFormValues\n onSubmit: (values: TeamMemberFormValues) => Promise<void>\n onDelete?: () => Promise<void>\n isLoading?: boolean\n loadingMessage?: string\n tagsSection?: TeamMemberTagsSectionConfig\n}\n\ntype TeamRoleRow = {\n id: string\n name: string\n teamId: string | null\n}\n\ntype TeamRolesResponse = {\n items?: Array<Record<string, unknown>>\n}\n\ntype UsersResponse = {\n items?: Array<{ id?: string; email?: string; organizationName?: string | null }>\n}\n\ntype TeamsResponse = {\n items?: Array<{ id?: string; name?: string }>\n}\n\ntype TeamMemberTagsSectionConfig = {\n title: string\n tags: TagOption[]\n onChange: (next: TagOption[]) => void\n loadOptions: (query?: string) => Promise<TagOption[]>\n createTag: (label: string) => Promise<TagOption>\n onSave: (payload: { next: TagOption[]; added: TagOption[]; removed: TagOption[] }) => Promise<void>\n labels: TagsSectionLabels\n}\n\nconst normalizeCustomFieldSubmitValue = (value: unknown): unknown => {\n const normalized = normalizeCustomFieldValues({ value })\n return normalized.value\n}\n\nexport const buildTeamMemberPayload = (\n values: TeamMemberFormValues,\n options: { id?: string } = {},\n): Record<string, unknown> => {\n const roleIds = Array.isArray(values.roleIds)\n ? values.roleIds.filter((item): item is string => typeof item === 'string')\n : []\n const tags = Array.isArray(values.tags)\n ? values.tags.filter((item): item is string => typeof item === 'string' && item.trim().length > 0)\n : null\n const customFields = collectCustomFieldValues(values, { transform: normalizeCustomFieldSubmitValue })\n return {\n ...(options.id ? { id: options.id } : {}),\n teamId: values.teamId ? String(values.teamId) : null,\n userId: values.userId ? String(values.userId) : null,\n displayName: typeof values.displayName === 'string' ? values.displayName : '',\n description: typeof values.description === 'string' && values.description.trim().length ? values.description : null,\n roleIds,\n ...(tags ? { tags } : {}),\n isActive: values.isActive ?? true,\n ...(Object.keys(customFields).length ? { customFields } : {}),\n }\n}\n\nexport function TeamMemberForm(props: TeamMemberFormProps) {\n const {\n title,\n submitLabel,\n backHref,\n cancelHref,\n embedded = false,\n initialValues,\n onSubmit,\n onDelete,\n isLoading,\n loadingMessage,\n tagsSection,\n } = props\n const translate = useT()\n const recordId = typeof initialValues?.id === 'string' ? initialValues.id : null\n const router = useRouter()\n const scopeVersion = useOrganizationScopeVersion()\n const [roles, setRoles] = React.useState<TeamRoleRow[]>([])\n const [selectedTeamId, setSelectedTeamId] = React.useState<string | null>(null)\n const [userOptions, setUserOptions] = React.useState<LookupSelectItem[]>([])\n const [teamOptions, setTeamOptions] = React.useState<Array<{ value: string; label: string }>>([])\n const [roleSearch, setRoleSearch] = React.useState('')\n\n const resolvedTeamId = typeof initialValues.teamId === 'string' && initialValues.teamId.trim().length\n ? initialValues.teamId\n : null\n const resolvedUserId = typeof initialValues.userId === 'string' && initialValues.userId.trim().length\n ? initialValues.userId\n : null\n\n React.useEffect(() => {\n setSelectedTeamId(resolvedTeamId)\n }, [resolvedTeamId])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadRoles() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: String(DEFAULT_PAGE_SIZE) })\n const call = await apiCall<TeamRolesResponse>(`/api/staff/team-roles?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const nextRoles = items\n .map(mapTeamRole)\n .filter((role): role is TeamRoleRow => role !== null)\n if (!cancelled) setRoles(nextRoles)\n } catch {\n if (!cancelled) setRoles([])\n }\n }\n loadRoles()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n let cancelled = false\n async function loadTeams() {\n try {\n const params = new URLSearchParams({ page: '1', pageSize: '100' })\n const call = await apiCall<TeamsResponse>(`/api/staff/teams?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((team) => {\n const id = typeof team.id === 'string' ? team.id : null\n const name = typeof team.name === 'string' ? team.name : null\n if (!id || !name) return null\n return { value: id, label: name }\n })\n .filter((entry): entry is { value: string; label: string } => entry !== null)\n if (!cancelled) setTeamOptions(options)\n } catch {\n if (!cancelled) setTeamOptions([])\n }\n }\n loadTeams()\n return () => { cancelled = true }\n }, [scopeVersion])\n\n React.useEffect(() => {\n if (!resolvedTeamId) return\n if (teamOptions.some((option) => option.value === resolvedTeamId)) return\n const selectedTeamId = resolvedTeamId\n let cancelled = false\n async function loadSelectedTeam() {\n try {\n const call = await apiCall<TeamsResponse>(`/api/staff/teams?ids=${encodeURIComponent(selectedTeamId)}&pageSize=1`)\n const entry = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const entryId = typeof entry?.id === 'string' ? entry.id : null\n const entryName = typeof entry?.name === 'string' ? entry.name : null\n if (!entryId || !entryName) return\n if (!cancelled) {\n setTeamOptions((prev) => {\n if (prev.some((option) => option.value === entryId)) return prev\n return [{ value: entryId, label: entryName }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setTeamOptions((prev) => prev)\n }\n }\n loadSelectedTeam()\n return () => { cancelled = true }\n }, [resolvedTeamId, teamOptions])\n\n React.useEffect(() => {\n if (!resolvedUserId) return\n const userId = resolvedUserId\n if (userOptions.some((option) => option.id === resolvedUserId)) return\n let cancelled = false\n async function loadSelectedUser() {\n try {\n const call = await apiCall<UsersResponse>(`/api/auth/users?id=${encodeURIComponent(userId)}`)\n const entry = Array.isArray(call.result?.items) ? call.result.items[0] : null\n const entryId = typeof entry?.id === 'string' ? entry.id : null\n const entryEmail = typeof entry?.email === 'string' ? entry.email : null\n if (!entryId || !entryEmail) return\n if (!cancelled) {\n setUserOptions((prev) => {\n if (prev.some((option) => option.id === entryId)) return prev\n return [{ id: entryId, title: entryEmail, subtitle: entry?.organizationName ?? null }, ...prev]\n })\n }\n } catch {\n if (!cancelled) setUserOptions((prev) => prev)\n }\n }\n loadSelectedUser()\n return () => { cancelled = true }\n }, [resolvedUserId, userOptions])\n\n const filteredRoles = React.useMemo(\n () => filterRolesByTeam(roles, selectedTeamId),\n [roles, selectedTeamId],\n )\n const roleOptions = React.useMemo(\n () => filteredRoles.map((role) => ({ value: role.id, label: role.name })),\n [filteredRoles],\n )\n const filteredRoleOptions = React.useMemo(() => {\n const query = roleSearch.trim().toLowerCase()\n if (!query) return roleOptions\n return roleOptions.filter((option) => (\n option.label.toLowerCase().includes(query) || option.value.toLowerCase().includes(query)\n ))\n }, [roleOptions, roleSearch])\n const createRoleHref = React.useMemo(() => {\n const params = new URLSearchParams()\n if (selectedTeamId) params.set('teamId', selectedTeamId)\n const query = params.toString()\n return `/backend/staff/team-roles/create${query ? `?${query}` : ''}`\n }, [selectedTeamId])\n const createTeamHref = '/backend/staff/teams/create'\n\n React.useEffect(() => {\n setRoleSearch('')\n }, [selectedTeamId])\n\n const fetchUserOptions = React.useCallback(async (query?: string): Promise<LookupSelectItem[]> => {\n const params = new URLSearchParams({ page: '1', pageSize: '50' })\n if (query && query.trim().length) params.set('search', query.trim())\n const call = await apiCall<UsersResponse>(`/api/auth/users?${params.toString()}`)\n const items = Array.isArray(call.result?.items) ? call.result.items : []\n const options = items\n .map((user): LookupSelectItem | null => {\n const id = typeof user?.id === 'string' ? user.id : null\n const email = typeof user?.email === 'string' ? user.email : null\n if (!id || !email) return null\n return {\n id,\n title: email,\n subtitle: user.organizationName ?? null,\n }\n })\n .filter((option): option is LookupSelectItem => option !== null)\n setUserOptions(options)\n return options\n }, [])\n\n const fields = React.useMemo<CrudField[]>(() => {\n const baseFields: CrudField[] = [\n {\n id: 'userId',\n label: translate('staff.teamMembers.form.fields.user', 'User'),\n type: 'custom',\n component: ({ value, setValue }) => (\n <LookupSelect\n value={typeof value === 'string' ? value : null}\n onChange={(next) => setValue(next)}\n options={userOptions}\n fetchOptions={fetchUserOptions}\n placeholder={translate('staff.teamMembers.form.fields.user.placeholder', 'Select a user')}\n searchPlaceholder={translate('staff.teamMembers.form.fields.user.search', 'Search users')}\n emptyLabel={translate('staff.teamMembers.form.fields.user.empty', 'No users found')}\n selectedHintLabel={(id) => translate('staff.teamMembers.form.fields.user.selected', 'Selected user: {{id}}', { id })}\n />\n ),\n },\n {\n id: 'teamId',\n label: '',\n type: 'custom',\n component: ({ value, setValue, setFormValue, values, disabled }) => {\n const currentValue = typeof value === 'string' ? value : ''\n const selectedOption = teamOptions.find((option) => option.value === currentValue)\n const optionsKey = teamOptions.map((option) => `${option.value}:${option.label}`).join('\\0')\n return (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between gap-3\">\n <span className=\"text-sm font-medium\">\n {translate('staff.teamMembers.form.fields.team', 'Team')}\n </span>\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n onClick={() => router.push(createTeamHref)}\n disabled={disabled}\n >\n <Plus className=\"mr-2 h-4 w-4\" aria-hidden />\n {translate('staff.teamMembers.form.actions.createTeam', 'Create new team')}\n </Button>\n </div>\n <Select\n key={`team:${currentValue}:${optionsKey}`}\n value={currentValue}\n onValueChange={(value) => {\n const nextValue = value || undefined\n const nextTeamId = value || null\n setValue(nextValue)\n setSelectedTeamId(nextTeamId)\n if (!setFormValue) return\n const roleIds = Array.isArray(values?.roleIds)\n ? values?.roleIds.filter((item): item is string => typeof item === 'string')\n : []\n const allowedRoleIds = buildAllowedRoleIdSet(roles, nextTeamId)\n const nextRoleIds = roleIds.filter((roleId) => allowedRoleIds.has(roleId))\n if (nextRoleIds.length !== roleIds.length) {\n setFormValue('roleIds', nextRoleIds)\n }\n }}\n disabled={disabled}\n >\n <SelectTrigger data-crud-focus-target=\"\">\n <SelectValue placeholder={translate('ui.forms.select.emptyOption', '\u2014')}>\n {selectedOption?.label}\n </SelectValue>\n </SelectTrigger>\n <SelectContent>\n {teamOptions.map((option) => (\n <SelectItem key={option.value} value={option.value}>\n {option.label}\n </SelectItem>\n ))}\n </SelectContent>\n </Select>\n </div>\n )\n },\n },\n {\n id: 'displayName',\n label: translate('staff.teamMembers.form.fields.displayName', 'Display name'),\n type: 'text',\n required: true,\n },\n {\n id: 'description',\n label: translate('staff.teamMembers.form.fields.description', 'Description'),\n type: 'richtext',\n editor: 'uiw',\n },\n {\n id: 'roleIds',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => {\n const selectedValues = Array.isArray(value)\n ? value.filter((item): item is string => typeof item === 'string')\n : []\n return (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between gap-3\">\n <span className=\"text-sm font-medium\">\n {translate('staff.teamMembers.form.fields.roles', 'Roles')}\n </span>\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n onClick={() => router.push(createRoleHref)}\n disabled={disabled}\n >\n <Plus className=\"mr-2 h-4 w-4\" aria-hidden />\n {translate('staff.teamMembers.form.actions.defineRole', 'Define new role')}\n </Button>\n </div>\n <input\n className=\"w-full h-8 rounded border px-2 text-sm\"\n placeholder={translate('ui.forms.listbox.searchPlaceholder', 'Search...')}\n value={roleSearch}\n onChange={(event) => setRoleSearch(event.target.value)}\n data-crud-focus-target=\"\"\n disabled={disabled}\n />\n <div className=\"rounded border max-h-48 overflow-auto divide-y\">\n {filteredRoleOptions.map((option) => {\n const isSelected = selectedValues.includes(option.value)\n return (\n <button\n key={option.value}\n type=\"button\"\n onClick={() => {\n const next = new Set(selectedValues)\n if (isSelected) {\n next.delete(option.value)\n } else {\n next.add(option.value)\n }\n setValue(Array.from(next))\n }}\n className={`w-full text-left px-3 py-2 text-sm hover:bg-muted ${isSelected ? 'bg-muted' : ''}`}\n disabled={disabled}\n >\n <span className=\"inline-flex items-center gap-2\">\n <input type=\"checkbox\" className=\"size-4\" readOnly checked={isSelected} />\n <span>{option.label}</span>\n </span>\n </button>\n )\n })}\n {!filteredRoleOptions.length ? (\n <div className=\"px-3 py-2 text-sm text-muted-foreground\">\n {translate('ui.forms.listbox.noMatches', 'No matches')}\n </div>\n ) : null}\n </div>\n </div>\n )\n },\n },\n {\n id: 'isActive',\n label: translate('staff.teamMembers.form.fields.active', 'Active'),\n type: 'checkbox',\n },\n ]\n\n if (!tagsSection) {\n baseFields.splice(5, 0, {\n id: 'tags',\n label: translate('staff.teamMembers.form.fields.tags', 'Tags'),\n type: 'tags',\n placeholder: translate('staff.teamMembers.form.fields.tags.placeholder', 'Add tags'),\n })\n }\n\n return baseFields\n }, [\n createRoleHref,\n fetchUserOptions,\n filteredRoleOptions,\n roleSearch,\n roles,\n router,\n tagsSection,\n translate,\n teamOptions,\n userOptions,\n ])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n const baseGroups: CrudFormGroup[] = [\n {\n id: 'details',\n column: 1,\n fields: ['userId', 'teamId', 'displayName', 'description', 'roleIds', 'isActive'],\n },\n {\n id: 'custom',\n title: translate('entities.customFields.title', 'Custom Attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n\n if (tagsSection) {\n baseGroups.push({\n id: 'tags',\n column: 2,\n bare: true,\n component: () => (\n <TagsSection\n title={tagsSection.title}\n tags={tagsSection.tags}\n onChange={tagsSection.onChange}\n loadOptions={tagsSection.loadOptions}\n createTag={tagsSection.createTag}\n onSave={tagsSection.onSave}\n labels={tagsSection.labels}\n />\n ),\n })\n } else {\n baseGroups.splice(1, 0, {\n id: 'tags',\n title: translate('staff.teamMembers.form.fields.tags', 'Tags'),\n column: 2,\n fields: ['tags'],\n })\n }\n\n baseGroups.push({\n id: 'attachments',\n title: translate('attachments.library.title', 'Attachments'),\n column: 1,\n component: () => (\n <AttachmentsSection\n entityId={E.staff.staff_team_member}\n recordId={recordId}\n showHeader={false}\n />\n ),\n })\n\n return baseGroups\n }, [recordId, tagsSection, translate])\n\n return (\n <CrudForm<TeamMemberFormValues>\n embedded={embedded}\n title={title}\n backHref={backHref}\n cancelHref={cancelHref}\n versionHistory={initialValues.id\n ? { resourceKind: 'staff.teamMember', resourceId: String(initialValues.id) }\n : undefined}\n submitLabel={submitLabel}\n fields={fields}\n groups={groups}\n entityId={E.staff.staff_team_member}\n initialValues={initialValues}\n optimisticLockUpdatedAt={initialValues.updatedAt}\n onSubmit={onSubmit}\n onDelete={onDelete}\n isLoading={isLoading}\n loadingMessage={loadingMessage}\n />\n )\n}\n\nfunction mapTeamRole(item: Record<string, unknown>): TeamRoleRow | null {\n const id = typeof item.id === 'string' ? item.id : ''\n if (!id) return null\n const name = typeof item.name === 'string' && item.name.trim().length ? item.name.trim() : id\n const teamId = typeof item.teamId === 'string'\n ? item.teamId\n : typeof item.team_id === 'string'\n ? item.team_id\n : null\n return { id, name, teamId }\n}\n\nfunction filterRolesByTeam(roles: TeamRoleRow[], teamId: string | null): TeamRoleRow[] {\n if (!teamId) return roles.filter((role) => role.teamId == null)\n return roles.filter((role) => role.teamId == null || role.teamId === teamId)\n}\n\nfunction buildAllowedRoleIdSet(roles: TeamRoleRow[], teamId: string | null): Set<string> {\n return new Set(filterRolesByTeam(roles, teamId).map((role) => role.id))\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["selectedTeamId", "value"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@open-mercato/core",
3
- "version": "0.6.5-develop.5116.1.f0af9e5080",
3
+ "version": "0.6.5-develop.5155.1.148d10a46d",
4
4
  "type": "module",
5
5
  "main": "./dist/index.js",
6
6
  "scripts": {
@@ -227,41 +227,41 @@
227
227
  }
228
228
  },
229
229
  "dependencies": {
230
- "@mikro-orm/core": "^7.1.3",
231
- "@mikro-orm/decorators": "^7.1.3",
232
- "@mikro-orm/postgresql": "^7.1.3",
230
+ "@mikro-orm/core": "^7.1.4",
231
+ "@mikro-orm/decorators": "^7.1.4",
232
+ "@mikro-orm/postgresql": "^7.1.4",
233
233
  "@xyflow/react": "^12.11.0",
234
- "ai": "^6.0.194",
234
+ "ai": "^6.0.197",
235
235
  "date-fns": "4.4.0",
236
236
  "date-fns-tz": "^3.2.0",
237
237
  "html-to-text": "^10.0.0",
238
238
  "mammoth": "^1.9.0",
239
239
  "pdfjs-dist": "^6.0.227",
240
- "resend": "^6.12.3",
240
+ "resend": "^6.12.4",
241
241
  "sanitize-html": "^2.13.0",
242
- "semver": "^7.8.1",
242
+ "semver": "^7.8.2",
243
243
  "svix": "^1.95.1",
244
244
  "ts-pattern": "^5.0.0",
245
245
  "zod": "^4.4.3"
246
246
  },
247
247
  "peerDependencies": {
248
- "@open-mercato/ai-assistant": "0.6.5-develop.5116.1.f0af9e5080",
249
- "@open-mercato/shared": "0.6.5-develop.5116.1.f0af9e5080",
250
- "@open-mercato/ui": "0.6.5-develop.5116.1.f0af9e5080",
248
+ "@open-mercato/ai-assistant": "0.6.5-develop.5155.1.148d10a46d",
249
+ "@open-mercato/shared": "0.6.5-develop.5155.1.148d10a46d",
250
+ "@open-mercato/ui": "0.6.5-develop.5155.1.148d10a46d",
251
251
  "react": "^19.0.0",
252
252
  "react-dom": "^19.0.0"
253
253
  },
254
254
  "devDependencies": {
255
- "@open-mercato/ai-assistant": "0.6.5-develop.5116.1.f0af9e5080",
256
- "@open-mercato/shared": "0.6.5-develop.5116.1.f0af9e5080",
257
- "@open-mercato/ui": "0.6.5-develop.5116.1.f0af9e5080",
255
+ "@open-mercato/ai-assistant": "0.6.5-develop.5155.1.148d10a46d",
256
+ "@open-mercato/shared": "0.6.5-develop.5155.1.148d10a46d",
257
+ "@open-mercato/ui": "0.6.5-develop.5155.1.148d10a46d",
258
258
  "@testing-library/dom": "^10.4.1",
259
259
  "@testing-library/jest-dom": "^6.9.1",
260
260
  "@testing-library/react": "^16.3.1",
261
261
  "@types/chance": "^1.1.8",
262
262
  "@types/html-to-text": "^9.0.4",
263
263
  "@types/jest": "^30.0.0",
264
- "@types/react": "^19.2.16",
264
+ "@types/react": "^19.2.17",
265
265
  "@types/react-dom": "^19.2.3",
266
266
  "@types/sanitize-html": "^2.13.0",
267
267
  "@types/semver": "^7.5.8",
@@ -71,15 +71,42 @@ function stripTagBlocks(html: string, tag: string): string {
71
71
  return current
72
72
  }
73
73
 
74
+ /**
75
+ * Drop HTML comments (`<!-- … -->`), including an unterminated comment running
76
+ * to end-of-input. Comments are stripped first because they can wrap content
77
+ * that would otherwise survive tag removal (`<!--<script-->`), and a naive
78
+ * filter that ignores them leaves a tag fragment behind.
79
+ */
80
+ function stripHtmlComments(html: string): string {
81
+ return html.replace(/<!--[\s\S]*?(?:-->|$)/g, ' ')
82
+ }
83
+
84
+ const BASIC_HTML_ENTITIES: Record<string, string> = {
85
+ '&nbsp;': ' ',
86
+ '&lt;': '<',
87
+ '&gt;': '>',
88
+ '&quot;': '"',
89
+ '&amp;': '&',
90
+ }
91
+
92
+ /**
93
+ * Decode the handful of HTML entities we surface in plaintext in ONE
94
+ * left-to-right pass. A single pass cannot double-unescape: characters produced
95
+ * by a replacement (e.g. the `&` from `&amp;`) are never re-scanned, so
96
+ * `&amp;lt;` decodes to the literal `&lt;` rather than collapsing into `<`.
97
+ */
98
+ function decodeBasicEntities(input: string): string {
99
+ return input.replace(
100
+ /&(?:nbsp|lt|gt|quot|amp);/gi,
101
+ (match) => BASIC_HTML_ENTITIES[match.toLowerCase()] ?? match,
102
+ )
103
+ }
104
+
74
105
  export function htmlToText(html: string): string {
75
- return stripTagBlocks(stripTagBlocks(html, 'style'), 'script')
106
+ const stripped = stripTagBlocks(stripTagBlocks(stripHtmlComments(html), 'style'), 'script')
76
107
  .replace(/<br\s*\/?>(?=\s*)/gi, '\n')
77
108
  .replace(/<\/p\s*>/gi, '\n\n')
78
- .replace(/&nbsp;/gi, ' ')
79
- .replace(/&lt;/gi, '<')
80
- .replace(/&gt;/gi, '>')
81
- .replace(/&quot;/gi, '"')
82
- .replace(/&amp;/gi, '&')
109
+ return decodeBasicEntities(stripped)
83
110
  .replace(/<[^>]+>/g, '')
84
111
  .replace(/\n{3,}/g, '\n\n')
85
112
  .trim()
@@ -1464,11 +1464,12 @@ export function buildCompanyPayload(
1464
1464
 
1465
1465
  // URL/email/phone fields are clearable on edit: blanking a previously-set value transmits null,
1466
1466
  // so the edit-form value types widen to `string | null` to match the edit-schema output. See #2526.
1467
- export type CompanyEditFormValues = Omit<CompanyFormValues, 'addresses' | 'primaryEmail' | 'primaryPhone' | 'websiteUrl'> & {
1467
+ export type CompanyEditFormValues = Omit<CompanyFormValues, 'addresses' | 'primaryEmail' | 'primaryPhone' | 'websiteUrl' | 'domain'> & {
1468
1468
  id: string
1469
1469
  primaryEmail?: string | null
1470
1470
  primaryPhone?: string | null
1471
1471
  websiteUrl?: string | null
1472
+ domain?: string | null
1472
1473
  }
1473
1474
 
1474
1475
  export type PersonEditFormValues = Omit<PersonFormValues, 'addresses' | 'primaryEmail' | 'primaryPhone'> & {
@@ -1517,6 +1518,18 @@ const clearableEmailField = () =>
1517
1518
  .transform((val) => (val === '' ? null : val))
1518
1519
  .optional()
1519
1520
 
1521
+ // Domain maps to a nullable column; on edit a blanked value must transmit null
1522
+ // (not undefined) so it actually clears — mirroring the website field. See #2529.
1523
+ const clearableDomainField = () =>
1524
+ z
1525
+ .string()
1526
+ .trim()
1527
+ .max(200)
1528
+ .optional()
1529
+ .or(z.literal(''))
1530
+ .transform((val) => (val === '' ? null : val))
1531
+ .optional()
1532
+
1520
1533
  const clearablePhoneField = () =>
1521
1534
  z
1522
1535
  .string()
@@ -1534,6 +1547,7 @@ export const createCompanyEditSchema = () =>
1534
1547
  primaryEmail: clearableEmailField(),
1535
1548
  primaryPhone: clearablePhoneField(),
1536
1549
  websiteUrl: clearableUrlField(),
1550
+ domain: clearableDomainField(),
1537
1551
  })
1538
1552
 
1539
1553
  export const createPersonEditSchema = () =>
@@ -1844,6 +1858,7 @@ export function buildCompanyEditPayload(values: CompanyEditFormValues, organizat
1844
1858
  assignClearable(payload, 'primaryEmail', values.primaryEmail)
1845
1859
  assignClearable(payload, 'primaryPhone', values.primaryPhone)
1846
1860
  assignClearable(payload, 'websiteUrl', values.websiteUrl)
1861
+ assignClearable(payload, 'domain', typeof values.domain === 'string' ? values.domain.toLowerCase() : values.domain)
1847
1862
 
1848
1863
  return payload
1849
1864
  }
@@ -37,6 +37,13 @@ const clearableUrlSchema = z.preprocess(
37
37
  z.string().url().max(300).nullable().optional(),
38
38
  )
39
39
 
40
+ // Domain is a plain (non-URL) string that maps to a nullable column, so blanking
41
+ // a previously-set value on edit must transmit null to clear it. See #2529.
42
+ const clearableDomainSchema = z.preprocess(
43
+ emptyStringToNull,
44
+ z.string().trim().max(200).nullable().optional(),
45
+ )
46
+
40
47
  const interactionPhoneNumberSchema = z.string().trim().max(50).optional().nullable()
41
48
 
42
49
  const scopedSchema = z.object({
@@ -94,7 +101,7 @@ const personLastNameSchema = z.string().trim().min(1).max(120)
94
101
  const companyDetailsSchema = {
95
102
  legalName: z.string().trim().max(200).optional(),
96
103
  brandName: z.string().trim().max(200).optional(),
97
- domain: z.string().trim().max(200).optional(),
104
+ domain: clearableDomainSchema,
98
105
  websiteUrl: clearableUrlSchema,
99
106
  industry: z.string().trim().max(150).optional(),
100
107
  sizeBucket: z.string().trim().max(100).optional(),
@@ -597,7 +597,7 @@ const createChannelCommand: CommandHandler<ChannelCreateInput, { channelId: stri
597
597
  ensureTenantScope(ctx, parsed.tenantId)
598
598
  ensureOrganizationScope(ctx, parsed.organizationId)
599
599
  const em = (ctx.container.resolve('em') as EntityManager).fork()
600
- const statusValue = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null)
600
+ const statusValue = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null, { tenantId: parsed.tenantId })
601
601
  const record = em.create(SalesChannel, {
602
602
  organizationId: parsed.organizationId,
603
603
  tenantId: parsed.tenantId,
@@ -760,7 +760,7 @@ const updateChannelCommand: CommandHandler<ChannelUpdateInput, { channelId: stri
760
760
  if (parsed.description !== undefined) record.description = parsed.description ?? null
761
761
  if (parsed.statusEntryId !== undefined) {
762
762
  record.statusEntryId = parsed.statusEntryId ?? null
763
- record.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null)
763
+ record.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null, { tenantId: scope.tenantId })
764
764
  }
765
765
  if (parsed.websiteUrl !== undefined) record.websiteUrl = parsed.websiteUrl ?? null
766
766
  if (parsed.contactEmail !== undefined) record.contactEmail = parsed.contactEmail ?? null
@@ -1075,6 +1075,7 @@ async function applyDocumentUpdate({
1075
1075
  const statusValue = await resolveDictionaryEntryValue(
1076
1076
  em,
1077
1077
  input.statusEntryId,
1078
+ { tenantId },
1078
1079
  );
1079
1080
  if (input.statusEntryId && !statusValue) {
1080
1081
  throw new CrudHttpError(400, {
@@ -3029,7 +3030,7 @@ async function applyOrderLineResults(params: {
3029
3030
  const resolveStatus = async (entryId?: string | null) => {
3030
3031
  if (!entryId) return null;
3031
3032
  if (statusCache.has(entryId)) return statusCache.get(entryId) ?? null;
3032
- const value = await resolveDictionaryEntryValue(em, entryId);
3033
+ const value = await resolveDictionaryEntryValue(em, entryId, { tenantId: order.tenantId });
3033
3034
  statusCache.set(entryId, value);
3034
3035
  return value;
3035
3036
  };
@@ -3103,7 +3104,7 @@ async function applyQuoteLineResults(params: {
3103
3104
  const resolveStatus = async (entryId?: string | null) => {
3104
3105
  if (!entryId) return null;
3105
3106
  if (statusCache.has(entryId)) return statusCache.get(entryId) ?? null;
3106
- const value = await resolveDictionaryEntryValue(em, entryId);
3107
+ const value = await resolveDictionaryEntryValue(em, entryId, { tenantId: quote.tenantId });
3107
3108
  statusCache.set(entryId, value);
3108
3109
  return value;
3109
3110
  };
@@ -3170,7 +3171,7 @@ async function replaceQuoteLines(
3170
3171
  const resolveStatus = async (entryId?: string | null) => {
3171
3172
  if (!entryId) return null;
3172
3173
  if (statusCache.has(entryId)) return statusCache.get(entryId) ?? null;
3173
- const value = await resolveDictionaryEntryValue(em, entryId);
3174
+ const value = await resolveDictionaryEntryValue(em, entryId, { tenantId: quote.tenantId });
3174
3175
  statusCache.set(entryId, value);
3175
3176
  return value;
3176
3177
  };
@@ -3298,7 +3299,7 @@ async function replaceOrderLines(
3298
3299
  const resolveStatus = async (entryId?: string | null) => {
3299
3300
  if (!entryId) return null;
3300
3301
  if (statusCache.has(entryId)) return statusCache.get(entryId) ?? null;
3301
- const value = await resolveDictionaryEntryValue(em, entryId);
3302
+ const value = await resolveDictionaryEntryValue(em, entryId, { tenantId: order.tenantId });
3302
3303
  statusCache.set(entryId, value);
3303
3304
  return value;
3304
3305
  };
@@ -4449,6 +4450,7 @@ const createQuoteCommand: CommandHandler<
4449
4450
  const quoteStatus = await resolveDictionaryEntryValue(
4450
4451
  em,
4451
4452
  parsed.statusEntryId ?? null,
4453
+ { tenantId: parsed.tenantId },
4452
4454
  );
4453
4455
  const quoteId = randomUUID();
4454
4456
  const quote = em.create(SalesQuote, {
@@ -5413,9 +5415,9 @@ const createOrderCommand: CommandHandler<
5413
5415
  ensureOrderScope(ctx, parsed.organizationId, parsed.tenantId);
5414
5416
  const em = (ctx.container.resolve("em") as EntityManager).fork();
5415
5417
  const [status, fulfillmentStatus, paymentStatus] = await Promise.all([
5416
- resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null),
5417
- resolveDictionaryEntryValue(em, parsed.fulfillmentStatusEntryId ?? null),
5418
- resolveDictionaryEntryValue(em, parsed.paymentStatusEntryId ?? null),
5418
+ resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null, { tenantId: parsed.tenantId }),
5419
+ resolveDictionaryEntryValue(em, parsed.fulfillmentStatusEntryId ?? null, { tenantId: parsed.tenantId }),
5420
+ resolveDictionaryEntryValue(em, parsed.paymentStatusEntryId ?? null, { tenantId: parsed.tenantId }),
5419
5421
  ]);
5420
5422
  const {
5421
5423
  customerSnapshot: resolvedCustomerSnapshot,
@@ -8362,6 +8364,7 @@ const createInvoiceCommand: CommandHandler<
8362
8364
  const status = await resolveDictionaryEntryValue(
8363
8365
  em,
8364
8366
  parsed.statusEntryId ?? null,
8367
+ { tenantId: parsed.tenantId },
8365
8368
  );
8366
8369
 
8367
8370
  // Validate orderId belongs to same org/tenant
@@ -8573,7 +8576,7 @@ const updateInvoiceCommand: CommandHandler<
8573
8576
  ]);
8574
8577
 
8575
8578
  if (parsed.statusEntryId !== undefined) {
8576
- invoice.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null);
8579
+ invoice.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null, { tenantId: invoice.tenantId });
8577
8580
  }
8578
8581
 
8579
8582
  Object.assign(invoice, changes);
@@ -8847,6 +8850,7 @@ const createCreditMemoCommand: CommandHandler<
8847
8850
  const status = await resolveDictionaryEntryValue(
8848
8851
  em,
8849
8852
  parsed.statusEntryId ?? null,
8853
+ { tenantId: parsed.tenantId },
8850
8854
  );
8851
8855
 
8852
8856
  // Validate orderId belongs to same org/tenant
@@ -9063,7 +9067,7 @@ const updateCreditMemoCommand: CommandHandler<
9063
9067
  ]);
9064
9068
 
9065
9069
  if (parsed.statusEntryId !== undefined) {
9066
- creditMemo.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null);
9070
+ creditMemo.status = await resolveDictionaryEntryValue(em, parsed.statusEntryId ?? null, { tenantId: creditMemo.tenantId });
9067
9071
  }
9068
9072
 
9069
9073
  Object.assign(creditMemo, changes);
@@ -375,7 +375,7 @@ const createPaymentCommand: CommandHandler<
375
375
  tx.persist(order)
376
376
  }
377
377
  if (input.documentStatusEntryId !== undefined) {
378
- const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null)
378
+ const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null, { tenantId: input.tenantId })
379
379
  if (input.documentStatusEntryId && !orderStatus) {
380
380
  throw new CrudHttpError(400, {
381
381
  error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.'),
@@ -387,7 +387,7 @@ const createPaymentCommand: CommandHandler<
387
387
  tx.persist(order)
388
388
  }
389
389
  if (input.lineStatusEntryId !== undefined) {
390
- const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null)
390
+ const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null, { tenantId: input.tenantId })
391
391
  if (input.lineStatusEntryId && !lineStatus) {
392
392
  throw new CrudHttpError(400, {
393
393
  error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.'),
@@ -401,7 +401,7 @@ const createPaymentCommand: CommandHandler<
401
401
  })
402
402
  orderLines.forEach((line) => tx.persist(line))
403
403
  }
404
- const status = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null)
404
+ const status = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null, { tenantId: input.tenantId })
405
405
  const payment = tx.create(SalesPayment, {
406
406
  organizationId: input.organizationId,
407
407
  tenantId: input.tenantId,
@@ -779,7 +779,7 @@ const updatePaymentCommand: CommandHandler<
779
779
  throw new CrudHttpError(400, { error: translate('sales.payments.order_required', 'Order is required for payments.') })
780
780
  }
781
781
  if (currentOrder && input.documentStatusEntryId !== undefined) {
782
- const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null)
782
+ const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null, { tenantId: resolvedTenantId })
783
783
  if (input.documentStatusEntryId && !orderStatus) {
784
784
  throw new CrudHttpError(400, {
785
785
  error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.'),
@@ -791,7 +791,7 @@ const updatePaymentCommand: CommandHandler<
791
791
  tx.persist(currentOrder)
792
792
  }
793
793
  if (currentOrder && input.lineStatusEntryId !== undefined) {
794
- const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null)
794
+ const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null, { tenantId: resolvedTenantId })
795
795
  if (input.lineStatusEntryId && !lineStatus) {
796
796
  throw new CrudHttpError(400, {
797
797
  error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.'),
@@ -808,7 +808,7 @@ const updatePaymentCommand: CommandHandler<
808
808
  if (input.paymentReference !== undefined) payment.paymentReference = input.paymentReference ?? null
809
809
  if (input.statusEntryId !== undefined) {
810
810
  payment.statusEntryId = input.statusEntryId ?? null
811
- payment.status = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null)
811
+ payment.status = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null, { tenantId: resolvedTenantId })
812
812
  }
813
813
  if (input.amount !== undefined) payment.amount = toNumericString(input.amount) ?? '0'
814
814
  if (input.currencyCode !== undefined) payment.currencyCode = input.currencyCode
@@ -444,7 +444,7 @@ const createShipmentCommand: CommandHandler<ShipmentCreateInput, { shipmentId: s
444
444
  items: input.items,
445
445
  lockOrderLines: true,
446
446
  })
447
- const statusValue = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null)
447
+ const statusValue = await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null, { tenantId: input.tenantId })
448
448
  const trackingNumbers = parseTrackingNumbers(input.trackingNumbers) ?? null
449
449
  const metadata =
450
450
  mergeAddressSnapshot(
@@ -502,7 +502,7 @@ const createShipmentCommand: CommandHandler<ShipmentCreateInput, { shipmentId: s
502
502
  })
503
503
  }
504
504
  if (input.documentStatusEntryId !== undefined) {
505
- const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null)
505
+ const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null, { tenantId: input.tenantId })
506
506
  if (input.documentStatusEntryId && !orderStatus) {
507
507
  throw new CrudHttpError(400, { error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.') })
508
508
  }
@@ -511,7 +511,7 @@ const createShipmentCommand: CommandHandler<ShipmentCreateInput, { shipmentId: s
511
511
  order.updatedAt = new Date()
512
512
  }
513
513
  if (input.lineStatusEntryId !== undefined) {
514
- const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null)
514
+ const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null, { tenantId: input.tenantId })
515
515
  if (input.lineStatusEntryId && !lineStatus) {
516
516
  throw new CrudHttpError(400, { error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.') })
517
517
  }
@@ -702,7 +702,7 @@ const updateShipmentCommand: CommandHandler<ShipmentUpdateInput, { shipmentId: s
702
702
  // dictionary read does not interleave with (and drop) the pending
703
703
  // shipment changeset under MikroORM v7 (SPEC-018 / #2453 class).
704
704
  const resolvedShipmentStatus = input.statusEntryId !== undefined
705
- ? await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null)
705
+ ? await resolveDictionaryEntryValue(tx, input.statusEntryId ?? null, { tenantId: shipmentEntity.tenantId })
706
706
  : undefined
707
707
  if (input.shipmentNumber !== undefined) shipmentEntity.shipmentNumber = input.shipmentNumber ?? null
708
708
  if (input.shippingMethodId !== undefined) shipmentEntity.shippingMethodId = input.shippingMethodId ?? null
@@ -770,7 +770,7 @@ const updateShipmentCommand: CommandHandler<ShipmentUpdateInput, { shipmentId: s
770
770
  })
771
771
  }
772
772
  if (input.documentStatusEntryId !== undefined) {
773
- const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null)
773
+ const orderStatus = await resolveDictionaryEntryValue(tx, input.documentStatusEntryId ?? null, { tenantId: shipmentEntity.tenantId })
774
774
  if (input.documentStatusEntryId && !orderStatus) {
775
775
  throw new CrudHttpError(400, { error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.') })
776
776
  }
@@ -779,7 +779,7 @@ const updateShipmentCommand: CommandHandler<ShipmentUpdateInput, { shipmentId: s
779
779
  order.updatedAt = new Date()
780
780
  }
781
781
  if (input.lineStatusEntryId !== undefined) {
782
- const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null)
782
+ const lineStatus = await resolveDictionaryEntryValue(tx, input.lineStatusEntryId ?? null, { tenantId: shipmentEntity.tenantId })
783
783
  if (input.lineStatusEntryId && !lineStatus) {
784
784
  throw new CrudHttpError(400, { error: translate('sales.documents.detail.statusInvalid', 'Selected status could not be found.') })
785
785
  }
@@ -113,10 +113,11 @@ export async function ensureSalesDictionary(params: {
113
113
 
114
114
  export async function resolveDictionaryEntryValue(
115
115
  em: EntityManager,
116
- entryId: string | null | undefined
116
+ entryId: string | null | undefined,
117
+ scope: { tenantId: string }
117
118
  ): Promise<string | null> {
118
119
  if (!entryId) return null
119
- const entry = await em.findOne(DictionaryEntry, entryId)
120
+ const entry = await em.findOne(DictionaryEntry, { id: entryId, tenantId: scope.tenantId })
120
121
  if (!entry) return null
121
122
  return entry.value?.trim() || null
122
123
  }
@@ -309,6 +309,8 @@ export function TeamMemberForm(props: TeamMemberFormProps) {
309
309
  type: 'custom',
310
310
  component: ({ value, setValue, setFormValue, values, disabled }) => {
311
311
  const currentValue = typeof value === 'string' ? value : ''
312
+ const selectedOption = teamOptions.find((option) => option.value === currentValue)
313
+ const optionsKey = teamOptions.map((option) => `${option.value}:${option.label}`).join('\0')
312
314
  return (
313
315
  <div className="space-y-2">
314
316
  <div className="flex items-center justify-between gap-3">
@@ -327,6 +329,7 @@ export function TeamMemberForm(props: TeamMemberFormProps) {
327
329
  </Button>
328
330
  </div>
329
331
  <Select
332
+ key={`team:${currentValue}:${optionsKey}`}
330
333
  value={currentValue}
331
334
  onValueChange={(value) => {
332
335
  const nextValue = value || undefined
@@ -346,7 +349,9 @@ export function TeamMemberForm(props: TeamMemberFormProps) {
346
349
  disabled={disabled}
347
350
  >
348
351
  <SelectTrigger data-crud-focus-target="">
349
- <SelectValue placeholder={translate('ui.forms.select.emptyOption', '—')} />
352
+ <SelectValue placeholder={translate('ui.forms.select.emptyOption', '—')}>
353
+ {selectedOption?.label}
354
+ </SelectValue>
350
355
  </SelectTrigger>
351
356
  <SelectContent>
352
357
  {teamOptions.map((option) => (