@open-mercato/core 0.4.6-develop-e321a4e2a1 → 0.4.6-main-24e64eef39
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/AGENTS.md +0 -22
- package/dist/modules/api_docs/frontend/docs/api/page.js +1 -1
- package/dist/modules/api_docs/frontend/docs/api/page.js.map +2 -2
- package/dist/modules/attachments/api/library/[id]/route.js +1 -0
- package/dist/modules/attachments/api/library/[id]/route.js.map +2 -2
- package/dist/modules/attachments/components/AttachmentLibrary.js +1 -1
- package/dist/modules/attachments/components/AttachmentLibrary.js.map +2 -2
- package/dist/modules/attachments/lib/partitionEnv.js +1 -1
- package/dist/modules/attachments/lib/partitionEnv.js.map +2 -2
- package/dist/modules/auth/backend/users/page.js +1 -1
- package/dist/modules/auth/backend/users/page.js.map +2 -2
- package/dist/modules/auth/cli.js +1 -1
- package/dist/modules/auth/cli.js.map +2 -2
- package/dist/modules/auth/commands/users.js +1 -1
- package/dist/modules/auth/commands/users.js.map +2 -2
- package/dist/modules/business_rules/components/utils/formHelpers.js +1 -1
- package/dist/modules/business_rules/components/utils/formHelpers.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/create/page.js +1 -1
- package/dist/modules/catalog/backend/catalog/products/create/page.js.map +2 -2
- package/dist/modules/catalog/commands/products.js +1 -1
- package/dist/modules/catalog/commands/products.js.map +2 -2
- package/dist/modules/catalog/commands/shared.js +1 -1
- package/dist/modules/catalog/commands/shared.js.map +2 -2
- package/dist/modules/catalog/components/PriceKindSettings.js +1 -1
- package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
- package/dist/modules/catalog/components/products/productForm.js +1 -1
- package/dist/modules/catalog/components/products/productForm.js.map +2 -2
- package/dist/modules/configs/lib/upgrade-actions.js.map +1 -1
- package/dist/modules/currencies/services/providers/raiffeisen.js +1 -1
- package/dist/modules/currencies/services/providers/raiffeisen.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies/page.js +3 -3
- package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/page.js +3 -3
- package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people/page.js +3 -3
- package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
- package/dist/modules/customers/cli.js +2 -2
- package/dist/modules/customers/cli.js.map +2 -2
- package/dist/modules/customers/lib/detailHelpers.js +1 -1
- package/dist/modules/customers/lib/detailHelpers.js.map +2 -2
- package/dist/modules/entities/cli.js +1 -1
- package/dist/modules/entities/cli.js.map +2 -2
- package/dist/modules/entities/lib/field-definitions.js +1 -1
- package/dist/modules/entities/lib/field-definitions.js.map +2 -2
- package/dist/modules/entities/lib/install-from-ce.js +1 -1
- package/dist/modules/entities/lib/install-from-ce.js.map +2 -2
- package/dist/modules/inbox_ops/lib/emailParser.js +1 -1
- package/dist/modules/inbox_ops/lib/emailParser.js.map +2 -2
- package/dist/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.js +0 -8
- package/dist/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.js.map +2 -2
- package/dist/modules/perspectives/services/perspectiveService.js +1 -1
- package/dist/modules/perspectives/services/perspectiveService.js.map +2 -2
- package/dist/modules/planner/backend/planner/availability-rulesets/page.js +3 -3
- package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
- package/dist/modules/query_index/components/QueryIndexesTable.js +7 -7
- package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
- package/dist/modules/query_index/lib/engine.js +1 -1
- package/dist/modules/query_index/lib/engine.js.map +2 -2
- package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
- package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
- package/dist/modules/resources/backend/resources/resources/page.js +3 -3
- package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
- package/dist/modules/resources/commands/resources.js +1 -1
- package/dist/modules/resources/commands/resources.js.map +2 -2
- package/dist/modules/resources/lib/seeds.js +1 -1
- package/dist/modules/resources/lib/seeds.js.map +2 -2
- package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js +1 -1
- package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js.map +2 -2
- package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js +1 -1
- package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js.map +2 -2
- package/dist/modules/sales/backend/sales/channels/page.js +3 -3
- package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
- package/dist/modules/sales/commands/documents.js +2 -2
- package/dist/modules/sales/commands/documents.js.map +2 -2
- package/dist/modules/sales/components/channels/offerTableUtils.js +2 -3
- package/dist/modules/sales/components/channels/offerTableUtils.js.map +2 -2
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js +4 -4
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
- package/dist/modules/sales/components/documents/ShipmentDialog.js +1 -1
- package/dist/modules/sales/components/documents/ShipmentDialog.js.map +2 -2
- package/dist/modules/sales/lib/shipments/snapshots.js.map +1 -1
- package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +0 -8
- package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +2 -2
- package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +0 -8
- package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +2 -2
- package/dist/modules/staff/backend/staff/leave-requests/page.js +3 -3
- package/dist/modules/staff/backend/staff/leave-requests/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/my-leave-requests/page.js +3 -3
- package/dist/modules/staff/backend/staff/my-leave-requests/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-members/page.js +3 -3
- package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
- package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +3 -3
- package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/teams/page.js +2 -2
- package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
- package/dist/modules/workflows/backend/instances/page.js +2 -2
- package/dist/modules/workflows/backend/instances/page.js.map +2 -2
- package/dist/modules/workflows/backend/tasks/page.js +1 -1
- package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js +1 -1
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +2 -2
- package/dist/modules/workflows/lib/graph-utils.js +1 -1
- package/dist/modules/workflows/lib/graph-utils.js.map +2 -2
- package/package.json +2 -2
- package/src/modules/api_docs/frontend/docs/api/page.tsx +1 -1
- package/src/modules/attachments/api/library/[id]/route.ts +1 -0
- package/src/modules/attachments/components/AttachmentLibrary.tsx +1 -1
- package/src/modules/attachments/lib/partitionEnv.ts +1 -1
- package/src/modules/auth/backend/users/page.tsx +1 -1
- package/src/modules/auth/cli.ts +1 -1
- package/src/modules/auth/commands/users.ts +1 -1
- package/src/modules/business_rules/components/utils/formHelpers.ts +1 -1
- package/src/modules/catalog/backend/catalog/products/create/page.tsx +1 -1
- package/src/modules/catalog/commands/products.ts +1 -1
- package/src/modules/catalog/commands/shared.ts +1 -1
- package/src/modules/catalog/components/PriceKindSettings.tsx +1 -1
- package/src/modules/catalog/components/products/productForm.ts +1 -1
- package/src/modules/configs/lib/upgrade-actions.ts +1 -1
- package/src/modules/currencies/services/providers/raiffeisen.ts +1 -1
- package/src/modules/customers/backend/customers/companies/page.tsx +3 -3
- package/src/modules/customers/backend/customers/deals/page.tsx +3 -3
- package/src/modules/customers/backend/customers/people/page.tsx +3 -3
- package/src/modules/customers/cli.ts +2 -2
- package/src/modules/customers/lib/detailHelpers.ts +1 -1
- package/src/modules/entities/cli.ts +1 -1
- package/src/modules/entities/lib/field-definitions.ts +1 -1
- package/src/modules/entities/lib/install-from-ce.ts +1 -1
- package/src/modules/inbox_ops/lib/emailParser.ts +1 -1
- package/src/modules/inbox_ops/widgets/notifications/ProposalCreatedRenderer.tsx +0 -8
- package/src/modules/perspectives/services/perspectiveService.ts +1 -1
- package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +4 -3
- package/src/modules/query_index/components/QueryIndexesTable.tsx +7 -7
- package/src/modules/query_index/lib/engine.ts +1 -1
- package/src/modules/resources/backend/resources/resource-types/page.tsx +3 -2
- package/src/modules/resources/backend/resources/resources/page.tsx +3 -3
- package/src/modules/resources/commands/resources.ts +1 -1
- package/src/modules/resources/lib/seeds.ts +1 -1
- package/src/modules/sales/api/dashboard/widgets/new-orders/route.ts +1 -1
- package/src/modules/sales/api/dashboard/widgets/new-quotes/route.ts +1 -1
- package/src/modules/sales/backend/sales/channels/page.tsx +3 -3
- package/src/modules/sales/commands/documents.ts +2 -2
- package/src/modules/sales/components/channels/offerTableUtils.tsx +2 -3
- package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +4 -4
- package/src/modules/sales/components/documents/ShipmentDialog.tsx +1 -1
- package/src/modules/sales/lib/shipments/snapshots.ts +1 -1
- package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +0 -8
- package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +0 -8
- package/src/modules/staff/backend/staff/leave-requests/page.tsx +3 -3
- package/src/modules/staff/backend/staff/my-leave-requests/page.tsx +3 -3
- package/src/modules/staff/backend/staff/team-members/page.tsx +3 -3
- package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
- package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +4 -3
- package/src/modules/staff/backend/staff/teams/page.tsx +3 -2
- package/src/modules/workflows/backend/instances/page.tsx +2 -2
- package/src/modules/workflows/backend/tasks/page.tsx +1 -1
- package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +1 -1
- package/src/modules/workflows/lib/graph-utils.ts +1 -1
- package/dist/modules/integrations/acl.js +0 -8
- package/dist/modules/integrations/acl.js.map +0 -7
- package/dist/modules/integrations/data/enrichers.js +0 -72
- package/dist/modules/integrations/data/enrichers.js.map +0 -7
- package/dist/modules/integrations/data/entities.js +0 -63
- package/dist/modules/integrations/data/entities.js.map +0 -7
- package/dist/modules/integrations/index.js +0 -9
- package/dist/modules/integrations/index.js.map +0 -7
- package/dist/modules/integrations/setup.js +0 -13
- package/dist/modules/integrations/setup.js.map +0 -7
- package/dist/modules/integrations/widgets/injection/external-ids/widget.client.js +0 -69
- package/dist/modules/integrations/widgets/injection/external-ids/widget.client.js.map +0 -7
- package/dist/modules/integrations/widgets/injection-table.js +0 -13
- package/dist/modules/integrations/widgets/injection-table.js.map +0 -7
- package/src/modules/integrations/acl.ts +0 -4
- package/src/modules/integrations/data/enrichers.ts +0 -98
- package/src/modules/integrations/data/entities.ts +0 -46
- package/src/modules/integrations/index.ts +0 -5
- package/src/modules/integrations/setup.ts +0 -11
- package/src/modules/integrations/widgets/injection/external-ids/widget.client.tsx +0 -94
- package/src/modules/integrations/widgets/injection-table.ts +0 -17
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/catalog/components/PriceKindSettings.tsx"],
|
|
4
|
-
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { Label } from '@open-mercato/ui/primitives/label'\nimport {\n Dialog,\n DialogContent,\n DialogDescription,\n DialogFooter,\n DialogHeader,\n DialogTitle,\n} from '@open-mercato/ui/primitives/dialog'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { apiCall, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { raiseCrudError } from '@open-mercato/ui/backend/utils/serverErrors'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport { DictionaryEntrySelect } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport { useCurrencyDictionary } from '@open-mercato/core/modules/customers/components/detail/hooks/useCurrencyDictionary'\nimport type { DictionaryOption } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport type { CatalogPriceDisplayMode } from '../data/types'\n\ntype PriceKind = {\n id: string\n code: string\n title: string\n displayMode: CatalogPriceDisplayMode\n currencyCode: string | null\n isPromotion: boolean\n isActive: boolean\n createdAt: string\n updatedAt: string\n}\n\ntype DialogState =\n | { mode: 'create' }\n | { mode: 'edit'; entry: PriceKind }\n\ntype PriceKindApiPayload = Partial<PriceKind> & {\n display_mode?: PriceKind['displayMode']\n currency_code?: string | null\n is_promotion?: boolean\n is_active?: boolean\n created_at?: string\n updated_at?: string\n}\n\nconst DISPLAY_MODES: Array<{ value: 'including-tax' | 'excluding-tax'; label: string }> = [\n { value: 'excluding-tax', label: 'Excluding tax' },\n { value: 'including-tax', label: 'Including tax' },\n]\n\nconst PAGE_SIZE = 100\n\ntype PriceKindFormState = {\n code: string\n title: string\n displayMode: CatalogPriceDisplayMode\n currencyCode: string\n isPromotion: boolean\n isActive: boolean\n}\n\nconst DEFAULT_FORM: PriceKindFormState = {\n code: '',\n title: '',\n displayMode: 'excluding-tax' as const,\n currencyCode: '',\n isPromotion: false,\n isActive: true,\n}\n\nconst normalizePriceKind = (input: PriceKindApiPayload | null | undefined): PriceKind => {\n const raw = input ?? {}\n const toStringValue = (value: unknown): string | null => {\n if (typeof value === 'string') return value\n if (typeof value === 'number' || typeof value === 'bigint') return String(value)\n return null\n }\n const toBooleanValue = (value: unknown): boolean | null => (typeof value === 'boolean' ? value : null)\n const resolveDisplayMode = (value: string | null): PriceKind['displayMode'] =>\n value === 'including-tax' ? 'including-tax' : 'excluding-tax'\n\n const displayMode = resolveDisplayMode(\n toStringValue(raw.displayMode) ?? toStringValue(raw.display_mode),\n )\n const currencyCode = toStringValue(raw.currencyCode) ?? toStringValue(raw.currency_code)\n const isPromotion = toBooleanValue(raw.isPromotion) ?? toBooleanValue(raw.is_promotion)\n const isActive = toBooleanValue(raw.isActive) ?? toBooleanValue(raw.is_active)\n\n return {\n id: toStringValue(raw.id) ?? '',\n code: toStringValue(raw.code) ?? '',\n title: toStringValue(raw.title) ?? '',\n displayMode,\n currencyCode: currencyCode ?? null,\n isPromotion: isPromotion ?? false,\n isActive: isActive ?? true,\n createdAt: toStringValue(raw.createdAt) ?? toStringValue(raw.created_at) ?? '',\n updatedAt: toStringValue(raw.updatedAt) ?? toStringValue(raw.updated_at) ?? '',\n }\n}\n\nexport function PriceKindSettings() {\n const t = useT()\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n const scopeVersion = useOrganizationScopeVersion()\n const [items, setItems] = React.useState<PriceKind[]>([])\n const [loading, setLoading] = React.useState(false)\n const [search, setSearch] = React.useState('')\n const [dialog, setDialog] = React.useState<DialogState | null>(null)\n const [form, setForm] = React.useState<PriceKindFormState>(DEFAULT_FORM)\n const [submitting, setSubmitting] = React.useState(false)\n const [error, setError] = React.useState<string | null>(null)\n const { data: currencyDictionary, refetch: refetchCurrencyDictionary } = useCurrencyDictionary()\n\n const currencyOptionsLoader = React.useCallback(async (): Promise<DictionaryOption[]> => {\n if (currencyDictionary && Array.isArray(currencyDictionary.entries)) {\n return currencyDictionary.entries.map((entry) => ({\n value: entry.value,\n label: entry.label,\n color: entry.color ?? null,\n icon: entry.icon ?? null,\n }))\n }\n const payload = await refetchCurrencyDictionary()\n return payload.entries.map((entry) => ({\n value: entry.value,\n label: entry.label,\n color: entry.color ?? null,\n icon: entry.icon ?? null,\n }))\n }, [currencyDictionary, refetchCurrencyDictionary])\n\n const loadItems = React.useCallback(async () => {\n setLoading(true)\n const loadErrorMessage = t('catalog.priceKinds.errors.load', 'Failed to load price kinds.')\n try {\n const payload = await readApiResultOrThrow<{ items?: PriceKindApiPayload[] }>(\n `/api/catalog/price-kinds?pageSize=${PAGE_SIZE}`,\n undefined,\n { errorMessage: loadErrorMessage },\n )\n const normalized = Array.isArray(payload.items) ? payload.items.map((item) => normalizePriceKind(item)) : []\n setItems(normalized)\n } catch (err) {\n console.error('catalog.price-kinds.list failed', err)\n flash(loadErrorMessage, 'error')\n } finally {\n setLoading(false)\n }\n }, [t])\n\n React.useEffect(() => {\n loadItems().catch(() => {})\n }, [loadItems, scopeVersion])\n\n const openDialog = React.useCallback((state: DialogState) => {\n if (state.mode === 'edit') {\n setForm({\n code: state.entry.code,\n title: state.entry.title,\n displayMode: state.entry.displayMode,\n currencyCode: state.entry.currencyCode ?? '',\n isPromotion: state.entry.isPromotion,\n isActive: state.entry.isActive,\n })\n } else {\n setForm(DEFAULT_FORM)\n }\n setError(null)\n setDialog(state)\n }, [])\n\n const closeDialog = React.useCallback(() => {\n setDialog(null)\n setError(null)\n setSubmitting(false)\n setForm(DEFAULT_FORM)\n }, [])\n\n const handleSubmit = React.useCallback(async () => {\n if (!dialog) return\n const trimmedCode = form.code.trim().toLowerCase()\n const trimmedTitle = form.title.trim()\n if (!trimmedCode || !trimmedTitle) {\n setError(t('catalog.priceKinds.errors.required', 'Code and title are required.'))\n return\n }\n setSubmitting(true)\n setError(null)\n try {\n const payload = {\n code: trimmedCode,\n title: trimmedTitle,\n displayMode: form.displayMode,\n currencyCode: form.currencyCode.trim() || undefined,\n isPromotion: form.isPromotion,\n isActive: form.isActive,\n }\n const path = '/api/catalog/price-kinds'\n const method = dialog.mode === 'create' ? 'POST' : 'PUT'\n const body =\n dialog.mode === 'edit'\n ? JSON.stringify({ id: dialog.entry.id, ...payload })\n : JSON.stringify(payload)\n const call = await apiCall(path, {\n method,\n headers: { 'content-type': 'application/json' },\n body,\n })\n if (!call.ok) {\n await raiseCrudError(call.response, t('catalog.priceKinds.errors.save', 'Failed to save price kind.'))\n }\n flash(\n dialog.mode === 'create'\n ? t('catalog.priceKinds.messages.created', 'Price kind created.')\n : t('catalog.priceKinds.messages.updated', 'Price kind updated.'),\n 'success',\n )\n closeDialog()\n await loadItems()\n } catch (err) {\n console.error('catalog.price-kinds.save failed', err)\n const message =\n err instanceof Error ? err.message : t('catalog.priceKinds.errors.save', 'Failed to save price kind.')\n setError(message)\n } finally {\n setSubmitting(false)\n }\n }, [dialog, form, t, closeDialog, loadItems])\n\n const handleDelete = React.useCallback(\n async (entry: PriceKind) => {\n const confirmMessage = t('catalog.priceKinds.confirm.delete', 'Delete price kind \"{{code}}\"?').replace('{{code}}', entry.code)\n const confirmed = await confirm({\n title: confirmMessage,\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n const call = await apiCall('/api/catalog/price-kinds', {\n method: 'DELETE',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({ id: entry.id }),\n })\n if (!call.ok) {\n await raiseCrudError(call.response, t('catalog.priceKinds.errors.delete', 'Failed to delete price kind.'))\n }\n flash(t('catalog.priceKinds.messages.deleted', 'Price kind deleted.'), 'success')\n await loadItems()\n } catch (err) {\n console.error('catalog.price-kinds.delete failed', err)\n const message =\n err instanceof Error ? err.message : t('catalog.priceKinds.errors.delete', 'Failed to delete price kind.')\n flash(message, 'error')\n }\n },\n [confirm, loadItems, t],\n )\n\n const formKeyHandler = React.useCallback(\n (event: React.KeyboardEvent<HTMLFormElement>) => {\n if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {\n event.preventDefault()\n void handleSubmit()\n }\n },\n [handleSubmit],\n )\n\n const displayModeLabels = React.useMemo(() => ({\n 'including-tax': t('catalog.priceKinds.form.displayMode.include', 'Including tax'),\n 'excluding-tax': t('catalog.priceKinds.form.displayMode.exclude', 'Excluding tax'),\n }), [t])\n\n const displayModeOptions = React.useMemo(\n () =>\n DISPLAY_MODES.map((mode) => ({\n ...mode,\n label: displayModeLabels[mode.value],\n })),\n [displayModeLabels],\n )\n\n const currencyLabels = React.useMemo(() => ({\n placeholder: t('catalog.priceKinds.form.currency.placeholder', 'Select currency\u2026'),\n addLabel: t('catalog.priceKinds.form.currency.add', 'Add currency'),\n addPrompt: t('catalog.priceKinds.form.currency.addPrompt', 'Provide a currency code.'),\n dialogTitle: t('catalog.priceKinds.form.currency.dialogTitle', 'Add currency'),\n valueLabel: t('catalog.priceKinds.form.currency.valueLabel', 'Currency code'),\n valuePlaceholder: t('catalog.priceKinds.form.currency.valuePlaceholder', 'e.g. USD'),\n labelLabel: t('catalog.priceKinds.form.currency.labelLabel', 'Display label (optional)'),\n labelPlaceholder: t('catalog.priceKinds.form.currency.labelPlaceholder', 'e.g. US Dollar'),\n emptyError: t('catalog.priceKinds.form.currency.required', 'Currency code is required.'),\n cancelLabel: t('catalog.priceKinds.form.currency.cancel', 'Cancel'),\n saveLabel: t('catalog.priceKinds.form.currency.save', 'Save'),\n saveShortcutHint: t('catalog.priceKinds.form.currency.saveShortcut', 'Press Enter to save'),\n successCreateLabel: t('catalog.priceKinds.form.currency.success', 'Currency added.'),\n errorLoad: t('catalog.priceKinds.form.currency.loadError', 'Unable to load currencies.'),\n errorSave: t('catalog.priceKinds.form.currency.createError', 'Unable to add currency.'),\n loadingLabel: t('catalog.priceKinds.form.currency.loading', 'Loading currencies\u2026'),\n manageTitle: t('catalog.priceKinds.form.currency.manage', 'Manage currencies'),\n }), [t])\n\n const tableLabels = React.useMemo(() => ({\n code: t('catalog.priceKinds.table.code', 'Code'),\n title: t('catalog.priceKinds.table.title', 'Title'),\n displayMode: t('catalog.priceKinds.table.displayMode', 'Display mode'),\n currency: t('catalog.priceKinds.table.currency', 'Currency'),\n promotion: t('catalog.priceKinds.table.promotion', 'Promotion'),\n promotionYes: t('catalog.priceKinds.table.promotionYes', 'Yes'),\n promotionNo: t('catalog.priceKinds.table.promotionNo', 'No'),\n active: t('catalog.priceKinds.table.active', 'Active'),\n activeYes: t('catalog.priceKinds.table.activeYes', 'Active'),\n activeNo: t('catalog.priceKinds.table.activeNo', 'Inactive'),\n search: t('catalog.priceKinds.search.placeholder', 'Search by code or title\u2026'),\n empty: t('catalog.priceKinds.table.empty', 'No price kinds yet.'),\n }), [t])\n\n const columns = React.useMemo<ColumnDef<PriceKind>[]>(() => [\n {\n accessorKey: 'code',\n header: tableLabels.code,\n cell: ({ row }) => <span className=\"font-mono uppercase\">{row.original.code}</span>,\n },\n {\n accessorKey: 'title',\n header: tableLabels.title,\n cell: ({ row }) => <span className=\"font-medium\">{row.original.title}</span>,\n },\n {\n accessorKey: 'displayMode',\n header: tableLabels.displayMode,\n cell: ({ row }) => displayModeLabels[row.original.displayMode] ?? row.original.displayMode,\n },\n {\n accessorKey: 'currencyCode',\n header: tableLabels.currency,\n cell: ({ row }) => (row.original.currencyCode ? row.original.currencyCode.toUpperCase() : '\u2014'),\n },\n {\n id: 'promotion',\n header: tableLabels.promotion,\n cell: ({ row }) =>\n row.original.isPromotion ? (\n <span className=\"inline-flex items-center rounded-full border border-amber-200 bg-amber-50 px-2 py-0.5 text-xs font-medium text-amber-900 dark:border-amber-500/50 dark:bg-amber-500/10 dark:text-amber-100\">\n {tableLabels.promotionYes}\n </span>\n ) : (\n <span className=\"inline-flex items-center rounded-full border px-2 py-0.5 text-xs font-medium\">\n {tableLabels.promotionNo}\n </span>\n ),\n },\n {\n id: 'active',\n header: tableLabels.active,\n cell: ({ row }) =>\n row.original.isActive ? (\n <span className=\"inline-flex items-center rounded-full border border-emerald-200 bg-emerald-50 px-2 py-0.5 text-xs font-medium text-emerald-900 dark:border-emerald-500/50 dark:bg-emerald-500/10 dark:text-emerald-100\">\n {tableLabels.activeYes}\n </span>\n ) : (\n <span className=\"inline-flex items-center rounded-full border px-2 py-0.5 text-xs font-medium\">\n {tableLabels.activeNo}\n </span>\n ),\n },\n ], [displayModeLabels, tableLabels])\n\n const filteredItems = React.useMemo(() => {\n const term = search.trim().toLowerCase()\n if (!term) return items\n return items.filter((item) => {\n const currency = (item.currencyCode ?? '').toLowerCase()\n const modeLabel = (displayModeLabels[item.displayMode] ?? item.displayMode).toLowerCase()\n return (\n item.code.toLowerCase().includes(term) ||\n item.title.toLowerCase().includes(term) ||\n currency.includes(term) ||\n modeLabel.includes(term)\n )\n })\n }, [displayModeLabels, items, search])\n\n const handleRowClick = React.useCallback((entry: PriceKind) => {\n openDialog({ mode: 'edit', entry })\n }, [openDialog])\n\n return (\n <>\n <section className=\"border bg-card text-card-foreground shadow-sm\">\n <div className=\"border-b px-6 py-4 space-y-1\">\n <h2 className=\"text-lg font-semibold\">{t('catalog.priceKinds.title', 'Price kinds')}</h2>\n <p className=\"text-sm text-muted-foreground\">\n {t('catalog.priceKinds.description', 'Configure reusable price kinds that control pricing columns and tax display.')}\n </p>\n </div>\n <div className=\"px-2 py-4 sm:px-4\">\n <DataTable<PriceKind>\n data={filteredItems}\n columns={columns}\n embedded\n isLoading={loading}\n searchValue={search}\n onSearchChange={setSearch}\n searchPlaceholder={tableLabels.search}\n emptyState={<p className=\"py-8 text-center text-sm text-muted-foreground\">{tableLabels.empty}</p>}\n actions={(\n <Button size=\"sm\" onClick={() => openDialog({ mode: 'create' })}>\n {t('catalog.priceKinds.actions.add', 'Add price kind')}\n </Button>\n )}\n refreshButton={{\n label: t('catalog.priceKinds.actions.refresh', 'Refresh'),\n onRefresh: () => { void loadItems() },\n isRefreshing: loading,\n }}\n rowActions={(entry) => (\n <RowActions\n items={[\n {\n id: 'edit',\n label: t('catalog.priceKinds.actions.edit', 'Edit'),\n onSelect: () => openDialog({ mode: 'edit', entry }),\n },\n {\n id: 'delete',\n label: t('catalog.priceKinds.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => { void handleDelete(entry) },\n },\n ]}\n />\n )}\n onRowClick={handleRowClick}\n />\n </div>\n <Dialog open={dialog !== null} onOpenChange={(open) => { if (!open) closeDialog(); }}>\n <DialogContent>\n <DialogHeader>\n <DialogTitle>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.dialog.editTitle', 'Edit price kind')\n : t('catalog.priceKinds.dialog.createTitle', 'Create price kind')}\n </DialogTitle>\n <DialogDescription>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.dialog.editDescription', 'Update labels or tax behavior for this price kind.')\n : t('catalog.priceKinds.dialog.createDescription', 'Define a reusable price kind for product pricing.')}\n </DialogDescription>\n </DialogHeader>\n <form className=\"space-y-4\" onKeyDown={formKeyHandler} onSubmit={(event) => { event.preventDefault(); void handleSubmit() }}>\n <div className=\"space-y-2\">\n <Label htmlFor=\"price-kind-code\">{t('catalog.priceKinds.form.codeLabel', 'Code')}</Label>\n <Input\n id=\"price-kind-code\"\n value={form.code}\n onChange={(event) => setForm((prev) => ({ ...prev, code: event.target.value }))}\n placeholder={t('catalog.priceKinds.form.codePlaceholder', 'e.g. regular')}\n className=\"font-mono uppercase\"\n disabled={dialog?.mode === 'edit'}\n />\n </div>\n <div className=\"space-y-2\">\n <Label htmlFor=\"price-kind-title\">{t('catalog.priceKinds.form.titleLabel', 'Title')}</Label>\n <Input\n id=\"price-kind-title\"\n value={form.title}\n onChange={(event) => setForm((prev) => ({ ...prev, title: event.target.value }))}\n placeholder={t('catalog.priceKinds.form.titlePlaceholder', 'e.g. Regular price')}\n />\n </div>\n <div className=\"space-y-2\">\n <Label>{t('catalog.priceKinds.form.displayModeLabel', 'Display mode')}</Label>\n <div className=\"grid gap-2 md:grid-cols-2\">\n {displayModeOptions.map((mode) => (\n <label\n key={mode.value}\n className={`flex cursor-pointer items-center gap-2 rounded-md border p-3 text-sm ${\n form.displayMode === mode.value ? 'border-primary bg-primary/5' : 'border-border'\n }`}\n >\n <input\n type=\"radio\"\n name=\"displayMode\"\n value={mode.value}\n checked={form.displayMode === mode.value}\n onChange={() => setForm((prev) => ({ ...prev, displayMode: mode.value }))}\n />\n <span>{mode.label}</span>\n </label>\n ))}\n </div>\n </div>\n <div className=\"space-y-2\">\n <Label>{t('catalog.priceKinds.form.currencyLabel', 'Currency (optional)')}</Label>\n <DictionaryEntrySelect\n value={form.currencyCode || undefined}\n onChange={(value) => setForm((prev) => ({ ...prev, currencyCode: value ?? '' }))}\n fetchOptions={currencyOptionsLoader}\n labels={currencyLabels}\n allowInlineCreate={false}\n />\n </div>\n <div className=\"flex flex-col gap-2\">\n <label className=\"flex items-center gap-2 text-sm font-medium\">\n <input\n type=\"checkbox\"\n className=\"h-4 w-4 rounded border\"\n checked={form.isPromotion}\n onChange={(event) => setForm((prev) => ({ ...prev, isPromotion: event.target.checked }))}\n />\n {t('catalog.priceKinds.form.promotionLabel', 'Mark as promotion')}\n </label>\n <label className=\"flex items-center gap-2 text-sm font-medium\">\n <input\n type=\"checkbox\"\n className=\"h-4 w-4 rounded border\"\n checked={form.isActive}\n onChange={(event) => setForm((prev) => ({ ...prev, isActive: event.target.checked }))}\n />\n {t('catalog.priceKinds.form.activeLabel', 'Active')}\n </label>\n </div>\n {error ? <p className=\"text-sm text-red-600\">{error}</p> : null}\n </form>\n <DialogFooter>\n <Button variant=\"ghost\" onClick={closeDialog}>\n {t('catalog.priceKinds.actions.cancel', 'Cancel')}\n </Button>\n <Button onClick={() => void handleSubmit()} disabled={submitting}>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.actions.saveChanges', 'Save changes')\n : t('catalog.priceKinds.actions.create', 'Create')}\n </Button>\n </DialogFooter>\n </DialogContent>\n </Dialog>\n </section>\n {ConfirmDialogElement}\n </>\n )\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { Label } from '@open-mercato/ui/primitives/label'\nimport {\n Dialog,\n DialogContent,\n DialogDescription,\n DialogFooter,\n DialogHeader,\n DialogTitle,\n} from '@open-mercato/ui/primitives/dialog'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { apiCall, readApiResultOrThrow } from '@open-mercato/ui/backend/utils/apiCall'\nimport { raiseCrudError } from '@open-mercato/ui/backend/utils/serverErrors'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport { DictionaryEntrySelect } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport { useCurrencyDictionary } from '@open-mercato/core/modules/customers/components/detail/hooks/useCurrencyDictionary'\nimport type { DictionaryOption } from '@open-mercato/core/modules/dictionaries/components/DictionaryEntrySelect'\nimport type { CatalogPriceDisplayMode } from '../data/types'\n\ntype PriceKind = {\n id: string\n code: string\n title: string\n displayMode: CatalogPriceDisplayMode\n currencyCode: string | null\n isPromotion: boolean\n isActive: boolean\n createdAt: string\n updatedAt: string\n}\n\ntype DialogState =\n | { mode: 'create' }\n | { mode: 'edit'; entry: PriceKind }\n\ntype PriceKindApiPayload = Partial<PriceKind> & {\n display_mode?: PriceKind['displayMode']\n currency_code?: string | null\n is_promotion?: boolean\n is_active?: boolean\n created_at?: string\n updated_at?: string\n}\n\nconst DISPLAY_MODES: Array<{ value: 'including-tax' | 'excluding-tax'; label: string }> = [\n { value: 'excluding-tax', label: 'Excluding tax' },\n { value: 'including-tax', label: 'Including tax' },\n]\n\nconst PAGE_SIZE = 100\n\ntype PriceKindFormState = {\n code: string\n title: string\n displayMode: CatalogPriceDisplayMode\n currencyCode: string\n isPromotion: boolean\n isActive: boolean\n}\n\nconst DEFAULT_FORM: PriceKindFormState = {\n code: '',\n title: '',\n displayMode: 'excluding-tax' as const,\n currencyCode: '',\n isPromotion: false,\n isActive: true,\n}\n\nconst normalizePriceKind = (input: PriceKindApiPayload | null | undefined): PriceKind => {\n const raw = input ?? {}\n const toStringValue = (value: unknown): string | null => {\n if (typeof value === 'string') return value\n if (typeof value === 'number' || typeof value === 'bigint') return String(value)\n return null\n }\n const toBooleanValue = (value: unknown): boolean | null => (typeof value === 'boolean' ? value : null)\n const resolveDisplayMode = (value: string | null): PriceKind['displayMode'] =>\n value === 'including-tax' ? 'including-tax' : value === 'excluding-tax' ? 'excluding-tax' : 'excluding-tax'\n\n const displayMode = resolveDisplayMode(\n toStringValue(raw.displayMode) ?? toStringValue(raw.display_mode),\n )\n const currencyCode = toStringValue(raw.currencyCode) ?? toStringValue(raw.currency_code)\n const isPromotion = toBooleanValue(raw.isPromotion) ?? toBooleanValue(raw.is_promotion)\n const isActive = toBooleanValue(raw.isActive) ?? toBooleanValue(raw.is_active)\n\n return {\n id: toStringValue(raw.id) ?? '',\n code: toStringValue(raw.code) ?? '',\n title: toStringValue(raw.title) ?? '',\n displayMode,\n currencyCode: currencyCode ?? null,\n isPromotion: isPromotion ?? false,\n isActive: isActive ?? true,\n createdAt: toStringValue(raw.createdAt) ?? toStringValue(raw.created_at) ?? '',\n updatedAt: toStringValue(raw.updatedAt) ?? toStringValue(raw.updated_at) ?? '',\n }\n}\n\nexport function PriceKindSettings() {\n const t = useT()\n const { confirm, ConfirmDialogElement } = useConfirmDialog()\n const scopeVersion = useOrganizationScopeVersion()\n const [items, setItems] = React.useState<PriceKind[]>([])\n const [loading, setLoading] = React.useState(false)\n const [search, setSearch] = React.useState('')\n const [dialog, setDialog] = React.useState<DialogState | null>(null)\n const [form, setForm] = React.useState<PriceKindFormState>(DEFAULT_FORM)\n const [submitting, setSubmitting] = React.useState(false)\n const [error, setError] = React.useState<string | null>(null)\n const { data: currencyDictionary, refetch: refetchCurrencyDictionary } = useCurrencyDictionary()\n\n const currencyOptionsLoader = React.useCallback(async (): Promise<DictionaryOption[]> => {\n if (currencyDictionary && Array.isArray(currencyDictionary.entries)) {\n return currencyDictionary.entries.map((entry) => ({\n value: entry.value,\n label: entry.label,\n color: entry.color ?? null,\n icon: entry.icon ?? null,\n }))\n }\n const payload = await refetchCurrencyDictionary()\n return payload.entries.map((entry) => ({\n value: entry.value,\n label: entry.label,\n color: entry.color ?? null,\n icon: entry.icon ?? null,\n }))\n }, [currencyDictionary, refetchCurrencyDictionary])\n\n const loadItems = React.useCallback(async () => {\n setLoading(true)\n const loadErrorMessage = t('catalog.priceKinds.errors.load', 'Failed to load price kinds.')\n try {\n const payload = await readApiResultOrThrow<{ items?: PriceKindApiPayload[] }>(\n `/api/catalog/price-kinds?pageSize=${PAGE_SIZE}`,\n undefined,\n { errorMessage: loadErrorMessage },\n )\n const normalized = Array.isArray(payload.items) ? payload.items.map((item) => normalizePriceKind(item)) : []\n setItems(normalized)\n } catch (err) {\n console.error('catalog.price-kinds.list failed', err)\n flash(loadErrorMessage, 'error')\n } finally {\n setLoading(false)\n }\n }, [t])\n\n React.useEffect(() => {\n loadItems().catch(() => {})\n }, [loadItems, scopeVersion])\n\n const openDialog = React.useCallback((state: DialogState) => {\n if (state.mode === 'edit') {\n setForm({\n code: state.entry.code,\n title: state.entry.title,\n displayMode: state.entry.displayMode,\n currencyCode: state.entry.currencyCode ?? '',\n isPromotion: state.entry.isPromotion,\n isActive: state.entry.isActive,\n })\n } else {\n setForm(DEFAULT_FORM)\n }\n setError(null)\n setDialog(state)\n }, [])\n\n const closeDialog = React.useCallback(() => {\n setDialog(null)\n setError(null)\n setSubmitting(false)\n setForm(DEFAULT_FORM)\n }, [])\n\n const handleSubmit = React.useCallback(async () => {\n if (!dialog) return\n const trimmedCode = form.code.trim().toLowerCase()\n const trimmedTitle = form.title.trim()\n if (!trimmedCode || !trimmedTitle) {\n setError(t('catalog.priceKinds.errors.required', 'Code and title are required.'))\n return\n }\n setSubmitting(true)\n setError(null)\n try {\n const payload = {\n code: trimmedCode,\n title: trimmedTitle,\n displayMode: form.displayMode,\n currencyCode: form.currencyCode.trim() || undefined,\n isPromotion: form.isPromotion,\n isActive: form.isActive,\n }\n const path = '/api/catalog/price-kinds'\n const method = dialog.mode === 'create' ? 'POST' : 'PUT'\n const body =\n dialog.mode === 'edit'\n ? JSON.stringify({ id: dialog.entry.id, ...payload })\n : JSON.stringify(payload)\n const call = await apiCall(path, {\n method,\n headers: { 'content-type': 'application/json' },\n body,\n })\n if (!call.ok) {\n await raiseCrudError(call.response, t('catalog.priceKinds.errors.save', 'Failed to save price kind.'))\n }\n flash(\n dialog.mode === 'create'\n ? t('catalog.priceKinds.messages.created', 'Price kind created.')\n : t('catalog.priceKinds.messages.updated', 'Price kind updated.'),\n 'success',\n )\n closeDialog()\n await loadItems()\n } catch (err) {\n console.error('catalog.price-kinds.save failed', err)\n const message =\n err instanceof Error ? err.message : t('catalog.priceKinds.errors.save', 'Failed to save price kind.')\n setError(message)\n } finally {\n setSubmitting(false)\n }\n }, [dialog, form, t, closeDialog, loadItems])\n\n const handleDelete = React.useCallback(\n async (entry: PriceKind) => {\n const confirmMessage = t('catalog.priceKinds.confirm.delete', 'Delete price kind \"{{code}}\"?').replace('{{code}}', entry.code)\n const confirmed = await confirm({\n title: confirmMessage,\n variant: 'destructive',\n })\n if (!confirmed) return\n try {\n const call = await apiCall('/api/catalog/price-kinds', {\n method: 'DELETE',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify({ id: entry.id }),\n })\n if (!call.ok) {\n await raiseCrudError(call.response, t('catalog.priceKinds.errors.delete', 'Failed to delete price kind.'))\n }\n flash(t('catalog.priceKinds.messages.deleted', 'Price kind deleted.'), 'success')\n await loadItems()\n } catch (err) {\n console.error('catalog.price-kinds.delete failed', err)\n const message =\n err instanceof Error ? err.message : t('catalog.priceKinds.errors.delete', 'Failed to delete price kind.')\n flash(message, 'error')\n }\n },\n [confirm, loadItems, t],\n )\n\n const formKeyHandler = React.useCallback(\n (event: React.KeyboardEvent<HTMLFormElement>) => {\n if ((event.metaKey || event.ctrlKey) && event.key === 'Enter') {\n event.preventDefault()\n void handleSubmit()\n }\n },\n [handleSubmit],\n )\n\n const displayModeLabels = React.useMemo(() => ({\n 'including-tax': t('catalog.priceKinds.form.displayMode.include', 'Including tax'),\n 'excluding-tax': t('catalog.priceKinds.form.displayMode.exclude', 'Excluding tax'),\n }), [t])\n\n const displayModeOptions = React.useMemo(\n () =>\n DISPLAY_MODES.map((mode) => ({\n ...mode,\n label: displayModeLabels[mode.value],\n })),\n [displayModeLabels],\n )\n\n const currencyLabels = React.useMemo(() => ({\n placeholder: t('catalog.priceKinds.form.currency.placeholder', 'Select currency\u2026'),\n addLabel: t('catalog.priceKinds.form.currency.add', 'Add currency'),\n addPrompt: t('catalog.priceKinds.form.currency.addPrompt', 'Provide a currency code.'),\n dialogTitle: t('catalog.priceKinds.form.currency.dialogTitle', 'Add currency'),\n valueLabel: t('catalog.priceKinds.form.currency.valueLabel', 'Currency code'),\n valuePlaceholder: t('catalog.priceKinds.form.currency.valuePlaceholder', 'e.g. USD'),\n labelLabel: t('catalog.priceKinds.form.currency.labelLabel', 'Display label (optional)'),\n labelPlaceholder: t('catalog.priceKinds.form.currency.labelPlaceholder', 'e.g. US Dollar'),\n emptyError: t('catalog.priceKinds.form.currency.required', 'Currency code is required.'),\n cancelLabel: t('catalog.priceKinds.form.currency.cancel', 'Cancel'),\n saveLabel: t('catalog.priceKinds.form.currency.save', 'Save'),\n saveShortcutHint: t('catalog.priceKinds.form.currency.saveShortcut', 'Press Enter to save'),\n successCreateLabel: t('catalog.priceKinds.form.currency.success', 'Currency added.'),\n errorLoad: t('catalog.priceKinds.form.currency.loadError', 'Unable to load currencies.'),\n errorSave: t('catalog.priceKinds.form.currency.createError', 'Unable to add currency.'),\n loadingLabel: t('catalog.priceKinds.form.currency.loading', 'Loading currencies\u2026'),\n manageTitle: t('catalog.priceKinds.form.currency.manage', 'Manage currencies'),\n }), [t])\n\n const tableLabels = React.useMemo(() => ({\n code: t('catalog.priceKinds.table.code', 'Code'),\n title: t('catalog.priceKinds.table.title', 'Title'),\n displayMode: t('catalog.priceKinds.table.displayMode', 'Display mode'),\n currency: t('catalog.priceKinds.table.currency', 'Currency'),\n promotion: t('catalog.priceKinds.table.promotion', 'Promotion'),\n promotionYes: t('catalog.priceKinds.table.promotionYes', 'Yes'),\n promotionNo: t('catalog.priceKinds.table.promotionNo', 'No'),\n active: t('catalog.priceKinds.table.active', 'Active'),\n activeYes: t('catalog.priceKinds.table.activeYes', 'Active'),\n activeNo: t('catalog.priceKinds.table.activeNo', 'Inactive'),\n search: t('catalog.priceKinds.search.placeholder', 'Search by code or title\u2026'),\n empty: t('catalog.priceKinds.table.empty', 'No price kinds yet.'),\n }), [t])\n\n const columns = React.useMemo<ColumnDef<PriceKind>[]>(() => [\n {\n accessorKey: 'code',\n header: tableLabels.code,\n cell: ({ row }) => <span className=\"font-mono uppercase\">{row.original.code}</span>,\n },\n {\n accessorKey: 'title',\n header: tableLabels.title,\n cell: ({ row }) => <span className=\"font-medium\">{row.original.title}</span>,\n },\n {\n accessorKey: 'displayMode',\n header: tableLabels.displayMode,\n cell: ({ row }) => displayModeLabels[row.original.displayMode] ?? row.original.displayMode,\n },\n {\n accessorKey: 'currencyCode',\n header: tableLabels.currency,\n cell: ({ row }) => (row.original.currencyCode ? row.original.currencyCode.toUpperCase() : '\u2014'),\n },\n {\n id: 'promotion',\n header: tableLabels.promotion,\n cell: ({ row }) =>\n row.original.isPromotion ? (\n <span className=\"inline-flex items-center rounded-full border border-amber-200 bg-amber-50 px-2 py-0.5 text-xs font-medium text-amber-900 dark:border-amber-500/50 dark:bg-amber-500/10 dark:text-amber-100\">\n {tableLabels.promotionYes}\n </span>\n ) : (\n <span className=\"inline-flex items-center rounded-full border px-2 py-0.5 text-xs font-medium\">\n {tableLabels.promotionNo}\n </span>\n ),\n },\n {\n id: 'active',\n header: tableLabels.active,\n cell: ({ row }) =>\n row.original.isActive ? (\n <span className=\"inline-flex items-center rounded-full border border-emerald-200 bg-emerald-50 px-2 py-0.5 text-xs font-medium text-emerald-900 dark:border-emerald-500/50 dark:bg-emerald-500/10 dark:text-emerald-100\">\n {tableLabels.activeYes}\n </span>\n ) : (\n <span className=\"inline-flex items-center rounded-full border px-2 py-0.5 text-xs font-medium\">\n {tableLabels.activeNo}\n </span>\n ),\n },\n ], [displayModeLabels, tableLabels])\n\n const filteredItems = React.useMemo(() => {\n const term = search.trim().toLowerCase()\n if (!term) return items\n return items.filter((item) => {\n const currency = (item.currencyCode ?? '').toLowerCase()\n const modeLabel = (displayModeLabels[item.displayMode] ?? item.displayMode).toLowerCase()\n return (\n item.code.toLowerCase().includes(term) ||\n item.title.toLowerCase().includes(term) ||\n currency.includes(term) ||\n modeLabel.includes(term)\n )\n })\n }, [displayModeLabels, items, search])\n\n const handleRowClick = React.useCallback((entry: PriceKind) => {\n openDialog({ mode: 'edit', entry })\n }, [openDialog])\n\n return (\n <>\n <section className=\"border bg-card text-card-foreground shadow-sm\">\n <div className=\"border-b px-6 py-4 space-y-1\">\n <h2 className=\"text-lg font-semibold\">{t('catalog.priceKinds.title', 'Price kinds')}</h2>\n <p className=\"text-sm text-muted-foreground\">\n {t('catalog.priceKinds.description', 'Configure reusable price kinds that control pricing columns and tax display.')}\n </p>\n </div>\n <div className=\"px-2 py-4 sm:px-4\">\n <DataTable<PriceKind>\n data={filteredItems}\n columns={columns}\n embedded\n isLoading={loading}\n searchValue={search}\n onSearchChange={setSearch}\n searchPlaceholder={tableLabels.search}\n emptyState={<p className=\"py-8 text-center text-sm text-muted-foreground\">{tableLabels.empty}</p>}\n actions={(\n <Button size=\"sm\" onClick={() => openDialog({ mode: 'create' })}>\n {t('catalog.priceKinds.actions.add', 'Add price kind')}\n </Button>\n )}\n refreshButton={{\n label: t('catalog.priceKinds.actions.refresh', 'Refresh'),\n onRefresh: () => { void loadItems() },\n isRefreshing: loading,\n }}\n rowActions={(entry) => (\n <RowActions\n items={[\n {\n id: 'edit',\n label: t('catalog.priceKinds.actions.edit', 'Edit'),\n onSelect: () => openDialog({ mode: 'edit', entry }),\n },\n {\n id: 'delete',\n label: t('catalog.priceKinds.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => { void handleDelete(entry) },\n },\n ]}\n />\n )}\n onRowClick={handleRowClick}\n />\n </div>\n <Dialog open={dialog !== null} onOpenChange={(open) => { if (!open) closeDialog(); }}>\n <DialogContent>\n <DialogHeader>\n <DialogTitle>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.dialog.editTitle', 'Edit price kind')\n : t('catalog.priceKinds.dialog.createTitle', 'Create price kind')}\n </DialogTitle>\n <DialogDescription>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.dialog.editDescription', 'Update labels or tax behavior for this price kind.')\n : t('catalog.priceKinds.dialog.createDescription', 'Define a reusable price kind for product pricing.')}\n </DialogDescription>\n </DialogHeader>\n <form className=\"space-y-4\" onKeyDown={formKeyHandler} onSubmit={(event) => { event.preventDefault(); void handleSubmit() }}>\n <div className=\"space-y-2\">\n <Label htmlFor=\"price-kind-code\">{t('catalog.priceKinds.form.codeLabel', 'Code')}</Label>\n <Input\n id=\"price-kind-code\"\n value={form.code}\n onChange={(event) => setForm((prev) => ({ ...prev, code: event.target.value }))}\n placeholder={t('catalog.priceKinds.form.codePlaceholder', 'e.g. regular')}\n className=\"font-mono uppercase\"\n disabled={dialog?.mode === 'edit'}\n />\n </div>\n <div className=\"space-y-2\">\n <Label htmlFor=\"price-kind-title\">{t('catalog.priceKinds.form.titleLabel', 'Title')}</Label>\n <Input\n id=\"price-kind-title\"\n value={form.title}\n onChange={(event) => setForm((prev) => ({ ...prev, title: event.target.value }))}\n placeholder={t('catalog.priceKinds.form.titlePlaceholder', 'e.g. Regular price')}\n />\n </div>\n <div className=\"space-y-2\">\n <Label>{t('catalog.priceKinds.form.displayModeLabel', 'Display mode')}</Label>\n <div className=\"grid gap-2 md:grid-cols-2\">\n {displayModeOptions.map((mode) => (\n <label\n key={mode.value}\n className={`flex cursor-pointer items-center gap-2 rounded-md border p-3 text-sm ${\n form.displayMode === mode.value ? 'border-primary bg-primary/5' : 'border-border'\n }`}\n >\n <input\n type=\"radio\"\n name=\"displayMode\"\n value={mode.value}\n checked={form.displayMode === mode.value}\n onChange={() => setForm((prev) => ({ ...prev, displayMode: mode.value }))}\n />\n <span>{mode.label}</span>\n </label>\n ))}\n </div>\n </div>\n <div className=\"space-y-2\">\n <Label>{t('catalog.priceKinds.form.currencyLabel', 'Currency (optional)')}</Label>\n <DictionaryEntrySelect\n value={form.currencyCode || undefined}\n onChange={(value) => setForm((prev) => ({ ...prev, currencyCode: value ?? '' }))}\n fetchOptions={currencyOptionsLoader}\n labels={currencyLabels}\n allowInlineCreate={false}\n />\n </div>\n <div className=\"flex flex-col gap-2\">\n <label className=\"flex items-center gap-2 text-sm font-medium\">\n <input\n type=\"checkbox\"\n className=\"h-4 w-4 rounded border\"\n checked={form.isPromotion}\n onChange={(event) => setForm((prev) => ({ ...prev, isPromotion: event.target.checked }))}\n />\n {t('catalog.priceKinds.form.promotionLabel', 'Mark as promotion')}\n </label>\n <label className=\"flex items-center gap-2 text-sm font-medium\">\n <input\n type=\"checkbox\"\n className=\"h-4 w-4 rounded border\"\n checked={form.isActive}\n onChange={(event) => setForm((prev) => ({ ...prev, isActive: event.target.checked }))}\n />\n {t('catalog.priceKinds.form.activeLabel', 'Active')}\n </label>\n </div>\n {error ? <p className=\"text-sm text-red-600\">{error}</p> : null}\n </form>\n <DialogFooter>\n <Button variant=\"ghost\" onClick={closeDialog}>\n {t('catalog.priceKinds.actions.cancel', 'Cancel')}\n </Button>\n <Button onClick={() => void handleSubmit()} disabled={submitting}>\n {dialog?.mode === 'edit'\n ? t('catalog.priceKinds.actions.saveChanges', 'Save changes')\n : t('catalog.priceKinds.actions.create', 'Create')}\n </Button>\n </DialogFooter>\n </DialogContent>\n </Dialog>\n </section>\n {ConfirmDialogElement}\n </>\n )\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -136,7 +136,7 @@ const createProductUnitConversionDraft = (overrides = {}) => ({
|
|
|
136
136
|
});
|
|
137
137
|
const buildOptionValuesKey = (optionValues) => {
|
|
138
138
|
if (!optionValues) return "";
|
|
139
|
-
return Object.keys(optionValues).sort(
|
|
139
|
+
return Object.keys(optionValues).sort().map((key) => `${key}:${optionValues[key] ?? ""}`).join("|");
|
|
140
140
|
};
|
|
141
141
|
const haveSameOptionValues = (current, next) => {
|
|
142
142
|
const a = current ?? {};
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../../src/modules/catalog/components/products/productForm.ts"],
|
|
4
|
-
"sourcesContent": ["import { z } from \"zod\";\nimport { slugify } from \"@open-mercato/shared/lib/slugify\";\nimport { parseObjectLike } from \"@open-mercato/shared/lib/json/parseObjectLike\";\nimport type { ReferenceUnitCode } from \"@open-mercato/shared/lib/units/unitCodes\";\nimport type { CatalogProductOptionSchema } from \"../../data/types\";\nimport type { ProductMediaItem } from \"./ProductMediaManager\";\n\nexport { slugify };\n\nexport type PriceKindSummary = {\n id: string;\n code: string;\n title: string;\n currencyCode: string | null;\n displayMode: \"including-tax\" | \"excluding-tax\";\n};\n\nexport type PriceKindApiPayload = {\n id?: string | number;\n code?: string;\n title?: string;\n currencyCode?: string | null;\n currency_code?: string | null;\n displayMode?: string | null;\n display_mode?: string | null;\n};\n\nexport type TaxRateSummary = {\n id: string;\n name: string;\n code: string | null;\n rate: number | null;\n isDefault: boolean;\n};\n\nexport type ProductOptionInput = {\n id: string;\n title: string;\n values: Array<{ id: string; label: string }>;\n};\n\nexport type ProductDimensions = {\n width?: number;\n height?: number;\n depth?: number;\n unit?: string | null;\n} | null;\n\nexport type ProductWeight = {\n value?: number;\n unit?: string | null;\n} | null;\n\nexport type VariantPriceValue = {\n amount: string;\n};\n\nexport type ProductUnitRoundingMode = \"half_up\" | \"down\" | \"up\";\nexport type ProductUnitPriceReferenceUnit = ReferenceUnitCode;\n\nexport type ProductUnitConversionDraft = {\n id: string | null;\n unitCode: string;\n toBaseFactor: string;\n sortOrder: string;\n isActive: boolean;\n};\n\nexport type VariantDraft = {\n id: string;\n title: string;\n sku: string;\n isDefault: boolean;\n taxRateId: string | null;\n manageInventory: boolean;\n allowBackorder: boolean;\n hasInventoryKit: boolean;\n optionValues: Record<string, string>;\n prices: Record<string, VariantPriceValue>;\n};\n\nexport type ProductFormValues = {\n title: string;\n subtitle: string;\n handle: string;\n description: string;\n useMarkdown: boolean;\n taxRateId: string | null;\n mediaDraftId: string;\n mediaItems: ProductMediaItem[];\n defaultMediaId: string | null;\n defaultMediaUrl: string;\n hasVariants: boolean;\n options: ProductOptionInput[];\n variants: VariantDraft[];\n metadata?: Record<string, unknown> | null;\n dimensions?: ProductDimensions;\n weight?: ProductWeight;\n defaultUnit: string | null;\n defaultSalesUnit: string | null;\n defaultSalesUnitQuantity: string;\n uomRoundingScale: string;\n uomRoundingMode: ProductUnitRoundingMode;\n unitPriceEnabled: boolean;\n unitPriceReferenceUnit: string | null;\n unitPriceBaseQuantity: string;\n unitConversions: ProductUnitConversionDraft[];\n customFieldsetCode?: string | null;\n categoryIds: string[];\n channelIds: string[];\n tags: string[];\n optionSchemaId?: string | null;\n};\n\nconst optionalPositiveNumberInput = z.preprocess((value) => {\n if (value === null || value === undefined) return undefined;\n if (typeof value === \"string\" && value.trim().length === 0) return undefined;\n return value;\n}, z.coerce.number().positive().optional());\n\nconst optionalBoundedIntegerInput = (min: number, max: number) =>\n z.preprocess((value) => {\n if (value === null || value === undefined) return undefined;\n if (typeof value === \"string\" && value.trim().length === 0)\n return undefined;\n return value;\n }, z.coerce.number().int().min(min).max(max).optional());\n\nexport const productFormSchema = z\n .object({\n title: z.string().trim().min(1, \"catalog.products.validation.titleRequired\"),\n subtitle: z.string().optional(),\n handle: z\n .string()\n .trim()\n .regex(\n /^[a-z0-9\\-_]*$/,\n \"catalog.products.validation.handleFormat\",\n )\n .max(150)\n .optional(),\n description: z.string().optional(),\n useMarkdown: z.boolean().optional(),\n taxRateId: z.string().uuid().nullable().optional(),\n hasVariants: z.boolean().optional(),\n mediaDraftId: z.string().optional(),\n mediaItems: z.any().optional(),\n defaultMediaId: z.string().uuid().nullable().optional(),\n defaultMediaUrl: z.string().trim().max(500).nullable().optional(),\n options: z.any().optional(),\n variants: z.any().optional(),\n // Use a permissive schema to avoid zod classic `_zod` runtime crashes on records in edge builds.\n metadata: z\n .custom<Record<string, unknown>>(() => true)\n .nullable()\n .optional(),\n dimensions: z\n .object({\n width: z.coerce.number().min(0).optional(),\n height: z.coerce.number().min(0).optional(),\n depth: z.coerce.number().min(0).optional(),\n unit: z.string().trim().max(25).optional(),\n })\n .nullable()\n .optional(),\n weight: z\n .object({\n value: z.coerce.number().min(0).optional(),\n unit: z.string().trim().max(25).optional(),\n })\n .nullable()\n .optional(),\n defaultUnit: z.string().trim().max(50).nullable().optional(),\n defaultSalesUnit: z.string().trim().max(50).nullable().optional(),\n defaultSalesUnitQuantity: optionalPositiveNumberInput,\n uomRoundingScale: optionalBoundedIntegerInput(0, 6),\n uomRoundingMode: z.enum([\"half_up\", \"down\", \"up\"]).optional(),\n unitPriceEnabled: z.boolean().optional(),\n unitPriceReferenceUnit: z.string().trim().max(50).nullable().optional(),\n unitPriceBaseQuantity: optionalPositiveNumberInput,\n unitConversions: z\n .array(\n z.object({\n id: z.string().nullable().optional(),\n unitCode: z.string().trim().max(50),\n toBaseFactor: z.coerce.number().positive(),\n sortOrder: z.coerce.number().int().min(0).max(100000).optional(),\n isActive: z.boolean().optional(),\n }),\n )\n .optional(),\n customFieldsetCode: z.string().optional().nullable(),\n categoryIds: z.array(z.string().uuid()).optional(),\n channelIds: z.array(z.string().uuid()).optional(),\n tags: z.array(z.string().trim().min(1).max(100)).optional(),\n optionSchemaId: z.string().uuid().nullable().optional(),\n })\n .passthrough()\n .refine(\n (data) => !data.unitPriceEnabled || (data.unitPriceReferenceUnit != null && data.unitPriceReferenceUnit.length > 0),\n { message: 'catalog.products.validation.referenceUnitRequired', path: ['unitPriceReferenceUnit'] }\n )\n .refine(\n (data) => !data.defaultSalesUnit || (data.defaultUnit != null && data.defaultUnit.length > 0),\n { message: 'catalog.products.validation.baseUnitRequired', path: ['defaultUnit'] }\n );\n\nexport const PRODUCT_FORM_STEPS = [\n \"general\",\n \"organize\",\n \"uom\",\n \"variants\",\n] as const;\n\nexport const BASE_INITIAL_VALUES: ProductFormValues = {\n title: \"\",\n subtitle: \"\",\n handle: \"\",\n description: \"\",\n useMarkdown: false,\n mediaDraftId: \"\",\n mediaItems: [],\n defaultMediaId: null,\n defaultMediaUrl: \"\",\n taxRateId: null,\n hasVariants: false,\n options: [],\n variants: [],\n metadata: {},\n dimensions: null,\n weight: null,\n defaultUnit: null,\n defaultSalesUnit: null,\n defaultSalesUnitQuantity: \"1\",\n uomRoundingScale: \"4\",\n uomRoundingMode: \"half_up\",\n unitPriceEnabled: false,\n unitPriceReferenceUnit: null,\n unitPriceBaseQuantity: \"\",\n unitConversions: [],\n customFieldsetCode: null,\n categoryIds: [],\n channelIds: [],\n tags: [],\n optionSchemaId: null,\n};\n\nexport const createInitialProductFormValues = (): ProductFormValues => ({\n ...BASE_INITIAL_VALUES,\n mediaDraftId: createLocalId(),\n variants: [createVariantDraft(null, { isDefault: true })],\n});\n\nexport const createVariantDraft = (\n productTaxRateId: string | null,\n overrides: Partial<VariantDraft> = {},\n): VariantDraft => ({\n id: createLocalId(),\n title: \"Default variant\",\n sku: \"\",\n isDefault: false,\n taxRateId: productTaxRateId ?? null,\n manageInventory: false,\n allowBackorder: false,\n hasInventoryKit: false,\n optionValues: {},\n prices: {},\n ...overrides,\n});\n\nexport const createProductUnitConversionDraft = (\n overrides: Partial<ProductUnitConversionDraft> = {},\n): ProductUnitConversionDraft => ({\n id: null,\n unitCode: \"\",\n toBaseFactor: \"\",\n sortOrder: \"\",\n isActive: true,\n ...overrides,\n});\n\nexport const buildOptionValuesKey = (\n optionValues?: Record<string, string>,\n): string => {\n if (!optionValues) return \"\";\n return Object.keys(optionValues)\n .sort((a, b) => a.localeCompare(b))\n .map((key) => `${key}:${optionValues[key] ?? \"\"}`)\n .join(\"|\");\n};\n\nexport const haveSameOptionValues = (\n current: Record<string, string> | undefined,\n next: Record<string, string>,\n): boolean => {\n const a = current ?? {};\n const keys = new Set([...Object.keys(a), ...Object.keys(next)]);\n for (const key of keys) {\n if ((a[key] ?? \"\") !== (next[key] ?? \"\")) return false;\n }\n return true;\n};\n\nconst parseNumeric = (input: unknown): number | null => {\n const numeric = typeof input === \"number\" ? input : Number(input);\n if (!Number.isFinite(numeric) || numeric < 0) return null;\n return numeric;\n};\n\nexport const normalizeProductDimensions = (raw: unknown): ProductDimensions => {\n const source = parseObjectLike(raw);\n if (!source) return null;\n const width = parseNumeric(source.width);\n const height = parseNumeric(source.height);\n const depth = parseNumeric(source.depth);\n const unit =\n typeof source.unit === \"string\" && source.unit.trim().length\n ? source.unit.trim()\n : null;\n const clean: Record<string, unknown> = {};\n if (width !== null) clean.width = width;\n if (height !== null) clean.height = height;\n if (depth !== null) clean.depth = depth;\n if (unit) clean.unit = unit;\n return Object.keys(clean).length ? (clean as ProductDimensions) : null;\n};\n\nexport const normalizeProductWeight = (raw: unknown): ProductWeight => {\n const source = parseObjectLike(raw);\n if (!source) return null;\n const value = parseNumeric(source.value);\n const unit =\n typeof source.unit === \"string\" && source.unit.trim().length\n ? source.unit.trim()\n : null;\n if (value === null && !unit) return null;\n const clean: Record<string, unknown> = {};\n if (value !== null) clean.value = value;\n if (unit) clean.unit = unit;\n return clean as ProductWeight;\n};\n\nexport const sanitizeProductDimensions = (\n raw: ProductDimensions,\n): ProductDimensions => {\n return normalizeProductDimensions(raw ?? null);\n};\n\nexport const sanitizeProductWeight = (raw: ProductWeight): ProductWeight => {\n return normalizeProductWeight(raw ?? null);\n};\n\nexport const updateDimensionValue = (\n current: ProductDimensions,\n field: \"width\" | \"height\" | \"depth\" | \"unit\",\n raw: string,\n): ProductDimensions => {\n const base = normalizeProductDimensions(current) ?? {};\n if (field === \"unit\") {\n base.unit = raw;\n } else {\n const numeric = parseNumeric(raw);\n if (numeric === null) {\n delete base[field];\n } else {\n base[field] = numeric;\n }\n }\n return sanitizeProductDimensions(base);\n};\n\nexport const updateWeightValue = (\n current: ProductWeight,\n field: \"value\" | \"unit\",\n raw: string,\n): ProductWeight => {\n const base = normalizeProductWeight(current) ?? {};\n if (field === \"unit\") {\n base.unit = raw;\n } else {\n const numeric = parseNumeric(raw);\n if (numeric === null) {\n delete (base as Record<string, unknown>).value;\n } else {\n base.value = numeric;\n }\n }\n return sanitizeProductWeight(base);\n};\n\nexport const normalizePriceKindSummary = (\n input: PriceKindApiPayload | undefined | null,\n): PriceKindSummary | null => {\n if (!input) return null;\n const getString = (value: unknown): string | null => {\n if (typeof value === \"string\" && value.trim().length) return value.trim();\n if (typeof value === \"number\" || typeof value === \"bigint\")\n return String(value);\n return null;\n };\n const id = getString(input.id);\n const code = getString(input.code);\n const title = getString(input.title);\n if (!id || !code || !title) return null;\n const currency =\n getString(input.currencyCode) ?? getString(input.currency_code);\n const displayRaw =\n getString(input.displayMode) ?? getString(input.display_mode);\n const displayMode: PriceKindSummary[\"displayMode\"] =\n displayRaw === \"including-tax\" ? \"including-tax\" : \"excluding-tax\";\n return {\n id,\n code,\n title,\n currencyCode: currency,\n displayMode,\n };\n};\n\nexport const formatTaxRateLabel = (rate: TaxRateSummary): string => {\n const extras: string[] = [];\n if (typeof rate.rate === \"number\" && Number.isFinite(rate.rate)) {\n extras.push(`${rate.rate}%`);\n }\n if (rate.code) {\n extras.push(rate.code.toUpperCase());\n }\n if (!extras.length) return rate.name;\n return `${rate.name} \u2022 ${extras.join(\" \u00B7 \")}`;\n};\n\nexport function createLocalId(): string {\n return Math.random().toString(36).slice(2, 10);\n}\n\nexport function buildOptionSchemaDefinition(\n options: ProductOptionInput[] | undefined,\n name: string,\n): CatalogProductOptionSchema | null {\n const list = Array.isArray(options) ? options : [];\n if (!list.length) return null;\n const normalizedName =\n name && name.trim().length ? name.trim() : \"Product options\";\n const schemaOptions = list\n .map((option) => {\n const title = option.title?.trim() || \"\";\n const code = resolveOptionCode(option);\n const values = Array.isArray(option.values) ? option.values : [];\n return {\n code: code || slugify(createLocalId()),\n label: title || code || \"Option\",\n inputType: \"select\" as const,\n choices: values\n .map((value) => {\n const label = value.label?.trim() || \"\";\n const valueCode = slugify(label || value.id || createLocalId());\n if (!label && !valueCode) return null;\n return {\n code: valueCode || slugify(createLocalId()),\n label: label || valueCode || \"Choice\",\n };\n })\n .filter((entry): entry is { code: string; label: string } => !!entry),\n };\n })\n .filter((entry) => entry.label.trim().length);\n if (!schemaOptions.length) return null;\n return {\n version: 1,\n name: normalizedName,\n options: schemaOptions,\n };\n}\n\nexport function convertSchemaToProductOptions(\n schema: CatalogProductOptionSchema | null | undefined,\n): ProductOptionInput[] {\n if (!schema || !Array.isArray(schema.options)) return [];\n return schema.options.map((option) => ({\n id: createLocalId(),\n title: option.label ?? option.code ?? \"Option\",\n values: Array.isArray(option.choices)\n ? option.choices.map((choice) => ({\n id: createLocalId(),\n label: choice.label ?? choice.code ?? \"\",\n }))\n : [],\n }));\n}\n\nfunction resolveOptionCode(option: ProductOptionInput): string {\n const base = option.title?.trim() || option.id?.trim() || \"\";\n const slugged = slugify(base);\n if (slugged.length) return slugged;\n if (base.length) return base;\n return createLocalId();\n}\n\nexport function buildVariantCombinations(\n options: ProductOptionInput[],\n): Record<string, string>[] {\n if (!options.length) return [];\n const [first, ...rest] = options;\n if (!first || !Array.isArray(first.values) || !first.values.length) return [];\n const firstKey = resolveOptionCode(first);\n const initial = first.values.map((value) => ({ [firstKey]: value.label }));\n return rest.reduce<Record<string, string>[]>((acc, option) => {\n if (!Array.isArray(option.values) || !option.values.length) return [];\n const optionKey = resolveOptionCode(option);\n const combos: Record<string, string>[] = [];\n acc.forEach((partial) => {\n option.values.forEach((value) => {\n combos.push({ ...partial, [optionKey]: value.label });\n });\n });\n return combos;\n }, initial);\n}\n"],
|
|
5
|
-
"mappings": "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
|
|
4
|
+
"sourcesContent": ["import { z } from \"zod\";\nimport { slugify } from \"@open-mercato/shared/lib/slugify\";\nimport { parseObjectLike } from \"@open-mercato/shared/lib/json/parseObjectLike\";\nimport type { ReferenceUnitCode } from \"@open-mercato/shared/lib/units/unitCodes\";\nimport type { CatalogProductOptionSchema } from \"../../data/types\";\nimport type { ProductMediaItem } from \"./ProductMediaManager\";\n\nexport { slugify };\n\nexport type PriceKindSummary = {\n id: string;\n code: string;\n title: string;\n currencyCode: string | null;\n displayMode: \"including-tax\" | \"excluding-tax\";\n};\n\nexport type PriceKindApiPayload = {\n id?: string | number;\n code?: string;\n title?: string;\n currencyCode?: string | null;\n currency_code?: string | null;\n displayMode?: string | null;\n display_mode?: string | null;\n};\n\nexport type TaxRateSummary = {\n id: string;\n name: string;\n code: string | null;\n rate: number | null;\n isDefault: boolean;\n};\n\nexport type ProductOptionInput = {\n id: string;\n title: string;\n values: Array<{ id: string; label: string }>;\n};\n\nexport type ProductDimensions = {\n width?: number;\n height?: number;\n depth?: number;\n unit?: string | null;\n} | null;\n\nexport type ProductWeight = {\n value?: number;\n unit?: string | null;\n} | null;\n\nexport type VariantPriceValue = {\n amount: string;\n};\n\nexport type ProductUnitRoundingMode = \"half_up\" | \"down\" | \"up\";\nexport type ProductUnitPriceReferenceUnit = ReferenceUnitCode;\n\nexport type ProductUnitConversionDraft = {\n id: string | null;\n unitCode: string;\n toBaseFactor: string;\n sortOrder: string;\n isActive: boolean;\n};\n\nexport type VariantDraft = {\n id: string;\n title: string;\n sku: string;\n isDefault: boolean;\n taxRateId: string | null;\n manageInventory: boolean;\n allowBackorder: boolean;\n hasInventoryKit: boolean;\n optionValues: Record<string, string>;\n prices: Record<string, VariantPriceValue>;\n};\n\nexport type ProductFormValues = {\n title: string;\n subtitle: string;\n handle: string;\n description: string;\n useMarkdown: boolean;\n taxRateId: string | null;\n mediaDraftId: string;\n mediaItems: ProductMediaItem[];\n defaultMediaId: string | null;\n defaultMediaUrl: string;\n hasVariants: boolean;\n options: ProductOptionInput[];\n variants: VariantDraft[];\n metadata?: Record<string, unknown> | null;\n dimensions?: ProductDimensions;\n weight?: ProductWeight;\n defaultUnit: string | null;\n defaultSalesUnit: string | null;\n defaultSalesUnitQuantity: string;\n uomRoundingScale: string;\n uomRoundingMode: ProductUnitRoundingMode;\n unitPriceEnabled: boolean;\n unitPriceReferenceUnit: string | null;\n unitPriceBaseQuantity: string;\n unitConversions: ProductUnitConversionDraft[];\n customFieldsetCode?: string | null;\n categoryIds: string[];\n channelIds: string[];\n tags: string[];\n optionSchemaId?: string | null;\n};\n\nconst optionalPositiveNumberInput = z.preprocess((value) => {\n if (value === null || value === undefined) return undefined;\n if (typeof value === \"string\" && value.trim().length === 0) return undefined;\n return value;\n}, z.coerce.number().positive().optional());\n\nconst optionalBoundedIntegerInput = (min: number, max: number) =>\n z.preprocess((value) => {\n if (value === null || value === undefined) return undefined;\n if (typeof value === \"string\" && value.trim().length === 0)\n return undefined;\n return value;\n }, z.coerce.number().int().min(min).max(max).optional());\n\nexport const productFormSchema = z\n .object({\n title: z.string().trim().min(1, \"catalog.products.validation.titleRequired\"),\n subtitle: z.string().optional(),\n handle: z\n .string()\n .trim()\n .regex(\n /^[a-z0-9\\-_]*$/,\n \"catalog.products.validation.handleFormat\",\n )\n .max(150)\n .optional(),\n description: z.string().optional(),\n useMarkdown: z.boolean().optional(),\n taxRateId: z.string().uuid().nullable().optional(),\n hasVariants: z.boolean().optional(),\n mediaDraftId: z.string().optional(),\n mediaItems: z.any().optional(),\n defaultMediaId: z.string().uuid().nullable().optional(),\n defaultMediaUrl: z.string().trim().max(500).nullable().optional(),\n options: z.any().optional(),\n variants: z.any().optional(),\n // Use a permissive schema to avoid zod classic `_zod` runtime crashes on records in edge builds.\n metadata: z\n .custom<Record<string, unknown>>(() => true)\n .nullable()\n .optional(),\n dimensions: z\n .object({\n width: z.coerce.number().min(0).optional(),\n height: z.coerce.number().min(0).optional(),\n depth: z.coerce.number().min(0).optional(),\n unit: z.string().trim().max(25).optional(),\n })\n .nullable()\n .optional(),\n weight: z\n .object({\n value: z.coerce.number().min(0).optional(),\n unit: z.string().trim().max(25).optional(),\n })\n .nullable()\n .optional(),\n defaultUnit: z.string().trim().max(50).nullable().optional(),\n defaultSalesUnit: z.string().trim().max(50).nullable().optional(),\n defaultSalesUnitQuantity: optionalPositiveNumberInput,\n uomRoundingScale: optionalBoundedIntegerInput(0, 6),\n uomRoundingMode: z.enum([\"half_up\", \"down\", \"up\"]).optional(),\n unitPriceEnabled: z.boolean().optional(),\n unitPriceReferenceUnit: z.string().trim().max(50).nullable().optional(),\n unitPriceBaseQuantity: optionalPositiveNumberInput,\n unitConversions: z\n .array(\n z.object({\n id: z.string().nullable().optional(),\n unitCode: z.string().trim().max(50),\n toBaseFactor: z.coerce.number().positive(),\n sortOrder: z.coerce.number().int().min(0).max(100000).optional(),\n isActive: z.boolean().optional(),\n }),\n )\n .optional(),\n customFieldsetCode: z.string().optional().nullable(),\n categoryIds: z.array(z.string().uuid()).optional(),\n channelIds: z.array(z.string().uuid()).optional(),\n tags: z.array(z.string().trim().min(1).max(100)).optional(),\n optionSchemaId: z.string().uuid().nullable().optional(),\n })\n .passthrough()\n .refine(\n (data) => !data.unitPriceEnabled || (data.unitPriceReferenceUnit != null && data.unitPriceReferenceUnit.length > 0),\n { message: 'catalog.products.validation.referenceUnitRequired', path: ['unitPriceReferenceUnit'] }\n )\n .refine(\n (data) => !data.defaultSalesUnit || (data.defaultUnit != null && data.defaultUnit.length > 0),\n { message: 'catalog.products.validation.baseUnitRequired', path: ['defaultUnit'] }\n );\n\nexport const PRODUCT_FORM_STEPS = [\n \"general\",\n \"organize\",\n \"uom\",\n \"variants\",\n] as const;\n\nexport const BASE_INITIAL_VALUES: ProductFormValues = {\n title: \"\",\n subtitle: \"\",\n handle: \"\",\n description: \"\",\n useMarkdown: false,\n mediaDraftId: \"\",\n mediaItems: [],\n defaultMediaId: null,\n defaultMediaUrl: \"\",\n taxRateId: null,\n hasVariants: false,\n options: [],\n variants: [],\n metadata: {},\n dimensions: null,\n weight: null,\n defaultUnit: null,\n defaultSalesUnit: null,\n defaultSalesUnitQuantity: \"1\",\n uomRoundingScale: \"4\",\n uomRoundingMode: \"half_up\",\n unitPriceEnabled: false,\n unitPriceReferenceUnit: null,\n unitPriceBaseQuantity: \"\",\n unitConversions: [],\n customFieldsetCode: null,\n categoryIds: [],\n channelIds: [],\n tags: [],\n optionSchemaId: null,\n};\n\nexport const createInitialProductFormValues = (): ProductFormValues => ({\n ...BASE_INITIAL_VALUES,\n mediaDraftId: createLocalId(),\n variants: [createVariantDraft(null, { isDefault: true })],\n});\n\nexport const createVariantDraft = (\n productTaxRateId: string | null,\n overrides: Partial<VariantDraft> = {},\n): VariantDraft => ({\n id: createLocalId(),\n title: \"Default variant\",\n sku: \"\",\n isDefault: false,\n taxRateId: productTaxRateId ?? null,\n manageInventory: false,\n allowBackorder: false,\n hasInventoryKit: false,\n optionValues: {},\n prices: {},\n ...overrides,\n});\n\nexport const createProductUnitConversionDraft = (\n overrides: Partial<ProductUnitConversionDraft> = {},\n): ProductUnitConversionDraft => ({\n id: null,\n unitCode: \"\",\n toBaseFactor: \"\",\n sortOrder: \"\",\n isActive: true,\n ...overrides,\n});\n\nexport const buildOptionValuesKey = (\n optionValues?: Record<string, string>,\n): string => {\n if (!optionValues) return \"\";\n return Object.keys(optionValues)\n .sort()\n .map((key) => `${key}:${optionValues[key] ?? \"\"}`)\n .join(\"|\");\n};\n\nexport const haveSameOptionValues = (\n current: Record<string, string> | undefined,\n next: Record<string, string>,\n): boolean => {\n const a = current ?? {};\n const keys = new Set([...Object.keys(a), ...Object.keys(next)]);\n for (const key of keys) {\n if ((a[key] ?? \"\") !== (next[key] ?? \"\")) return false;\n }\n return true;\n};\n\nconst parseNumeric = (input: unknown): number | null => {\n const numeric = typeof input === \"number\" ? input : Number(input);\n if (!Number.isFinite(numeric) || numeric < 0) return null;\n return numeric;\n};\n\nexport const normalizeProductDimensions = (raw: unknown): ProductDimensions => {\n const source = parseObjectLike(raw);\n if (!source) return null;\n const width = parseNumeric(source.width);\n const height = parseNumeric(source.height);\n const depth = parseNumeric(source.depth);\n const unit =\n typeof source.unit === \"string\" && source.unit.trim().length\n ? source.unit.trim()\n : null;\n const clean: Record<string, unknown> = {};\n if (width !== null) clean.width = width;\n if (height !== null) clean.height = height;\n if (depth !== null) clean.depth = depth;\n if (unit) clean.unit = unit;\n return Object.keys(clean).length ? (clean as ProductDimensions) : null;\n};\n\nexport const normalizeProductWeight = (raw: unknown): ProductWeight => {\n const source = parseObjectLike(raw);\n if (!source) return null;\n const value = parseNumeric(source.value);\n const unit =\n typeof source.unit === \"string\" && source.unit.trim().length\n ? source.unit.trim()\n : null;\n if (value === null && !unit) return null;\n const clean: Record<string, unknown> = {};\n if (value !== null) clean.value = value;\n if (unit) clean.unit = unit;\n return clean as ProductWeight;\n};\n\nexport const sanitizeProductDimensions = (\n raw: ProductDimensions,\n): ProductDimensions => {\n return normalizeProductDimensions(raw ?? null);\n};\n\nexport const sanitizeProductWeight = (raw: ProductWeight): ProductWeight => {\n return normalizeProductWeight(raw ?? null);\n};\n\nexport const updateDimensionValue = (\n current: ProductDimensions,\n field: \"width\" | \"height\" | \"depth\" | \"unit\",\n raw: string,\n): ProductDimensions => {\n const base = normalizeProductDimensions(current) ?? {};\n if (field === \"unit\") {\n base.unit = raw;\n } else {\n const numeric = parseNumeric(raw);\n if (numeric === null) {\n delete base[field];\n } else {\n base[field] = numeric;\n }\n }\n return sanitizeProductDimensions(base);\n};\n\nexport const updateWeightValue = (\n current: ProductWeight,\n field: \"value\" | \"unit\",\n raw: string,\n): ProductWeight => {\n const base = normalizeProductWeight(current) ?? {};\n if (field === \"unit\") {\n base.unit = raw;\n } else {\n const numeric = parseNumeric(raw);\n if (numeric === null) {\n delete (base as Record<string, unknown>).value;\n } else {\n base.value = numeric;\n }\n }\n return sanitizeProductWeight(base);\n};\n\nexport const normalizePriceKindSummary = (\n input: PriceKindApiPayload | undefined | null,\n): PriceKindSummary | null => {\n if (!input) return null;\n const getString = (value: unknown): string | null => {\n if (typeof value === \"string\" && value.trim().length) return value.trim();\n if (typeof value === \"number\" || typeof value === \"bigint\")\n return String(value);\n return null;\n };\n const id = getString(input.id);\n const code = getString(input.code);\n const title = getString(input.title);\n if (!id || !code || !title) return null;\n const currency =\n getString(input.currencyCode) ?? getString(input.currency_code);\n const displayRaw =\n getString(input.displayMode) ?? getString(input.display_mode);\n const displayMode: PriceKindSummary[\"displayMode\"] =\n displayRaw === \"including-tax\" ? \"including-tax\" : \"excluding-tax\";\n return {\n id,\n code,\n title,\n currencyCode: currency,\n displayMode,\n };\n};\n\nexport const formatTaxRateLabel = (rate: TaxRateSummary): string => {\n const extras: string[] = [];\n if (typeof rate.rate === \"number\" && Number.isFinite(rate.rate)) {\n extras.push(`${rate.rate}%`);\n }\n if (rate.code) {\n extras.push(rate.code.toUpperCase());\n }\n if (!extras.length) return rate.name;\n return `${rate.name} \u2022 ${extras.join(\" \u00B7 \")}`;\n};\n\nexport function createLocalId(): string {\n return Math.random().toString(36).slice(2, 10);\n}\n\nexport function buildOptionSchemaDefinition(\n options: ProductOptionInput[] | undefined,\n name: string,\n): CatalogProductOptionSchema | null {\n const list = Array.isArray(options) ? options : [];\n if (!list.length) return null;\n const normalizedName =\n name && name.trim().length ? name.trim() : \"Product options\";\n const schemaOptions = list\n .map((option) => {\n const title = option.title?.trim() || \"\";\n const code = resolveOptionCode(option);\n const values = Array.isArray(option.values) ? option.values : [];\n return {\n code: code || slugify(createLocalId()),\n label: title || code || \"Option\",\n inputType: \"select\" as const,\n choices: values\n .map((value) => {\n const label = value.label?.trim() || \"\";\n const valueCode = slugify(label || value.id || createLocalId());\n if (!label && !valueCode) return null;\n return {\n code: valueCode || slugify(createLocalId()),\n label: label || valueCode || \"Choice\",\n };\n })\n .filter((entry): entry is { code: string; label: string } => !!entry),\n };\n })\n .filter((entry) => entry.label.trim().length);\n if (!schemaOptions.length) return null;\n return {\n version: 1,\n name: normalizedName,\n options: schemaOptions,\n };\n}\n\nexport function convertSchemaToProductOptions(\n schema: CatalogProductOptionSchema | null | undefined,\n): ProductOptionInput[] {\n if (!schema || !Array.isArray(schema.options)) return [];\n return schema.options.map((option) => ({\n id: createLocalId(),\n title: option.label ?? option.code ?? \"Option\",\n values: Array.isArray(option.choices)\n ? option.choices.map((choice) => ({\n id: createLocalId(),\n label: choice.label ?? choice.code ?? \"\",\n }))\n : [],\n }));\n}\n\nfunction resolveOptionCode(option: ProductOptionInput): string {\n const base = option.title?.trim() || option.id?.trim() || \"\";\n const slugged = slugify(base);\n if (slugged.length) return slugged;\n if (base.length) return base;\n return createLocalId();\n}\n\nexport function buildVariantCombinations(\n options: ProductOptionInput[],\n): Record<string, string>[] {\n if (!options.length) return [];\n const [first, ...rest] = options;\n if (!first || !Array.isArray(first.values) || !first.values.length) return [];\n const firstKey = resolveOptionCode(first);\n const initial = first.values.map((value) => ({ [firstKey]: value.label }));\n return rest.reduce<Record<string, string>[]>((acc, option) => {\n if (!Array.isArray(option.values) || !option.values.length) return [];\n const optionKey = resolveOptionCode(option);\n const combos: Record<string, string>[] = [];\n acc.forEach((partial) => {\n option.values.forEach((value) => {\n combos.push({ ...partial, [optionKey]: value.label });\n });\n });\n return combos;\n }, initial);\n}\n"],
|
|
5
|
+
"mappings": "AAAA,SAAS,SAAS;AAClB,SAAS,eAAe;AACxB,SAAS,uBAAuB;AAgHhC,MAAM,8BAA8B,EAAE,WAAW,CAAC,UAAU;AAC1D,MAAI,UAAU,QAAQ,UAAU,OAAW,QAAO;AAClD,MAAI,OAAO,UAAU,YAAY,MAAM,KAAK,EAAE,WAAW,EAAG,QAAO;AACnE,SAAO;AACT,GAAG,EAAE,OAAO,OAAO,EAAE,SAAS,EAAE,SAAS,CAAC;AAE1C,MAAM,8BAA8B,CAAC,KAAa,QAChD,EAAE,WAAW,CAAC,UAAU;AACtB,MAAI,UAAU,QAAQ,UAAU,OAAW,QAAO;AAClD,MAAI,OAAO,UAAU,YAAY,MAAM,KAAK,EAAE,WAAW;AACvD,WAAO;AACT,SAAO;AACT,GAAG,EAAE,OAAO,OAAO,EAAE,IAAI,EAAE,IAAI,GAAG,EAAE,IAAI,GAAG,EAAE,SAAS,CAAC;AAElD,MAAM,oBAAoB,EAC9B,OAAO;AAAA,EACN,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,GAAG,2CAA2C;AAAA,EAC3E,UAAU,EAAE,OAAO,EAAE,SAAS;AAAA,EAC9B,QAAQ,EACL,OAAO,EACP,KAAK,EACL;AAAA,IACC;AAAA,IACA;AAAA,EACF,EACC,IAAI,GAAG,EACP,SAAS;AAAA,EACZ,aAAa,EAAE,OAAO,EAAE,SAAS;AAAA,EACjC,aAAa,EAAE,QAAQ,EAAE,SAAS;AAAA,EAClC,WAAW,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,EACjD,aAAa,EAAE,QAAQ,EAAE,SAAS;AAAA,EAClC,cAAc,EAAE,OAAO,EAAE,SAAS;AAAA,EAClC,YAAY,EAAE,IAAI,EAAE,SAAS;AAAA,EAC7B,gBAAgB,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,EACtD,iBAAiB,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,GAAG,EAAE,SAAS,EAAE,SAAS;AAAA,EAChE,SAAS,EAAE,IAAI,EAAE,SAAS;AAAA,EAC1B,UAAU,EAAE,IAAI,EAAE,SAAS;AAAA;AAAA,EAE3B,UAAU,EACP,OAAgC,MAAM,IAAI,EAC1C,SAAS,EACT,SAAS;AAAA,EACZ,YAAY,EACT,OAAO;AAAA,IACN,OAAO,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,SAAS;AAAA,IACzC,QAAQ,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,SAAS;AAAA,IAC1C,OAAO,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,SAAS;AAAA,IACzC,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,SAAS;AAAA,EAC3C,CAAC,EACA,SAAS,EACT,SAAS;AAAA,EACZ,QAAQ,EACL,OAAO;AAAA,IACN,OAAO,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,SAAS;AAAA,IACzC,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,SAAS;AAAA,EAC3C,CAAC,EACA,SAAS,EACT,SAAS;AAAA,EACZ,aAAa,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,SAAS,EAAE,SAAS;AAAA,EAC3D,kBAAkB,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,SAAS,EAAE,SAAS;AAAA,EAChE,0BAA0B;AAAA,EAC1B,kBAAkB,4BAA4B,GAAG,CAAC;AAAA,EAClD,iBAAiB,EAAE,KAAK,CAAC,WAAW,QAAQ,IAAI,CAAC,EAAE,SAAS;AAAA,EAC5D,kBAAkB,EAAE,QAAQ,EAAE,SAAS;AAAA,EACvC,wBAAwB,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,SAAS,EAAE,SAAS;AAAA,EACtE,uBAAuB;AAAA,EACvB,iBAAiB,EACd;AAAA,IACC,EAAE,OAAO;AAAA,MACP,IAAI,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,MACnC,UAAU,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE;AAAA,MAClC,cAAc,EAAE,OAAO,OAAO,EAAE,SAAS;AAAA,MACzC,WAAW,EAAE,OAAO,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,IAAI,GAAM,EAAE,SAAS;AAAA,MAC/D,UAAU,EAAE,QAAQ,EAAE,SAAS;AAAA,IACjC,CAAC;AAAA,EACH,EACC,SAAS;AAAA,EACZ,oBAAoB,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,EACnD,aAAa,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,CAAC,EAAE,SAAS;AAAA,EACjD,YAAY,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,CAAC,EAAE,SAAS;AAAA,EAChD,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,CAAC,EAAE,IAAI,GAAG,CAAC,EAAE,SAAS;AAAA,EAC1D,gBAAgB,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AACxD,CAAC,EACA,YAAY,EACZ;AAAA,EACC,CAAC,SAAS,CAAC,KAAK,oBAAqB,KAAK,0BAA0B,QAAQ,KAAK,uBAAuB,SAAS;AAAA,EACjH,EAAE,SAAS,qDAAqD,MAAM,CAAC,wBAAwB,EAAE;AACnG,EACC;AAAA,EACC,CAAC,SAAS,CAAC,KAAK,oBAAqB,KAAK,eAAe,QAAQ,KAAK,YAAY,SAAS;AAAA,EAC3F,EAAE,SAAS,gDAAgD,MAAM,CAAC,aAAa,EAAE;AACnF;AAEK,MAAM,qBAAqB;AAAA,EAChC;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;AAEO,MAAM,sBAAyC;AAAA,EACpD,OAAO;AAAA,EACP,UAAU;AAAA,EACV,QAAQ;AAAA,EACR,aAAa;AAAA,EACb,aAAa;AAAA,EACb,cAAc;AAAA,EACd,YAAY,CAAC;AAAA,EACb,gBAAgB;AAAA,EAChB,iBAAiB;AAAA,EACjB,WAAW;AAAA,EACX,aAAa;AAAA,EACb,SAAS,CAAC;AAAA,EACV,UAAU,CAAC;AAAA,EACX,UAAU,CAAC;AAAA,EACX,YAAY;AAAA,EACZ,QAAQ;AAAA,EACR,aAAa;AAAA,EACb,kBAAkB;AAAA,EAClB,0BAA0B;AAAA,EAC1B,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,wBAAwB;AAAA,EACxB,uBAAuB;AAAA,EACvB,iBAAiB,CAAC;AAAA,EAClB,oBAAoB;AAAA,EACpB,aAAa,CAAC;AAAA,EACd,YAAY,CAAC;AAAA,EACb,MAAM,CAAC;AAAA,EACP,gBAAgB;AAClB;AAEO,MAAM,iCAAiC,OAA0B;AAAA,EACtE,GAAG;AAAA,EACH,cAAc,cAAc;AAAA,EAC5B,UAAU,CAAC,mBAAmB,MAAM,EAAE,WAAW,KAAK,CAAC,CAAC;AAC1D;AAEO,MAAM,qBAAqB,CAChC,kBACA,YAAmC,CAAC,OAClB;AAAA,EAClB,IAAI,cAAc;AAAA,EAClB,OAAO;AAAA,EACP,KAAK;AAAA,EACL,WAAW;AAAA,EACX,WAAW,oBAAoB;AAAA,EAC/B,iBAAiB;AAAA,EACjB,gBAAgB;AAAA,EAChB,iBAAiB;AAAA,EACjB,cAAc,CAAC;AAAA,EACf,QAAQ,CAAC;AAAA,EACT,GAAG;AACL;AAEO,MAAM,mCAAmC,CAC9C,YAAiD,CAAC,OAClB;AAAA,EAChC,IAAI;AAAA,EACJ,UAAU;AAAA,EACV,cAAc;AAAA,EACd,WAAW;AAAA,EACX,UAAU;AAAA,EACV,GAAG;AACL;AAEO,MAAM,uBAAuB,CAClC,iBACW;AACX,MAAI,CAAC,aAAc,QAAO;AAC1B,SAAO,OAAO,KAAK,YAAY,EAC5B,KAAK,EACL,IAAI,CAAC,QAAQ,GAAG,GAAG,IAAI,aAAa,GAAG,KAAK,EAAE,EAAE,EAChD,KAAK,GAAG;AACb;AAEO,MAAM,uBAAuB,CAClC,SACA,SACY;AACZ,QAAM,IAAI,WAAW,CAAC;AACtB,QAAM,OAAO,oBAAI,IAAI,CAAC,GAAG,OAAO,KAAK,CAAC,GAAG,GAAG,OAAO,KAAK,IAAI,CAAC,CAAC;AAC9D,aAAW,OAAO,MAAM;AACtB,SAAK,EAAE,GAAG,KAAK,SAAS,KAAK,GAAG,KAAK,IAAK,QAAO;AAAA,EACnD;AACA,SAAO;AACT;AAEA,MAAM,eAAe,CAAC,UAAkC;AACtD,QAAM,UAAU,OAAO,UAAU,WAAW,QAAQ,OAAO,KAAK;AAChE,MAAI,CAAC,OAAO,SAAS,OAAO,KAAK,UAAU,EAAG,QAAO;AACrD,SAAO;AACT;AAEO,MAAM,6BAA6B,CAAC,QAAoC;AAC7E,QAAM,SAAS,gBAAgB,GAAG;AAClC,MAAI,CAAC,OAAQ,QAAO;AACpB,QAAM,QAAQ,aAAa,OAAO,KAAK;AACvC,QAAM,SAAS,aAAa,OAAO,MAAM;AACzC,QAAM,QAAQ,aAAa,OAAO,KAAK;AACvC,QAAM,OACJ,OAAO,OAAO,SAAS,YAAY,OAAO,KAAK,KAAK,EAAE,SAClD,OAAO,KAAK,KAAK,IACjB;AACN,QAAM,QAAiC,CAAC;AACxC,MAAI,UAAU,KAAM,OAAM,QAAQ;AAClC,MAAI,WAAW,KAAM,OAAM,SAAS;AACpC,MAAI,UAAU,KAAM,OAAM,QAAQ;AAClC,MAAI,KAAM,OAAM,OAAO;AACvB,SAAO,OAAO,KAAK,KAAK,EAAE,SAAU,QAA8B;AACpE;AAEO,MAAM,yBAAyB,CAAC,QAAgC;AACrE,QAAM,SAAS,gBAAgB,GAAG;AAClC,MAAI,CAAC,OAAQ,QAAO;AACpB,QAAM,QAAQ,aAAa,OAAO,KAAK;AACvC,QAAM,OACJ,OAAO,OAAO,SAAS,YAAY,OAAO,KAAK,KAAK,EAAE,SAClD,OAAO,KAAK,KAAK,IACjB;AACN,MAAI,UAAU,QAAQ,CAAC,KAAM,QAAO;AACpC,QAAM,QAAiC,CAAC;AACxC,MAAI,UAAU,KAAM,OAAM,QAAQ;AAClC,MAAI,KAAM,OAAM,OAAO;AACvB,SAAO;AACT;AAEO,MAAM,4BAA4B,CACvC,QACsB;AACtB,SAAO,2BAA2B,OAAO,IAAI;AAC/C;AAEO,MAAM,wBAAwB,CAAC,QAAsC;AAC1E,SAAO,uBAAuB,OAAO,IAAI;AAC3C;AAEO,MAAM,uBAAuB,CAClC,SACA,OACA,QACsB;AACtB,QAAM,OAAO,2BAA2B,OAAO,KAAK,CAAC;AACrD,MAAI,UAAU,QAAQ;AACpB,SAAK,OAAO;AAAA,EACd,OAAO;AACL,UAAM,UAAU,aAAa,GAAG;AAChC,QAAI,YAAY,MAAM;AACpB,aAAO,KAAK,KAAK;AAAA,IACnB,OAAO;AACL,WAAK,KAAK,IAAI;AAAA,IAChB;AAAA,EACF;AACA,SAAO,0BAA0B,IAAI;AACvC;AAEO,MAAM,oBAAoB,CAC/B,SACA,OACA,QACkB;AAClB,QAAM,OAAO,uBAAuB,OAAO,KAAK,CAAC;AACjD,MAAI,UAAU,QAAQ;AACpB,SAAK,OAAO;AAAA,EACd,OAAO;AACL,UAAM,UAAU,aAAa,GAAG;AAChC,QAAI,YAAY,MAAM;AACpB,aAAQ,KAAiC;AAAA,IAC3C,OAAO;AACL,WAAK,QAAQ;AAAA,IACf;AAAA,EACF;AACA,SAAO,sBAAsB,IAAI;AACnC;AAEO,MAAM,4BAA4B,CACvC,UAC4B;AAC5B,MAAI,CAAC,MAAO,QAAO;AACnB,QAAM,YAAY,CAAC,UAAkC;AACnD,QAAI,OAAO,UAAU,YAAY,MAAM,KAAK,EAAE,OAAQ,QAAO,MAAM,KAAK;AACxE,QAAI,OAAO,UAAU,YAAY,OAAO,UAAU;AAChD,aAAO,OAAO,KAAK;AACrB,WAAO;AAAA,EACT;AACA,QAAM,KAAK,UAAU,MAAM,EAAE;AAC7B,QAAM,OAAO,UAAU,MAAM,IAAI;AACjC,QAAM,QAAQ,UAAU,MAAM,KAAK;AACnC,MAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAO,QAAO;AACnC,QAAM,WACJ,UAAU,MAAM,YAAY,KAAK,UAAU,MAAM,aAAa;AAChE,QAAM,aACJ,UAAU,MAAM,WAAW,KAAK,UAAU,MAAM,YAAY;AAC9D,QAAM,cACJ,eAAe,kBAAkB,kBAAkB;AACrD,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA,cAAc;AAAA,IACd;AAAA,EACF;AACF;AAEO,MAAM,qBAAqB,CAAC,SAAiC;AAClE,QAAM,SAAmB,CAAC;AAC1B,MAAI,OAAO,KAAK,SAAS,YAAY,OAAO,SAAS,KAAK,IAAI,GAAG;AAC/D,WAAO,KAAK,GAAG,KAAK,IAAI,GAAG;AAAA,EAC7B;AACA,MAAI,KAAK,MAAM;AACb,WAAO,KAAK,KAAK,KAAK,YAAY,CAAC;AAAA,EACrC;AACA,MAAI,CAAC,OAAO,OAAQ,QAAO,KAAK;AAChC,SAAO,GAAG,KAAK,IAAI,WAAM,OAAO,KAAK,QAAK,CAAC;AAC7C;AAEO,SAAS,gBAAwB;AACtC,SAAO,KAAK,OAAO,EAAE,SAAS,EAAE,EAAE,MAAM,GAAG,EAAE;AAC/C;AAEO,SAAS,4BACd,SACA,MACmC;AACnC,QAAM,OAAO,MAAM,QAAQ,OAAO,IAAI,UAAU,CAAC;AACjD,MAAI,CAAC,KAAK,OAAQ,QAAO;AACzB,QAAM,iBACJ,QAAQ,KAAK,KAAK,EAAE,SAAS,KAAK,KAAK,IAAI;AAC7C,QAAM,gBAAgB,KACnB,IAAI,CAAC,WAAW;AACf,UAAM,QAAQ,OAAO,OAAO,KAAK,KAAK;AACtC,UAAM,OAAO,kBAAkB,MAAM;AACrC,UAAM,SAAS,MAAM,QAAQ,OAAO,MAAM,IAAI,OAAO,SAAS,CAAC;AAC/D,WAAO;AAAA,MACL,MAAM,QAAQ,QAAQ,cAAc,CAAC;AAAA,MACrC,OAAO,SAAS,QAAQ;AAAA,MACxB,WAAW;AAAA,MACX,SAAS,OACN,IAAI,CAAC,UAAU;AACd,cAAM,QAAQ,MAAM,OAAO,KAAK,KAAK;AACrC,cAAM,YAAY,QAAQ,SAAS,MAAM,MAAM,cAAc,CAAC;AAC9D,YAAI,CAAC,SAAS,CAAC,UAAW,QAAO;AACjC,eAAO;AAAA,UACL,MAAM,aAAa,QAAQ,cAAc,CAAC;AAAA,UAC1C,OAAO,SAAS,aAAa;AAAA,QAC/B;AAAA,MACF,CAAC,EACA,OAAO,CAAC,UAAoD,CAAC,CAAC,KAAK;AAAA,IACxE;AAAA,EACF,CAAC,EACA,OAAO,CAAC,UAAU,MAAM,MAAM,KAAK,EAAE,MAAM;AAC9C,MAAI,CAAC,cAAc,OAAQ,QAAO;AAClC,SAAO;AAAA,IACL,SAAS;AAAA,IACT,MAAM;AAAA,IACN,SAAS;AAAA,EACX;AACF;AAEO,SAAS,8BACd,QACsB;AACtB,MAAI,CAAC,UAAU,CAAC,MAAM,QAAQ,OAAO,OAAO,EAAG,QAAO,CAAC;AACvD,SAAO,OAAO,QAAQ,IAAI,CAAC,YAAY;AAAA,IACrC,IAAI,cAAc;AAAA,IAClB,OAAO,OAAO,SAAS,OAAO,QAAQ;AAAA,IACtC,QAAQ,MAAM,QAAQ,OAAO,OAAO,IAChC,OAAO,QAAQ,IAAI,CAAC,YAAY;AAAA,MAC9B,IAAI,cAAc;AAAA,MAClB,OAAO,OAAO,SAAS,OAAO,QAAQ;AAAA,IACxC,EAAE,IACF,CAAC;AAAA,EACP,EAAE;AACJ;AAEA,SAAS,kBAAkB,QAAoC;AAC7D,QAAM,OAAO,OAAO,OAAO,KAAK,KAAK,OAAO,IAAI,KAAK,KAAK;AAC1D,QAAM,UAAU,QAAQ,IAAI;AAC5B,MAAI,QAAQ,OAAQ,QAAO;AAC3B,MAAI,KAAK,OAAQ,QAAO;AACxB,SAAO,cAAc;AACvB;AAEO,SAAS,yBACd,SAC0B;AAC1B,MAAI,CAAC,QAAQ,OAAQ,QAAO,CAAC;AAC7B,QAAM,CAAC,OAAO,GAAG,IAAI,IAAI;AACzB,MAAI,CAAC,SAAS,CAAC,MAAM,QAAQ,MAAM,MAAM,KAAK,CAAC,MAAM,OAAO,OAAQ,QAAO,CAAC;AAC5E,QAAM,WAAW,kBAAkB,KAAK;AACxC,QAAM,UAAU,MAAM,OAAO,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,GAAG,MAAM,MAAM,EAAE;AACzE,SAAO,KAAK,OAAiC,CAAC,KAAK,WAAW;AAC5D,QAAI,CAAC,MAAM,QAAQ,OAAO,MAAM,KAAK,CAAC,OAAO,OAAO,OAAQ,QAAO,CAAC;AACpE,UAAM,YAAY,kBAAkB,MAAM;AAC1C,UAAM,SAAmC,CAAC;AAC1C,QAAI,QAAQ,CAAC,YAAY;AACvB,aAAO,OAAO,QAAQ,CAAC,UAAU;AAC/B,eAAO,KAAK,EAAE,GAAG,SAAS,CAAC,SAAS,GAAG,MAAM,MAAM,CAAC;AAAA,MACtD,CAAC;AAAA,IACH,CAAC;AACD,WAAO;AAAA,EACT,GAAG,OAAO;AACZ;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/configs/lib/upgrade-actions.ts"],
|
|
4
|
-
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport * as semver from 'semver'\nimport type { AppContainer } from '@open-mercato/shared/lib/di/container'\n\nexport type UpgradeActionContext = {\n tenantId: string\n organizationId: string\n container: AppContainer\n em: EntityManager\n}\n\nexport type UpgradeActionDefinition = {\n id: string\n version: string\n messageKey: string\n ctaKey: string\n successKey: string\n loadingKey?: string\n run: (ctx: UpgradeActionContext) => Promise<void>\n}\n\n/**\n * Compare two semantic version strings.\n * Uses the semver library for robust version comparison.\n * Returns negative if a < b, positive if a > b, 0 if equal.\n * Throws an error if either version string is invalid.\n */\nexport function compareVersions(a: string, b: string): number {\n const cleanA = semver.valid(semver.coerce(a))\n const cleanB = semver.valid(semver.coerce(b))\n if (!cleanA) {\n throw new Error(`Invalid version string: \"${a}\". Expected a valid semver format (e.g., \"1.2.3\").`)\n }\n if (!cleanB) {\n throw new Error(`Invalid version string: \"${b}\". Expected a valid semver format (e.g., \"1.2.3\").`)\n }\n return semver.compare(cleanA, cleanB)\n}\n\nexport const upgradeActions: UpgradeActionDefinition[] = []\n\nexport function actionsUpToVersion(version: string): UpgradeActionDefinition[] {\n return upgradeActions
|
|
4
|
+
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport * as semver from 'semver'\nimport type { AppContainer } from '@open-mercato/shared/lib/di/container'\n\nexport type UpgradeActionContext = {\n tenantId: string\n organizationId: string\n container: AppContainer\n em: EntityManager\n}\n\nexport type UpgradeActionDefinition = {\n id: string\n version: string\n messageKey: string\n ctaKey: string\n successKey: string\n loadingKey?: string\n run: (ctx: UpgradeActionContext) => Promise<void>\n}\n\n/**\n * Compare two semantic version strings.\n * Uses the semver library for robust version comparison.\n * Returns negative if a < b, positive if a > b, 0 if equal.\n * Throws an error if either version string is invalid.\n */\nexport function compareVersions(a: string, b: string): number {\n const cleanA = semver.valid(semver.coerce(a))\n const cleanB = semver.valid(semver.coerce(b))\n if (!cleanA) {\n throw new Error(`Invalid version string: \"${a}\". Expected a valid semver format (e.g., \"1.2.3\").`)\n }\n if (!cleanB) {\n throw new Error(`Invalid version string: \"${b}\". Expected a valid semver format (e.g., \"1.2.3\").`)\n }\n return semver.compare(cleanA, cleanB)\n}\n\nexport const upgradeActions: UpgradeActionDefinition[] = []\n\nexport function actionsUpToVersion(version: string): UpgradeActionDefinition[] {\n return upgradeActions\n .filter((action) => compareVersions(action.version, version) <= 0)\n .sort((a, b) => compareVersions(a.version, b.version) || a.id.localeCompare(b.id))\n}\n\nexport function findUpgradeAction(actionId: string, maxVersion: string): UpgradeActionDefinition | undefined {\n const matches = actionsUpToVersion(maxVersion).filter((action) => action.id === actionId)\n if (!matches.length) return undefined\n return matches[matches.length - 1]\n}\n"],
|
|
5
5
|
"mappings": "AACA,YAAY,YAAY;AA0BjB,SAAS,gBAAgB,GAAW,GAAmB;AAC5D,QAAM,SAAS,OAAO,MAAM,OAAO,OAAO,CAAC,CAAC;AAC5C,QAAM,SAAS,OAAO,MAAM,OAAO,OAAO,CAAC,CAAC;AAC5C,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,4BAA4B,CAAC,oDAAoD;AAAA,EACnG;AACA,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,4BAA4B,CAAC,oDAAoD;AAAA,EACnG;AACA,SAAO,OAAO,QAAQ,QAAQ,MAAM;AACtC;AAEO,MAAM,iBAA4C,CAAC;AAEnD,SAAS,mBAAmB,SAA4C;AAC7E,SAAO,eACJ,OAAO,CAAC,WAAW,gBAAgB,OAAO,SAAS,OAAO,KAAK,CAAC,EAChE,KAAK,CAAC,GAAG,MAAM,gBAAgB,EAAE,SAAS,EAAE,OAAO,KAAK,EAAE,GAAG,cAAc,EAAE,EAAE,CAAC;AACrF;AAEO,SAAS,kBAAkB,UAAkB,YAAyD;AAC3G,QAAM,UAAU,mBAAmB,UAAU,EAAE,OAAO,CAAC,WAAW,OAAO,OAAO,QAAQ;AACxF,MAAI,CAAC,QAAQ,OAAQ,QAAO;AAC5B,SAAO,QAAQ,QAAQ,SAAS,CAAC;AACnC;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -31,7 +31,7 @@ class RaiffeisenPolandProvider {
|
|
|
31
31
|
if (!data.rates || Object.keys(data.rates).length === 0) {
|
|
32
32
|
return [];
|
|
33
33
|
}
|
|
34
|
-
const times = Object.keys(data.rates).sort(
|
|
34
|
+
const times = Object.keys(data.rates).sort();
|
|
35
35
|
const firstTime = times[0];
|
|
36
36
|
const rates = data.rates[firstTime];
|
|
37
37
|
const results = [];
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../../src/modules/currencies/services/providers/raiffeisen.ts"],
|
|
4
|
-
"sourcesContent": ["import { RateProvider, RateProviderResult } from './base'\nimport { fromZonedTime } from 'date-fns-tz'\n\ninterface RaiffeisenResponse {\n date: string\n rates: Record<\n string,\n Array<{\n code: string\n units: number\n buy: string\n sell: string\n spread: string\n date: string\n time: string\n }>\n >\n range: {\n minRateDate: string\n maxRateDate: string\n }\n}\n\n// \nexport class RaiffeisenPolandProvider implements RateProvider {\n readonly name = 'Raiffeisen Bank Polska'\n readonly source = 'Raiffeisen Bank Polska'\n readonly providerBaseCurrency = 'PLN'\n\n private readonly baseUrl = 'https://www.rbinternational.com.pl/rest/rates/'\n\n isAvailable(): boolean {\n return true\n }\n\n async fetchRates(\n date: Date,\n scope: { tenantId: string; organizationId: string },\n availableCurrencies: Set<string>\n ): Promise<RateProviderResult[]> {\n // Check if PLN is available (required as base currency for Raiffeisen)\n if (!availableCurrencies.has(this.providerBaseCurrency)) {\n console.debug('[Raiffeisen] Skipping: PLN not found in available currencies')\n return []\n }\n\n const dateStr = this.formatDate(date)\n const url = `${this.baseUrl}?type=kursywalut&range=all&date=${dateStr}`\n\n try {\n const response = await fetch(url)\n\n if (!response.ok) {\n if (response.status === 404) {\n console.log(`[Raiffeisen] No data available for ${dateStr}`)\n return []\n }\n throw new Error(\n `Raiffeisen API error: ${response.status} ${response.statusText}`\n )\n }\n\n const data: RaiffeisenResponse = await response.json()\n\n if (!data.rates || Object.keys(data.rates).length === 0) {\n return []\n }\n\n // Get FIRST time slot (opening rates)\n const times = Object.keys(data.rates).sort(
|
|
5
|
-
"mappings": "AACA,SAAS,qBAAqB;AAuBvB,MAAM,yBAAiD;AAAA,EAAvD;AACL,SAAS,OAAO;AAChB,SAAS,SAAS;AAClB,SAAS,uBAAuB;AAEhC,SAAiB,UAAU;AAAA;AAAA,EAE3B,cAAuB;AACrB,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,WACJ,MACA,OACA,qBAC+B;AAE/B,QAAI,CAAC,oBAAoB,IAAI,KAAK,oBAAoB,GAAG;AACvD,cAAQ,MAAM,8DAA8D;AAC5E,aAAO,CAAC;AAAA,IACV;AAEA,UAAM,UAAU,KAAK,WAAW,IAAI;AACpC,UAAM,MAAM,GAAG,KAAK,OAAO,mCAAmC,OAAO;AAErE,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,GAAG;AAEhC,UAAI,CAAC,SAAS,IAAI;AAChB,YAAI,SAAS,WAAW,KAAK;AAC3B,kBAAQ,IAAI,sCAAsC,OAAO,EAAE;AAC3D,iBAAO,CAAC;AAAA,QACV;AACA,cAAM,IAAI;AAAA,UACR,yBAAyB,SAAS,MAAM,IAAI,SAAS,UAAU;AAAA,QACjE;AAAA,MACF;AAEA,YAAM,OAA2B,MAAM,SAAS,KAAK;AAErD,UAAI,CAAC,KAAK,SAAS,OAAO,KAAK,KAAK,KAAK,EAAE,WAAW,GAAG;AACvD,eAAO,CAAC;AAAA,MACV;AAGA,YAAM,QAAQ,OAAO,KAAK,KAAK,KAAK,EAAE,KAAK
|
|
4
|
+
"sourcesContent": ["import { RateProvider, RateProviderResult } from './base'\nimport { fromZonedTime } from 'date-fns-tz'\n\ninterface RaiffeisenResponse {\n date: string\n rates: Record<\n string,\n Array<{\n code: string\n units: number\n buy: string\n sell: string\n spread: string\n date: string\n time: string\n }>\n >\n range: {\n minRateDate: string\n maxRateDate: string\n }\n}\n\n// \nexport class RaiffeisenPolandProvider implements RateProvider {\n readonly name = 'Raiffeisen Bank Polska'\n readonly source = 'Raiffeisen Bank Polska'\n readonly providerBaseCurrency = 'PLN'\n\n private readonly baseUrl = 'https://www.rbinternational.com.pl/rest/rates/'\n\n isAvailable(): boolean {\n return true\n }\n\n async fetchRates(\n date: Date,\n scope: { tenantId: string; organizationId: string },\n availableCurrencies: Set<string>\n ): Promise<RateProviderResult[]> {\n // Check if PLN is available (required as base currency for Raiffeisen)\n if (!availableCurrencies.has(this.providerBaseCurrency)) {\n console.debug('[Raiffeisen] Skipping: PLN not found in available currencies')\n return []\n }\n\n const dateStr = this.formatDate(date)\n const url = `${this.baseUrl}?type=kursywalut&range=all&date=${dateStr}`\n\n try {\n const response = await fetch(url)\n\n if (!response.ok) {\n if (response.status === 404) {\n console.log(`[Raiffeisen] No data available for ${dateStr}`)\n return []\n }\n throw new Error(\n `Raiffeisen API error: ${response.status} ${response.statusText}`\n )\n }\n\n const data: RaiffeisenResponse = await response.json()\n\n if (!data.rates || Object.keys(data.rates).length === 0) {\n return []\n }\n\n // Get FIRST time slot (opening rates)\n const times = Object.keys(data.rates).sort()\n const firstTime = times[0]\n const rates = data.rates[firstTime]\n\n const results: RateProviderResult[] = []\n \n // Extract time from first rate entry (all rates in same time slot have same time)\n const firstRate = rates[0]\n if (!firstRate) {\n console.log(`[Raiffeisen] No rates available in time slot ${firstTime}`)\n return []\n }\n\n // Combine date + time and parse as Europe/Warsaw timezone\n const rateDate = fromZonedTime(\n `${data.date} ${firstRate.time}`,\n 'Europe/Warsaw'\n )\n\n for (const rateData of rates) {\n // Raiffeisen rates are from bank's perspective:\n // - SELL: bank sells XXX for PLN \u2192 1 XXX = sell PLN\n // - BUY: bank buys XXX for PLN \u2192 1 XXX = buy PLN\n \n // Rate 1: PLN \u2192 XXX (inverse of SELL) - this is when bank SELLS foreign currency\n // If sell = 4.5 (1 EUR costs 4.5 PLN), then 1 PLN = 1/4.5 EUR\n const sellRate = parseFloat(rateData.sell)\n results.push({\n fromCurrencyCode: this.providerBaseCurrency,\n toCurrencyCode: rateData.code,\n rate: (1 / sellRate).toString(),\n source: this.source,\n date: rateDate,\n type: 'sell', // Bank sells foreign currency (from their perspective)\n })\n\n // Rate 2: XXX \u2192 PLN (using BUY) - this is when bank BUYS foreign currency\n // If buy = 4.3 (1 EUR gives 4.3 PLN), then 1 EUR = 4.3 PLN\n results.push({\n fromCurrencyCode: rateData.code,\n toCurrencyCode: this.providerBaseCurrency,\n rate: rateData.buy,\n source: this.source,\n date: rateDate,\n type: 'buy', // Bank buys foreign currency (from their perspective)\n })\n }\n\n console.log(\n `[Raiffeisen] Fetched ${results.length} rates for ${dateStr} at ${firstRate.time} Warsaw time (${rateDate.toISOString()})`\n )\n return results\n } catch (err: any) {\n console.error(`[Raiffeisen] Fetch error for ${dateStr}:`, err)\n throw new Error(`Failed to fetch Raiffeisen rates: ${err.message}`)\n }\n }\n\n private formatDate(date: Date): string {\n return date.toISOString().split('T')[0] // YYYY-MM-DD\n }\n}\n"],
|
|
5
|
+
"mappings": "AACA,SAAS,qBAAqB;AAuBvB,MAAM,yBAAiD;AAAA,EAAvD;AACL,SAAS,OAAO;AAChB,SAAS,SAAS;AAClB,SAAS,uBAAuB;AAEhC,SAAiB,UAAU;AAAA;AAAA,EAE3B,cAAuB;AACrB,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,WACJ,MACA,OACA,qBAC+B;AAE/B,QAAI,CAAC,oBAAoB,IAAI,KAAK,oBAAoB,GAAG;AACvD,cAAQ,MAAM,8DAA8D;AAC5E,aAAO,CAAC;AAAA,IACV;AAEA,UAAM,UAAU,KAAK,WAAW,IAAI;AACpC,UAAM,MAAM,GAAG,KAAK,OAAO,mCAAmC,OAAO;AAErE,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,GAAG;AAEhC,UAAI,CAAC,SAAS,IAAI;AAChB,YAAI,SAAS,WAAW,KAAK;AAC3B,kBAAQ,IAAI,sCAAsC,OAAO,EAAE;AAC3D,iBAAO,CAAC;AAAA,QACV;AACA,cAAM,IAAI;AAAA,UACR,yBAAyB,SAAS,MAAM,IAAI,SAAS,UAAU;AAAA,QACjE;AAAA,MACF;AAEA,YAAM,OAA2B,MAAM,SAAS,KAAK;AAErD,UAAI,CAAC,KAAK,SAAS,OAAO,KAAK,KAAK,KAAK,EAAE,WAAW,GAAG;AACvD,eAAO,CAAC;AAAA,MACV;AAGA,YAAM,QAAQ,OAAO,KAAK,KAAK,KAAK,EAAE,KAAK;AAC3C,YAAM,YAAY,MAAM,CAAC;AACzB,YAAM,QAAQ,KAAK,MAAM,SAAS;AAElC,YAAM,UAAgC,CAAC;AAGvC,YAAM,YAAY,MAAM,CAAC;AACzB,UAAI,CAAC,WAAW;AACd,gBAAQ,IAAI,gDAAgD,SAAS,EAAE;AACvE,eAAO,CAAC;AAAA,MACV;AAGA,YAAM,WAAW;AAAA,QACf,GAAG,KAAK,IAAI,IAAI,UAAU,IAAI;AAAA,QAC9B;AAAA,MACF;AAEA,iBAAW,YAAY,OAAO;AAO5B,cAAM,WAAW,WAAW,SAAS,IAAI;AACzC,gBAAQ,KAAK;AAAA,UACX,kBAAkB,KAAK;AAAA,UACvB,gBAAgB,SAAS;AAAA,UACzB,OAAO,IAAI,UAAU,SAAS;AAAA,UAC9B,QAAQ,KAAK;AAAA,UACb,MAAM;AAAA,UACN,MAAM;AAAA;AAAA,QACR,CAAC;AAID,gBAAQ,KAAK;AAAA,UACX,kBAAkB,SAAS;AAAA,UAC3B,gBAAgB,KAAK;AAAA,UACrB,MAAM,SAAS;AAAA,UACf,QAAQ,KAAK;AAAA,UACb,MAAM;AAAA,UACN,MAAM;AAAA;AAAA,QACR,CAAC;AAAA,MACH;AAEA,cAAQ;AAAA,QACN,wBAAwB,QAAQ,MAAM,cAAc,OAAO,OAAO,UAAU,IAAI,iBAAiB,SAAS,YAAY,CAAC;AAAA,MACzH;AACA,aAAO;AAAA,IACT,SAAS,KAAU;AACjB,cAAQ,MAAM,gCAAgC,OAAO,KAAK,GAAG;AAC7D,YAAM,IAAI,MAAM,qCAAqC,IAAI,OAAO,EAAE;AAAA,IACpE;AAAA,EACF;AAAA,EAEQ,WAAW,MAAoB;AACrC,WAAO,KAAK,YAAY,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,EACxC;AACF;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -4,7 +4,7 @@ import * as React from "react";
|
|
|
4
4
|
import Link from "next/link";
|
|
5
5
|
import { useRouter } from "next/navigation";
|
|
6
6
|
import { Page, PageBody } from "@open-mercato/ui/backend/Page";
|
|
7
|
-
import { DataTable
|
|
7
|
+
import { DataTable } from "@open-mercato/ui/backend/DataTable";
|
|
8
8
|
import { Button } from "@open-mercato/ui/primitives/button";
|
|
9
9
|
import { RowActions } from "@open-mercato/ui/backend/RowActions";
|
|
10
10
|
import { apiCall, apiCallOrThrow, readApiResultOrThrow } from "@open-mercato/ui/backend/utils/apiCall";
|
|
@@ -52,7 +52,7 @@ function mapApiItem(item) {
|
|
|
52
52
|
customFields[key] = value;
|
|
53
53
|
}
|
|
54
54
|
}
|
|
55
|
-
return
|
|
55
|
+
return {
|
|
56
56
|
id,
|
|
57
57
|
name,
|
|
58
58
|
description,
|
|
@@ -67,7 +67,7 @@ function mapApiItem(item) {
|
|
|
67
67
|
organizationId,
|
|
68
68
|
source,
|
|
69
69
|
...customFields
|
|
70
|
-
}
|
|
70
|
+
};
|
|
71
71
|
}
|
|
72
72
|
function CustomersCompaniesPage() {
|
|
73
73
|
const { confirm, ConfirmDialogElement } = useConfirmDialog();
|